From 738833a5f086f9f751fb45a267af2d7786c6e7f6 Mon Sep 17 00:00:00 2001
From: Spencer Lepine <60903378+spencerlepine@users.noreply.github.com>
Date: Sat, 26 Oct 2024 22:55:48 +0300
Subject: [PATCH] refactor: use Stripe embedded checkout form
---
next.config.mjs | 9 +-
package.json | 15 +-
src/app/api/v1/checkout/route.ts | 92 +----
src/app/api/v1/webhook/checkout/route.ts | 58 ++--
src/app/cart/page.tsx | 39 ++-
src/app/order-confirmation/page.tsx | 38 +-
src/components/CheckoutBtn.tsx | 46 ---
src/envVars.js | 16 +
src/lib/printify.ts | 98 ++++--
src/lib/products.ts | 1 +
src/lib/stripe.ts | 134 +++----
src/types/index.ts | 52 +++
yarn.lock | 425 +++++++++++++++--------
13 files changed, 572 insertions(+), 451 deletions(-)
delete mode 100644 src/components/CheckoutBtn.tsx
create mode 100644 src/envVars.js
diff --git a/next.config.mjs b/next.config.mjs
index 4678774..12d8a12 100644
--- a/next.config.mjs
+++ b/next.config.mjs
@@ -1,4 +1,11 @@
+import './src/envVars.js';
+
/** @type {import('next').NextConfig} */
-const nextConfig = {};
+const nextConfig = {
+ reactStrictMode: true,
+ swcMinify: true,
+ output: 'standalone', // for Docker
+ trailingSlash: false,
+};
export default nextConfig;
diff --git a/package.json b/package.json
index 1ace801..6905afe 100644
--- a/package.json
+++ b/package.json
@@ -16,20 +16,25 @@
],
"scripts": {
"dev": "next dev",
- "build": "NODE_ENV=production next build",
+ "build": "NODE_ENV=production next build && npm run copy:assets",
+ "comment": "TODO_DEPLOYMENT - serve assets from CDN instead",
+ "copy:assets": "cp -rn public .next/standalone && cp -rn .next/static .next/standalone/.next/static",
"test": "echo 'No tests added yet' && exit 0",
- "start": "next start",
+ "start": "node .next/standalone/server.js",
"lint": "next lint"
},
"dependencies": {
+ "@stripe/react-stripe-js": "^2.8.1",
"fuse.js": "^7.0.0",
"next": "14.2.11",
- "printify-sdk-js": "^1.0.1",
+ "printify-sdk-js": "1.0.2",
"react": "^18",
"react-dom": "^18",
- "sharp": "^0.33.5",
+ "sharp": "0.32.6",
"stripe": "^16.9.0",
- "use-shopping-cart": "^3.2.0"
+ "use-shopping-cart": "^3.2.0",
+ "winston": "^3.15.0",
+ "zod": "^3.23.8"
},
"devDependencies": {
"@types/node": "^20",
diff --git a/src/app/api/v1/checkout/route.ts b/src/app/api/v1/checkout/route.ts
index dd92414..d49e5ac 100644
--- a/src/app/api/v1/checkout/route.ts
+++ b/src/app/api/v1/checkout/route.ts
@@ -1,98 +1,28 @@
-import logger from '@/lib/logger';
-import { createDraftOrder } from '@/lib/printify';
+import { NextResponse } from 'next/server';
import { createCheckoutSession } from '@/lib/stripe';
import { UserError } from '@/utils/errors';
+import logger from '@/lib/logger';
+import { retrieveShippingCost } from '@/lib/printify';
import validateCartItems from '@/utils/validateCartItems';
-import { NextRequest, NextResponse } from 'next/server';
-/**
- * @openapi
- * /v1/checkout:
- * post:
- * summary: Creates a Stripe checkout session.
- * tags:
- * - Checkout
- * requestBody:
- * required: true
- * content:
- * application/json:
- * schema:
- * type: object
- * properties:
- * cartItems:
- * type: array
- * items:
- * type: object
- * properties:
- * id:
- * type: string
- * name:
- * type: string
- * description:
- * type: string
- * quantity:
- * type: integer
- * price:
- * type: number
- * image:
- * type: string
- * currency:
- * type: string
- * price_data:
- * type: object
- * product_data:
- * type: object
- * properties:
- * size:
- * type: string
- * productId:
- * type: string
- * category:
- * type: string
- * type: string
- * responses:
- * 200:
- * description: Checkout session created successfully.
- * content:
- * application/json:
- * schema:
- * type: object
- * properties:
- * checkoutUrl:
- * type: string
- * 400:
- * description: Error creating checkout session.
- * content:
- * application/json:
- * schema:
- * type: object
- * properties:
- * message:
- * type: string
- */
-export const POST = async (request: NextRequest) => {
+export async function POST(request: Request) {
const correlationId = request.headers.get('x-correlation-id');
try {
- logger.info('[Checkout] Processing checkout request', { correlationId });
-
const body = await request.json();
const { cartItems: clientCartItems } = body;
logger.info('[Checkout] Validating cart items', { correlationId });
const cartItems = validateCartItems(clientCartItems);
- // TODO_PRINTIFY (move this to final webhook)
- logger.info('[Printify] Creating Printify draft order', { correlationId });
- const { id: printifyOrderId } = await createDraftOrder(cartItems);
-
- // TODO_PRINTIFY (calulateShipping())
+ logger.info('[Printify] calculating shipping methods', { correlationId });
+ const shippingMethods = await retrieveShippingCost();
- logger.info('[Stripe] Creating checkout session', { correlationId, printifyOrderId });
- const session = await createCheckoutSession(cartItems, { printifyOrderId });
+ const swagOrderId = crypto.randomUUID();
+ logger.info('[Stripe] Creating checkout session', { correlationId, swagOrderId });
+ const session = await createCheckoutSession(cartItems, shippingMethods, { swagOrderId });
- logger.info('[Checkout] Checkout session created successfully', { checkoutUrl: session.url, correlationId, printifyOrderId, sessionId: session.id });
- return NextResponse.json({ checkoutUrl: session.url });
+ return NextResponse.json({ id: session.id, client_secret: session.client_secret });
} catch (error) {
if (error instanceof UserError) {
return NextResponse.json({ error: error.message }, { status: 400 });
@@ -101,4 +31,4 @@ export const POST = async (request: NextRequest) => {
logger.error('[Checkout] Error processing checkout request', { error });
return NextResponse.json({ error: 'Internal server error' }, { status: 500 });
}
-};
+}
diff --git a/src/app/api/v1/webhook/checkout/route.ts b/src/app/api/v1/webhook/checkout/route.ts
index 9a6defa..8bb0ee8 100644
--- a/src/app/api/v1/webhook/checkout/route.ts
+++ b/src/app/api/v1/webhook/checkout/route.ts
@@ -1,24 +1,14 @@
import type { Stripe as StripeType } from 'stripe';
-import { retrieveCheckoutSession, stripe } from '@/lib/stripe';
+import { stripe } from '@/lib/stripe';
import { NextRequest, NextResponse } from 'next/server';
-import { sendOrderToProduction } from '@/lib/printify';
import logger from '@/lib/logger';
+import { createDraftOrder } from '@/lib/printify';
export const POST = async (request: NextRequest) => {
- const correlationId = request.headers.get('x-correlation-id');
-
try {
- logger.info('[Stripe Webhook] Processing Stripe webhook request');
-
- const secret = process.env.STRIPE_WEBHOOK_SECRET || '';
- if (!secret) {
- logger.error('[Stripe Webhook] Missing STRIPE_WEBHOOK_SECRET environment variable');
- throw new Error('Missing STRIPE_WEBHOOK_SECRET environment variable');
- }
-
const body = await (await request.blob()).text();
const signature = request.headers.get('stripe-signature') as string;
- const event: StripeType.Event = stripe.webhooks.constructEvent(body, signature, secret);
+ const event: StripeType.Event = stripe.webhooks.constructEvent(body, signature, process.env.STRIPE_WEBHOOK_SECRET as string);
const permittedEvents: string[] = ['checkout.session.completed', 'payment_intent.succeeded', 'payment_intent.payment_failed'];
if (permittedEvents.includes(event.type)) {
@@ -27,45 +17,49 @@ export const POST = async (request: NextRequest) => {
switch (event.type) {
case 'checkout.session.completed':
data = event.data.object as StripeType.Checkout.Session;
+ logger.info('[Stripe Webhook] Checkout succeeded', { eventType: event.type, sessionId: data.id, eventId: event.id });
- const printifyOrderId = event.data.object?.metadata?.printifyOrderId;
- if (!printifyOrderId) {
- logger.warn(`[Stripe Webhook] Missing printifyOrderId in metadata for session ${data.id}. Unable to fullfil order`, { sessionId: data.id, correlationId });
- throw new Error(`missing printifyOrderId on metadata, ${data.id}`);
- }
+ logger.info('[Stripe Webhook] Retrieving line_items', { sessionId: data.id });
+ const line_items = await stripe.checkout.sessions.listLineItems(data.id, {
+ expand: ['data.price.product'],
+ });
+ const email = data?.customer_details?.email || '';
+ const phone = data?.customer_details?.phone || '';
- logger.info('[Stripe Webhook] Verifying payment status for Checkout Session', { sessionId: data.id, correlationId });
- const checkoutSession = await retrieveCheckoutSession(data.id);
- if (checkoutSession.payment_status === 'unpaid') {
- logger.warn('[Stripe Webhook] Cannot fulfill an unpaid order', { sessionId: data.id, correlationId });
- return NextResponse.json({ message: 'Cannot fullfil an unpaid order' }, { status: 400 });
+ const swagOrderId = data.metadata?.swagOrderId;
+ if (!swagOrderId) {
+ logger.warn(`[Stripe Webhook] Missing swagOrderId in metadata for session ${data.id}. Unable to fullfil order`, { sessionId: data.id, eventId: event.id });
+ throw new Error(`missing swagOrderId on metadata, ${data.id}`);
}
- logger.info('[Printify] Sending order to production', { sessionId: data.id, correlationId, printifyOrderId });
- await sendOrderToProduction(printifyOrderId);
+ logger.info('[Printify] Creating Printify draft order');
+ const { id: printifyOrderId } = await createDraftOrder(line_items.data, data.shipping_details, swagOrderId, data.id, email, phone);
- logger.info('[Stripe Webhook] Successfully fulfilled order', { sessionId: data.id, correlationId, printifyOrderId });
- return NextResponse.json({ result: event, ok: true });
+ await new Promise(r => setTimeout(r, 3000));
+
+ // TODO_PRINTIFY - validate publish endpoint! - curl request always works..
+ logger.info('[Printify] Fullfilling order', { eventType: event.type, sessionId: data.id, printifyOrderId, eventId: event.id });
+ // await sendOrderToProduction(printifyOrderId);
+ break;
case 'payment_intent.payment_failed':
data = event.data.object as StripeType.PaymentIntent;
- logger.error('[Stripe Webhook] Payment failed', { message: data.last_payment_error?.message, sessionId: data.id });
+ logger.error('[Stripe Webhook] Payment failed', { message: data.last_payment_error?.message, eventType: event.type, sessionId: data.id, eventId: event.id });
break;
case 'payment_intent.succeeded':
data = event.data.object as StripeType.PaymentIntent;
- logger.info('[Stripe Webhook] PaymentIntent succeeded', { status: data.status, sessionId: data.id });
+ logger.info('[Stripe Webhook] PaymentIntent succeeded', { eventType: event.type, status: data.status, sessionId: data.id, eventId: event.id });
break;
default:
+ // fallback, not used
data = (event.data.object as unknown) || {};
- // @ts-expect-error - ignore "Property 'id' does not exist on type '{}'.ts(2339)"
- logger.warn('[Stripe Webhook] Unhandled event type', { eventType: event.type, sessionId: data?.id });
return NextResponse.json({ result: event, ok: true });
}
}
- logger.info('[Stripe Webhook] Webhook processing complete', { eventId: event.id });
+ logger.info('[Stripe Webhook] Processed request', { eventType: event.type });
return NextResponse.json({ result: event, ok: true });
} catch (error) {
logger.error('[Stripe Webhook] Error processing webhook request', { error });
diff --git a/src/app/cart/page.tsx b/src/app/cart/page.tsx
index e2fa624..53bf47d 100644
--- a/src/app/cart/page.tsx
+++ b/src/app/cart/page.tsx
@@ -1,15 +1,20 @@
'use client';
+import React, { useState } from 'react';
+import { loadStripe } from '@stripe/stripe-js';
+import { EmbeddedCheckoutProvider, EmbeddedCheckout } from '@stripe/react-stripe-js';
import CartItemCard from '@/components/CartItemCard';
import formatPriceForDisplay from '@/utils/formatPriceForDisplay';
import { useShoppingCart } from 'use-shopping-cart';
import { CartItem } from '@/types';
-import CheckoutButton from '@/components/CheckoutBtn';
export default function CartPage() {
const { cartCount, cartDetails, removeItem, totalPrice, addItem, decrementItem } = useShoppingCart();
const cartItems = Object.values(cartDetails ?? {});
+ const stripePromise = loadStripe(process.env.NEXT_PUBLIC_STRIPE_KEY!);
+ const [showCheckout, setShowCheckout] = useState(false);
+
const handleRemove = (cartItem: CartItem) => {
if (cartItem.quantity === 1) {
removeItem(cartItem.id);
@@ -18,12 +23,42 @@ export default function CartPage() {
}
};
+ const fetchClientSecret = () => {
+ return fetch('/api/v1/checkout', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json',
+ },
+ body: JSON.stringify({ cartItems }),
+ })
+ .then(res => res.json())
+ .then(data => data.client_secret);
+ };
+
+ const options = { fetchClientSecret };
+
+ const handleCheckoutClick = () => {
+ setShowCheckout(true);
+ };
+
+ if (showCheckout) {
+ return (
+
+
+
+
+
+ );
+ }
+
const subtotal = formatPriceForDisplay(totalPrice);
return (
Subtotal: {subtotal}
-
+
{!cartItems ||
diff --git a/src/app/order-confirmation/page.tsx b/src/app/order-confirmation/page.tsx
index 4c64e6f..c0e44f5 100644
--- a/src/app/order-confirmation/page.tsx
+++ b/src/app/order-confirmation/page.tsx
@@ -1,20 +1,38 @@
-import { memo } from 'react';
import OrderConfirmation from '@/components/OrderConfirmation';
-import { validateStripeSession } from '@/lib/stripe';
+import { stripe } from '@/lib/stripe';
import { notFound } from 'next/navigation';
+async function getSession(sessionId: string) {
+ try {
+ const session = await stripe.checkout.sessions.retrieve(sessionId!);
+ return session;
+ } catch (error) {
+ return null;
+ }
+}
+
// Redirect page after successful checkout
//
/order-confirmation?session_id=cs_test_b1FKoQomBOaQFgMqW1lU6oYXRwIruD6AbGV804gMZRrptJr1bF91sDmK5T
const OrderConfirmationPage: React.FC<{ searchParams: { [key: string]: string | undefined } }> = async ({ searchParams }) => {
const sessionId = searchParams['session_id'];
- const { validSession } = await validateStripeSession(sessionId);
- if (!validSession) return notFound();
+ if (!sessionId) return notFound();
+
+ const session = await getSession(sessionId);
+ if (!session) return notFound();
+
+ if (session?.status === 'open') {
+ return Payment did not work.
;
+ }
+
+ if (session?.status === 'complete') {
+ return (
+
+
+
+ );
+ }
- return (
-
-
-
- );
+ return notFound();
};
-export default memo(OrderConfirmationPage);
+export default OrderConfirmationPage;
diff --git a/src/components/CheckoutBtn.tsx b/src/components/CheckoutBtn.tsx
deleted file mode 100644
index 116500a..0000000
--- a/src/components/CheckoutBtn.tsx
+++ /dev/null
@@ -1,46 +0,0 @@
-'use client';
-
-import { useRouter } from 'next/navigation';
-import { useRef, useState } from 'react';
-import { CartEntry } from 'use-shopping-cart/core';
-
-const CheckoutButton: React.FC<{ cartCount?: number; cartItems: CartEntry[] }> = ({ cartCount, cartItems }) => {
- const router = useRouter();
- const isProcessing = useRef(false);
- const [error, setError] = useState('');
-
- const handleCheckoutClick = async () => {
- if (!cartCount || cartCount === 0 || isProcessing.current) {
- return;
- }
-
- isProcessing.current = true;
-
- try {
- const res = await fetch('/api/v1/checkout', {
- method: 'POST',
- body: JSON.stringify({ cartItems }),
- });
-
- const { checkoutUrl } = await res.json();
- if (!checkoutUrl) {
- setError('Unable to checkout at this time. Please try again later.');
- return;
- }
- router.push(checkoutUrl);
- } catch (err) {
- console.error('Checkout request threw an error', err, error);
- setError('An error occurred during checkout. Please try again later.');
- } finally {
- isProcessing.current = false;
- }
- };
-
- return (
-
- );
-};
-
-export default CheckoutButton;
diff --git a/src/envVars.js b/src/envVars.js
new file mode 100644
index 0000000..66d58f9
--- /dev/null
+++ b/src/envVars.js
@@ -0,0 +1,16 @@
+/* eslint @typescript-eslint/no-var-requires: 0 */
+const { z } = require('zod');
+
+const EnvVars = z.object({
+ NEXT_PUBLIC_URL: z.string(),
+ NEXT_PUBLIC_STRIPE_KEY: z.string(),
+ STRIPE_SECRET_KEY: z.string(),
+ STRIPE_WEBHOOK_SECRET: z.string(),
+ PRINTIFY_API_TOKEN: z.string(),
+ PRINTIFY_SHOP_ID: z.string(),
+});
+
+const secrets = EnvVars.parse(process.env);
+console.log('Validated .env secrets');
+
+module.exports.config = secrets;
diff --git a/src/lib/printify.ts b/src/lib/printify.ts
index e7776cd..979e9f6 100644
--- a/src/lib/printify.ts
+++ b/src/lib/printify.ts
@@ -1,6 +1,7 @@
import Printify from 'printify-sdk-js';
-import { retrieveStickerPNGFileUrl, STICKER_SIZES } from '@/lib/products';
-import { CartItem, PrintifyLineItem, SubmitOrderData } from '@/types';
+import { PRODUCT_CONFIG, retrieveStickerPNGFileUrl, STICKER_SIZES } from '@/lib/products';
+import { PrintifyLineItem, PrintifyShippingProfile, StripeShippingDetails, SubmitOrderData, VariantShippingData } from '@/types';
+import type { Stripe as StripeType } from 'stripe';
import logger from './logger';
// docs: https://developers.printify.com/
@@ -21,55 +22,88 @@ const PRINTIFY_VARIANT_IDS = {
[STICKER_SIZES.FOUR_BY_FOUR_IN]: 95745,
};
-export const formatCartItemsForPrintify = (cartItems: CartItem[]): PrintifyLineItem[] => {
- return cartItems.map(cartItem => ({
+// export const formatCartItemsForPrintify = (cartItems: CartItem[]): PrintifyLineItem[] => {
+// return cartItems.map(cartItem => ({
+// print_provider_id: PRINTIFY_PRINT_PROVIDER_ID,
+// blueprint_id: PRINTIFY_BLUEPRINT_ID,
+// variant_id: PRINTIFY_VARIANT_IDS[cartItem.product_data.size],
+// print_areas: {
+// front: retrieveStickerPNGFileUrl(cartItem.product_data.productId),
+// },
+// quantity: cartItem.quantity,
+// }));
+// };
+
+export const formatCartItemsForPrintify = (lineItems: StripeType.LineItem[]): PrintifyLineItem[] => {
+ return lineItems.map(item => ({
print_provider_id: PRINTIFY_PRINT_PROVIDER_ID,
blueprint_id: PRINTIFY_BLUEPRINT_ID,
- variant_id: PRINTIFY_VARIANT_IDS[cartItem.product_data.size],
+ // @ts-expect-error - metadata is valid, hard to find stripe type tho
+ variant_id: PRINTIFY_VARIANT_IDS[item.price?.product?.metadata?.size],
print_areas: {
- front: retrieveStickerPNGFileUrl(cartItem.product_data.productId),
+ // @ts-expect-error - metadata is valid, hard to find stripe type tho
+ front: retrieveStickerPNGFileUrl(item.price?.product?.metadata?.productId),
},
- quantity: cartItem.quantity,
+ quantity: item.quantity || 1,
}));
};
-export async function createDraftOrder(cartItems: CartItem[]): Promise<{ id: string }> {
- logger.info('[Printify] Formatting cart items for Printify');
- const printifyLineItems: PrintifyLineItem[] = formatCartItemsForPrintify(cartItems);
-
- const randomId = crypto.randomUUID();
- const randomLabel = Math.floor(Math.random() * 100000)
- .toString()
- .padStart(5, '0');
+const formatStripeAddress = (stripeAddress: StripeShippingDetails, email: string | null, phone: string | null) => ({
+ first_name: stripeAddress.name.split(' ')[0],
+ last_name: stripeAddress.name.split(' ')[1],
+ email: email || 'placeholder@gmail.com',
+ phone: phone || '0574 69 21 90',
+ country: stripeAddress.address.country,
+ region: stripeAddress.address.state,
+ // state: stripeAddress.address.state, // Printify is European startup, they don't use "state"
+ address1: stripeAddress.address.line1,
+ address2: stripeAddress.address.line2,
+ city: stripeAddress.address.city,
+ zip: stripeAddress.address.postal_code,
+});
+export async function createDraftOrder(
+ cartItems: StripeType.LineItem[],
+ stripeShippingDetails: StripeType.Checkout.Session.ShippingDetails | null,
+ swagOrderId: string,
+ stripeSessionId: string,
+ email: string | null,
+ phone: string | null
+): Promise<{ id: string }> {
+ const printifyLineItems: PrintifyLineItem[] = formatCartItemsForPrintify(cartItems);
+ console.log(printifyLineItems);
+ const address = formatStripeAddress(stripeShippingDetails as StripeShippingDetails, email, phone);
const orderData: SubmitOrderData = {
- external_id: randomId,
- label: `shipment_${randomLabel}`,
- // TODO_PRINTIFY (pull/format from stripe)
+ external_id: stripeSessionId,
+ label: `swagOrderId_${swagOrderId}`,
line_items: printifyLineItems,
shipping_method: 1,
is_printify_express: false,
is_economy_shipping: false,
- send_shipping_notification: false,
- // TODO_PRINTIFY (pull address from stripe)
- address_to: {
- first_name: 'John',
- last_name: 'Doe',
- email: 'testing@beta.com',
- phone: '0574 69 21 90',
- country: 'BE',
- region: '',
- address1: 'ExampleBaan 121',
- address2: '45',
- city: 'Retie',
- zip: '2470',
- },
+ send_shipping_notification: true,
+ address_to: address,
};
const order = await printify.orders.submit(orderData);
return order;
}
+export async function retrieveShippingCost(): Promise {
+ logger.info('[Printify] retrieving shipping options');
+ const shipping: VariantShippingData = await printify.catalog.getVariantShipping(PRINTIFY_BLUEPRINT_ID.toString(), PRINTIFY_PRINT_PROVIDER_ID.toString());
+
+ const fallbackShippingOptions = {
+ variant_ids: [95743, 95744, 95745, 95746],
+ first_item: { cost: 509, currency: 'USD' },
+ additional_items: { cost: 9, currency: 'USD' },
+ countries: ['US'],
+ };
+
+ // US-only
+ // 1 option, "Standard" only
+ return shipping.profiles.find(profile => profile.countries.includes(PRODUCT_CONFIG.allowCountries[0])) || fallbackShippingOptions;
+}
+
export async function sendOrderToProduction(printifyOrderId: string) {
logger.info('[Printify] Sending order to production', { printifyOrderId });
await printify.orders.sendToProduction(printifyOrderId);
diff --git a/src/lib/products.ts b/src/lib/products.ts
index 7b90106..2bac911 100644
--- a/src/lib/products.ts
+++ b/src/lib/products.ts
@@ -7,6 +7,7 @@ export const STICKER_SIZES = {
};
export const PRODUCT_CONFIG = {
+ // US-only
language: 'en-US',
allowCountries: ['US'], // e.g. ['US', 'GB', 'CA'];
currency: 'USD',
diff --git a/src/lib/stripe.ts b/src/lib/stripe.ts
index 0e6402c..61ad13a 100644
--- a/src/lib/stripe.ts
+++ b/src/lib/stripe.ts
@@ -1,8 +1,7 @@
import Stripe from 'stripe';
import { PRODUCT_CONFIG } from '@/lib/products';
-import { CartItem } from '@/types';
+import { CartItem, PrintifyShippingProfile, StripeShippingMethod } from '@/types';
import logger from './logger';
-import { UserError } from '@/utils/errors';
// docs: https://docs.stripe.com
// keys: https://dashboard.stripe.com/apikeys
@@ -22,8 +21,9 @@ export const formatCartItemsForStripe = (cartItems: CartItem[]): Stripe.Checkout
currency: cartItem.currency,
product_data: {
name: cartItem.name,
- description: cartItem.description,
- images: [`${process.env.NEXT_PUBLIC_URL}${cartItem.image}`], // up to 8 images
+ description: 'cartItem.description', // cartItem.description,
+ // images: [`${process.env.NEXT_PUBLIC_URL}${cartItem.image}`], // up to 8 images
+ images: [`https://swagsticker.com${cartItem.image}`], // up to 8 images
metadata: {
// pass metadata to stripe, (productId, size, category, ...)
...(cartItem.product_data || {}),
@@ -42,101 +42,59 @@ export const formatCartItemsForStripe = (cartItems: CartItem[]): Stripe.Checkout
});
};
+const calculateStripeShipping = (cartItemCount: number, shippingMethod: PrintifyShippingProfile): StripeShippingMethod => {
+ const additionalItemCount = cartItemCount - 1;
+ const totalShippingCost = shippingMethod.first_item.cost + additionalItemCount * shippingMethod.additional_items.cost;
+
+ return {
+ shipping_rate_data: {
+ type: 'fixed_amount',
+ fixed_amount: {
+ amount: totalShippingCost,
+ currency: shippingMethod.first_item.currency,
+ },
+ display_name: `Shipping to US (Standard)`,
+ delivery_estimate: {
+ minimum: {
+ unit: 'business_day',
+ value: 2, // hard-coded, based on website (could pull from shippingMethod.handling_time)
+ },
+ maximum: {
+ unit: 'business_day',
+ value: 5, // hard-coded, based on website (could pull from shippingMethod.handling_time)
+ },
+ },
+ },
+ };
+};
+
// docs: https://docs.stripe.com/api/checkout/sessions/create
-export async function createCheckoutSession(cartItems: CartItem[], metadata: { [key: string]: string } = {}): Promise> {
+export async function createCheckoutSession(
+ cartItems: CartItem[],
+ shippingMethod: PrintifyShippingProfile,
+ metadata: { [key: string]: string } = {}
+): Promise> {
+ logger.info('[Stripe] Formatting shipping methods for Stripe');
+ const stripeShippingMethod: StripeShippingMethod = calculateStripeShipping(cartItems.length, shippingMethod);
+
logger.info('[Stripe] Formatting cart items for Stripe');
const stripeLineItems: Stripe.Checkout.SessionCreateParams.LineItem[] = formatCartItemsForStripe(cartItems);
-
const session = await stripe.checkout.sessions.create({
+ ui_mode: 'embedded',
+ payment_method_types: ['card'],
line_items: stripeLineItems,
metadata: metadata,
mode: 'payment',
- success_url: `${process.env.NEXT_PUBLIC_URL}/order-confirmation?session_id={CHECKOUT_SESSION_ID}`,
- cancel_url: `${process.env.NEXT_PUBLIC_URL}/cart`,
- // customer: 'customerId',
- // customer_email: 'customer@gmail.com',
+ return_url: `${process.env.NEXT_PUBLIC_URL}/order-confirmation?session_id={CHECKOUT_SESSION_ID}`,
shipping_address_collection: {
+ // US-only
allowed_countries: PRODUCT_CONFIG.allowCountries as Stripe.Checkout.SessionCreateParams.ShippingAddressCollection.AllowedCountry[],
},
- // TODO_PRINTIFY - calculate this dynamically with Printify request + USD 0.09 per item!
- shipping_options: [
- {
- shipping_rate_data: {
- type: 'fixed_amount',
- fixed_amount: {
- amount: 459,
- currency: PRODUCT_CONFIG.currency,
- },
- display_name: 'Standard',
- delivery_estimate: {
- minimum: {
- unit: 'business_day',
- value: 2,
- },
- maximum: {
- unit: 'business_day',
- value: 5,
- },
- },
- },
- },
- // {
- // shipping_rate_data: {
- // type: 'fixed_amount',
- // fixed_amount: {
- // amount: 429,
- // currency: 'usd',
- // },
- // display_name: 'Economy',
- // delivery_estimate: {
- // minimum: {
- // unit: 'business_day',
- // value: 4,
- // },
- // maximum: {
- // unit: 'business_day',
- // value: 8,
- // },
- // },
- // },
- // },
- ],
- automatic_tax: {
- enabled: true, // Enable tax based on location
- },
- });
-
- if (!session || !session.url) {
- logger.error('[Stripe] Error creating checkout session', { session });
- throw new UserError('Unable to process checkout request');
- }
-
- return session;
-}
-
-export async function retrieveCheckoutSession(sessionId: string) {
- const session = await stripe.checkout.sessions.retrieve(sessionId, {
- expand: ['line_items'],
+ shipping_options: [stripeShippingMethod],
+ automatic_tax: { enabled: true },
+ // customer: 'customerId',
+ // customer_email: 'customer@gmail.com',
});
- if (!session) {
- logger.error('[Stripe] Failed to process stripe checkout session');
- throw new UserError('Unable to process stripe checkout session');
- }
-
return session;
}
-
-export async function validateStripeSession(sessionId?: string) {
- if (!sessionId) return { validSession: false };
-
- try {
- const session = await retrieveCheckoutSession(sessionId);
-
- if (session.object !== 'checkout.session') return { validSession: false };
-
- return { validSession: true };
- } catch (error) {
- return { validSession: false };
- }
-}
diff --git a/src/types/index.ts b/src/types/index.ts
index 417f9db..02e2452 100644
--- a/src/types/index.ts
+++ b/src/types/index.ts
@@ -100,3 +100,55 @@ export interface SubmitOrderData {
send_shipping_notification: boolean;
address_to: PrintifyAddress;
}
+
+export interface PrintifyShippingProfile {
+ variant_ids: number[];
+ first_item: {
+ cost: number;
+ currency: string;
+ };
+ additional_items: {
+ cost: number;
+ currency: string;
+ };
+ countries: string[];
+}
+
+export interface VariantShippingData {
+ profiles: PrintifyShippingProfile[];
+}
+
+export interface StripeShippingMethod {
+ shipping_rate_data: {
+ type: 'fixed_amount';
+ fixed_amount: {
+ amount: number;
+ currency: string; // Assuming PRODUCT_CONFIG.currency is of type string
+ };
+ display_name: string;
+ delivery_estimate: {
+ minimum: {
+ unit: 'business_day'; // You could also make this a union type if there are other units
+ value: number;
+ };
+ maximum: {
+ unit: 'business_day'; // Same as above
+ value: number;
+ };
+ };
+ };
+}
+
+interface Address {
+ city: string;
+ country: string;
+ line1: string;
+ line2?: string;
+ postal_code: string;
+ state: string;
+}
+
+export interface StripeShippingDetails {
+ address: Address;
+ name: string;
+}
diff --git a/yarn.lock b/yarn.lock
index a698718..9df6068 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -51,13 +51,6 @@
enabled "2.0.x"
kuler "^2.0.0"
-"@emnapi/runtime@^1.2.0":
- version "1.3.1"
- resolved "https://registry.yarnpkg.com/@emnapi/runtime/-/runtime-1.3.1.tgz#0fcaa575afc31f455fd33534c19381cfce6c6f60"
- integrity sha512-kEBmG8KyqtxJZv+ygbEim+KCGtIq1fC22Ms3S4ziXmYKm8uyoLX0MHONVKwp+9opg390VaKRNt4a7A9NwmpNhw==
- dependencies:
- tslib "^2.4.0"
-
"@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0":
version "4.4.0"
resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59"
@@ -109,119 +102,6 @@
resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz#4a2868d75d6d6963e423bcf90b7fd1be343409d3"
integrity sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==
-"@img/sharp-darwin-arm64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-darwin-arm64/-/sharp-darwin-arm64-0.33.5.tgz#ef5b5a07862805f1e8145a377c8ba6e98813ca08"
- integrity sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==
- optionalDependencies:
- "@img/sharp-libvips-darwin-arm64" "1.0.4"
-
-"@img/sharp-darwin-x64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-darwin-x64/-/sharp-darwin-x64-0.33.5.tgz#e03d3451cd9e664faa72948cc70a403ea4063d61"
- integrity sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==
- optionalDependencies:
- "@img/sharp-libvips-darwin-x64" "1.0.4"
-
-"@img/sharp-libvips-darwin-arm64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-darwin-arm64/-/sharp-libvips-darwin-arm64-1.0.4.tgz#447c5026700c01a993c7804eb8af5f6e9868c07f"
- integrity sha512-XblONe153h0O2zuFfTAbQYAX2JhYmDHeWikp1LM9Hul9gVPjFY427k6dFEcOL72O01QxQsWi761svJ/ev9xEDg==
-
-"@img/sharp-libvips-darwin-x64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-darwin-x64/-/sharp-libvips-darwin-x64-1.0.4.tgz#e0456f8f7c623f9dbfbdc77383caa72281d86062"
- integrity sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==
-
-"@img/sharp-libvips-linux-arm64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linux-arm64/-/sharp-libvips-linux-arm64-1.0.4.tgz#979b1c66c9a91f7ff2893556ef267f90ebe51704"
- integrity sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==
-
-"@img/sharp-libvips-linux-arm@1.0.5":
- version "1.0.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linux-arm/-/sharp-libvips-linux-arm-1.0.5.tgz#99f922d4e15216ec205dcb6891b721bfd2884197"
- integrity sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==
-
-"@img/sharp-libvips-linux-s390x@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linux-s390x/-/sharp-libvips-linux-s390x-1.0.4.tgz#f8a5eb1f374a082f72b3f45e2fb25b8118a8a5ce"
- integrity sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==
-
-"@img/sharp-libvips-linux-x64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linux-x64/-/sharp-libvips-linux-x64-1.0.4.tgz#d4c4619cdd157774906e15770ee119931c7ef5e0"
- integrity sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==
-
-"@img/sharp-libvips-linuxmusl-arm64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linuxmusl-arm64/-/sharp-libvips-linuxmusl-arm64-1.0.4.tgz#166778da0f48dd2bded1fa3033cee6b588f0d5d5"
- integrity sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==
-
-"@img/sharp-libvips-linuxmusl-x64@1.0.4":
- version "1.0.4"
- resolved "https://registry.yarnpkg.com/@img/sharp-libvips-linuxmusl-x64/-/sharp-libvips-linuxmusl-x64-1.0.4.tgz#93794e4d7720b077fcad3e02982f2f1c246751ff"
- integrity sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==
-
-"@img/sharp-linux-arm64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linux-arm64/-/sharp-linux-arm64-0.33.5.tgz#edb0697e7a8279c9fc829a60fc35644c4839bb22"
- integrity sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==
- optionalDependencies:
- "@img/sharp-libvips-linux-arm64" "1.0.4"
-
-"@img/sharp-linux-arm@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linux-arm/-/sharp-linux-arm-0.33.5.tgz#422c1a352e7b5832842577dc51602bcd5b6f5eff"
- integrity sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==
- optionalDependencies:
- "@img/sharp-libvips-linux-arm" "1.0.5"
-
-"@img/sharp-linux-s390x@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linux-s390x/-/sharp-linux-s390x-0.33.5.tgz#f5c077926b48e97e4a04d004dfaf175972059667"
- integrity sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==
- optionalDependencies:
- "@img/sharp-libvips-linux-s390x" "1.0.4"
-
-"@img/sharp-linux-x64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linux-x64/-/sharp-linux-x64-0.33.5.tgz#d806e0afd71ae6775cc87f0da8f2d03a7c2209cb"
- integrity sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==
- optionalDependencies:
- "@img/sharp-libvips-linux-x64" "1.0.4"
-
-"@img/sharp-linuxmusl-arm64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linuxmusl-arm64/-/sharp-linuxmusl-arm64-0.33.5.tgz#252975b915894fb315af5deea174651e208d3d6b"
- integrity sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==
- optionalDependencies:
- "@img/sharp-libvips-linuxmusl-arm64" "1.0.4"
-
-"@img/sharp-linuxmusl-x64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-linuxmusl-x64/-/sharp-linuxmusl-x64-0.33.5.tgz#3f4609ac5d8ef8ec7dadee80b560961a60fd4f48"
- integrity sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==
- optionalDependencies:
- "@img/sharp-libvips-linuxmusl-x64" "1.0.4"
-
-"@img/sharp-wasm32@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-wasm32/-/sharp-wasm32-0.33.5.tgz#6f44f3283069d935bb5ca5813153572f3e6f61a1"
- integrity sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==
- dependencies:
- "@emnapi/runtime" "^1.2.0"
-
-"@img/sharp-win32-ia32@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-win32-ia32/-/sharp-win32-ia32-0.33.5.tgz#1a0c839a40c5351e9885628c85f2e5dfd02b52a9"
- integrity sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==
-
-"@img/sharp-win32-x64@0.33.5":
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/@img/sharp-win32-x64/-/sharp-win32-x64-0.33.5.tgz#56f00962ff0c4e0eb93d34a047d29fa995e3e342"
- integrity sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==
-
"@isaacs/cliui@^8.0.2":
version "8.0.2"
resolved "https://registry.yarnpkg.com/@isaacs/cliui/-/cliui-8.0.2.tgz#b37667b7bc181c168782259bab42474fbf52b550"
@@ -374,6 +254,13 @@
resolved "https://registry.yarnpkg.com/@rushstack/eslint-patch/-/eslint-patch-1.10.4.tgz#427d5549943a9c6fce808e39ea64dbe60d4047f1"
integrity sha512-WJgX9nzTqknM393q1QJDJmoW28kUfEnybeTfVNcNAPnIx210RXm2DiXiHzfNPJNIUUb1tJnz/l4QGtJ30PgWmA==
+"@stripe/react-stripe-js@^2.8.1":
+ version "2.8.1"
+ resolved "https://registry.yarnpkg.com/@stripe/react-stripe-js/-/react-stripe-js-2.8.1.tgz#63d9a666749f818b1bf4eb788eb1bc57eb496f3d"
+ integrity sha512-C410jVKOATinXLalWotab6E6jlWAlbqUDWL9q1km0p5UHrvnihjjYzA8imYXc4xc4Euf9GeKDQc4n35HKZvgwg==
+ dependencies:
+ prop-types "^15.7.2"
+
"@stripe/stripe-js@^1.54.2":
version "1.54.2"
resolved "https://registry.yarnpkg.com/@stripe/stripe-js/-/stripe-js-1.54.2.tgz#0665848e22cbda936cfd05256facdfbba121438d"
@@ -792,11 +679,49 @@ axobject-query@^4.1.0:
resolved "https://registry.yarnpkg.com/axobject-query/-/axobject-query-4.1.0.tgz#28768c76d0e3cff21bc62a9e2d0b6ac30042a1ee"
integrity sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==
+b4a@^1.6.4:
+ version "1.6.7"
+ resolved "https://registry.yarnpkg.com/b4a/-/b4a-1.6.7.tgz#a99587d4ebbfbd5a6e3b21bdb5d5fa385767abe4"
+ integrity sha512-OnAYlL5b7LEkALw87fUVafQw5rVR9RjwGd4KUwNQ6DrrNmaVaUCgLipfVlzrPQ4tWOR9P0IXGNOx50jYCCdSJg==
+
balanced-match@^1.0.0:
version "1.0.2"
resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee"
integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==
+bare-events@^2.0.0, bare-events@^2.2.0:
+ version "2.5.0"
+ resolved "https://registry.yarnpkg.com/bare-events/-/bare-events-2.5.0.tgz#305b511e262ffd8b9d5616b056464f8e1b3329cc"
+ integrity sha512-/E8dDe9dsbLyh2qrZ64PEPadOQ0F4gbl1sUJOrmph7xOiIxfY8vwab/4bFLh4Y88/Hk/ujKcrQKc+ps0mv873A==
+
+bare-fs@^2.1.1:
+ version "2.3.5"
+ resolved "https://registry.yarnpkg.com/bare-fs/-/bare-fs-2.3.5.tgz#05daa8e8206aeb46d13c2fe25a2cd3797b0d284a"
+ integrity sha512-SlE9eTxifPDJrT6YgemQ1WGFleevzwY+XAP1Xqgl56HtcrisC2CHCZ2tq6dBpcH2TnNxwUEUGhweo+lrQtYuiw==
+ dependencies:
+ bare-events "^2.0.0"
+ bare-path "^2.0.0"
+ bare-stream "^2.0.0"
+
+bare-os@^2.1.0:
+ version "2.4.4"
+ resolved "https://registry.yarnpkg.com/bare-os/-/bare-os-2.4.4.tgz#01243392eb0a6e947177bb7c8a45123d45c9b1a9"
+ integrity sha512-z3UiI2yi1mK0sXeRdc4O1Kk8aOa/e+FNWZcTiPB/dfTWyLypuE99LibgRaQki914Jq//yAWylcAt+mknKdixRQ==
+
+bare-path@^2.0.0, bare-path@^2.1.0:
+ version "2.1.3"
+ resolved "https://registry.yarnpkg.com/bare-path/-/bare-path-2.1.3.tgz#594104c829ef660e43b5589ec8daef7df6cedb3e"
+ integrity sha512-lh/eITfU8hrj9Ru5quUp0Io1kJWIk1bTjzo7JH1P5dWmQ2EL4hFUlfI8FonAhSlgIfhn63p84CDY/x+PisgcXA==
+ dependencies:
+ bare-os "^2.1.0"
+
+bare-stream@^2.0.0:
+ version "2.3.2"
+ resolved "https://registry.yarnpkg.com/bare-stream/-/bare-stream-2.3.2.tgz#3bc62b429bcf850d2f265719b7a49ee0630a3ae4"
+ integrity sha512-EFZHSIBkDgSHIwj2l2QZfP4U5OcD4xFAOwhSb/vlr9PIqyGJGvB/nfClJbcnh3EY4jtPE4zsb5ztae96bVF79A==
+ dependencies:
+ streamx "^2.20.0"
+
base64-js@^1.3.1:
version "1.5.1"
resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
@@ -807,6 +732,15 @@ binary-extensions@^2.0.0:
resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.3.0.tgz#f6e14a97858d327252200242d4ccfe522c445522"
integrity sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==
+bl@^4.0.3:
+ version "4.1.0"
+ resolved "https://registry.yarnpkg.com/bl/-/bl-4.1.0.tgz#451535264182bec2fbbc83a62ab98cf11d9f7b3a"
+ integrity sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==
+ dependencies:
+ buffer "^5.5.0"
+ inherits "^2.0.4"
+ readable-stream "^3.4.0"
+
brace-expansion@^1.1.7:
version "1.1.11"
resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
@@ -829,6 +763,14 @@ braces@^3.0.3, braces@~3.0.2:
dependencies:
fill-range "^7.1.1"
+buffer@^5.5.0:
+ version "5.7.1"
+ resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0"
+ integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==
+ dependencies:
+ base64-js "^1.3.1"
+ ieee754 "^1.1.13"
+
buffer@^6.0.3:
version "6.0.3"
resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6"
@@ -902,6 +844,11 @@ chokidar@^3.5.3:
optionalDependencies:
fsevents "~2.3.2"
+chownr@^1.1.1:
+ version "1.1.4"
+ resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b"
+ integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==
+
ci-info@^2.0.0:
version "2.0.0"
resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-2.0.0.tgz#67a9e964be31a51e15e5010d58e6f12834002f46"
@@ -1096,6 +1043,13 @@ debug@^4.2.0, debug@^4.3.1, debug@^4.3.2, debug@^4.3.4, debug@^4.3.5:
dependencies:
ms "^2.1.3"
+decompress-response@^6.0.0:
+ version "6.0.0"
+ resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-6.0.0.tgz#ca387612ddb7e104bd16d85aab00d5ecf09c66fc"
+ integrity sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==
+ dependencies:
+ mimic-response "^3.1.0"
+
dedent@^0.7.0:
version "0.7.0"
resolved "https://registry.yarnpkg.com/dedent/-/dedent-0.7.0.tgz#2495ddbaf6eb874abb0e1be9df22d2e5a544326c"
@@ -1125,6 +1079,11 @@ deep-equal@^2.0.5:
which-collection "^1.0.1"
which-typed-array "^1.1.13"
+deep-extend@^0.6.0:
+ version "0.6.0"
+ resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac"
+ integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==
+
deep-is@^0.1.3:
version "0.1.4"
resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831"
@@ -1148,7 +1107,7 @@ define-properties@^1.1.3, define-properties@^1.2.0, define-properties@^1.2.1:
has-property-descriptors "^1.0.0"
object-keys "^1.1.1"
-detect-libc@^2.0.3:
+detect-libc@^2.0.0, detect-libc@^2.0.2:
version "2.0.3"
resolved "https://registry.yarnpkg.com/detect-libc/-/detect-libc-2.0.3.tgz#f0cd503b40f9939b894697d19ad50895e30cf700"
integrity sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==
@@ -1204,7 +1163,7 @@ enabled@2.0.x:
resolved "https://registry.yarnpkg.com/enabled/-/enabled-2.0.0.tgz#f9dd92ec2d6f4bbc0d5d1e64e21d61cd4665e7c2"
integrity sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==
-end-of-stream@^1.1.0:
+end-of-stream@^1.1.0, end-of-stream@^1.4.1:
version "1.4.4"
resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0"
integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==
@@ -1611,11 +1570,21 @@ execa@^4.1.0:
signal-exit "^3.0.2"
strip-final-newline "^2.0.0"
+expand-template@^2.0.3:
+ version "2.0.3"
+ resolved "https://registry.yarnpkg.com/expand-template/-/expand-template-2.0.3.tgz#6e14b3fcee0f3a6340ecb57d2e8918692052a47c"
+ integrity sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==
+
fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3:
version "3.1.3"
resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525"
integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==
+fast-fifo@^1.2.0, fast-fifo@^1.3.2:
+ version "1.3.2"
+ resolved "https://registry.yarnpkg.com/fast-fifo/-/fast-fifo-1.3.2.tgz#286e31de96eb96d38a97899815740ba2a4f3640c"
+ integrity sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==
+
fast-glob@^3.2.9, fast-glob@^3.3.0, fast-glob@^3.3.2:
version "3.3.2"
resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.3.2.tgz#a904501e57cfdd2ffcded45e99a54fef55e46129"
@@ -1712,6 +1681,11 @@ foreground-child@^3.1.0:
cross-spawn "^7.0.0"
signal-exit "^4.0.1"
+fs-constants@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/fs-constants/-/fs-constants-1.0.0.tgz#6be0de9be998ce16af8afc24497b9ee9b7ccd9ad"
+ integrity sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==
+
fs.realpath@^1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
@@ -1786,6 +1760,11 @@ get-tsconfig@^4.7.5:
dependencies:
resolve-pkg-maps "^1.0.0"
+github-from-package@0.0.0:
+ version "0.0.0"
+ resolved "https://registry.yarnpkg.com/github-from-package/-/github-from-package-0.0.0.tgz#97fb5d96bfde8973313f20e8288ef9a167fa64ce"
+ integrity sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==
+
glob-parent@^5.1.2, glob-parent@~5.1.2:
version "5.1.2"
resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4"
@@ -1953,7 +1932,7 @@ husky@4.3.8:
slash "^3.0.0"
which-pm-runs "^1.0.0"
-ieee754@^1.2.1:
+ieee754@^1.1.13, ieee754@^1.2.1:
version "1.2.1"
resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
@@ -1994,11 +1973,16 @@ inflight@^1.0.4:
once "^1.3.0"
wrappy "1"
-inherits@2, inherits@^2.0.3:
+inherits@2, inherits@^2.0.3, inherits@^2.0.4:
version "2.0.4"
resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+ini@~1.3.0:
+ version "1.3.8"
+ resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c"
+ integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==
+
internal-slot@^1.0.4, internal-slot@^1.0.7:
version "1.0.7"
resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.7.tgz#c06dcca3ed874249881007b0a5523b172a190802"
@@ -2487,6 +2471,11 @@ mimic-fn@^2.1.0:
resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b"
integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==
+mimic-response@^3.1.0:
+ version "3.1.0"
+ resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-3.1.0.tgz#2d1d59af9c1b129815accc2c46a022a5ce1fa3c9"
+ integrity sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==
+
minimatch@9.0.3:
version "9.0.3"
resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.3.tgz#a6e00c3de44c3a542bfaae70abfc22420a6da825"
@@ -2508,7 +2497,7 @@ minimatch@^9.0.1, minimatch@^9.0.4:
dependencies:
brace-expansion "^2.0.1"
-minimist@^1.2.0, minimist@^1.2.6:
+minimist@^1.2.0, minimist@^1.2.3, minimist@^1.2.6:
version "1.2.8"
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c"
integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==
@@ -2518,6 +2507,11 @@ minimist@^1.2.0, minimist@^1.2.6:
resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.1.2.tgz#93a9626ce5e5e66bd4db86849e7515e92340a707"
integrity sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==
+mkdirp-classic@^0.5.2, mkdirp-classic@^0.5.3:
+ version "0.5.3"
+ resolved "https://registry.yarnpkg.com/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz#fa10c9115cc6d8865be221ba47ee9bed78601113"
+ integrity sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==
+
ms@^2.1.1, ms@^2.1.3:
version "2.1.3"
resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2"
@@ -2537,6 +2531,11 @@ nanoid@^3.3.6, nanoid@^3.3.7:
resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8"
integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==
+napi-build-utils@^1.0.1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/napi-build-utils/-/napi-build-utils-1.0.2.tgz#b1fddc0b2c46e380a0b7a76f984dd47c41a13806"
+ integrity sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==
+
natural-compare@^1.4.0:
version "1.4.0"
resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7"
@@ -2565,6 +2564,18 @@ next@14.2.11:
"@next/swc-win32-ia32-msvc" "14.2.11"
"@next/swc-win32-x64-msvc" "14.2.11"
+node-abi@^3.3.0:
+ version "3.71.0"
+ resolved "https://registry.yarnpkg.com/node-abi/-/node-abi-3.71.0.tgz#52d84bbcd8575efb71468fbaa1f9a49b2c242038"
+ integrity sha512-SZ40vRiy/+wRTf21hxkkEjPJZpARzUMVcJoQse2EF8qkUWbbO2z7vd5oA/H6bVH6SZQ5STGcu0KRDS7biNRfxw==
+ dependencies:
+ semver "^7.3.5"
+
+node-addon-api@^6.1.0:
+ version "6.1.0"
+ resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-6.1.0.tgz#ac8470034e58e67d0c6f1204a18ae6995d9c0d76"
+ integrity sha512-+eawOlIgy680F0kBzPUNFhMZGtJ1YmqM6l4+Crf4IkImjYrO/mqPwRMh352g23uIaQKFItcQ64I7KMaJxHgAVA==
+
normalize-path@^3.0.0, normalize-path@~3.0.0:
version "3.0.0"
resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65"
@@ -2867,6 +2878,24 @@ postcss@^8, postcss@^8.4.23:
picocolors "^1.1.0"
source-map-js "^1.2.1"
+prebuild-install@^7.1.1:
+ version "7.1.2"
+ resolved "https://registry.yarnpkg.com/prebuild-install/-/prebuild-install-7.1.2.tgz#a5fd9986f5a6251fbc47e1e5c65de71e68c0a056"
+ integrity sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==
+ dependencies:
+ detect-libc "^2.0.0"
+ expand-template "^2.0.3"
+ github-from-package "0.0.0"
+ minimist "^1.2.3"
+ mkdirp-classic "^0.5.3"
+ napi-build-utils "^1.0.1"
+ node-abi "^3.3.0"
+ pump "^3.0.0"
+ rc "^1.2.7"
+ simple-get "^4.0.0"
+ tar-fs "^2.0.0"
+ tunnel-agent "^0.6.0"
+
prelude-ls@^1.2.1:
version "1.2.1"
resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396"
@@ -2921,6 +2950,21 @@ queue-microtask@^1.2.2:
resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243"
integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==
+queue-tick@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/queue-tick/-/queue-tick-1.0.1.tgz#f6f07ac82c1fd60f82e098b417a80e52f1f4c142"
+ integrity sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag==
+
+rc@^1.2.7:
+ version "1.2.8"
+ resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz#cd924bf5200a075b83c188cd6b9e211b7fc0d3ed"
+ integrity sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==
+ dependencies:
+ deep-extend "^0.6.0"
+ ini "~1.3.0"
+ minimist "^1.2.0"
+ strip-json-comments "~2.0.1"
+
react-dom@^17.0.2:
version "17.0.2"
resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-17.0.2.tgz#ecffb6845e3ad8dbfcdc498f0d0a939736502c23"
@@ -2974,7 +3018,7 @@ read-cache@^1.0.0:
dependencies:
pify "^2.3.0"
-readable-stream@^3.4.0:
+readable-stream@^3.1.1, readable-stream@^3.4.0:
version "3.6.2"
resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967"
integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==
@@ -3128,7 +3172,7 @@ safe-array-concat@^1.1.2:
has-symbols "^1.0.3"
isarray "^2.0.5"
-safe-buffer@~5.2.0:
+safe-buffer@^5.0.1, safe-buffer@~5.2.0:
version "5.2.1"
resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
@@ -3177,7 +3221,7 @@ semver@^6.3.1:
resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4"
integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==
-semver@^7.5.4, semver@^7.6.3:
+semver@^7.3.5, semver@^7.5.4, semver@^7.6.3:
version "7.6.3"
resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143"
integrity sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==
@@ -3204,34 +3248,19 @@ set-function-name@^2.0.1, set-function-name@^2.0.2:
functions-have-names "^1.2.3"
has-property-descriptors "^1.0.2"
-sharp@^0.33.5:
- version "0.33.5"
- resolved "https://registry.yarnpkg.com/sharp/-/sharp-0.33.5.tgz#13e0e4130cc309d6a9497596715240b2ec0c594e"
- integrity sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==
+sharp@0.32.6:
+ version "0.32.6"
+ resolved "https://registry.yarnpkg.com/sharp/-/sharp-0.32.6.tgz#6ad30c0b7cd910df65d5f355f774aa4fce45732a"
+ integrity sha512-KyLTWwgcR9Oe4d9HwCwNM2l7+J0dUQwn/yf7S0EnTtb0eVS4RxO0eUSvxPtzT4F3SY+C4K6fqdv/DO27sJ/v/w==
dependencies:
color "^4.2.3"
- detect-libc "^2.0.3"
- semver "^7.6.3"
- optionalDependencies:
- "@img/sharp-darwin-arm64" "0.33.5"
- "@img/sharp-darwin-x64" "0.33.5"
- "@img/sharp-libvips-darwin-arm64" "1.0.4"
- "@img/sharp-libvips-darwin-x64" "1.0.4"
- "@img/sharp-libvips-linux-arm" "1.0.5"
- "@img/sharp-libvips-linux-arm64" "1.0.4"
- "@img/sharp-libvips-linux-s390x" "1.0.4"
- "@img/sharp-libvips-linux-x64" "1.0.4"
- "@img/sharp-libvips-linuxmusl-arm64" "1.0.4"
- "@img/sharp-libvips-linuxmusl-x64" "1.0.4"
- "@img/sharp-linux-arm" "0.33.5"
- "@img/sharp-linux-arm64" "0.33.5"
- "@img/sharp-linux-s390x" "0.33.5"
- "@img/sharp-linux-x64" "0.33.5"
- "@img/sharp-linuxmusl-arm64" "0.33.5"
- "@img/sharp-linuxmusl-x64" "0.33.5"
- "@img/sharp-wasm32" "0.33.5"
- "@img/sharp-win32-ia32" "0.33.5"
- "@img/sharp-win32-x64" "0.33.5"
+ detect-libc "^2.0.2"
+ node-addon-api "^6.1.0"
+ prebuild-install "^7.1.1"
+ semver "^7.5.4"
+ simple-get "^4.0.1"
+ tar-fs "^3.0.4"
+ tunnel-agent "^0.6.0"
shebang-command@^2.0.0:
version "2.0.0"
@@ -3265,6 +3294,20 @@ signal-exit@^4.0.1:
resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04"
integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==
+simple-concat@^1.0.0:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/simple-concat/-/simple-concat-1.0.1.tgz#f46976082ba35c2263f1c8ab5edfe26c41c9552f"
+ integrity sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==
+
+simple-get@^4.0.0, simple-get@^4.0.1:
+ version "4.0.1"
+ resolved "https://registry.yarnpkg.com/simple-get/-/simple-get-4.0.1.tgz#4a39db549287c979d352112fa03fd99fd6bc3543"
+ integrity sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==
+ dependencies:
+ decompress-response "^6.0.0"
+ once "^1.3.1"
+ simple-concat "^1.0.0"
+
simple-swizzle@^0.2.2:
version "0.2.2"
resolved "https://registry.yarnpkg.com/simple-swizzle/-/simple-swizzle-0.2.2.tgz#a4da6b635ffcccca33f70d17cb92592de95e557a"
@@ -3317,6 +3360,17 @@ streamsearch@^1.1.0:
resolved "https://registry.yarnpkg.com/streamsearch/-/streamsearch-1.1.0.tgz#404dd1e2247ca94af554e841a8ef0eaa238da764"
integrity sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==
+streamx@^2.15.0, streamx@^2.20.0:
+ version "2.20.1"
+ resolved "https://registry.yarnpkg.com/streamx/-/streamx-2.20.1.tgz#471c4f8b860f7b696feb83d5b125caab2fdbb93c"
+ integrity sha512-uTa0mU6WUC65iUvzKH4X9hEdvSW7rbPxPtwfWiLMSj3qTdQbAiUboZTxauKfpFuGIGa1C2BYijZ7wgdUXICJhA==
+ dependencies:
+ fast-fifo "^1.3.2"
+ queue-tick "^1.0.1"
+ text-decoder "^1.1.0"
+ optionalDependencies:
+ bare-events "^2.2.0"
+
string-argv@0.3.1:
version "0.3.1"
resolved "https://registry.yarnpkg.com/string-argv/-/string-argv-0.3.1.tgz#95e2fbec0427ae19184935f816d74aaa4c5c19da"
@@ -3463,6 +3517,11 @@ strip-json-comments@^3.1.1:
resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006"
integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==
+strip-json-comments@~2.0.1:
+ version "2.0.1"
+ resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a"
+ integrity sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==
+
stripe@^16.9.0:
version "16.12.0"
resolved "https://registry.yarnpkg.com/stripe/-/stripe-16.12.0.tgz#75e3d8f0f35bea14885cb3605a41d6afc182aa66"
@@ -3543,6 +3602,52 @@ tapable@^2.2.0:
resolved "https://registry.yarnpkg.com/tapable/-/tapable-2.2.1.tgz#1967a73ef4060a82f12ab96af86d52fdb76eeca0"
integrity sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==
+tar-fs@^2.0.0:
+ version "2.1.1"
+ resolved "https://registry.yarnpkg.com/tar-fs/-/tar-fs-2.1.1.tgz#489a15ab85f1f0befabb370b7de4f9eb5cbe8784"
+ integrity sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==
+ dependencies:
+ chownr "^1.1.1"
+ mkdirp-classic "^0.5.2"
+ pump "^3.0.0"
+ tar-stream "^2.1.4"
+
+tar-fs@^3.0.4:
+ version "3.0.6"
+ resolved "https://registry.yarnpkg.com/tar-fs/-/tar-fs-3.0.6.tgz#eaccd3a67d5672f09ca8e8f9c3d2b89fa173f217"
+ integrity sha512-iokBDQQkUyeXhgPYaZxmczGPhnhXZ0CmrqI+MOb/WFGS9DW5wnfrLgtjUJBvz50vQ3qfRwJ62QVoCFu8mPVu5w==
+ dependencies:
+ pump "^3.0.0"
+ tar-stream "^3.1.5"
+ optionalDependencies:
+ bare-fs "^2.1.1"
+ bare-path "^2.1.0"
+
+tar-stream@^2.1.4:
+ version "2.2.0"
+ resolved "https://registry.yarnpkg.com/tar-stream/-/tar-stream-2.2.0.tgz#acad84c284136b060dc3faa64474aa9aebd77287"
+ integrity sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==
+ dependencies:
+ bl "^4.0.3"
+ end-of-stream "^1.4.1"
+ fs-constants "^1.0.0"
+ inherits "^2.0.3"
+ readable-stream "^3.1.1"
+
+tar-stream@^3.1.5:
+ version "3.1.7"
+ resolved "https://registry.yarnpkg.com/tar-stream/-/tar-stream-3.1.7.tgz#24b3fb5eabada19fe7338ed6d26e5f7c482e792b"
+ integrity sha512-qJj60CXt7IU1Ffyc3NJMjh6EkuCFej46zUqJ4J7pqYlThyd9bO0XBTmcOIhSzZJVWfsLks0+nle/j538YAW9RQ==
+ dependencies:
+ b4a "^1.6.4"
+ fast-fifo "^1.2.0"
+ streamx "^2.15.0"
+
+text-decoder@^1.1.0:
+ version "1.2.1"
+ resolved "https://registry.yarnpkg.com/text-decoder/-/text-decoder-1.2.1.tgz#e173f5121d97bfa3ff8723429ad5ba92e1ead67e"
+ integrity sha512-x9v3H/lTKIJKQQe7RPQkLfKAnc9lUTkWDypIQgTzPJAq+5/GCDHonmshfvlsNSj58yyshbIJJDLmU15qNERrXQ==
+
text-hex@1.0.x:
version "1.0.0"
resolved "https://registry.yarnpkg.com/text-hex/-/text-hex-1.0.0.tgz#69dc9c1b17446ee79a92bf5b884bb4b9127506f5"
@@ -3609,6 +3714,13 @@ tslib@^2.1.0, tslib@^2.4.0:
resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.7.0.tgz#d9b40c5c40ab59e8738f297df3087bf1a2690c01"
integrity sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==
+tunnel-agent@^0.6.0:
+ version "0.6.0"
+ resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd"
+ integrity sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==
+ dependencies:
+ safe-buffer "^5.0.1"
+
type-check@^0.4.0, type-check@~0.4.0:
version "0.4.0"
resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.4.0.tgz#07b8203bfa7056c0657050e3ccd2c37730bab8f1"
@@ -3874,3 +3986,8 @@ yocto-queue@^0.1.0:
version "0.1.0"
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
+
+zod@^3.23.8:
+ version "3.23.8"
+ resolved "https://registry.yarnpkg.com/zod/-/zod-3.23.8.tgz#e37b957b5d52079769fb8097099b592f0ef4067d"
+ integrity sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==