mirror of
https://github.com/vercel/commerce.git
synced 2025-06-19 05:31:22 +00:00
Remove unused utils
This commit is contained in:
parent
1fd5764f86
commit
a59f38baa5
@ -1 +0,0 @@
|
|||||||
export default function () {}
|
|
@ -1 +0,0 @@
|
|||||||
export default function () {}
|
|
@ -1 +0,0 @@
|
|||||||
export default function () {}
|
|
@ -1,58 +0,0 @@
|
|||||||
import type { NextApiHandler, NextApiRequest, NextApiResponse } from 'next'
|
|
||||||
import { ShopifyConfig, getConfig } from '..'
|
|
||||||
|
|
||||||
export type ShopifyApiHandler<
|
|
||||||
T = any,
|
|
||||||
H extends ShopifyHandlers = {},
|
|
||||||
Options extends {} = {}
|
|
||||||
> = (
|
|
||||||
req: NextApiRequest,
|
|
||||||
res: NextApiResponse<ShopifyApiResponse<T>>,
|
|
||||||
config: ShopifyConfig,
|
|
||||||
handlers: H,
|
|
||||||
// Custom configs that may be used by a particular handler
|
|
||||||
options: Options
|
|
||||||
) => void | Promise<void>
|
|
||||||
|
|
||||||
export type ShopifyHandler<T = any, Body = null> = (options: {
|
|
||||||
req: NextApiRequest
|
|
||||||
res: NextApiResponse<ShopifyApiResponse<T>>
|
|
||||||
config: ShopifyConfig
|
|
||||||
body: Body
|
|
||||||
}) => void | Promise<void>
|
|
||||||
|
|
||||||
export type ShopifyHandlers<T = any> = {
|
|
||||||
[k: string]: ShopifyHandler<T, any>
|
|
||||||
}
|
|
||||||
|
|
||||||
export type ShopifyApiResponse<T> = {
|
|
||||||
data: T | null
|
|
||||||
errors?: { message: string; code?: string }[]
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function createApiHandler<
|
|
||||||
T = any,
|
|
||||||
H extends ShopifyHandlers = {},
|
|
||||||
Options extends {} = {}
|
|
||||||
>(
|
|
||||||
handler: ShopifyApiHandler<T, H, Options>,
|
|
||||||
handlers: H,
|
|
||||||
defaultOptions: Options
|
|
||||||
) {
|
|
||||||
return function getApiHandler({
|
|
||||||
config,
|
|
||||||
operations,
|
|
||||||
options,
|
|
||||||
}: {
|
|
||||||
config?: ShopifyConfig
|
|
||||||
operations?: Partial<H>
|
|
||||||
options?: Options extends {} ? Partial<Options> : never
|
|
||||||
} = {}): NextApiHandler {
|
|
||||||
const ops = { ...operations, ...handlers }
|
|
||||||
const opts = { ...defaultOptions, ...options }
|
|
||||||
|
|
||||||
return function apiHandler(req, res) {
|
|
||||||
return handler(req, res, getConfig(config), ops, opts)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,41 +0,0 @@
|
|||||||
import { ProductEdge } from '../../schema'
|
|
||||||
import { ShopifyConfig } from '..'
|
|
||||||
|
|
||||||
const fetchAllProducts = async ({
|
|
||||||
config,
|
|
||||||
query,
|
|
||||||
variables,
|
|
||||||
acc = [],
|
|
||||||
cursor,
|
|
||||||
}: {
|
|
||||||
config: ShopifyConfig
|
|
||||||
query: string
|
|
||||||
acc?: ProductEdge[]
|
|
||||||
variables?: any
|
|
||||||
cursor?: string
|
|
||||||
}): Promise<ProductEdge[]> => {
|
|
||||||
const { data } = await config.fetch(query, {
|
|
||||||
variables: { ...variables, cursor },
|
|
||||||
})
|
|
||||||
|
|
||||||
const edges: ProductEdge[] = data.products?.edges ?? []
|
|
||||||
const hasNextPage = data.products?.pageInfo?.hasNextPage
|
|
||||||
acc = acc.concat(edges)
|
|
||||||
|
|
||||||
if (hasNextPage) {
|
|
||||||
const cursor = edges.pop()?.cursor
|
|
||||||
if (cursor) {
|
|
||||||
return fetchAllProducts({
|
|
||||||
config,
|
|
||||||
query,
|
|
||||||
variables,
|
|
||||||
acc,
|
|
||||||
cursor,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return acc
|
|
||||||
}
|
|
||||||
|
|
||||||
export default fetchAllProducts
|
|
@ -1,34 +0,0 @@
|
|||||||
import type { GraphQLFetcher } from '@commerce/api'
|
|
||||||
import fetch from './fetch'
|
|
||||||
|
|
||||||
import { API_URL, API_TOKEN } from '../../const'
|
|
||||||
import { getError } from '../../utils/handle-fetch-response'
|
|
||||||
|
|
||||||
const fetchGraphqlApi: GraphQLFetcher = async (
|
|
||||||
query: string,
|
|
||||||
{ variables } = {},
|
|
||||||
fetchOptions
|
|
||||||
) => {
|
|
||||||
const res = await fetch(API_URL, {
|
|
||||||
...fetchOptions,
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'X-Shopify-Storefront-Access-Token': API_TOKEN!,
|
|
||||||
...fetchOptions?.headers,
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
query,
|
|
||||||
variables,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
|
|
||||||
const { data, errors, status } = await res.json()
|
|
||||||
|
|
||||||
if (errors) {
|
|
||||||
throw getError(errors, status)
|
|
||||||
}
|
|
||||||
|
|
||||||
return { data, res }
|
|
||||||
}
|
|
||||||
export default fetchGraphqlApi
|
|
@ -1,2 +0,0 @@
|
|||||||
import zeitFetch from '@vercel/fetch'
|
|
||||||
export default zeitFetch()
|
|
@ -1,28 +0,0 @@
|
|||||||
import type { NextApiRequest, NextApiResponse } from 'next'
|
|
||||||
|
|
||||||
export default function isAllowedMethod(
|
|
||||||
req: NextApiRequest,
|
|
||||||
res: NextApiResponse,
|
|
||||||
allowedMethods: string[]
|
|
||||||
) {
|
|
||||||
const methods = allowedMethods.includes('OPTIONS')
|
|
||||||
? allowedMethods
|
|
||||||
: [...allowedMethods, 'OPTIONS']
|
|
||||||
|
|
||||||
if (!req.method || !methods.includes(req.method)) {
|
|
||||||
res.status(405)
|
|
||||||
res.setHeader('Allow', methods.join(', '))
|
|
||||||
res.end()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (req.method === 'OPTIONS') {
|
|
||||||
res.status(200)
|
|
||||||
res.setHeader('Allow', methods.join(', '))
|
|
||||||
res.setHeader('Content-Length', '0')
|
|
||||||
res.end()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
export const checkoutCreate = async (): Promise<void> => {
|
|
||||||
return Promise.resolve()
|
|
||||||
}
|
|
||||||
|
|
||||||
export default checkoutCreate
|
|
@ -1,48 +0,0 @@
|
|||||||
import { Cart } from '../types'
|
|
||||||
import { CommerceError } from '@commerce/utils/errors'
|
|
||||||
|
|
||||||
import {
|
|
||||||
CheckoutLineItemsAddPayload,
|
|
||||||
CheckoutLineItemsRemovePayload,
|
|
||||||
CheckoutLineItemsUpdatePayload,
|
|
||||||
CheckoutCreatePayload,
|
|
||||||
CheckoutUserError,
|
|
||||||
Checkout,
|
|
||||||
Maybe,
|
|
||||||
} from '../schema'
|
|
||||||
|
|
||||||
import { normalizeCart } from './normalize'
|
|
||||||
import throwUserErrors from './throw-user-errors'
|
|
||||||
|
|
||||||
export type CheckoutQuery = {
|
|
||||||
checkout: Checkout
|
|
||||||
checkoutUserErrors?: Array<CheckoutUserError>
|
|
||||||
}
|
|
||||||
|
|
||||||
export type CheckoutPayload =
|
|
||||||
| CheckoutLineItemsAddPayload
|
|
||||||
| CheckoutLineItemsUpdatePayload
|
|
||||||
| CheckoutLineItemsRemovePayload
|
|
||||||
| CheckoutCreatePayload
|
|
||||||
| CheckoutQuery
|
|
||||||
|
|
||||||
const checkoutToCart = (checkoutPayload?: Maybe<CheckoutPayload>): Cart => {
|
|
||||||
if (!checkoutPayload) {
|
|
||||||
throw new CommerceError({
|
|
||||||
message: 'Missing checkout payload from response',
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const checkout = checkoutPayload?.checkout
|
|
||||||
throwUserErrors(checkoutPayload?.checkoutUserErrors)
|
|
||||||
|
|
||||||
if (!checkout) {
|
|
||||||
throw new CommerceError({
|
|
||||||
message: 'Missing checkout object from response',
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return normalizeCart(checkout)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default checkoutToCart
|
|
@ -1,2 +0,0 @@
|
|||||||
export const getCustomerToken = () => null
|
|
||||||
export const setCustomerToken = () => null
|
|
@ -1,29 +0,0 @@
|
|||||||
import { ShopifyConfig } from '../api'
|
|
||||||
import { CollectionEdge } from '../schema'
|
|
||||||
import getSiteCollectionsQuery from './queries/get-all-collections-query'
|
|
||||||
|
|
||||||
export type Category = {
|
|
||||||
entityId: string
|
|
||||||
name: string
|
|
||||||
path: string
|
|
||||||
}
|
|
||||||
|
|
||||||
const getCategories = async (config: ShopifyConfig): Promise<Category[]> => {
|
|
||||||
const { data } = await config.fetch(getSiteCollectionsQuery, {
|
|
||||||
variables: {
|
|
||||||
first: 250,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
return (
|
|
||||||
data.collections?.edges?.map(
|
|
||||||
({ node: { id: entityId, title: name, handle } }: CollectionEdge) => ({
|
|
||||||
entityId,
|
|
||||||
name,
|
|
||||||
path: `/${handle}`,
|
|
||||||
})
|
|
||||||
) ?? []
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getCategories
|
|
@ -1,5 +0,0 @@
|
|||||||
const getCheckoutId = (id?: string) => {
|
|
||||||
return id
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getCheckoutId
|
|
@ -1,27 +0,0 @@
|
|||||||
import getSortVariables from './get-sort-variables'
|
|
||||||
import type { SearchProductsInput } from '../product/use-search'
|
|
||||||
|
|
||||||
export const getSearchVariables = ({
|
|
||||||
brandId,
|
|
||||||
search,
|
|
||||||
categoryId,
|
|
||||||
sort,
|
|
||||||
}: SearchProductsInput) => {
|
|
||||||
let query = ''
|
|
||||||
|
|
||||||
if (search) {
|
|
||||||
query += `product_type:${search} OR title:${search} OR tag:${search}`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (brandId) {
|
|
||||||
query += `${search ? ' AND ' : ''}vendor:${brandId}`
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
categoryId,
|
|
||||||
query,
|
|
||||||
...getSortVariables(sort, !!categoryId),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getSearchVariables
|
|
@ -1,32 +0,0 @@
|
|||||||
const getSortVariables = (sort?: string, isCategory: boolean = false) => {
|
|
||||||
let output = {}
|
|
||||||
switch (sort) {
|
|
||||||
case 'price-asc':
|
|
||||||
output = {
|
|
||||||
sortKey: 'PRICE',
|
|
||||||
reverse: false,
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'price-desc':
|
|
||||||
output = {
|
|
||||||
sortKey: 'PRICE',
|
|
||||||
reverse: true,
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'trending-desc':
|
|
||||||
output = {
|
|
||||||
sortKey: 'BEST_SELLING',
|
|
||||||
reverse: false,
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'latest-desc':
|
|
||||||
output = {
|
|
||||||
sortKey: isCategory ? 'CREATED' : 'CREATED_AT',
|
|
||||||
reverse: true,
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getSortVariables
|
|
@ -1,40 +0,0 @@
|
|||||||
import { ShopifyConfig } from '../api'
|
|
||||||
import fetchAllProducts from '../api/utils/fetch-all-products'
|
|
||||||
import getAllProductVendors from './queries/get-all-product-vendors-query'
|
|
||||||
|
|
||||||
export type Brand = {
|
|
||||||
entityId: string
|
|
||||||
name: string
|
|
||||||
path: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export type BrandEdge = {
|
|
||||||
node: Brand
|
|
||||||
}
|
|
||||||
|
|
||||||
export type Brands = BrandEdge[]
|
|
||||||
|
|
||||||
const getVendors = async (config: ShopifyConfig): Promise<BrandEdge[]> => {
|
|
||||||
const vendors = await fetchAllProducts({
|
|
||||||
config,
|
|
||||||
query: getAllProductVendors,
|
|
||||||
variables: {
|
|
||||||
first: 250,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
let vendorsStrings = vendors.map(({ node: { vendor } }) => vendor)
|
|
||||||
|
|
||||||
return [...new Set(vendorsStrings)].map((v) => {
|
|
||||||
const id = v.replace(/\s+/g, '-').toLowerCase()
|
|
||||||
return {
|
|
||||||
node: {
|
|
||||||
entityId: id,
|
|
||||||
name: v,
|
|
||||||
path: `brands/${id}`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
export default getVendors
|
|
@ -1,30 +0,0 @@
|
|||||||
import { FetcherOptions } from '@commerce/utils/types'
|
|
||||||
import throwUserErrors from './throw-user-errors'
|
|
||||||
|
|
||||||
import {
|
|
||||||
MutationCustomerActivateArgs,
|
|
||||||
MutationCustomerActivateByUrlArgs,
|
|
||||||
} from '../schema'
|
|
||||||
import { Mutation } from '../schema'
|
|
||||||
import { customerActivateByUrlMutation } from './mutations'
|
|
||||||
|
|
||||||
const handleAccountActivation = async (
|
|
||||||
fetch: <T = any, B = Body>(options: FetcherOptions<B>) => Promise<T>,
|
|
||||||
input: MutationCustomerActivateByUrlArgs
|
|
||||||
) => {
|
|
||||||
try {
|
|
||||||
const { customerActivateByUrl } = await fetch<
|
|
||||||
Mutation,
|
|
||||||
MutationCustomerActivateArgs
|
|
||||||
>({
|
|
||||||
query: customerActivateByUrlMutation,
|
|
||||||
variables: {
|
|
||||||
input,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
throwUserErrors(customerActivateByUrl?.customerUserErrors)
|
|
||||||
} catch (error) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default handleAccountActivation
|
|
@ -1,27 +0,0 @@
|
|||||||
import { FetcherError } from '@commerce/utils/errors'
|
|
||||||
|
|
||||||
export function getError(errors: any[], status: number) {
|
|
||||||
errors = errors ?? [{ message: 'Failed to fetch Shopify API' }]
|
|
||||||
return new FetcherError({ errors, status })
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getAsyncError(res: Response) {
|
|
||||||
const data = await res.json()
|
|
||||||
return getError(data.errors, res.status)
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleFetchResponse = async (res: Response) => {
|
|
||||||
if (res.ok) {
|
|
||||||
const { data, errors } = await res.json()
|
|
||||||
|
|
||||||
if (errors && errors.length) {
|
|
||||||
throw getError(errors, res.status)
|
|
||||||
}
|
|
||||||
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
throw await getAsyncError(res)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default handleFetchResponse
|
|
@ -1,36 +0,0 @@
|
|||||||
import { FetcherOptions } from '@commerce/utils/types'
|
|
||||||
import { CustomerAccessTokenCreateInput } from '../schema'
|
|
||||||
import { setCustomerToken } from './customer-token'
|
|
||||||
import { customerAccessTokenCreateMutation } from './mutations'
|
|
||||||
import throwUserErrors from './throw-user-errors'
|
|
||||||
|
|
||||||
const handleLogin = (data: any) => {
|
|
||||||
const response = data.customerAccessTokenCreate
|
|
||||||
throwUserErrors(response?.customerUserErrors)
|
|
||||||
|
|
||||||
const customerAccessToken = response?.customerAccessToken
|
|
||||||
const accessToken = customerAccessToken?.accessToken
|
|
||||||
|
|
||||||
if (accessToken) {
|
|
||||||
setCustomerToken(accessToken)
|
|
||||||
}
|
|
||||||
|
|
||||||
return customerAccessToken
|
|
||||||
}
|
|
||||||
|
|
||||||
export const handleAutomaticLogin = async (
|
|
||||||
fetch: <T = any, B = Body>(options: FetcherOptions<B>) => Promise<T>,
|
|
||||||
input: CustomerAccessTokenCreateInput
|
|
||||||
) => {
|
|
||||||
try {
|
|
||||||
const loginData = await fetch({
|
|
||||||
query: customerAccessTokenCreateMutation,
|
|
||||||
variables: {
|
|
||||||
input,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
handleLogin(loginData)
|
|
||||||
} catch (error) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default handleLogin
|
|
@ -1,15 +0,0 @@
|
|||||||
export { default as handleFetchResponse } from './handle-fetch-response'
|
|
||||||
export { default as getSearchVariables } from './get-search-variables'
|
|
||||||
export { default as getSortVariables } from './get-sort-variables'
|
|
||||||
export { default as getVendors } from './get-vendors'
|
|
||||||
export { default as getCategories } from './get-categories'
|
|
||||||
export { default as getCheckoutId } from './get-checkout-id'
|
|
||||||
export { default as checkoutCreate } from './checkout-create'
|
|
||||||
export { default as checkoutToCart } from './checkout-to-cart'
|
|
||||||
export { default as handleLogin, handleAutomaticLogin } from './handle-login'
|
|
||||||
export { default as handleAccountActivation } from './handle-account-activation'
|
|
||||||
export { default as throwUserErrors } from './throw-user-errors'
|
|
||||||
export * from './queries'
|
|
||||||
export * from './mutations'
|
|
||||||
export * from './normalize'
|
|
||||||
export * from './customer-token'
|
|
@ -1,165 +0,0 @@
|
|||||||
import { Product } from '@commerce/types'
|
|
||||||
|
|
||||||
import {
|
|
||||||
Product as ShopifyProduct,
|
|
||||||
Checkout,
|
|
||||||
CheckoutLineItemEdge,
|
|
||||||
SelectedOption,
|
|
||||||
ImageConnection,
|
|
||||||
ProductVariantConnection,
|
|
||||||
MoneyV2,
|
|
||||||
ProductOption,
|
|
||||||
} from '../schema'
|
|
||||||
|
|
||||||
import type { Cart, LineItem } from '../types'
|
|
||||||
|
|
||||||
const money = ({ amount, currencyCode }: MoneyV2) => {
|
|
||||||
return {
|
|
||||||
value: +amount,
|
|
||||||
currencyCode,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const normalizeProductOption = ({
|
|
||||||
id,
|
|
||||||
name: displayName,
|
|
||||||
values,
|
|
||||||
}: ProductOption) => {
|
|
||||||
return {
|
|
||||||
__typename: 'MultipleChoiceOption',
|
|
||||||
id,
|
|
||||||
displayName,
|
|
||||||
values: values.map((value) => {
|
|
||||||
let output: any = {
|
|
||||||
label: value,
|
|
||||||
}
|
|
||||||
if (displayName.match(/colou?r/gi)) {
|
|
||||||
output = {
|
|
||||||
...output,
|
|
||||||
hexColors: [value],
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const normalizeProductImages = ({ edges }: ImageConnection) =>
|
|
||||||
edges?.map(({ node: { originalSrc: url, ...rest } }) => ({
|
|
||||||
url,
|
|
||||||
...rest,
|
|
||||||
}))
|
|
||||||
|
|
||||||
const normalizeProductVariants = ({ edges }: ProductVariantConnection) => {
|
|
||||||
return edges?.map(
|
|
||||||
({
|
|
||||||
node: { id, selectedOptions, sku, title, priceV2, compareAtPriceV2 },
|
|
||||||
}) => {
|
|
||||||
return {
|
|
||||||
id,
|
|
||||||
name: title,
|
|
||||||
sku: sku ?? id,
|
|
||||||
price: +priceV2.amount,
|
|
||||||
listPrice: +compareAtPriceV2?.amount,
|
|
||||||
requiresShipping: true,
|
|
||||||
options: selectedOptions.map(({ name, value }: SelectedOption) => {
|
|
||||||
const options = normalizeProductOption({
|
|
||||||
id,
|
|
||||||
name,
|
|
||||||
values: [value],
|
|
||||||
})
|
|
||||||
return options
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export function normalizeProduct(productNode: ShopifyProduct): Product {
|
|
||||||
const {
|
|
||||||
id,
|
|
||||||
title: name,
|
|
||||||
vendor,
|
|
||||||
images,
|
|
||||||
variants,
|
|
||||||
description,
|
|
||||||
descriptionHtml,
|
|
||||||
handle,
|
|
||||||
priceRange,
|
|
||||||
options,
|
|
||||||
...rest
|
|
||||||
} = productNode
|
|
||||||
|
|
||||||
const product = {
|
|
||||||
id,
|
|
||||||
name,
|
|
||||||
vendor,
|
|
||||||
path: `/${handle}`,
|
|
||||||
slug: handle?.replace(/^\/+|\/+$/g, ''),
|
|
||||||
price: money(priceRange?.minVariantPrice),
|
|
||||||
images: normalizeProductImages(images),
|
|
||||||
variants: variants ? normalizeProductVariants(variants) : [],
|
|
||||||
options: options
|
|
||||||
? options
|
|
||||||
.filter((o) => o.name !== 'Title') // By default Shopify adds a 'Title' name when there's only one option. We don't need it. https://community.shopify.com/c/Shopify-APIs-SDKs/Adding-new-product-variant-is-automatically-adding-quot-Default/td-p/358095
|
|
||||||
.map((o) => normalizeProductOption(o))
|
|
||||||
: [],
|
|
||||||
...(description && { description }),
|
|
||||||
...(descriptionHtml && { descriptionHtml }),
|
|
||||||
...rest,
|
|
||||||
}
|
|
||||||
|
|
||||||
return product
|
|
||||||
}
|
|
||||||
|
|
||||||
export function normalizeCart(checkout: Checkout): Cart {
|
|
||||||
return {
|
|
||||||
id: checkout.id,
|
|
||||||
customerId: '',
|
|
||||||
email: '',
|
|
||||||
createdAt: checkout.createdAt,
|
|
||||||
currency: {
|
|
||||||
code: checkout.totalPriceV2?.currencyCode,
|
|
||||||
},
|
|
||||||
taxesIncluded: checkout.taxesIncluded,
|
|
||||||
lineItems: checkout.lineItems?.edges.map(normalizeLineItem),
|
|
||||||
lineItemsSubtotalPrice: +checkout.subtotalPriceV2?.amount,
|
|
||||||
subtotalPrice: +checkout.subtotalPriceV2?.amount,
|
|
||||||
totalPrice: checkout.totalPriceV2?.amount,
|
|
||||||
discounts: [],
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function normalizeLineItem({
|
|
||||||
node: { id, title, variant, quantity, ...rest },
|
|
||||||
}: CheckoutLineItemEdge): LineItem {
|
|
||||||
return {
|
|
||||||
id,
|
|
||||||
variantId: String(variant?.id),
|
|
||||||
productId: String(variant?.id),
|
|
||||||
name: `${title}`,
|
|
||||||
quantity,
|
|
||||||
variant: {
|
|
||||||
id: String(variant?.id),
|
|
||||||
sku: variant?.sku ?? '',
|
|
||||||
name: variant?.title!,
|
|
||||||
image: {
|
|
||||||
url: variant?.image?.originalSrc ?? '/product-img-placeholder.svg',
|
|
||||||
},
|
|
||||||
requiresShipping: variant?.requiresShipping ?? false,
|
|
||||||
price: variant?.priceV2?.amount,
|
|
||||||
listPrice: variant?.compareAtPriceV2?.amount,
|
|
||||||
},
|
|
||||||
path: String(variant?.product?.handle),
|
|
||||||
discounts: [],
|
|
||||||
options:
|
|
||||||
// By default Shopify adds a default variant with default names, we're removing it. https://community.shopify.com/c/Shopify-APIs-SDKs/Adding-new-product-variant-is-automatically-adding-quot-Default/td-p/358095
|
|
||||||
variant?.title == 'Default Title'
|
|
||||||
? []
|
|
||||||
: [
|
|
||||||
{
|
|
||||||
value: variant?.title,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,38 +0,0 @@
|
|||||||
import { ValidationError } from '@commerce/utils/errors'
|
|
||||||
|
|
||||||
import {
|
|
||||||
CheckoutErrorCode,
|
|
||||||
CheckoutUserError,
|
|
||||||
CustomerErrorCode,
|
|
||||||
CustomerUserError,
|
|
||||||
} from '../schema'
|
|
||||||
|
|
||||||
export type UserErrors = Array<CheckoutUserError | CustomerUserError>
|
|
||||||
|
|
||||||
export type UserErrorCode =
|
|
||||||
| CustomerErrorCode
|
|
||||||
| CheckoutErrorCode
|
|
||||||
| null
|
|
||||||
| undefined
|
|
||||||
|
|
||||||
const getCustomMessage = (code: UserErrorCode, message: string) => {
|
|
||||||
switch (code) {
|
|
||||||
case 'UNIDENTIFIED_CUSTOMER':
|
|
||||||
message = 'Cannot find an account that matches the provided credentials'
|
|
||||||
break
|
|
||||||
}
|
|
||||||
return message
|
|
||||||
}
|
|
||||||
|
|
||||||
export const throwUserErrors = (errors?: UserErrors) => {
|
|
||||||
if (errors && errors.length) {
|
|
||||||
throw new ValidationError({
|
|
||||||
errors: errors.map(({ code, message }) => ({
|
|
||||||
code: code ?? 'validation_error',
|
|
||||||
message: getCustomMessage(code, message),
|
|
||||||
})),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default throwUserErrors
|
|
Loading…
x
Reference in New Issue
Block a user