import type { GetProductQuery, GetProductQueryVariables, } from 'lib/bigcommerce/schema'; import type { RecursivePartial, RecursiveRequired } from '../types'; import { getConfig, Images, ProductImageVariables } from '..'; export const getProductQuery = /* GraphQL */ ` query getProduct( $slug: String! $imgSmallWidth: Int = 320 $imgSmallHeight: Int $imgMediumWidth: Int = 640 $imgMediumHeight: Int $imgLargeWidth: Int = 960 $imgLargeHeight: Int $imgXLWidth: Int = 1280 $imgXLHeight: Int ) { site { route(path: $slug) { node { __typename ... on Product { entityId name path brand { name } description prices { price { currencyCode value } salePrice { currencyCode value } } images { edges { node { urlSmall: url(width: $imgSmallWidth, height: $imgSmallHeight) urlMedium: url( width: $imgMediumWidth height: $imgMediumHeight ) urlLarge: url(width: $imgLargeWidth, height: $imgLargeHeight) urlXL: url(width: $imgXLWidth, height: $imgXLHeight) } } } variants { edges { node { entityId } } } options { edges { node { entityId displayName isRequired values { edges { node { entityId label } } } } } } } } } } } `; export interface GetProductResult { product?: T extends GetProductQuery ? Extract : unknown; } export type ProductVariables = Images & Omit; async function getProduct(opts: { query?: string; variables: ProductVariables; }): Promise>; async function getProduct(opts: { query: string; variables: V; }): Promise>; async function getProduct({ query = getProductQuery, variables: vars, }: { query?: string; variables: ProductVariables; }): Promise> { const config = getConfig(); const variables: GetProductQueryVariables = { ...config.imageVariables, ...vars, }; const data = await config.fetch>(query, { variables, }); const product = data.site?.route?.node; if (product?.__typename === 'Product') { return { product: product as RecursiveRequired, }; } return {}; } export default getProduct;