@nexo-labs/payload-stripe-inventory 1.6.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.mjs","names":["prices","customers","existingCustomer: Customer | null","customer","products","headers","getHeaders","metadata: Stripe.MetadataParam","flowData: Stripe.BillingPortal.SessionCreateParams.FlowData | undefined","metadata: Stripe.MetadataParam","isAdmin: Access","isAnyone: Access","isAdminOrCurrentUser: Access","isAdminOrPublished: Access","isAdminOrStripeActive: Access","isAdminOrUserFieldMatchingCurrentUser: Access","loggedInOrPublished: Access","customers: CollectionConfig","permissionEvaluationField: Field","prices: CollectionConfig","products: CollectionConfig"],"sources":["../../src/server/utils/payload/upsert.ts","../../src/server/utils/stripe/stripe-builder.ts","../../src/server/actions/price.ts","../../src/server/actions/product.ts","../../src/server/utils/payload/sync-customer-by-email.ts","../../src/server/utils/payload/upsert-customer-inventory-and-sync-with-user.ts","../../src/server/utils/payload/get-userId-by-email.ts","../../src/server/utils/stripe/get-customer.ts","../../src/server/utils/payload/remove-customer-by-stripe-id.ts","../../src/server/utils/payload/find-or-create-customer.ts","../../src/server/actions/subscription.ts","../../src/server/actions/donation.ts","../../src/server/actions/invoice.ts","../../src/server/actions/customer.ts","../../src/server/access/get-current-user-query.ts","../../src/server/actions/unlock-item-for-user-action.ts","../../src/server/actions/update-products-and-prices-action.ts","../../src/server/utils/stripe/create-customer-at-stripe.ts","../../src/server/utils/stripe/get-customer-from-stripe-or-create.ts","../../src/server/api/handle-donation.ts","../../src/server/api/handle-update.ts","../../src/server/api/handle-portal.ts","../../src/server/api/handle-checkout.ts","../../src/server/api/index.ts","../../src/server/access/access-queries.ts","../../src/server/collections/customers.ts","../../src/server/collections/fields/permission-evaluation-field.ts","../../src/server/collections/prices.ts","../../src/server/collections/products.ts","../../src/server/plugin.ts"],"sourcesContent":["\"use server\";\n\nimport { Payload } from \"payload\"\n\ntype Config = {\n collections: {\n [key: string]: any\n }\n}\n\ninterface UpsertOptions<T extends keyof Config['collections']> {\n collection: T\n payload: Payload\n data: Omit<Config['collections'][T], 'createdAt' | 'id' | 'updatedAt' | 'sizes'>\n where: any\n}\n\nexport const payloadUpsert = async <T extends keyof Config['collections']>({\n payload,\n collection,\n data,\n where,\n}: UpsertOptions<T>): Promise<Config['collections'][T] | null> => {\n try {\n const existingDocs = await payload.find({\n collection: collection as any,\n where,\n pagination: false,\n limit: 1,\n })\n\n const existingDocId = existingDocs.docs?.at(0)?.id\n if (existingDocId) {\n const updatedDoc = await payload.update({\n collection: collection as any,\n id: existingDocId,\n data: data as any,\n })\n\n return updatedDoc || null\n }\n\n return await payload.create({\n collection,\n data,\n } as any)\n } catch (error) {\n console.error(`Error in payloadUpsert: ${error}`)\n throw new Error(`Failed to upsert document in collection ${collection} ${error}`)\n }\n}\n","import Stripe from \"stripe\";\nexport const stripeBuilder = (): Stripe => {\n return new Stripe(process.env.STRIPE_SECRET_KEY!, {\n apiVersion: '2024-09-30.acacia'\n })\n}\n","\"use server\";\n\nimport type { Payload } from \"payload\";\nimport type Stripe from \"stripe\";\nimport {\n COLLECTION_SLUG_PRICES,\n COLLECTION_SLUG_PRODUCTS,\n} from \"../../model/index.js\";\nimport { payloadUpsert } from \"../utils/payload/upsert.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\n\nexport const updatePrices = async (payload: Payload) => {\n const stripe = await stripeBuilder();\n const prices = await stripe.prices.list({ limit: 100, active: true });\n const promises = prices.data.map(price => priceUpsert(price, payload));\n const pricesUpserted = await Promise.all(promises);\n\n const pricesByProductId = pricesUpserted\n .mapNotNull(t => t)\n .reduce(\n (acc, { productId, priceId }) => {\n if (!acc[productId]) {\n acc[productId] = [];\n }\n acc[productId].push(priceId);\n return acc;\n },\n {} as Record<string, number[]>\n );\n\n Object.entries(pricesByProductId).map(async ([productId, prices]) => {\n await payload.update({\n collection: COLLECTION_SLUG_PRODUCTS,\n data: {\n prices,\n },\n where: {\n stripeID: { equals: productId },\n },\n });\n });\n};\n\ninterface PriceUpserted {\n productId: string;\n priceId: number;\n}\n\nexport async function priceUpsert(\n price: Stripe.Price,\n payload: Payload\n): Promise<PriceUpserted | null> {\n const stripeProductID =\n typeof price.product === \"string\" ? price.product : price.product.id;\n\n if (price.deleted !== undefined) {\n priceDeleted(price, payload);\n return null;\n }\n if (price.unit_amount == null) return null;\n const priceUpserted = await payloadUpsert({\n payload,\n collection: COLLECTION_SLUG_PRICES,\n data: {\n stripeID: price.id,\n stripeProductId: stripeProductID,\n active: price.active,\n unitAmount: price.unit_amount as number,\n currency: price.currency,\n type: price.type,\n interval: price.recurring?.interval,\n intervalCount: price.recurring?.interval_count,\n },\n where: {\n stripeID: { equals: price.id },\n },\n });\n if (!priceUpserted) return null;\n return { productId: stripeProductID, priceId: priceUpserted.id };\n}\n\nexport const priceDeleted = async (price: Stripe.Price, payload: Payload) => {\n const { id } = price;\n\n try {\n await payload.delete({\n collection: COLLECTION_SLUG_PRICES,\n where: {\n stripeID: { equals: id },\n },\n });\n } catch (error) {\n payload.logger.error(`- Error deleting price: ${error}`);\n throw error;\n }\n};\n","\"use server\";\n\nimport type { Payload } from \"payload\";\nimport type Stripe from \"stripe\";\nimport { COLLECTION_SLUG_PRODUCTS } from \"../../model/index.js\";\nimport { payloadUpsert } from \"../utils/payload/upsert.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\n\nconst logs = false;\n\nexport const updateProducts = async (payload: Payload) => {\n const stripe = await stripeBuilder();\n const products = await stripe.products.list({ limit: 100, active: true });\n products.data.forEach(product => productSync(product, payload));\n};\n\nexport const productSync = async (object: Stripe.Product, payload: Payload) => {\n const { id: stripeProductID, name, description, images } = object;\n if (object.deleted !== undefined) return productDeleted(object, payload);\n try {\n await payloadUpsert({\n payload,\n collection: COLLECTION_SLUG_PRODUCTS,\n data: {\n prices: [],\n stripeID: stripeProductID,\n active: true,\n metadata: object.metadata,\n type: object.type,\n name,\n description,\n images: images?.map(image => ({ url: image })) || [],\n },\n where: {\n stripeID: { equals: stripeProductID },\n },\n });\n } catch (error) {\n console.error(error);\n throw error;\n }\n};\n\nexport const productDeleted = async (\n object: Stripe.Product,\n payload: Payload\n) => {\n const { id: stripeProductID } = object;\n\n try {\n const productQuery = await payload.find({\n collection: COLLECTION_SLUG_PRODUCTS,\n where: {\n stripeID: { equals: stripeProductID },\n },\n });\n\n const payloadProductID = productQuery.docs?.[0]?.id;\n\n if (payloadProductID) {\n await payload.delete({\n collection: COLLECTION_SLUG_PRODUCTS,\n id: payloadProductID,\n });\n\n if (logs)\n payload.logger.info(\n `✅ Successfully deleted product with Stripe ID: ${stripeProductID}`\n );\n }\n } catch (error) {\n payload.logger.error(`- Error deleting product: ${error}`);\n throw error;\n }\n};\n","import { COLLECTION_SLUG_CUSTOMERS } from \"../../../model/index.js\";\nimport { Payload } from \"payload\";\n\nexport async function syncCustomerByEmail({ email, payload }: { email: string, payload: Payload }) {\n const customers = await payload.find({\n collection: COLLECTION_SLUG_CUSTOMERS,\n where: { email: { equals: email } },\n });\n const customerId = customers.docs?.[0]?.id;\n\n await payload.update({\n collection: \"users\",\n data: {\n customer: customerId,\n },\n where: { email: { equals: email } },\n });\n}\n","import { COLLECTION_SLUG_CUSTOMERS, generateCustomerInventory } from \"../../../model/index.js\";\nimport type { CustomerInventory } from \"../../../types/index.js\";\nimport { syncCustomerByEmail } from \"./sync-customer-by-email.js\";\nimport { payloadUpsert } from \"./upsert.js\";\nimport { Payload } from \"payload\";\n\nexport async function upsertCustomerInventoryAndSyncWithUser(\n payload: Payload,\n inventory: CustomerInventory | null | undefined,\n email: string,\n stripeCustomerId?: string | null\n) {\n await payloadUpsert({\n payload,\n collection: COLLECTION_SLUG_CUSTOMERS,\n data: {\n email: email,\n stripeId: stripeCustomerId,\n inventory: inventory ?? generateCustomerInventory(),\n },\n where: { email: { equals: email } },\n });\n await syncCustomerByEmail({ email, payload });\n}\n","import { COLLECTION_SLUG_USER } from \"../../../model/index.js\";\nimport { Payload } from \"payload\";\n\ninterface Props {\n email: string;\n payload: Payload;\n}\n\n/**\n * Gets a user ID by email address using Payload's find method\n * @param email - The email address to search for\n * @returns The user ID if found, null otherwise\n */\nexport async function getUserIdByEmail({email, payload}: Props): Promise<string | null | undefined> {\n const userQuery = await payload.find({\n collection: COLLECTION_SLUG_USER,\n where: {\n email: { equals: email },\n },\n });\n\n const user = userQuery.docs?.[0];\n return user?.id as string | null;\n}\n","import Stripe from \"stripe\";\nimport { stripeBuilder } from \"./stripe-builder.js\";\n\nexport async function getCustomer({\n stripe,\n email,\n}: {\n stripe?: Stripe;\n email: string;\n}): Promise<Stripe.Customer | null> {\n stripe = stripe ?? stripeBuilder();\n const customers = await stripe.customers.search({\n query: `email:'${email}'`,\n });\n return customers.data.length ? customers.data[0] as Stripe.Customer : null;\n \n}\n\nexport async function resolveStripeCustomer({ customer }: {\n customer: string | Stripe.Customer | Stripe.DeletedCustomer | null\n}): Promise<Stripe.Customer | Stripe.DeletedCustomer | null> {\n const stripe = stripeBuilder();\n if (typeof customer === \"string\") {\n return await stripe.customers.retrieve(customer);\n }\n return customer;\n}","import { Payload } from \"payload\";\nimport { COLLECTION_SLUG_CUSTOMERS } from \"../../../model/index.js\";\n\nexport async function removeCustomerByStripeId({\n stripeId,\n payload,\n}: {\n stripeId: string;\n payload: Payload;\n}) {\n await payload.delete({\n collection: COLLECTION_SLUG_CUSTOMERS,\n where: { stripeId: { equals: stripeId } },\n });\n payload.logger.info(`✅ Successfully removed customer with Stripe ID: ${stripeId}`);\n}","import { Payload } from \"payload\";\nimport {\n COLLECTION_SLUG_CUSTOMERS,\n generateCustomerInventory,\n} from \"../../../model/index.js\";\nimport { Customer, CustomerInventory } from \"../../../types/index.js\";\n\ninterface FindOrCreateCustomerProps {\n email: string;\n payload: Payload;\n stripeId?: string;\n}\n\n/**\n * Finds a customer by email address in the customers collection, or creates one if not found\n * @param email - The email address to search for\n * @param payload - Payload instance\n * @param stripeId - Optional Stripe customer ID to set when creating\n * @returns The customer document (found or created)\n */\nexport async function findOrCreateCustomer({\n email,\n payload,\n stripeId,\n}: FindOrCreateCustomerProps): Promise<Customer | null> {\n if (!email) {\n payload.logger.error(\"Email is required to find or create customer\");\n return null;\n }\n\n try {\n const userQuery = await payload.find({\n collection: COLLECTION_SLUG_CUSTOMERS,\n where: {\n email: { equals: email },\n },\n });\n\n let existingCustomer: Customer | null = userQuery.docs?.[0] as unknown as Customer | null;\n if (existingCustomer) {\n existingCustomer.inventory = existingCustomer?.inventory\n ? (existingCustomer.inventory as unknown as CustomerInventory)\n : generateCustomerInventory();\n return existingCustomer;\n }\n\n payload.logger.info(`Creating new customer for email: ${email}`);\n\n const newCustomer = await payload.create({\n collection: COLLECTION_SLUG_CUSTOMERS,\n data: {\n email,\n stripeId: stripeId || \"\",\n inventory: generateCustomerInventory() as unknown as [k: string],\n },\n });\n\n payload.logger.info(`✅ Successfully created customer for email: ${email}`);\n return newCustomer as unknown as Customer;\n } catch (error) {\n payload.logger.error(\n `Error finding or creating customer for email ${email}: ${error}`\n );\n throw error;\n }\n}\n","import { Payload } from \"payload\";\nimport type Stripe from \"stripe\";\nimport { COLLECTION_SLUG_PRODUCTS } from \"../../model/index.js\";\nimport { getPermissionsSlugs } from \"../../model/permissions.js\";\nimport type { CustomerInventory } from \"../../types/index.js\";\nimport { upsertCustomerInventoryAndSyncWithUser } from \"../utils/payload/upsert-customer-inventory-and-sync-with-user.js\";\nimport { getUserIdByEmail } from \"../utils/payload/get-userId-by-email.js\";\nimport { resolveStripeCustomer } from \"../utils/stripe/get-customer.js\";\nimport { removeCustomerByStripeId } from \"../utils/payload/remove-customer-by-stripe-id.js\";\nimport { findOrCreateCustomer } from \"../utils/payload/find-or-create-customer.js\";\n\nexport const subscriptionUpsert = async (\n subscription: Stripe.Subscription,\n payload: Payload,\n onSubscriptionUpdate: (\n type: \"create\" | \"delete\",\n userId: string\n ) => Promise<void>\n) => {\n const { id: stripeID, status, customer: stripeCustomer } = subscription;\n const customer = await resolveStripeCustomer({ customer: stripeCustomer });\n const error = (message: string) =>\n payload.logger.error(\"Subscription Upsert: \", message as any);\n const info = (message: string) =>\n payload.logger.info(\"Subscription Upsert: \", message as any);\n\n if (!customer) {\n error(\"No stripe customer found for subscription\");\n return;\n }\n if (customer.deleted) {\n await removeCustomerByStripeId({ stripeId: customer.id, payload });\n return;\n }\n if (!customer.email) {\n error(\"No email found for stripe customer\");\n return;\n }\n const email = customer.email;\n const stripeId = customer.id;\n\n try {\n const customer = await findOrCreateCustomer({\n email,\n payload,\n stripeId,\n });\n\n const item = subscription.items.data.at(0);\n if (!item || !customer) {\n error(`No item ${item} or customer ${customer} found`);\n return;\n }\n\n const { docs: products } = await payload.find({\n collection: COLLECTION_SLUG_PRODUCTS,\n where: { stripeID: { equals: item.price.product } },\n });\n const product = products.at(0);\n if (!product) return;\n\n const inventory = customer.inventory;\n inventory.subscriptions[stripeID] = {\n ...subscription,\n permissions: getPermissionsSlugs({ permissions: product.roles }),\n };\n info(`INVENTORY OF THE SUBSCRIPTION ${inventory}`);\n await upsertCustomerInventoryAndSyncWithUser(payload, inventory, email, stripeId);\n\n if ([\"active\", \"trialing\"].includes(status)) {\n const userId = await getUserIdByEmail({ email, payload });\n if (!userId) return;\n await onSubscriptionUpdate(\"create\", userId);\n }\n info(\n `✅ Successfully updated subscription with ID: ${stripeID} for user: ${email}`\n );\n } catch (e) {\n error(`- Error managing subscription: ${e}`);\n throw e;\n }\n};\n\nexport const subscriptionDeleted = async (\n subscription: Stripe.Subscription,\n payload: Payload,\n onSubscriptionUpdate: (\n type: \"create\" | \"delete\",\n userId: string\n ) => Promise<void>\n) => {\n const { id, customer: customerId } = subscription;\n const customer = await resolveStripeCustomer({ customer: customerId });\n const stripeId = customer?.id;\n if (!customer) {\n payload.logger.error(\"No stripe customer found for subscription\");\n return;\n } \n if (customer.deleted) {\n await removeCustomerByStripeId({ stripeId: customer.id, payload });\n return;\n }\n if (!customer.email) {\n payload.logger.error(\"No email found for stripe customer\");\n return;\n }\n const email = customer.email;\n try {\n const customer = await findOrCreateCustomer({\n email,\n payload,\n stripeId\n });\n if (!customer) {\n payload.logger.error(\"No customer found for subscription\");\n return;\n }\n\n const inventory = customer.inventory as unknown as CustomerInventory;\n delete inventory.subscriptions[id];\n\n await upsertCustomerInventoryAndSyncWithUser(payload, inventory, email, stripeId);\n const userId = await getUserIdByEmail({ email, payload });\n if (!userId) {\n payload.logger.error(\"No user found for subscription\");\n return;\n }\n await onSubscriptionUpdate(\"delete\", userId);\n\n payload.logger.info(\n `✅ Successfully deleted subscription: ${id} for user: ${email}`\n );\n } catch (error) {\n payload.logger.error(`- Error deleting subscription: ${error}`);\n throw error;\n }\n};\n","import { Payload } from \"payload\";\nimport type Stripe from \"stripe\";\nimport { COLLECTION_SLUG_CUSTOMERS } from \"../../model/index.js\";\nimport { resolveStripeCustomer } from \"../utils/stripe/get-customer.js\";\nimport { findOrCreateCustomer } from \"../utils/payload/find-or-create-customer.js\";\nimport { removeCustomerByStripeId } from \"../utils/payload/remove-customer-by-stripe-id.js\";\n\nexport const paymentSucceeded = async (\n paymentIntent: Stripe.PaymentIntent,\n payload: Payload\n) => {\n const { id, customer: paymentCustomer } = paymentIntent;\n const stripeCustomer = await resolveStripeCustomer({ customer: paymentCustomer });\n if (!stripeCustomer) {\n payload.logger.error(\"No stripe customer found for payment\");\n return\n }\n if (stripeCustomer.deleted) {\n await removeCustomerByStripeId({ stripeId: stripeCustomer.id, payload });\n return;\n }\n if (!stripeCustomer.email) {\n payload.logger.error(\"No email found for stripe customer\");\n return;\n }\n\n try {\n const customer = await findOrCreateCustomer({\n email: stripeCustomer.email,\n payload,\n stripeId: stripeCustomer.id,\n });\n if (!customer) return;\n\n if (!customer) {\n payload.logger.error(`User not found for payment: ${stripeCustomer.email}`);\n return;\n }\n\n let inventory = customer.inventory\n inventory.payments[id] = paymentIntent;\n\n await payload.update({\n collection: COLLECTION_SLUG_CUSTOMERS,\n data: { inventory: inventory as unknown as { [x: string]: {} } },\n where: { email: { equals: stripeCustomer.email } },\n });\n\n payload.logger.info(\n `✅ Successfully recorded ${stripeCustomer.metadata?.type ?? \"subscription\"} with Payment Intent ID: ${id} for user: ${stripeCustomer.email}`\n );\n } catch (error) {\n payload.logger.error(`- Error recording payment: ${error}`);\n throw error;\n }\n};\n","import { Payload } from \"payload\";\nimport type Stripe from \"stripe\";\nimport { COLLECTION_SLUG_CUSTOMERS } from \"../../model/index.js\";\nimport { findOrCreateCustomer } from \"../utils/payload/find-or-create-customer.js\";\nimport { resolveStripeCustomer } from \"../utils/stripe/get-customer.js\";\nimport { removeCustomerByStripeId } from \"../utils/payload/remove-customer-by-stripe-id.js\";\n\nexport const invoiceSucceeded = async (\n invoiceIntent: Stripe.Invoice,\n payload: Payload\n) => {\n const { id, customer: paymentCustomer } = invoiceIntent;\n const stripeCustomer = await resolveStripeCustomer({ customer: paymentCustomer });\n if (!stripeCustomer) {\n payload.logger.error(\"No stripe customer found for payment\");\n return\n }\n if (stripeCustomer.deleted) {\n await removeCustomerByStripeId({ stripeId: stripeCustomer.id, payload });\n return;\n }\n if (!stripeCustomer.email) {\n payload.logger.error(\"No email found for stripe customer\");\n return;\n }\n\n try {\n const customer = await findOrCreateCustomer({\n email: stripeCustomer.email,\n payload,\n stripeId: stripeCustomer.id,\n });\n if (!customer) return;\n\n let inventory = customer.inventory\n inventory.invoices[id] = invoiceIntent;\n\n await payload.update({\n collection: COLLECTION_SLUG_CUSTOMERS,\n data: { inventory: inventory as unknown as { [x: string]: {} } },\n where: { email: { equals: stripeCustomer.email } },\n });\n\n payload.logger.info(\n `✅ Successfully recorded ${stripeCustomer.metadata?.type ?? \"subscription\"} with Payment Intent ID: ${id} for user: ${stripeCustomer.email}`\n );\n } catch (error) {\n payload.logger.error(`- Error recording payment: ${error}`);\n throw error;\n }\n};\n","import { Payload } from \"payload\";\nimport Stripe from \"stripe\";\nimport { COLLECTION_SLUG_CUSTOMERS } from \"../../index.js\";\n\nexport const customerDeleted = async (\n customer: Stripe.Customer,\n payload: Payload,\n ) => {\n const { id, email } = customer;\n try { \n await payload.delete({\n collection: COLLECTION_SLUG_CUSTOMERS,\n where: { email: { equals: email } },\n });\n payload.logger.info(\n `✅ Successfully deleted customer with Stripe ID: ${id}`\n );\n } catch (error) {\n payload.logger.error(`- Error deleting subscription: ${error}`);\n throw error;\n }\n };\n ","'use server'\n\nimport { headers as getHeaders } from 'next/headers.js'\nimport { BasePayload } from 'payload'\nimport { BaseUser } from '../../index.js'\n\nexport async function getCurrentUserQuery(payload: BasePayload): Promise<BaseUser | null> {\n const headers = await getHeaders()\n return (await payload.auth({ headers })).user as BaseUser | null\n}\n","import { Result } from \"@nexo-labs/hegel\";\nimport { Payload } from \"payload\";\nimport {\n checkIfUserCanUnlockQuery,\n COLLECTION_SLUG_USER,\n countWeeklyUnlocksQuery,\n MAX_UNLOCKS_PER_WEEK,\n} from \"../../model/index.js\";\nimport { generateUserInventory } from \"../../model/builders.js\";\nimport { getPermissionsSlugs } from \"../../model/permissions.js\";\nimport type { UnlockItem, UserInventory } from \"../../types/index.js\";\nimport { getCurrentUserQuery } from \"../access/get-current-user-query.js\";\n\nconst addUniqueUnlock = (\n unlocks: UnlockItem[],\n collection: string,\n contentId: number\n): UnlockItem[] => {\n const isDuplicate = unlocks.some(\n unlock => unlock.collection === collection && unlock.id === contentId\n );\n\n if (isDuplicate) {\n return unlocks;\n }\n return [\n ...unlocks,\n {\n collection,\n id: contentId,\n dateUnlocked: new Date(),\n },\n ];\n};\n\nexport const unlockItemForUser = async (\n getPayload: () => Promise<Payload>,\n collection: string,\n contentId: number\n): Promise<Result<boolean>> => {\n const payload = await getPayload();\n const user = await getCurrentUserQuery(payload);\n if (!user || !user.id) {\n return { error: \"Usuario no válido\" };\n }\n const item = await payload.findByID({\n collection: collection as any,\n id: contentId.toString(),\n });\n\n if (!item) {\n return { error: \"Elemento no encontrado\" };\n }\n const permissions = getPermissionsSlugs({ permissions: item.permissions });\n\n if (!checkIfUserCanUnlockQuery(user, permissions)) {\n return { error: \"No tienes permisos para desbloquear este elemento\" };\n }\n\n const weeklyUnlocks = countWeeklyUnlocksQuery(user);\n if (weeklyUnlocks >= MAX_UNLOCKS_PER_WEEK) {\n return {\n error: `Has alcanzado el límite de ${MAX_UNLOCKS_PER_WEEK} desbloqueos para esta semana`,\n };\n }\n\n const inventory =\n (user.inventory as UserInventory) ?? generateUserInventory();\n\n const updatedUnlocks = addUniqueUnlock(\n inventory.unlocks,\n collection,\n contentId\n );\n\n if (updatedUnlocks.length === inventory.unlocks.length) {\n return { data: true };\n }\n\n try {\n await payload.update({\n collection: COLLECTION_SLUG_USER,\n id: user.id.toString(),\n data: {\n inventory: {\n ...inventory,\n unlocks: updatedUnlocks,\n },\n },\n });\n\n return { data: true };\n } catch (error) {\n console.error(\"Error al actualizar el inventario del usuario:\", error);\n return { error: \"Error al actualizar el inventario del usuario\" };\n }\n};\n","\"use server\";\n\nimport { updatePrices } from \"./price.js\";\nimport { updateProducts } from \"./product.js\";\nimport { Payload } from \"payload\";\n\nexport async function updateProductsAndPrices(payload: Payload) {\n await updateProducts(payload);\n await updatePrices(payload);\n}\n","import Stripe from \"stripe\";\nimport { stripeBuilder } from \"./stripe-builder.js\";\n\nexport async function createCustomerAtStripe({\n stripe,\n email,\n name,\n}: {\n stripe?: Stripe;\n email: string;\n name?: string;\n}) {\n stripe = stripe ?? stripeBuilder();\n return await stripe.customers.create({\n email: email,\n name: name || undefined,\n });\n}\n","import { createCustomerAtStripe } from \"./create-customer-at-stripe.js\";\nimport { getCustomer } from \"./get-customer.js\";\nimport { stripeBuilder } from \"./stripe-builder.js\";\n\nexport async function getCustomerFromStripeOrCreate(\n email: string,\n name?: string\n): Promise<string> {\n const stripe = stripeBuilder();\n let customer = await getCustomer({ stripe, email });\n if (!customer) {\n customer = await createCustomerAtStripe({ stripe, email, name });\n }\n return customer.id;\n}\n","import { NextResponse } from \"next/server.js\";\nimport Stripe from \"stripe\";\nimport { upsertCustomerInventoryAndSyncWithUser } from \"../utils/payload/upsert-customer-inventory-and-sync-with-user.js\";\nimport { getCustomerFromStripeOrCreate } from \"../utils/stripe/get-customer-from-stripe-or-create.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\nimport { getCurrentUserQuery } from \"../access/get-current-user-query.js\";\nimport { Payload } from \"payload\";\n\nexport async function handleDonation(\n request: Request,\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n) {\n const payload = await getPayload();\n const payloadUser = await getCurrentUserQuery(payload);\n if (!payloadUser || !payloadUser.email)\n throw new Error(\"You must be logged in to make a donation\");\n\n const url = new URL(request.url);\n const amountParam = url.searchParams.get(\"amount\");\n\n if (!amountParam) throw new Error(\"Amount is required\");\n\n const amount = parseInt(amountParam);\n if (amount < 100) throw new Error(\"Minimum donation amount is €1\");\n\n const stripe = stripeBuilder();\n const customerId = await getCustomerFromStripeOrCreate(\n payloadUser.email,\n payloadUser.name\n );\n await upsertCustomerInventoryAndSyncWithUser(\n payload,\n payloadUser.customer?.inventory,\n payloadUser.email,\n customerId\n );\n const metadata: Stripe.MetadataParam = {\n type: \"donation\",\n };\n const session = await stripe.checkout.sessions.create({\n customer: customerId,\n payment_method_types: [\"card\"],\n line_items: [\n {\n price_data: {\n currency: \"eur\",\n product_data: {\n name: \"Donación - Portal Escohotado\",\n description:\n \"Apoyo al mantenimiento del legado digital de Antonio Escohotado\",\n },\n unit_amount: amount,\n },\n quantity: 1,\n },\n ],\n mode: \"payment\",\n success_url: `${process.env.DOMAIN}${getRoutes().nextJS.subscriptionPageHref}?success=donation`,\n cancel_url: `${process.env.DOMAIN}${getRoutes().nextJS.subscriptionPageHref}?error=donation_cancelled`,\n metadata,\n payment_intent_data: { metadata },\n invoice_creation: { enabled: true, invoice_data: { metadata } },\n });\n\n return NextResponse.json({ url: session.url });\n}\n","import { NextResponse } from \"next/server.js\";\nimport type { Customer, CustomerInventory } from \"../../types/index.js\";\nimport { upsertCustomerInventoryAndSyncWithUser } from \"../utils/payload/upsert-customer-inventory-and-sync-with-user.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\nimport { getCurrentUserQuery } from \"../access/get-current-user-query.js\";\nimport { Payload } from \"payload\";\n\nexport async function handleUpdate(\n request: Request,\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n) {\n const payload = await getPayload();\n const payloadUser = await getCurrentUserQuery(payload);\n if (!payloadUser) {\n throw new Error(\"You must be logged in to access this page\");\n }\n const { searchParams } = new URL(request.url);\n const subscriptionId = searchParams.get(\"subscriptionId\");\n const cancelAtPeriodEnd = searchParams.get(\"cancelAtPeriodEnd\") === \"true\";\n if (!subscriptionId) throw Error(\"SubscriptionId could not be found.\");\n\n await stripeBuilder().subscriptions.update(subscriptionId, {\n cancel_at_period_end: cancelAtPeriodEnd,\n });\n const customer = payloadUser.customer as Customer;\n console.error(\"UPDATE: customer\", customer);\n const inventory = customer.inventory as CustomerInventory | null;\n if (inventory && inventory.subscriptions && inventory.subscriptions[subscriptionId]) {\n inventory.subscriptions[subscriptionId].cancel_at_period_end =\n cancelAtPeriodEnd;\n }\n\n await upsertCustomerInventoryAndSyncWithUser(\n payload,\n inventory,\n customer.email\n );\n\n const routes = getRoutes();\n return NextResponse.redirect(\n `${process.env.DOMAIN}${routes.nextJS.subscriptionPageHref}?refresh=${Date.now()}`,\n 303\n );\n}\n","import { NextResponse } from \"next/server.js\";\nimport Stripe from \"stripe\";\nimport { upsertCustomerInventoryAndSyncWithUser } from \"../utils/payload/upsert-customer-inventory-and-sync-with-user.js\";\nimport { getCustomerFromStripeOrCreate } from \"../utils/stripe/get-customer-from-stripe-or-create.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\nimport { getCurrentUserQuery } from \"../access/get-current-user-query.js\";\nimport { Payload } from \"payload\";\n\nexport async function handlePortal(\n request: Request,\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n) {\n const payload = await getPayload();\n const payloadUser = await getCurrentUserQuery(payload);\n if (!payloadUser || !payloadUser.email)\n throw new Error(\"You must be logged in to access this page\");\n const url = new URL(request.url);\n const cancelSubscriptionId = url.searchParams.get(\"cancelSubscriptionId\");\n const updateSubscriptionId = url.searchParams.get(\"updateSubscriptionId\");\n\n let flowData: Stripe.BillingPortal.SessionCreateParams.FlowData | undefined;\n if (cancelSubscriptionId)\n flowData = {\n type: \"subscription_cancel\",\n subscription_cancel: { subscription: cancelSubscriptionId },\n };\n else if (updateSubscriptionId)\n flowData = {\n type: \"subscription_update\",\n subscription_update: { subscription: updateSubscriptionId },\n };\n\n const stripe = stripeBuilder();\n const routes = getRoutes();\n const customerId = await getCustomerFromStripeOrCreate(\n payloadUser.email,\n payloadUser.name\n );\n await upsertCustomerInventoryAndSyncWithUser(\n payload,\n payloadUser.customer?.inventory,\n payloadUser.email,\n customerId\n );\n\n const session = await stripe.billingPortal.sessions.create({\n flow_data: flowData,\n customer: customerId,\n return_url: `${process.env.DOMAIN}${routes.nextJS.subscriptionPageHref}`,\n });\n\n return NextResponse.redirect(session.url, 303);\n}\n","import { NextResponse } from \"next/server.js\";\nimport Stripe from \"stripe\";\nimport { upsertCustomerInventoryAndSyncWithUser } from \"../utils/payload/upsert-customer-inventory-and-sync-with-user.js\";\nimport { getCustomerFromStripeOrCreate } from \"../utils/stripe/get-customer-from-stripe-or-create.js\";\nimport { stripeBuilder } from \"../utils/stripe/stripe-builder.js\";\nimport { getCurrentUserQuery } from \"../access/get-current-user-query.js\";\nimport { Payload } from \"payload\";\n\nexport async function handleCheckout(\n request: Request,\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n) {\n const payload = await getPayload();\n const payloadUser = await getCurrentUserQuery(payload);\n\n const url = new URL(request.url);\n const priceId = url.searchParams.get(\"priceId\");\n if (!priceId || !payloadUser || !payloadUser.email)\n throw new Error(\"Invalid request\");\n\n const stripe = stripeBuilder();\n const routes = getRoutes();\n const customerId = await getCustomerFromStripeOrCreate(\n payloadUser.email,\n payloadUser.name\n );\n await upsertCustomerInventoryAndSyncWithUser(\n payload,\n payloadUser.customer?.inventory,\n payloadUser.email,\n customerId\n );\n\n const metadata: Stripe.MetadataParam = {\n type: \"subscription\",\n };\n\n const checkoutResult = await stripe.checkout.sessions.create({\n success_url: `${process.env.DOMAIN}${routes.nextJS.subscriptionPageHref}?success=${Date.now()}`,\n cancel_url: `${process.env.DOMAIN}${routes.nextJS.subscriptionPageHref}?error=${Date.now()}`,\n mode: \"subscription\",\n customer: customerId,\n client_reference_id: String(payloadUser.id),\n line_items: [{ price: priceId, quantity: 1 }],\n metadata,\n tax_id_collection: { enabled: true },\n customer_update: {\n name: \"auto\",\n address: \"auto\",\n shipping: \"auto\",\n },\n subscription_data: { metadata },\n });\n if (checkoutResult.url) return NextResponse.redirect(checkoutResult.url, 303);\n else return NextResponse.json(\"Create checkout url failed\", { status: 406 });\n}\n","import { NextResponse } from \"next/server.js\";\nimport { handleDonation } from \"./handle-donation.js\";\nimport { handleUpdate } from \"./handle-update.js\";\nimport { handlePortal } from \"./handle-portal.js\";\nimport { handleCheckout } from \"./handle-checkout.js\";\nimport type { StripeHandlers } from \"../../types/index.js\";\nimport { Payload } from \"payload\";\nexport * from \"../utils/stripe/get-customer-from-stripe-or-create.js\";\n\nexport function createStripeInventoryHandlers(\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n): StripeHandlers {\n return {\n checkout: {\n GET: (request: Request) =>\n handleCheckout(request, getPayload, getRoutes),\n },\n portal: {\n GET: (request: Request) =>\n handlePortal(request, getPayload, getRoutes),\n },\n update: {\n GET: (request: Request) =>\n handleUpdate(request, getPayload, getRoutes),\n },\n donation: {\n GET: (request: Request) =>\n handleDonation(request, getPayload, getRoutes),\n },\n };\n}\n\nexport function createRouteHandlers(\n getPayload: () => Promise<Payload>,\n getRoutes: () => { nextJS: { subscriptionPageHref: string } }\n) {\n return {\n GET: async (\n request: Request,\n { params }: { params: Promise<{ stripe: string[] }> }\n ) => {\n const path = (await params).stripe[0];\n const handlers = createStripeInventoryHandlers(getPayload, getRoutes);\n\n if (\n path === \"checkout\" ||\n path === \"portal\" ||\n path === \"update\" ||\n path === \"donation\"\n ) {\n return handlers[path].GET(request);\n }\n\n return NextResponse.json({ error: \"Route not found\" }, { status: 404 });\n },\n };\n}\n","import { Access } from \"payload\";\nimport { permissionSlugs } from \"../../model/constants.js\";\n\nexport const isAdmin: Access = ({ req }) => {\n return req?.user?.roles?.includes(permissionSlugs.webAdmin) || false;\n};\n\nexport const isAnyone: Access = () => true;\n\nexport const isAdminOrCurrentUser: Access = ({ req }) => {\n if (req?.user?.roles?.includes(permissionSlugs.webAdmin)) return true;\n return { id: { equals: req.user?.id } };\n};\n\nexport const isAdminOrPublished: Access = ({ req: { user } }) => {\n if (user && user?.roles?.includes(permissionSlugs.webAdmin)) {\n return true;\n }\n\n return {\n _status: {\n equals: \"published\",\n },\n };\n};\n\nexport const isAdminOrStripeActive: Access = ({ req: { user } }) => {\n if (user && user?.roles?.includes(permissionSlugs.webAdmin)) {\n return true;\n }\n\n return {\n active: {\n equals: true,\n },\n };\n};\n\nexport const isAdminOrUserFieldMatchingCurrentUser: Access = ({\n req: { user },\n}) => {\n if (user) {\n if (user?.roles?.includes(permissionSlugs.webAdmin)) return true;\n return {\n user: {\n equals: user?.id,\n },\n };\n }\n return false;\n};\n\nexport const loggedInOrPublished: Access = ({ req: { user } }) => {\n if (user) {\n return true;\n }\n\n return {\n _status: {\n equals: \"published\",\n },\n };\n};\n","import { CollectionConfig } from \"payload\";\nimport { isAdmin } from \"../access/index.js\";\nimport { COLLECTION_SLUG_CUSTOMERS } from \"../../model/index.js\";\n\nexport const customers: CollectionConfig = {\n slug: COLLECTION_SLUG_CUSTOMERS,\n admin: {\n useAsTitle: \"email\",\n group: \"Stripe\",\n defaultColumns: [\"email\", \"stripeId\", \"createdAt\"],\n },\n access: {\n read: () => true,\n create: () => false,\n update: () => false,\n delete: isAdmin,\n },\n fields: [\n {\n name: \"email\",\n type: \"email\",\n required: true,\n unique: true,\n admin: {\n position: \"sidebar\",\n },\n },\n {\n name: \"stripeId\",\n type: \"text\",\n required: true,\n unique: true,\n admin: {\n position: \"sidebar\",\n readOnly: true,\n },\n },\n {\n name: \"inventory\",\n type: \"json\",\n label: \"Inventario\",\n admin: {\n description: \"Datos de inventario de Stripe almacenados como JSON\",\n readOnly: true,\n },\n },\n ],\n};\n","import { COLLECTION_SLUG_TAXONOMY } from \"@nexo-labs/payload-taxonomies\";\nimport { Field } from \"payload\";\n\nexport const permissionEvaluationField: Field = {\n type: 'row',\n fields: [\n {\n type: 'select',\n name: 'type_of_permissions',\n options: [\n { label: 'Todos', value: 'all' },\n { label: 'Permisos por roles', value: 'roles' },\n { label: 'Solo para usuarios sin roles', value: 'only_no_roles' },\n { label: 'Solo invitados', value: 'only_guess' },\n ],\n defaultValue: 'all',\n label: 'Tipo de permisos',\n },\n {\n type: 'relationship',\n name: 'permissions',\n relationTo: [COLLECTION_SLUG_TAXONOMY],\n hasMany: false,\n admin: {\n condition: (_, siblingData) => siblingData.type_of_permissions === 'roles',\n }\n }\n ]\n}","import { CollectionConfig } from \"payload\";\nimport {\n COLLECTION_SLUG_PRICES,\n COLLECTION_SLUG_PRODUCTS,\n PricingPlanInterval,\n PricingType,\n formatOptions,\n} from \"../../model/index.js\";\nimport { isAdmin, isAdminOrStripeActive } from \"../access/index.js\";\n\nexport const prices: CollectionConfig = {\n slug: COLLECTION_SLUG_PRICES,\n admin: {\n useAsTitle: \"unitAmount\",\n group: \"Stripe\",\n },\n access: {\n read: isAdminOrStripeActive,\n create: () => false,\n update: () => false,\n delete: isAdmin,\n },\n fields: [\n {\n name: \"stripeID\",\n label: \"Stripe ID\",\n type: \"text\",\n required: true,\n admin: { position: \"sidebar\", readOnly: true },\n },\n {\n name: \"stripeProductId\",\n type: \"text\",\n required: true,\n admin: { position: \"sidebar\", readOnly: true },\n },\n {\n name: \"product\",\n type: \"join\",\n collection: COLLECTION_SLUG_PRODUCTS,\n on: \"prices\",\n hasMany: false,\n },\n {\n name: \"active\",\n type: \"checkbox\",\n required: true,\n admin: { position: \"sidebar\" },\n },\n { name: \"description\", type: \"textarea\" },\n {\n type: \"row\",\n fields: [\n { name: \"unitAmount\", type: \"number\", required: true },\n { name: \"currency\", type: \"text\", required: true },\n {\n name: \"type\",\n type: \"select\",\n options: formatOptions(PricingType),\n required: true,\n },\n ],\n },\n {\n type: \"row\",\n fields: [\n {\n name: \"interval\",\n type: \"select\",\n options: formatOptions(PricingPlanInterval),\n },\n { name: \"intervalCount\", type: \"number\" },\n { name: \"trialPeriodDays\", type: \"number\" },\n ],\n },\n { name: \"metadata\", type: \"json\", label: \"Metadata\" },\n ],\n};\n","import { buildTaxonomyRelationship } from \"@nexo-labs/payload-taxonomies\";\nimport { CollectionConfig } from \"payload\";\nimport { COLLECTION_SLUG_PRICES, COLLECTION_SLUG_PRODUCTS } from \"../../model/index.js\";\nimport { isAdminOrStripeActive } from \"../access/index.js\";\n\nexport const products: CollectionConfig = {\n slug: COLLECTION_SLUG_PRODUCTS,\n admin: {\n useAsTitle: \"name\",\n group: \"Stripe\",\n },\n access: {\n read: isAdminOrStripeActive,\n create: () => false,\n },\n fields: [\n {\n name: \"stripeID\",\n label: \"Stripe ID\",\n type: \"text\",\n required: true,\n admin: { position: \"sidebar\", readOnly: true },\n },\n {\n name: \"type\",\n type: \"select\",\n options: [\n { value: \"good\", label: \"Bienes\" },\n { value: \"service\", label: \"Service\" },\n ],\n },\n {\n name: \"active\",\n type: \"checkbox\",\n required: true,\n admin: { position: \"sidebar\" },\n },\n {\n name: \"show\",\n type: \"checkbox\",\n defaultValue: false,\n admin: { position: \"sidebar\" },\n },\n { name: \"name\", type: \"text\", required: true },\n { name: \"description\", type: \"textarea\" },\n { name: \"images\", type: \"array\", fields: [{ type: \"text\", name: \"url\" }] },\n {\n name: \"prices\",\n type: \"relationship\",\n relationTo: COLLECTION_SLUG_PRICES,\n hasMany: true,\n required: false,\n },\n { name: \"metadata\", type: \"json\", label: \"Metadata\" },\n {\n type: \"array\",\n name: \"features\",\n fields: [{ type: \"text\", name: \"title\" }],\n },\n buildTaxonomyRelationship({\n name: \"roles\",\n label: \"Roles\",\n defaultValue: [],\n filterOptions: () => {\n return {\n \"payload.types\": { in: [\"role\"] },\n };\n },\n required: false,\n }),\n ],\n};\n","import { stripePlugin } from \"@payloadcms/plugin-stripe\";\nimport {\n priceDeleted,\n subscriptionUpsert,\n subscriptionDeleted,\n productDeleted,\n paymentSucceeded,\n invoiceSucceeded,\n customerDeleted,\n} from \"./actions/index.js\";\nimport { Plugin } from \"payload\";\n\nexport const plugin = (\n onSubscriptionUpdate: (\n type: \"create\" | \"delete\",\n userId: string\n ) => Promise<void>\n): Plugin => {\n return stripePlugin({\n isTestKey: process.env.STRIPE_SECRET_KEY?.includes(\"sk_test\"),\n stripeSecretKey: process.env.STRIPE_SECRET_KEY || \"\",\n stripeWebhooksEndpointSecret: process.env.STRIPE_WEBHOOK_SECRET,\n webhooks: {\n \"price.deleted\": async ({ event, payload }) =>\n await priceDeleted(event.data.object, payload),\n \"customer.subscription.created\": async ({ event, payload }) =>\n await subscriptionUpsert(\n event.data.object,\n payload,\n onSubscriptionUpdate\n ),\n \"customer.subscription.paused\": async ({ event, payload }) =>\n await subscriptionUpsert(\n event.data.object,\n payload,\n onSubscriptionUpdate\n ),\n \"customer.subscription.updated\": async ({ event, payload }) =>\n await subscriptionUpsert(\n event.data.object,\n payload,\n onSubscriptionUpdate\n ),\n \"customer.subscription.deleted\": async ({ event, payload }) =>\n await subscriptionDeleted(\n event.data.object,\n payload,\n onSubscriptionUpdate\n ),\n \"customer.deleted\": async ({ event, payload }) =>\n await customerDeleted(event.data.object, payload),\n \"product.deleted\": async ({ event, payload }) =>\n await productDeleted(event.data.object, payload),\n \"payment_intent.succeeded\": async ({ event, payload }) => {\n await paymentSucceeded(event.data.object, payload)\n },\n \"invoice.paid\": async ({ event, payload }) => {\n await invoiceSucceeded(event.data.object, payload)\n }\n },\n });\n};\n"],"mappings":"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"}
@@ -0,0 +1,193 @@
1
+ //#region src/model/builders.ts
2
+ const generateUserInventory = () => ({
3
+ unlocks: [],
4
+ favorites: []
5
+ });
6
+ const generateCustomerInventory = () => ({
7
+ subscriptions: {},
8
+ products: {},
9
+ payments: {},
10
+ invoices: {}
11
+ });
12
+
13
+ //#endregion
14
+ //#region src/model/constants.ts
15
+ const COLLECTION_SLUG_PRODUCTS = "products";
16
+ const COLLECTION_SLUG_PRICES = "prices";
17
+ const COLLECTION_SLUG_USER = "users";
18
+ const COLLECTION_SLUG_CUSTOMERS = "customers";
19
+ const QUERY_PERMISSION_TYPES = {
20
+ ALL: "all",
21
+ ROLES: "roles",
22
+ ONLY_NO_ROLES: "only_no_roles",
23
+ ONLY_GUESS: "only_guess"
24
+ };
25
+ const FREEMIUM_PERMISSION = "freemium";
26
+ const FREE_PERMISSION = "free";
27
+ const TESTER_PERMISSION = "tester";
28
+ const DEV_PERMISSION = "dev";
29
+ const BASIC_PERMISSION = "basic";
30
+ const ADMIN_PERMISSION = "web_admin";
31
+ const permissionSlugs = {
32
+ webAdmin: ADMIN_PERMISSION,
33
+ dev: DEV_PERMISSION,
34
+ tester: TESTER_PERMISSION,
35
+ free: FREE_PERMISSION,
36
+ freemium: FREEMIUM_PERMISSION,
37
+ basic: BASIC_PERMISSION
38
+ };
39
+ const PERMISSIONS = [
40
+ {
41
+ id: 5,
42
+ slug: FREE_PERMISSION,
43
+ title: "Free"
44
+ },
45
+ {
46
+ id: 6,
47
+ slug: FREEMIUM_PERMISSION,
48
+ title: "Freemium"
49
+ },
50
+ {
51
+ id: 3,
52
+ slug: TESTER_PERMISSION,
53
+ title: "Tester"
54
+ },
55
+ {
56
+ id: 2,
57
+ slug: DEV_PERMISSION,
58
+ title: "Developer"
59
+ },
60
+ {
61
+ id: 1,
62
+ slug: BASIC_PERMISSION,
63
+ title: "Basic"
64
+ },
65
+ {
66
+ id: 4,
67
+ slug: ADMIN_PERMISSION,
68
+ title: "Admin"
69
+ }
70
+ ];
71
+ const MAX_UNLOCKS_PER_WEEK = 3;
72
+ const PricingType = {
73
+ one_time: "One Time",
74
+ recurring: "Recurring"
75
+ };
76
+ const PricingPlanInterval = {
77
+ day: "Day",
78
+ week: "Week",
79
+ month: "Month",
80
+ year: "Year"
81
+ };
82
+ const formatOptions = (obj) => Object.entries(obj).map(([key, value]) => ({
83
+ value: key,
84
+ label: value
85
+ }));
86
+
87
+ //#endregion
88
+ //#region src/model/get-user-permissions.ts
89
+ /**
90
+ * Obtiene los permisos de un usuario basados en su inventario y suscripciones activas
91
+ */
92
+ const getUserPermissions = (user) => {
93
+ if (!user) return [];
94
+ const inventory = (user?.customer)?.inventory;
95
+ if (!inventory) return [];
96
+ return Object.values(inventory.subscriptions)?.filter((subscription) => subscription.status === "active" || subscription.status === "trialing")?.flatMap((subscription) => subscription.permissions);
97
+ };
98
+
99
+ //#endregion
100
+ //#region src/model/is-content-unlocked.ts
101
+ const isContentUnlocked = (user, contentId, collection) => {
102
+ if (!user?.inventory?.unlocks) return false;
103
+ return user.inventory.unlocks.some((unlock) => unlock.id === contentId && unlock.collection === collection);
104
+ };
105
+
106
+ //#endregion
107
+ //#region src/model/eval-permission-by-role-query.ts
108
+ const evalPermissionByRoleQuery = ({ user, permissions, content }) => {
109
+ const userPermissions = getUserPermissions(user);
110
+ if (!permissions || permissions.length === 0) return true;
111
+ if (permissions.includes(permissionSlugs.free)) return true;
112
+ if (user && content?.id ? isContentUnlocked(user, content.id, content.collection) : false) return true;
113
+ return permissions.some((permission) => permission && userPermissions.includes(permission));
114
+ };
115
+
116
+ //#endregion
117
+ //#region src/model/eval-advance-permissionQuery.ts
118
+ const evalAdvancePermissionQuery = ({ user, typeOfPermission, permissions }) => {
119
+ if (typeOfPermission === QUERY_PERMISSION_TYPES.ALL) return true;
120
+ else if (typeOfPermission === QUERY_PERMISSION_TYPES.ROLES) return evalPermissionByRoleQuery({
121
+ user,
122
+ permissions
123
+ });
124
+ else if (typeOfPermission === QUERY_PERMISSION_TYPES.ONLY_NO_ROLES) return getUserPermissions(user).length === 0;
125
+ else if (typeOfPermission === QUERY_PERMISSION_TYPES.ONLY_GUESS) return user === null;
126
+ return true;
127
+ };
128
+
129
+ //#endregion
130
+ //#region src/model/permissions.ts
131
+ const getPermissionsSlugs = ({ permissions }) => {
132
+ return permissions?.mapNotNull((p) => typeof p === "number" ? null : p.slug) ?? [];
133
+ };
134
+
135
+ //#endregion
136
+ //#region src/model/fetch-permitted-content-query.ts
137
+ /**
138
+ * Filtra contenido basado en los permisos del usuario
139
+ */
140
+ const fetchPermittedContentQuery = (user, permissions, content, freeContent = null) => {
141
+ const permissionsSlugs = getPermissionsSlugs({ permissions });
142
+ const isFreeContent = permissionsSlugs.includes(permissionSlugs.free);
143
+ const hasPermission = evalPermissionByRoleQuery({
144
+ user,
145
+ permissions: permissionsSlugs
146
+ });
147
+ if (isFreeContent) return content;
148
+ if (hasPermission) return content;
149
+ return freeContent;
150
+ };
151
+
152
+ //#endregion
153
+ //#region src/model/check-if-user-can-unlock-query.ts
154
+ /**
155
+ * Verifica si un usuario puede desbloquear un elemento basado en sus permisos y límites semanales
156
+ * @param user Usuario base
157
+ * @param permissions Permisos requeridos para el elemento
158
+ * @returns Booleano indicando si el usuario puede desbloquear el elemento
159
+ */
160
+ const checkIfUserCanUnlockQuery = (user, permissions) => {
161
+ return permissions.flatMap((item) => item.split(" ")).includes(permissionSlugs.freemium) && !permissions.includes(permissionSlugs.free);
162
+ };
163
+
164
+ //#endregion
165
+ //#region src/model/get-count-weekly-unlocks-query.ts
166
+ /**
167
+ * Cuenta cuántos elementos ha desbloqueado el usuario en los últimos 7 días
168
+ * @param user Usuario base
169
+ * @returns Número de elementos desbloqueados en los últimos 7 días
170
+ */
171
+ const countWeeklyUnlocksQuery = (user) => {
172
+ const inventory = user.inventory;
173
+ if (!inventory || !inventory.unlocks || inventory.unlocks.length === 0) return 0;
174
+ const sevenDaysAgo = /* @__PURE__ */ new Date();
175
+ sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);
176
+ return inventory.unlocks.filter((unlock) => new Date(unlock.dateUnlocked) >= sevenDaysAgo).length;
177
+ };
178
+
179
+ //#endregion
180
+ //#region src/model/get-next-unlock-date-query.ts
181
+ const getNextUnlockDateQuery = (user) => {
182
+ const inventory = user.inventory;
183
+ if (!inventory || !inventory.unlocks || inventory.unlocks.length === 0) return /* @__PURE__ */ new Date();
184
+ const lastUnlock = inventory.unlocks.sort((a, b) => new Date(a.dateUnlocked).getTime() - new Date(b.dateUnlocked).getTime())[inventory.unlocks.length - 1];
185
+ const lastUnlockDate = lastUnlock ? new Date(lastUnlock.dateUnlocked) : /* @__PURE__ */ new Date();
186
+ const nextUnlockDate = new Date(lastUnlockDate);
187
+ nextUnlockDate.setDate(lastUnlockDate.getDate() + 7);
188
+ return nextUnlockDate;
189
+ };
190
+
191
+ //#endregion
192
+ export { generateUserInventory as S, PricingType as _, getPermissionsSlugs as a, permissionSlugs as b, isContentUnlocked as c, COLLECTION_SLUG_PRICES as d, COLLECTION_SLUG_PRODUCTS as f, PricingPlanInterval as g, PERMISSIONS as h, fetchPermittedContentQuery as i, getUserPermissions as l, MAX_UNLOCKS_PER_WEEK as m, countWeeklyUnlocksQuery as n, evalAdvancePermissionQuery as o, COLLECTION_SLUG_USER as p, checkIfUserCanUnlockQuery as r, evalPermissionByRoleQuery as s, getNextUnlockDateQuery as t, COLLECTION_SLUG_CUSTOMERS as u, QUERY_PERMISSION_TYPES as v, generateCustomerInventory as x, formatOptions as y };
193
+ //# sourceMappingURL=src-BmlQoR4x.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"src-BmlQoR4x.mjs","names":[],"sources":["../src/model/builders.ts","../src/model/constants.ts","../src/model/get-user-permissions.ts","../src/model/is-content-unlocked.ts","../src/model/eval-permission-by-role-query.ts","../src/model/eval-advance-permissionQuery.ts","../src/model/permissions.ts","../src/model/fetch-permitted-content-query.ts","../src/model/check-if-user-can-unlock-query.ts","../src/model/get-count-weekly-unlocks-query.ts","../src/model/get-next-unlock-date-query.ts"],"sourcesContent":["import { CustomerInventory, UserInventory } from \"../types/index.js\";\n\nexport const generateUserInventory = (): UserInventory => ({\n unlocks: [],\n favorites: [],\n});\n\nexport const generateCustomerInventory = (): CustomerInventory => ({\n subscriptions: {},\n products: {},\n payments: {},\n invoices: {},\n});\n","export const COLLECTION_SLUG_PRODUCTS = \"products\" as const;\nexport const COLLECTION_SLUG_PRICES = \"prices\" as const;\nexport const COLLECTION_SLUG_USER = 'users' as const\nexport const COLLECTION_SLUG_CUSTOMERS = 'customers' as const\n\nexport const QUERY_PERMISSION_TYPES = {\n ALL: \"all\",\n ROLES: \"roles\",\n ONLY_NO_ROLES: \"only_no_roles\",\n ONLY_GUESS: \"only_guess\",\n} as const;\n\nconst FREEMIUM_PERMISSION = \"freemium\"\nconst FREE_PERMISSION = \"free\"\nconst TESTER_PERMISSION = \"tester\"\nconst DEV_PERMISSION = \"dev\"\nconst BASIC_PERMISSION = \"basic\"\nconst ADMIN_PERMISSION = \"web_admin\"\n\nexport const permissionSlugs = {\n webAdmin: ADMIN_PERMISSION,\n dev: DEV_PERMISSION,\n tester: TESTER_PERMISSION,\n free: FREE_PERMISSION,\n freemium: FREEMIUM_PERMISSION,\n basic: BASIC_PERMISSION,\n}\n\nexport const PERMISSIONS = [\n { id: 5, slug: FREE_PERMISSION, title: 'Free' },\n { id: 6, slug: FREEMIUM_PERMISSION, title: 'Freemium' },\n { id: 3, slug: TESTER_PERMISSION, title: 'Tester' },\n { id: 2, slug: DEV_PERMISSION, title: 'Developer' },\n { id: 1, slug: BASIC_PERMISSION, title: 'Basic' },\n { id: 4, slug: ADMIN_PERMISSION, title: 'Admin' },\n]\nexport const MAX_UNLOCKS_PER_WEEK = 3 as const;\n\nexport const PricingType = {\n one_time: \"One Time\",\n recurring: \"Recurring\",\n} as const;\n\nexport const PricingPlanInterval = {\n day: \"Day\",\n week: \"Week\",\n month: \"Month\",\n year: \"Year\",\n} as const;\n\nexport const formatOptions = (obj: Record<string, string>) =>\n Object.entries(obj).map(([key, value]) => ({ value: key, label: value }));\n\n","import { BaseUser, Customer } from \"../types/index.js\";\n\n/**\n * Obtiene los permisos de un usuario basados en su inventario y suscripciones activas\n */\nexport const getUserPermissions = (user?: BaseUser | null): string[] => {\n if (!user) return [];\n\n const customer = user?.customer as Customer;\n const inventory = customer?.inventory;\n if (!inventory) return [];\n\n const subscriptionPermissions = Object.values(inventory.subscriptions)\n ?.filter(\n subscription =>\n subscription.status === \"active\" || subscription.status === \"trialing\"\n )\n ?.flatMap(subscription => subscription.permissions);\n\n return subscriptionPermissions;\n};\n","import { BaseUser, UnlockItem, UserInventory } from \"../types/index.js\";\n\nexport const isContentUnlocked = (\n user: BaseUser<UserInventory>,\n contentId: number,\n collection: string\n): boolean => {\n if (!user?.inventory?.unlocks) return false;\n\n return user.inventory.unlocks.some(\n (unlock: UnlockItem) =>\n unlock.id === contentId && unlock.collection === collection\n );\n};\n","import type { BaseUser } from \"../types/index.js\";\nimport { permissionSlugs } from \"./constants.js\";\nimport { getUserPermissions } from \"./get-user-permissions.js\";\nimport { isContentUnlocked } from \"./is-content-unlocked.js\";\n/**\n * Evalúa si un usuario tiene los permisos necesarios basados en las semillas de permisos\n */\ninterface Props<T extends BaseUser> {\n user: T | null | undefined;\n permissions?: string[] | null;\n content?: {\n collection: string;\n id: number;\n };\n}\n\nexport const evalPermissionByRoleQuery = <T extends BaseUser>({\n user,\n permissions,\n content,\n}: Props<T>): boolean => {\n const userPermissions = getUserPermissions(user);\n\n if (!permissions || permissions.length === 0) return true;\n if (permissions.includes(permissionSlugs.free)) return true;\n const isUnlocked =\n user && content?.id\n ? isContentUnlocked(user, content.id, content.collection)\n : false;\n if (isUnlocked) return true;\n return permissions.some(\n permission => permission && userPermissions.includes(permission)\n );\n};\n","import type { BaseUser } from \"../types/index.js\";\nimport { QUERY_PERMISSION_TYPES } from \"./constants.js\";\nimport { evalPermissionByRoleQuery } from \"./eval-permission-by-role-query.js\";\nimport { getUserPermissions } from \"./get-user-permissions.js\";\n\n/**\n * Evalúa permisos avanzados basados en el tipo de permiso y usuario\n */\ninterface Props<T extends BaseUser> {\n user: T | null;\n typeOfPermission: keyof typeof QUERY_PERMISSION_TYPES | string;\n permissions?: string[] | undefined;\n}\n\nexport const evalAdvancePermissionQuery = <T extends BaseUser>({\n user,\n typeOfPermission,\n permissions,\n}: Props<T>): boolean => {\n if (typeOfPermission === QUERY_PERMISSION_TYPES.ALL) {\n return true;\n } else if (typeOfPermission === QUERY_PERMISSION_TYPES.ROLES) {\n return evalPermissionByRoleQuery({\n user,\n permissions,\n });\n } else if (typeOfPermission === QUERY_PERMISSION_TYPES.ONLY_NO_ROLES) {\n const userPermissions = getUserPermissions(user);\n return userPermissions.length === 0;\n } else if (typeOfPermission === QUERY_PERMISSION_TYPES.ONLY_GUESS) {\n return user === null;\n }\n\n return true;\n};\n","import type { Permission } from \"../types/index.js\";\n\nexport const getPermissionsSlugs = ({\n permissions,\n}: {\n permissions?: (number | Permission)[] | null | undefined;\n}) => {\n return (\n permissions?.mapNotNull(p => (typeof p === \"number\" ? null : p.slug)) ?? []\n );\n};\n","import { BaseUser, Permission } from \"../types/index.js\";\nimport { permissionSlugs } from \"./constants.js\";\nimport { evalPermissionByRoleQuery } from \"./eval-permission-by-role-query.js\";\nimport { getPermissionsSlugs } from \"./permissions.js\";\n\n/**\n * Filtra contenido basado en los permisos del usuario\n */\nexport const fetchPermittedContentQuery = <T extends BaseUser, C>(\n user: T | null | undefined,\n permissions: (Permission | number)[] | null | undefined,\n content: C,\n freeContent: C | null = null\n): C | null => {\n const permissionsSlugs = getPermissionsSlugs({ permissions });\n const isFreeContent = permissionsSlugs.includes(permissionSlugs.free);\n const hasPermission = evalPermissionByRoleQuery({\n user,\n permissions: permissionsSlugs,\n });\n\n if (isFreeContent) {\n return content;\n }\n\n if (hasPermission) {\n return content;\n }\n\n return freeContent;\n};\n","import { BaseUser } from \"../types/index.js\";\nimport { permissionSlugs } from \"./constants.js\";\n\n/**\n * Verifica si un usuario puede desbloquear un elemento basado en sus permisos y límites semanales\n * @param user Usuario base\n * @param permissions Permisos requeridos para el elemento\n * @returns Booleano indicando si el usuario puede desbloquear el elemento\n */\nexport const checkIfUserCanUnlockQuery = (\n user: BaseUser,\n permissions: string[]\n): boolean => {\n return (\n permissions\n .flatMap(item => item.split(\" \"))\n .includes(permissionSlugs.freemium) &&\n !permissions.includes(permissionSlugs.free)\n );\n};\n","import { BaseUser, UserInventory } from \"../types/index.js\";\n/**\n * Cuenta cuántos elementos ha desbloqueado el usuario en los últimos 7 días\n * @param user Usuario base\n * @returns Número de elementos desbloqueados en los últimos 7 días\n */\n\nexport const countWeeklyUnlocksQuery = (\n user: BaseUser<UserInventory>\n): number => {\n const inventory = user.inventory;\n if (!inventory || !inventory.unlocks || inventory.unlocks.length === 0) {\n return 0;\n }\n\n const sevenDaysAgo = new Date();\n sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);\n\n return inventory.unlocks.filter(\n unlock => new Date(unlock.dateUnlocked) >= sevenDaysAgo\n ).length;\n};\n","import { BaseUser, UserInventory } from \"../types/index.js\";\n\nexport const getNextUnlockDateQuery = (user: BaseUser<UserInventory>): Date => {\n const inventory = user.inventory;\n if (!inventory || !inventory.unlocks || inventory.unlocks.length === 0) {\n return new Date();\n }\n\n const lastUnlock = inventory.unlocks.sort(\n (a, b) =>\n new Date(a.dateUnlocked).getTime() - new Date(b.dateUnlocked).getTime()\n )[inventory.unlocks.length - 1];\n\n const lastUnlockDate = lastUnlock ? new Date(lastUnlock.dateUnlocked) : new Date();\n const nextUnlockDate = new Date(lastUnlockDate);\n nextUnlockDate.setDate(lastUnlockDate.getDate() + 7);\n\n return nextUnlockDate;\n};\n"],"mappings":";AAEA,MAAa,+BAA8C;CACzD,SAAS,EAAE;CACX,WAAW,EAAE;CACd;AAED,MAAa,mCAAsD;CACjE,eAAe,EAAE;CACjB,UAAU,EAAE;CACZ,UAAU,EAAE;CACZ,UAAU,EAAE;CACb;;;;ACZD,MAAa,2BAA2B;AACxC,MAAa,yBAAyB;AACtC,MAAa,uBAAuB;AACpC,MAAa,4BAA4B;AAEzC,MAAa,yBAAyB;CACpC,KAAK;CACL,OAAO;CACP,eAAe;CACf,YAAY;CACb;AAED,MAAM,sBAAsB;AAC5B,MAAM,kBAAkB;AACxB,MAAM,oBAAoB;AAC1B,MAAM,iBAAiB;AACvB,MAAM,mBAAmB;AACzB,MAAM,mBAAmB;AAEzB,MAAa,kBAAkB;CAC7B,UAAU;CACV,KAAK;CACL,QAAQ;CACR,MAAM;CACN,UAAU;CACV,OAAO;CACR;AAED,MAAa,cAAc;CACzB;EAAE,IAAI;EAAG,MAAM;EAAiB,OAAO;EAAQ;CAC/C;EAAE,IAAI;EAAG,MAAM;EAAqB,OAAO;EAAY;CACvD;EAAE,IAAI;EAAG,MAAM;EAAmB,OAAO;EAAU;CACnD;EAAE,IAAI;EAAG,MAAM;EAAgB,OAAO;EAAa;CACnD;EAAE,IAAI;EAAG,MAAM;EAAkB,OAAO;EAAS;CACjD;EAAE,IAAI;EAAG,MAAM;EAAkB,OAAO;EAAS;CAClD;AACD,MAAa,uBAAuB;AAEpC,MAAa,cAAc;CACzB,UAAU;CACV,WAAW;CACZ;AAED,MAAa,sBAAsB;CACjC,KAAK;CACL,MAAM;CACN,OAAO;CACP,MAAM;CACP;AAED,MAAa,iBAAiB,QAC5B,OAAO,QAAQ,IAAI,CAAC,KAAK,CAAC,KAAK,YAAY;CAAE,OAAO;CAAK,OAAO;CAAO,EAAE;;;;;;;AC9C3E,MAAa,sBAAsB,SAAqC;AACtE,KAAI,CAAC,KAAM,QAAO,EAAE;CAGpB,MAAM,aADW,MAAM,WACK;AAC5B,KAAI,CAAC,UAAW,QAAO,EAAE;AASzB,QAPgC,OAAO,OAAO,UAAU,cAAc,EAClE,QACA,iBACE,aAAa,WAAW,YAAY,aAAa,WAAW,WAC/D,EACC,SAAQ,iBAAgB,aAAa,YAAY;;;;;ACfvD,MAAa,qBACX,MACA,WACA,eACY;AACZ,KAAI,CAAC,MAAM,WAAW,QAAS,QAAO;AAEtC,QAAO,KAAK,UAAU,QAAQ,MAC3B,WACC,OAAO,OAAO,aAAa,OAAO,eAAe,WACpD;;;;;ACIH,MAAa,6BAAiD,EAC5D,MACA,aACA,cACuB;CACvB,MAAM,kBAAkB,mBAAmB,KAAK;AAEhD,KAAI,CAAC,eAAe,YAAY,WAAW,EAAG,QAAO;AACrD,KAAI,YAAY,SAAS,gBAAgB,KAAK,CAAE,QAAO;AAKvD,KAHE,QAAQ,SAAS,KACb,kBAAkB,MAAM,QAAQ,IAAI,QAAQ,WAAW,GACvD,MACU,QAAO;AACvB,QAAO,YAAY,MACjB,eAAc,cAAc,gBAAgB,SAAS,WAAW,CACjE;;;;;AClBH,MAAa,8BAAkD,EAC7D,MACA,kBACA,kBACuB;AACvB,KAAI,qBAAqB,uBAAuB,IAC9C,QAAO;UACE,qBAAqB,uBAAuB,MACrD,QAAO,0BAA0B;EAC/B;EACA;EACD,CAAC;UACO,qBAAqB,uBAAuB,cAErD,QADwB,mBAAmB,KAAK,CACzB,WAAW;UACzB,qBAAqB,uBAAuB,WACrD,QAAO,SAAS;AAGlB,QAAO;;;;;AC/BT,MAAa,uBAAuB,EAClC,kBAGI;AACJ,QACE,aAAa,YAAW,MAAM,OAAO,MAAM,WAAW,OAAO,EAAE,KAAM,IAAI,EAAE;;;;;;;;ACA/E,MAAa,8BACX,MACA,aACA,SACA,cAAwB,SACX;CACb,MAAM,mBAAmB,oBAAoB,EAAE,aAAa,CAAC;CAC7D,MAAM,gBAAgB,iBAAiB,SAAS,gBAAgB,KAAK;CACrE,MAAM,gBAAgB,0BAA0B;EAC9C;EACA,aAAa;EACd,CAAC;AAEF,KAAI,cACF,QAAO;AAGT,KAAI,cACF,QAAO;AAGT,QAAO;;;;;;;;;;;ACpBT,MAAa,6BACX,MACA,gBACY;AACZ,QACE,YACG,SAAQ,SAAQ,KAAK,MAAM,IAAI,CAAC,CAChC,SAAS,gBAAgB,SAAS,IACrC,CAAC,YAAY,SAAS,gBAAgB,KAAK;;;;;;;;;;ACV/C,MAAa,2BACX,SACW;CACX,MAAM,YAAY,KAAK;AACvB,KAAI,CAAC,aAAa,CAAC,UAAU,WAAW,UAAU,QAAQ,WAAW,EACnE,QAAO;CAGT,MAAM,+BAAe,IAAI,MAAM;AAC/B,cAAa,QAAQ,aAAa,SAAS,GAAG,EAAE;AAEhD,QAAO,UAAU,QAAQ,QACvB,WAAU,IAAI,KAAK,OAAO,aAAa,IAAI,aAC5C,CAAC;;;;;AClBJ,MAAa,0BAA0B,SAAwC;CAC7E,MAAM,YAAY,KAAK;AACvB,KAAI,CAAC,aAAa,CAAC,UAAU,WAAW,UAAU,QAAQ,WAAW,EACnE,wBAAO,IAAI,MAAM;CAGnB,MAAM,aAAa,UAAU,QAAQ,MAClC,GAAG,MACF,IAAI,KAAK,EAAE,aAAa,CAAC,SAAS,GAAG,IAAI,KAAK,EAAE,aAAa,CAAC,SAAS,CAC1E,CAAC,UAAU,QAAQ,SAAS;CAE7B,MAAM,iBAAiB,aAAa,IAAI,KAAK,WAAW,aAAa,mBAAG,IAAI,MAAM;CAClF,MAAM,iBAAiB,IAAI,KAAK,eAAe;AAC/C,gBAAe,QAAQ,eAAe,SAAS,GAAG,EAAE;AAEpD,QAAO"}
package/package.json ADDED
@@ -0,0 +1,63 @@
1
+ {
2
+ "name": "@nexo-labs/payload-stripe-inventory",
3
+ "version": "1.6.1",
4
+ "description": "Sistema de inventario y suscripciones con Stripe para Payload CMS",
5
+ "keywords": [
6
+ "payload-cms",
7
+ "stripe",
8
+ "payments",
9
+ "subscriptions",
10
+ "inventory",
11
+ "e-commerce"
12
+ ],
13
+ "license": "MIT",
14
+ "author": "",
15
+ "sideEffects": false,
16
+ "type": "module",
17
+ "exports": {
18
+ ".": {
19
+ "import": "./dist/index.js",
20
+ "types": "./dist/index.d.ts",
21
+ "default": "./dist/index.js"
22
+ },
23
+ "./server": {
24
+ "import": "./dist/server/index.js",
25
+ "types": "./dist/server/index.d.ts",
26
+ "default": "./dist/server/index.js"
27
+ }
28
+ },
29
+ "main": "./dist/index.js",
30
+ "types": "./dist/index.d.ts",
31
+ "files": [
32
+ "dist",
33
+ "README.md"
34
+ ],
35
+ "peerDependencies": {
36
+ "@payloadcms/plugin-stripe": "3.69.0",
37
+ "next": "15.4.8",
38
+ "payload": "3.69.0",
39
+ "react": "^19.0.0",
40
+ "stripe": "17.2.1",
41
+ "@nexo-labs/hegel": "1.6.1",
42
+ "@nexo-labs/payload-taxonomies": "1.6.1"
43
+ },
44
+ "devDependencies": {
45
+ "@types/node": "^22.0.0",
46
+ "@types/react": "npm:types-react@19.0.0-rc.1",
47
+ "copyfiles": "^2.4.1",
48
+ "next": "15.4.10",
49
+ "rimraf": "^5.0.5",
50
+ "tsdown": "^0.16.5",
51
+ "typescript": "^5.0.0"
52
+ },
53
+ "publishConfig": {
54
+ "registry": "https://registry.npmjs.org/",
55
+ "access": "public"
56
+ },
57
+ "scripts": {
58
+ "build": "pnpm copyfiles && tsdown",
59
+ "dev": "tsdown --watch",
60
+ "clean": "rimraf -g {dist,*.tsbuildinfo}",
61
+ "copyfiles": "copyfiles -u 1 \"src/**/*.{html,css,scss,ttf,woff,woff2,eot,svg,jpg,png,json}\" dist/ || true"
62
+ }
63
+ }