TypeError: (0 , react__WEBPACK_IMPORTED_MODULE_0__.cache) is not a function
Answered
Yellowstripe scad posted this in #help-forum
Yellowstripe scadOP
Hello! I'm having a problem with React's cache() function. I'm creating a project in Next.js 14 with App Router and Firebase, and in it I perform interactions with the bank in server actions, in which I use cache() to reduce the number of requests to the bank.
In the same application, I'm using Stripe to process payments, and I have an API route in which I have a webhook and other routes related to Stripe. Previously, I was performing several actions directly in the webhook, including using several actions that use cache in this API code. Everything was working fine. So, I decided to refactor the webhook, and I created a services folder with an index.js file that contains other functions, one for each type of Stripe event that can occur.
The problem is that in this new file, I get the error:
And the error occurs when importing the actions in the file (api/stripe/webhook/services/index.js). I tried moving this logic to other places, such as the src/services folder, but that didn't work either. I even tried keeping the functions in the webhook file, but the error persisted.
At this point, I don't know what else to try. If you need to provide more details, I'm at your disposal! I really appreciate any attempt to help.
In the same application, I'm using Stripe to process payments, and I have an API route in which I have a webhook and other routes related to Stripe. Previously, I was performing several actions directly in the webhook, including using several actions that use cache in this API code. Everything was working fine. So, I decided to refactor the webhook, and I created a services folder with an index.js file that contains other functions, one for each type of Stripe event that can occur.
The problem is that in this new file, I get the error:
TypeError: (0 , react__WEBPACK_IMPORTED_MODULE_0__.cache) is not a function
And the error occurs when importing the actions in the file (api/stripe/webhook/services/index.js). I tried moving this logic to other places, such as the src/services folder, but that didn't work either. I even tried keeping the functions in the webhook file, but the error persisted.
At this point, I don't know what else to try. If you need to provide more details, I'm at your disposal! I really appreciate any attempt to help.
Answered by Yellowstripe scad
I just moved this object to the webhook file, and imported the handlers.js functions directly into it, now the cache is being recognized with a function
93 Replies
Is this a route file you’re trying to create? It’s a somewhat precarious place to have an index.js file in nextjs
You should place your logic into a lib folder and not within the apps routing mechanism.
That’s my first suggestion. Can you also share the code in the index.js file referenced? And where you are calling it? You can give a generic example as to not include any of your data.
That error typically indicates your using something meant for the server in the client or something meant for the client on the server. Typically but not always.
@Jboncz Is this a route file you’re trying to create? It’s a somewhat precarious place to have an index.js file in nextjs
Yellowstripe scadOP
I'm not trying to create a route, I just used index.js as a convention (which may not be the best, actually).
Since it's something related to the API, I thought it would make more sense to keep it inside the api folder, or not necessarily?
In index.js, I have functions like these:
Since it's something related to the API, I thought it would make more sense to keep it inside the api folder, or not necessarily?
In index.js, I have functions like these:
export async function handleCheckoutCompleted(stripe, event) {
const session = event.data.object
const subscription = await stripe.subscriptions.retrieve(session.subscription)
const items = JSON.parse(session.metadata.items)
items.forEach((item) => {
item.ref = getDocRef(item.id, item_COLLECTION)
delete item.id
})
const order = await createOrder(
{
items,
subscriptionId: subscription.id,
status: subscription.status
},
session.metadata.userId
)
await stripe.subscriptions.update(subscription.id, {
metadata: {
orderId: order.orderId
}
})
}
And where are you calling said function?
Yellowstripe scadOP
In the webhook:
The STRIPE_EVENTS_HANDLERS is this object:
export const STRIPE_EVENTS_HANDLERS = {
'checkout.session.completed': handleCheckoutCompleted,
'invoice.payment_succeeded': handlePaymentSucceeded,
'invoice.payment_failed': handlePaymentFailed,
'customer.subscription.updated': handleSubscriptionUpdated,
'customer.subscription.deleted': handleSubscriptionDeleted
}
import { STRIPE_EVENTS_HANDLERS } from '@/store/constants'
import Stripe from 'stripe'
export async function POST(req) {
const stripe = new Stripe(process.env.STRIPE_SECRET_KEY)
const sig = req.headers.get('stripe-signature')
const payload = await req.text()
const endpointSecret = process.env.STRIPE_WEBHOOK_SECRET
try {
const event = stripe.webhooks.constructEvent(payload, sig, endpointSecret)
const handlerFunction = STRIPE_EVENTS_HANDLERS[event.type]
if (handlerFunction) await handlerFunction(stripe, event)
return new Response('Webhook handled', { status: 200 })
} catch (err) {
console.error('Webhook Error:', err.message)
return new Response(`Webhook Error: ${err.message}`, { status: 400 })
}
}
The STRIPE_EVENTS_HANDLERS is this object:
export const STRIPE_EVENTS_HANDLERS = {
'checkout.session.completed': handleCheckoutCompleted,
'invoice.payment_succeeded': handlePaymentSucceeded,
'invoice.payment_failed': handlePaymentFailed,
'customer.subscription.updated': handleSubscriptionUpdated,
'customer.subscription.deleted': handleSubscriptionDeleted
}
Index.js is abhorrently bade for organizations sake. But that’s an opinion, idk if there is anything bad that can happen from it but it’s just not great because it doesn’t tell you what the file is index.js is known as an “entry point” that’s why people use that in other environments.
You’re calling it as a function? But it’s in []s?
@Jboncz You’re calling it as a function? But it’s in []s?
Yellowstripe scadOP
Yes, event.type is a string, so I can select the object property based on the string, and each property has a specific function associated with it
like STRIPE_EVENTS_HANDLERS[ 'checkout.session.completed']
@Jboncz Index.js is abhorrently bade for organizations sake. But that’s an opinion, idk if there is anything bad that can happen from it but it’s just not great because it doesn’t tell you what the file is index.js is known as an “entry point” that’s why people use that in other environments.
Yellowstripe scadOP
I see, that makes sense. About the lib folder, I used it to make connections with Firebase and for additional logic in a utils file. Would you use it for larger logic like these functions as well?
@Atlantic horse mackerel you have used stripe what is this sorcery? nvm I see what hes doing... I just was trying to read on my phone and was not a good idea 🤣
I would never locate my logic inside of the routing system unless it’s a server action that’s part of nextjs call it stripe-utils.js or something.
I actually prefer using library as my folder name but industry standard is lib for whatever reason lol
Also where are you using cache?
@Jboncz Also where are you using cache?
Yellowstripe scadOP
Im using cache in the server actions
specially get actions
Give me some code to work with 🙂 Where you are using cache
@Yellowstripe scad hmm ok
Its mostly up to the user, but I just find it hard to locate things, this way I know where everything is example:
'use server'
import { cache } from 'react'
// Memoize a database query or expensive computation
const cachedQuery = cache(async (id) => {
// Database query or other expensive operation
return await db.findById(id)
})
export async function myServerAction(id) {
// The result will be cached based on the id parameter
const result = await cachedQuery(id)
// Do something with the result
return result
}
Should be something like this, very generic example
Yellowstripe scadOP
export const getCurrentGoal = cache(async () => {
try {
const collectionRef = collection(db, GOALS_COLLECTION)
const docSnap = await getDocs(collectionRef)
if (!docSnap.empty) {
const goalData = docSnap.docs[0].data()
if (goalData.validCauses) {
goalData.validCauses = await Promise.all(
goalData.validCauses.map(async (cause) => {
const causeData = await resolveDocument(cause)
return {
id: causeData.id,
name: causeData.name
}
})
)
}
return goalData
} else {
return null
}
} catch (error) {
console.error('Error fetching goals:', error)
throw error
}
})
This is an example of how I'm using the cache
@Jboncz Its mostly up to the user, but I just find it hard to locate things, this way I know where everything is example:
Yellowstripe scadOP
cool, I found a good structure, I was using services for something similar
What really bothered me about this was the fact that everything was working until I decided to refactor the webhook, and I just don't understand technically what's going on
And the file 100% has
use server
at the top right?Yellowstripe scadOP
yes 😅
Alright, can you show me the full error? Should be more to the error stack than what was provided in OP
export const getCurrentGoal = cache(async () => {
try {
return 'boo'
}
} catch (error) {
console.error('Error fetching goals:', error)
throw error
}
})
Switch this into your current code and see if it returns the same issue... that will better tell me where the issue is. Reverse engineer till you know exactly where the issue is.
Yellowstripe scadOP
Give me the call stack, hit show collapsed frames
@Jboncz Give me the call stack, hit show collapsed frames
Yellowstripe scadOP
oh ok
one last sanity check.... you have
import { cache } from 'react';
right?@Jboncz export const getCurrentGoal = cache(async () => {
try {
return 'boo'
}
} catch (error) {
console.error('Error fetching goals:', error)
throw error
}
})
Switch this into your current code and see if it returns the same issue... that will better tell me where the issue is. Reverse engineer till you know exactly where the issue is.
Yellowstripe scadOP
I can't even change the code of the functions to test because the error is in the import, so the functions don't even actually run
@Jboncz one last sanity check.... you have js
import { cache } from 'react';
right?
Yellowstripe scadOP
yes
Are you importing the function into middleware?
Yellowstripe scadOP
The cache works correctly in other places, such as in components for example
If I comment out the index.js file (the one with the handler functions), the project compiles normally
If I comment out the index.js file (the one with the handler functions), the project compiles normally
Wait wait, I just saw the stack you gave, your importing that into middleware yeah?
@Jboncz Are you importing the function into middleware?
Yellowstripe scadOP
No, this is my middleware:
import { NextResponse } from 'next/server'
import {
SIGN_UP_ROUTE,
SIGN_IN_ROUTE,
APP_ROUTE,
SESSION_COOKIE
} from './store/constants'
export default function middleware(request) {
const session = request.cookies.get(SESSION_COOKIE)?.value || ''
const pathname = request.nextUrl.pathname
if (!session && pathname.startsWith(APP_ROUTE)) {
const absoluteURL = new URL(SIGN_IN_ROUTE, request.nextUrl.origin)
return NextResponse.redirect(absoluteURL.toString())
}
if (session && (pathname === SIGN_IN_ROUTE || pathname === SIGN_UP_ROUTE)) {
const absoluteURL = new URL(APP_ROUTE, request.nextUrl.origin)
return NextResponse.redirect(absoluteURL.toString())
}
}
Hmmmmmm.... something just isnt adding up lol
Yellowstripe scadOP
yeah, I found it really strange too
How can it compile properly if the index.js file is gone completely? There are things importing from there, so it would break?
Shouldnt compile if it cant resolve the import at compilation
Can you please rename it to something other than index.js? Just in case there is some really weird functionality with that lol
@Jboncz How can it compile properly if the index.js file is gone completely? There are things importing from there, so it would break?
Yellowstripe scadOP
Yes, there are things imported from index.js, but for some reason the project compiles normally
same error
I tried handlers.js
Okay. one sec, lemme replicate.
Yellowstripe scadOP
ok thanks
Can you show me the file where your importing the server action?
can redact what you need, just need structure
@Jboncz Can you show me the file where your importing the server action?
Yellowstripe scadOP
the old index.js right? ok
No, where you are importing the getCurrentGoal function to
Yellowstripe scadOP
import {
getCurrentGoal
} from '@/app/actions/goals'
In index.js I'm importing all the actions like this, as well as in the other files where I use the actions
is that what you wanted? I didnt quite understand
okay.... I think we are getting to the root of the issue.
Where are you importing the stuff from index.js?
Yellowstripe scadOP
src/store/constants.js, to create this object:
export const STRIPE_EVENTS_HANDLERS = {
'checkout.session.completed': handleCheckoutCompleted,
'invoice.payment_succeeded': handlePaymentSucceeded,
'invoice.payment_failed': handlePaymentFailed,
'customer.subscription.updated': handleSubscriptionUpdated,
'customer.subscription.deleted': handleSubscriptionDeleted
}
import {
handleCheckoutCompleted,
handlePaymentFailed,
handlePaymentSucceeded,
handleSubscriptionDeleted,
handleSubscriptionUpdated
} from '@/app/api/stripe/webhook/services/handlers'
handleCheckoutCompleted,
handlePaymentFailed,
handlePaymentSucceeded,
handleSubscriptionDeleted,
handleSubscriptionUpdated
} from '@/app/api/stripe/webhook/services/handlers'
Yellowstripe scadOP
getCurrentGoal is an action that I use inside these functions
to get the data I need to handle Stripe payment events
I import it into the old index.js, as well as other actions
actually, I just realized something important
If I remove the imports of these functions from the store, the error also disappears, maybe in src/store I don't have access to the cache at the time it is compiled and this generates the error?
apparently it worked!!!
In my opinion your over complicating your setup you have. If that works great though lol
It seems really really complicated lol.
Yellowstripe scadOP
I just moved this object to the webhook file, and imported the handlers.js functions directly into it, now the cache is being recognized with a function
Answer
Hey if it works it works.... 😅
@Jboncz In my opinion your over complicating your setup you have. If that works great though lol
Yellowstripe scadOP
So, I'm using actions to interact with the database, like a CRUD
I use this object only to list the events that I need to handle and to know which functions will handle each event
I use actions in these handlers, but in other places too, like to display things in interfaces
I use this object only to list the events that I need to handle and to know which functions will handle each event
I use actions in these handlers, but in other places too, like to display things in interfaces
thanks for your help, it is always good to reconstruct reasoning with the help of a different point of view.
Sure thing, just makr whatever fixed it as your answer, your welcome to mark your own comment as an answer lol
Yellowstripe scadOP
can you tell me why I can't import a function that uses cache() in a folder like src/store?
But remember this
Your caching deeply nested data, so idk how or why its reacting the way it is
its ahrd to tell
@Jboncz You can.... This works perfectly fine
Yellowstripe scadOP
It's a little different from the structure I have, getCurrentGoal is an action in src/app/actions and then there would be another function that calls the action itself (but I don't know if that would generate a very different result from the structure you created)
@Jboncz Click to see attachment
Yellowstripe scadOP
hmm ok
Anyway, it worked
Thanks again for taking the time to help
Thanks again for taking the time to help
Yeah, without having it in front of me its hard to diagnose properly lol
@Jboncz Yeah, without having it in front of me its hard to diagnose properly lol
Yellowstripe scadOP
yeah
Dont forget to mark an answer to keep the forums clean 🙂
ty ty
Yellowstripe scadOP
nice
good night, thx again
np