mirror of
https://github.com/vercel/commerce.git
synced 2025-05-19 07:56:59 +00:00
implement account-tied carts and cart reconciliation
Signed-off-by: Loan Laux <loan@outgrow.io>
This commit is contained in:
parent
946545b091
commit
25ba1f1bae
@ -6,14 +6,14 @@ import {
|
|||||||
import getCartCookie from '@framework/api/utils/get-cart-cookie'
|
import getCartCookie from '@framework/api/utils/get-cart-cookie'
|
||||||
import {
|
import {
|
||||||
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
REACTION_CART_ID_COOKIE,
|
REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
} from '@framework/const'
|
} from '@framework/const'
|
||||||
|
|
||||||
const addItem: CartHandlers['addItem'] = async ({
|
const addItem: CartHandlers['addItem'] = async ({
|
||||||
req: {
|
req: {
|
||||||
cookies: {
|
cookies: {
|
||||||
[REACTION_ANONYMOUS_CART_TOKEN_COOKIE]: anonymousCartToken,
|
[REACTION_ANONYMOUS_CART_TOKEN_COOKIE]: anonymousCartToken,
|
||||||
[REACTION_CART_ID_COOKIE]: cartId,
|
[REACTION_ANONYMOUS_CART_ID_COOKIE]: cartId,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
res,
|
res,
|
||||||
@ -54,9 +54,13 @@ const addItem: CartHandlers['addItem'] = async ({
|
|||||||
console.log('created cart', createdCart.data.createCart.cart)
|
console.log('created cart', createdCart.data.createCart.cart)
|
||||||
|
|
||||||
res.setHeader('Set-Cookie', [
|
res.setHeader('Set-Cookie', [
|
||||||
getCartCookie(config.cartCookie, createdCart.data.createCart.token, 999),
|
|
||||||
getCartCookie(
|
getCartCookie(
|
||||||
config.cartIdCookie,
|
config.anonymousCartTokenCookie,
|
||||||
|
createdCart.data.createCart.token,
|
||||||
|
999
|
||||||
|
),
|
||||||
|
getCartCookie(
|
||||||
|
config.anonymousCartIdCookie,
|
||||||
createdCart.data.createCart.cart._id,
|
createdCart.data.createCart.cart._id,
|
||||||
999
|
999
|
||||||
),
|
),
|
||||||
|
@ -1,10 +1,14 @@
|
|||||||
import type { Cart } from '../../../types'
|
import type { Cart } from '../../../types'
|
||||||
import type { CartHandlers } from '../'
|
import type { CartHandlers } from '../'
|
||||||
import getAnomymousCartQuery from '@framework/utils/queries/get-anonymous-cart'
|
import getAnomymousCartQuery from '@framework/utils/queries/get-anonymous-cart'
|
||||||
|
import accountCartByAccountIdQuery from '@framework/utils/queries/account-cart-by-account-id'
|
||||||
|
import reconcileCartsMutation from '@framework/utils/mutations/reconcile-carts'
|
||||||
import getCartCookie from '@framework/api/utils/get-cart-cookie'
|
import getCartCookie from '@framework/api/utils/get-cart-cookie'
|
||||||
|
import getViewerId from '@framework/customer/get-viewer-id'
|
||||||
import {
|
import {
|
||||||
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
REACTION_CART_ID_COOKIE,
|
REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
|
REACTION_CUSTOMER_TOKEN_COOKIE,
|
||||||
} from '@framework/const.ts'
|
} from '@framework/const.ts'
|
||||||
import { normalizeCart } from '@framework/utils'
|
import { normalizeCart } from '@framework/utils'
|
||||||
|
|
||||||
@ -13,32 +17,83 @@ const getCart: CartHandlers['getCart'] = async ({ req, res, config }) => {
|
|||||||
const {
|
const {
|
||||||
cookies: {
|
cookies: {
|
||||||
[REACTION_ANONYMOUS_CART_TOKEN_COOKIE]: anonymousCartToken,
|
[REACTION_ANONYMOUS_CART_TOKEN_COOKIE]: anonymousCartToken,
|
||||||
[REACTION_CART_ID_COOKIE]: cartId,
|
[REACTION_ANONYMOUS_CART_ID_COOKIE]: anonymousCartId,
|
||||||
|
[REACTION_CUSTOMER_TOKEN_COOKIE]: reactionCustomerToken,
|
||||||
},
|
},
|
||||||
} = req
|
} = req
|
||||||
|
|
||||||
let normalizedCart
|
let normalizedCart
|
||||||
|
|
||||||
console.log('get-cart API')
|
if (anonymousCartId && anonymousCartToken && reactionCustomerToken) {
|
||||||
console.log('anonymousCartToken', anonymousCartToken)
|
|
||||||
console.log('cartId', cartId)
|
|
||||||
console.log('shopId', config.shopId)
|
|
||||||
|
|
||||||
if (cartId && anonymousCartToken) {
|
|
||||||
const {
|
const {
|
||||||
data: { cart: rawCart },
|
data: {
|
||||||
|
reconcileCarts: { cart: rawReconciledCart },
|
||||||
|
},
|
||||||
|
} = await config.fetch(
|
||||||
|
reconcileCartsMutation,
|
||||||
|
{
|
||||||
|
variables: {
|
||||||
|
input: {
|
||||||
|
anonymousCartId,
|
||||||
|
cartToken: anonymousCartToken,
|
||||||
|
shopId: config.shopId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${reactionCustomerToken}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
normalizedCart = normalizeCart(rawReconciledCart)
|
||||||
|
|
||||||
|
// Clear the anonymous cart cookies, as we're now using an account-tied cart
|
||||||
|
res.setHeader('Set-Cookie', [
|
||||||
|
getCartCookie(config.anonymousCartTokenCookie),
|
||||||
|
getCartCookie(config.anonymousCartIdCookie),
|
||||||
|
])
|
||||||
|
} else if (anonymousCartId && anonymousCartToken) {
|
||||||
|
const {
|
||||||
|
data: { cart: rawAnonymousCart },
|
||||||
} = await config.fetch(getAnomymousCartQuery, {
|
} = await config.fetch(getAnomymousCartQuery, {
|
||||||
variables: {
|
variables: {
|
||||||
cartId,
|
cartId: anonymousCartId,
|
||||||
cartToken: anonymousCartToken,
|
cartToken: anonymousCartToken,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
normalizedCart = normalizeCart(rawCart)
|
normalizedCart = normalizeCart(rawAnonymousCart)
|
||||||
|
} else if (reactionCustomerToken && !anonymousCartToken && !anonymousCartId) {
|
||||||
|
const accountId = await getViewerId({
|
||||||
|
customerToken: reactionCustomerToken,
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: { cart: rawAccountCart },
|
||||||
|
} = await config.fetch(
|
||||||
|
accountCartByAccountIdQuery,
|
||||||
|
{
|
||||||
|
variables: {
|
||||||
|
accountId,
|
||||||
|
shopId: config.shopId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${reactionCustomerToken}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
normalizedCart = normalizeCart(rawAccountCart)
|
||||||
} else {
|
} else {
|
||||||
|
// If there's no cart for now, return a dummy cart ID to keep Next Commerce happy
|
||||||
res.setHeader(
|
res.setHeader(
|
||||||
'Set-Cookie',
|
'Set-Cookie',
|
||||||
getCartCookie(config.cartCookie, config.dummyEmptyCartId, 999)
|
getCartCookie(config.anonymousCartIdCookie, config.dummyEmptyCartId, 999)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ const cartApi: ReactionCommerceApiHandler<Cart, CartHandlers> = async (
|
|||||||
if (!isAllowedMethod(req, res, METHODS)) return
|
if (!isAllowedMethod(req, res, METHODS)) return
|
||||||
|
|
||||||
const { cookies } = req
|
const { cookies } = req
|
||||||
const cartId = cookies[config.cartCookie]
|
const cartId = cookies[config.anonymousCartTokenCookie]
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Return current cart info
|
// Return current cart info
|
||||||
|
@ -3,7 +3,7 @@ import type { CommerceAPIConfig } from '@commerce/api'
|
|||||||
import {
|
import {
|
||||||
API_URL,
|
API_URL,
|
||||||
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
REACTION_CART_ID_COOKIE,
|
REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
REACTION_EMPTY_DUMMY_CART_ID,
|
REACTION_EMPTY_DUMMY_CART_ID,
|
||||||
REACTION_CUSTOMER_TOKEN_COOKIE,
|
REACTION_CUSTOMER_TOKEN_COOKIE,
|
||||||
REACTION_COOKIE_EXPIRE,
|
REACTION_COOKIE_EXPIRE,
|
||||||
@ -42,10 +42,10 @@ export class Config {
|
|||||||
const config = new Config({
|
const config = new Config({
|
||||||
locale: 'en-US',
|
locale: 'en-US',
|
||||||
commerceUrl: API_URL,
|
commerceUrl: API_URL,
|
||||||
cartCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
anonymousCartTokenCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
cartIdCookie: REACTION_CART_ID_COOKIE,
|
anonymousCartIdCookie: REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
dummyEmptyCartId: REACTION_EMPTY_DUMMY_CART_ID,
|
dummyEmptyCartId: REACTION_EMPTY_DUMMY_CART_ID,
|
||||||
cartCookieMaxAge: REACTION_COOKIE_EXPIRE,
|
anonymousCartTokenCookieMaxAge: REACTION_COOKIE_EXPIRE,
|
||||||
fetch: fetchGraphqlApi,
|
fetch: fetchGraphqlApi,
|
||||||
customerCookie: REACTION_CUSTOMER_TOKEN_COOKIE,
|
customerCookie: REACTION_CUSTOMER_TOKEN_COOKIE,
|
||||||
shopId: SHOP_ID,
|
shopId: SHOP_ID,
|
||||||
|
@ -39,8 +39,6 @@ export default function createApiHandler<
|
|||||||
handlers: H,
|
handlers: H,
|
||||||
defaultOptions: Options
|
defaultOptions: Options
|
||||||
) {
|
) {
|
||||||
console.log('next api handler', defaultOptions)
|
|
||||||
|
|
||||||
return function getApiHandler({
|
return function getApiHandler({
|
||||||
config,
|
config,
|
||||||
operations,
|
operations,
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
export const REACTION_ANONYMOUS_CART_TOKEN_COOKIE =
|
export const REACTION_ANONYMOUS_CART_TOKEN_COOKIE =
|
||||||
'reaction_anonymousCartToken'
|
'reaction_anonymousCartToken'
|
||||||
|
|
||||||
export const REACTION_CART_ID_COOKIE = 'reaction_cartId'
|
export const REACTION_ANONYMOUS_CART_ID_COOKIE = 'reaction_cartId'
|
||||||
|
|
||||||
export const REACTION_EMPTY_DUMMY_CART_ID = 'DUMMY_EMPTY_CART_ID'
|
export const REACTION_EMPTY_DUMMY_CART_ID = 'DUMMY_EMPTY_CART_ID'
|
||||||
|
|
||||||
|
@ -1,24 +0,0 @@
|
|||||||
import { getConfig, ReactionCommerceConfig } from '../api'
|
|
||||||
import getCustomerIdQuery from '../utils/queries/get-customer-id-query'
|
|
||||||
import Cookies from 'js-cookie'
|
|
||||||
|
|
||||||
async function getCustomerId({
|
|
||||||
customerToken: customerAccesToken,
|
|
||||||
config,
|
|
||||||
}: {
|
|
||||||
customerToken: string
|
|
||||||
config?: ReactionCommerceConfig
|
|
||||||
}): Promise<number | undefined> {
|
|
||||||
config = getConfig(config)
|
|
||||||
|
|
||||||
const { data } = await config.fetch(getCustomerIdQuery, {
|
|
||||||
variables: {
|
|
||||||
customerAccesToken:
|
|
||||||
customerAccesToken || Cookies.get(config.customerCookie),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
return data.customer?.id
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getCustomerId
|
|
26
framework/reactioncommerce/customer/get-viewer-id.ts
Normal file
26
framework/reactioncommerce/customer/get-viewer-id.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import { getConfig, ReactionCommerceConfig } from '../api'
|
||||||
|
import getViewerIdQuery from '../utils/queries/get-customer-id-query'
|
||||||
|
|
||||||
|
async function getViewerId({
|
||||||
|
customerToken: customerAccessToken,
|
||||||
|
config,
|
||||||
|
}: {
|
||||||
|
customerToken: string
|
||||||
|
config?: ReactionCommerceConfig
|
||||||
|
}): Promise<number | undefined> {
|
||||||
|
config = getConfig(config)
|
||||||
|
|
||||||
|
const { data } = await config.fetch(
|
||||||
|
getViewerIdQuery,
|
||||||
|
{},
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${customerAccessToken}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
return data.viewer?._id
|
||||||
|
}
|
||||||
|
|
||||||
|
export default getViewerId
|
@ -15,7 +15,7 @@ export type { ReactionCommerceProvider }
|
|||||||
|
|
||||||
export const reactionCommerceConfig: CommerceConfig = {
|
export const reactionCommerceConfig: CommerceConfig = {
|
||||||
locale: 'en-us',
|
locale: 'en-us',
|
||||||
cartCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
anonymousCartTokenCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
shopId: SHOP_ID,
|
shopId: SHOP_ID,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
REACTION_CART_ID_COOKIE,
|
REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
STORE_DOMAIN,
|
STORE_DOMAIN,
|
||||||
} from './const'
|
} from './const'
|
||||||
|
|
||||||
@ -20,8 +20,8 @@ import fetcher from './fetcher'
|
|||||||
|
|
||||||
export const reactionCommerceProvider = {
|
export const reactionCommerceProvider = {
|
||||||
locale: 'en-us',
|
locale: 'en-us',
|
||||||
cartCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
anonymousCartTokenCookie: REACTION_ANONYMOUS_CART_TOKEN_COOKIE,
|
||||||
cartIdCookie: REACTION_CART_ID_COOKIE,
|
anonymousCartIdCookie: REACTION_ANONYMOUS_CART_ID_COOKIE,
|
||||||
storeDomain: STORE_DOMAIN,
|
storeDomain: STORE_DOMAIN,
|
||||||
fetcher,
|
fetcher,
|
||||||
cart: { useCart, useAddItem, useUpdateItem, useRemoveItem },
|
cart: { useCart, useAddItem, useUpdateItem, useRemoveItem },
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import Cookies from 'js-cookie'
|
import Cookies from 'js-cookie'
|
||||||
import { REACTION_CART_ID_COOKIE } from '../const'
|
import { REACTION_ANONYMOUS_CART_ID_COOKIE } from '../const'
|
||||||
|
|
||||||
const getCartId = (id?: string) => {
|
const getCartId = (id?: string) => {
|
||||||
return id ?? Cookies.get(REACTION_CART_ID_COOKIE)
|
return id ?? Cookies.get(REACTION_ANONYMOUS_CART_ID_COOKIE)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default getCartId
|
export default getCartId
|
||||||
|
@ -0,0 +1,13 @@
|
|||||||
|
import { cartPayloadFragment } from '@framework/utils/queries/get-checkout-query'
|
||||||
|
|
||||||
|
const reconcileCartsMutation = `
|
||||||
|
mutation reconcileCartsMutation($input: ReconcileCartsInput!) {
|
||||||
|
reconcileCarts(input: $input) {
|
||||||
|
cart {
|
||||||
|
${cartPayloadFragment}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
export default reconcileCartsMutation
|
@ -0,0 +1,11 @@
|
|||||||
|
import { cartQueryFragment } from '@framework/utils/queries/get-checkout-query'
|
||||||
|
|
||||||
|
const accountCartByAccountIdQuery = `
|
||||||
|
query accountCartByAccountIdQuery($accountId: ID!, $shopId: ID!, $itemsAfterCursor: ConnectionCursor) {
|
||||||
|
cart: accountCartByAccountId(accountId: $accountId, shopId: $shopId) {
|
||||||
|
${cartQueryFragment}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
export default accountCartByAccountIdQuery
|
@ -1,8 +1,8 @@
|
|||||||
export const viewerQuery = /* GraphQL */ `
|
export const viewerIdQuery = /* GraphQL */ `
|
||||||
query getCustomerId($customerAccessToken: String!) {
|
query viewer {
|
||||||
customer(customerAccessToken: $customerAccessToken) {
|
viewer {
|
||||||
id
|
_id
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
export default viewerQuery
|
export default viewerIdQuery
|
||||||
|
Loading…
x
Reference in New Issue
Block a user