forked from crowetic/commerce
Merge branch 'master' of github.com:okbel/e-comm-example
This commit is contained in:
commit
b9ca1f275d
@ -8,7 +8,7 @@
|
|||||||
},
|
},
|
||||||
"documents": [
|
"documents": [
|
||||||
{
|
{
|
||||||
"./lib/bigcommerce/api/{operations,fragments}/**/*.ts": {
|
"./lib/bigcommerce/api/**/*.ts": {
|
||||||
"noRequire": true
|
"noRequire": true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -80,19 +80,19 @@ const cartApi: BigcommerceApiHandler<Cart, CartHandlers> = async (
|
|||||||
|
|
||||||
// Create or add an item to the cart
|
// Create or add an item to the cart
|
||||||
if (req.method === 'POST') {
|
if (req.method === 'POST') {
|
||||||
const body = { cartId, ...req.body }
|
const body = { ...req.body, cartId }
|
||||||
return await handlers['addItem']({ req, res, config, body })
|
return await handlers['addItem']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update item in cart
|
// Update item in cart
|
||||||
if (req.method === 'PUT') {
|
if (req.method === 'PUT') {
|
||||||
const body = { cartId, ...req.body }
|
const body = { ...req.body, cartId }
|
||||||
return await handlers['updateItem']({ req, res, config, body })
|
return await handlers['updateItem']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove an item from the cart
|
// Remove an item from the cart
|
||||||
if (req.method === 'DELETE') {
|
if (req.method === 'DELETE') {
|
||||||
const body = { cartId, ...req.body }
|
const body = { ...req.body, cartId }
|
||||||
return await handlers['removeItem']({ req, res, config, body })
|
return await handlers['removeItem']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
import { GetLoggedInCustomerQuery } from '@lib/bigcommerce/schema'
|
||||||
|
import type { CustomersHandlers } from '..'
|
||||||
|
|
||||||
|
export const getLoggedInCustomerQuery = /* GraphQL */ `
|
||||||
|
query getLoggedInCustomer {
|
||||||
|
customer {
|
||||||
|
entityId
|
||||||
|
firstName
|
||||||
|
lastName
|
||||||
|
email
|
||||||
|
company
|
||||||
|
customerGroupId
|
||||||
|
notes
|
||||||
|
phone
|
||||||
|
addressCount
|
||||||
|
attributeCount
|
||||||
|
storeCredit {
|
||||||
|
value
|
||||||
|
currencyCode
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
const getLoggedInCustomer: CustomersHandlers['getLoggedInCustomer'] = async ({
|
||||||
|
res,
|
||||||
|
config,
|
||||||
|
}) => {
|
||||||
|
const data = await config.fetch<GetLoggedInCustomerQuery>(
|
||||||
|
getLoggedInCustomerQuery
|
||||||
|
)
|
||||||
|
const { customer } = data
|
||||||
|
|
||||||
|
if (!customer) {
|
||||||
|
return res.status(400).json({
|
||||||
|
data: null,
|
||||||
|
errors: [{ message: 'Customer not found', code: 'not_found' }],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
res.status(200).json({ data: { customer } })
|
||||||
|
}
|
||||||
|
|
||||||
|
export default getLoggedInCustomer
|
26
lib/bigcommerce/api/customers/handlers/login.ts
Normal file
26
lib/bigcommerce/api/customers/handlers/login.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import login from '../../operations/login'
|
||||||
|
import type { LoginHandlers } from '../login'
|
||||||
|
|
||||||
|
const loginHandler: LoginHandlers['login'] = async ({
|
||||||
|
res,
|
||||||
|
body: { email, password },
|
||||||
|
config,
|
||||||
|
}) => {
|
||||||
|
// TODO: Add proper validations with something like Ajv
|
||||||
|
if (!(email && password)) {
|
||||||
|
return res.status(400).json({
|
||||||
|
data: null,
|
||||||
|
errors: [{ message: 'Invalid request' }],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// TODO: validate the password and email
|
||||||
|
// Passwords must be at least 7 characters and contain both alphabetic
|
||||||
|
// and numeric characters.
|
||||||
|
|
||||||
|
// TODO: Currently not working, fix this asap.
|
||||||
|
const loginData = await login({ variables: { email, password }, config })
|
||||||
|
|
||||||
|
res.status(200).json({ data: null })
|
||||||
|
}
|
||||||
|
|
||||||
|
export default loginHandler
|
@ -1,8 +1,8 @@
|
|||||||
import { BigcommerceApiError } from '../../utils/errors'
|
import { BigcommerceApiError } from '../../utils/errors'
|
||||||
import login from '../../operations/login'
|
import login from '../../operations/login'
|
||||||
import type { CustomersHandlers } from '..'
|
import { SignupHandlers } from '../signup'
|
||||||
|
|
||||||
const createCustomer: CustomersHandlers['createCustomer'] = async ({
|
const signup: SignupHandlers['signup'] = async ({
|
||||||
res,
|
res,
|
||||||
body: { firstName, lastName, email, password },
|
body: { firstName, lastName, email, password },
|
||||||
config,
|
config,
|
||||||
@ -57,6 +57,7 @@ const createCustomer: CustomersHandlers['createCustomer'] = async ({
|
|||||||
|
|
||||||
console.log('DATA', result.data)
|
console.log('DATA', result.data)
|
||||||
|
|
||||||
|
// TODO: Currently not working, fix this asap.
|
||||||
const loginData = await login({ variables: { email, password }, config })
|
const loginData = await login({ variables: { email, password }, config })
|
||||||
|
|
||||||
console.log('LOGIN DATA', loginData)
|
console.log('LOGIN DATA', loginData)
|
||||||
@ -64,4 +65,4 @@ const createCustomer: CustomersHandlers['createCustomer'] = async ({
|
|||||||
res.status(200).json({ data: result.data ?? null })
|
res.status(200).json({ data: result.data ?? null })
|
||||||
}
|
}
|
||||||
|
|
||||||
export default createCustomer
|
export default signup
|
@ -4,42 +4,30 @@ import createApiHandler, {
|
|||||||
} from '../utils/create-api-handler'
|
} from '../utils/create-api-handler'
|
||||||
import isAllowedMethod from '../utils/is-allowed-method'
|
import isAllowedMethod from '../utils/is-allowed-method'
|
||||||
import { BigcommerceApiError } from '../utils/errors'
|
import { BigcommerceApiError } from '../utils/errors'
|
||||||
import createCustomer from './handlers/create-customer'
|
import getLoggedInCustomer from './handlers/get-logged-in-customer'
|
||||||
|
|
||||||
type Body<T> = Partial<T> | undefined
|
export type Customer = any
|
||||||
|
|
||||||
export type Customer = null
|
export type CustomerData = {
|
||||||
|
customer: Customer
|
||||||
export type CreateCustomerBody = {
|
|
||||||
firstName: string
|
|
||||||
lastName: string
|
|
||||||
email: string
|
|
||||||
password: string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export type CustomersHandlers = {
|
export type CustomersHandlers = {
|
||||||
createCustomer: BigcommerceHandler<
|
getLoggedInCustomer: BigcommerceHandler<CustomerData, null>
|
||||||
Customer,
|
|
||||||
{ cartId?: string } & Body<CreateCustomerBody>
|
|
||||||
>
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const METHODS = ['POST']
|
const METHODS = ['GET']
|
||||||
|
|
||||||
const customersApi: BigcommerceApiHandler<Customer, CustomersHandlers> = async (
|
const customersApi: BigcommerceApiHandler<
|
||||||
req,
|
CustomerData,
|
||||||
res,
|
CustomersHandlers
|
||||||
config
|
> = async (req, res, config, handlers) => {
|
||||||
) => {
|
|
||||||
if (!isAllowedMethod(req, res, METHODS)) return
|
if (!isAllowedMethod(req, res, METHODS)) return
|
||||||
|
|
||||||
const { cookies } = req
|
|
||||||
const cartId = cookies[config.cartCookie]
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (req.method === 'POST') {
|
if (req.method === 'GET') {
|
||||||
const body = { cartId, ...req.body }
|
const body = null
|
||||||
return await handlers['createCustomer']({ req, res, config, body })
|
return await handlers['getLoggedInCustomer']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error)
|
console.error(error)
|
||||||
@ -53,6 +41,6 @@ const customersApi: BigcommerceApiHandler<Customer, CustomersHandlers> = async (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const handlers = { createCustomer }
|
const handlers = { getLoggedInCustomer }
|
||||||
|
|
||||||
export default createApiHandler(customersApi, handlers, {})
|
export default createApiHandler(customersApi, handlers, {})
|
||||||
|
45
lib/bigcommerce/api/customers/login.ts
Normal file
45
lib/bigcommerce/api/customers/login.ts
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import createApiHandler, {
|
||||||
|
BigcommerceApiHandler,
|
||||||
|
BigcommerceHandler,
|
||||||
|
} from '../utils/create-api-handler'
|
||||||
|
import isAllowedMethod from '../utils/is-allowed-method'
|
||||||
|
import { BigcommerceApiError } from '../utils/errors'
|
||||||
|
import login from './handlers/login'
|
||||||
|
|
||||||
|
export type LoginBody = {
|
||||||
|
email: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export type LoginHandlers = {
|
||||||
|
login: BigcommerceHandler<null, Partial<LoginBody>>
|
||||||
|
}
|
||||||
|
|
||||||
|
const METHODS = ['POST']
|
||||||
|
|
||||||
|
const loginApi: BigcommerceApiHandler<null, LoginHandlers> = async (
|
||||||
|
req,
|
||||||
|
res,
|
||||||
|
config,
|
||||||
|
handlers
|
||||||
|
) => {
|
||||||
|
if (!isAllowedMethod(req, res, METHODS)) return
|
||||||
|
|
||||||
|
try {
|
||||||
|
const body = req.body ?? {}
|
||||||
|
return await handlers['login']({ req, res, config, body })
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error)
|
||||||
|
|
||||||
|
const message =
|
||||||
|
error instanceof BigcommerceApiError
|
||||||
|
? 'An unexpected error ocurred with the Bigcommerce API'
|
||||||
|
: 'An unexpected error ocurred'
|
||||||
|
|
||||||
|
res.status(500).json({ data: null, errors: [{ message }] })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlers = { login }
|
||||||
|
|
||||||
|
export default createApiHandler(loginApi, handlers, {})
|
50
lib/bigcommerce/api/customers/signup.ts
Normal file
50
lib/bigcommerce/api/customers/signup.ts
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
import createApiHandler, {
|
||||||
|
BigcommerceApiHandler,
|
||||||
|
BigcommerceHandler,
|
||||||
|
} from '../utils/create-api-handler'
|
||||||
|
import isAllowedMethod from '../utils/is-allowed-method'
|
||||||
|
import { BigcommerceApiError } from '../utils/errors'
|
||||||
|
import signup from './handlers/signup'
|
||||||
|
|
||||||
|
export type SignupBody = {
|
||||||
|
firstName: string
|
||||||
|
lastName: string
|
||||||
|
email: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SignupHandlers = {
|
||||||
|
signup: BigcommerceHandler<null, { cartId?: string } & Partial<SignupBody>>
|
||||||
|
}
|
||||||
|
|
||||||
|
const METHODS = ['POST']
|
||||||
|
|
||||||
|
const signupApi: BigcommerceApiHandler<null, SignupHandlers> = async (
|
||||||
|
req,
|
||||||
|
res,
|
||||||
|
config,
|
||||||
|
handlers
|
||||||
|
) => {
|
||||||
|
if (!isAllowedMethod(req, res, METHODS)) return
|
||||||
|
|
||||||
|
const { cookies } = req
|
||||||
|
const cartId = cookies[config.cartCookie]
|
||||||
|
|
||||||
|
try {
|
||||||
|
const body = { ...req.body, cartId }
|
||||||
|
return await handlers['signup']({ req, res, config, body })
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error)
|
||||||
|
|
||||||
|
const message =
|
||||||
|
error instanceof BigcommerceApiError
|
||||||
|
? 'An unexpected error ocurred with the Bigcommerce API'
|
||||||
|
: 'An unexpected error ocurred'
|
||||||
|
|
||||||
|
res.status(500).json({ data: null, errors: [{ message }] })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlers = { signup }
|
||||||
|
|
||||||
|
export default createApiHandler(signupApi, handlers, {})
|
@ -3,7 +3,7 @@ import { BigcommerceConfig, getConfig } from '..'
|
|||||||
|
|
||||||
export type BigcommerceApiHandler<
|
export type BigcommerceApiHandler<
|
||||||
T = any,
|
T = any,
|
||||||
H extends BigcommerceHandlers = {},
|
H extends BigcommerceHandlers<T> = {},
|
||||||
Options extends {} = {}
|
Options extends {} = {}
|
||||||
> = (
|
> = (
|
||||||
req: NextApiRequest,
|
req: NextApiRequest,
|
||||||
|
@ -47,7 +47,7 @@ export type Wishlist = {
|
|||||||
export type WishlistList = Wishlist[]
|
export type WishlistList = Wishlist[]
|
||||||
|
|
||||||
export type WishlistHandlers = {
|
export type WishlistHandlers = {
|
||||||
getAllWishlists: BigcommerceHandler<WishlistList, { customerId?: string }>
|
getAllWishlists: BigcommerceHandler<WishlistList, { customerId?: string }>
|
||||||
getWishlist: BigcommerceHandler<Wishlist, { wishlistId?: string }>
|
getWishlist: BigcommerceHandler<Wishlist, { wishlistId?: string }>
|
||||||
addWishlist: BigcommerceHandler<
|
addWishlist: BigcommerceHandler<
|
||||||
Wishlist,
|
Wishlist,
|
||||||
@ -57,7 +57,10 @@ export type WishlistHandlers = {
|
|||||||
Wishlist,
|
Wishlist,
|
||||||
{ wishlistId: string } & Body<AddWishlistBody>
|
{ wishlistId: string } & Body<AddWishlistBody>
|
||||||
>
|
>
|
||||||
addItem: BigcommerceHandler<Wishlist, { wishlistId: string } & Body<AddItemBody>>
|
addItem: BigcommerceHandler<
|
||||||
|
Wishlist,
|
||||||
|
{ wishlistId: string } & Body<AddItemBody>
|
||||||
|
>
|
||||||
removeItem: BigcommerceHandler<
|
removeItem: BigcommerceHandler<
|
||||||
Wishlist,
|
Wishlist,
|
||||||
{ wishlistId: string } & Body<RemoveItemBody>
|
{ wishlistId: string } & Body<RemoveItemBody>
|
||||||
@ -86,13 +89,13 @@ const wishlistApi: BigcommerceApiHandler<Wishlist, WishlistHandlers> = async (
|
|||||||
|
|
||||||
// Add an item to the wishlist
|
// Add an item to the wishlist
|
||||||
if (req.method === 'POST' && wishlistId) {
|
if (req.method === 'POST' && wishlistId) {
|
||||||
const body = { wishlistId, ...req.body }
|
const body = { ...req.body, wishlistId }
|
||||||
return await handlers['addItem']({ req, res, config, body })
|
return await handlers['addItem']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update a wishlist
|
// Update a wishlist
|
||||||
if (req.method === 'PUT' && wishlistId) {
|
if (req.method === 'PUT' && wishlistId) {
|
||||||
const body = { wishlistId, ...req.body }
|
const body = { ...req.body, wishlistId }
|
||||||
return await handlers['updateWishlist']({ req, res, config, body })
|
return await handlers['updateWishlist']({ req, res, config, body })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,8 +26,8 @@ export const fetcher: HookFetcher<Cart, AddItemBody> = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url ?? defaultOpts.url,
|
...defaultOpts,
|
||||||
method: options?.method ?? defaultOpts.method,
|
...options,
|
||||||
body: { item },
|
body: { item },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ import type { Cart } from '../api/cart'
|
|||||||
|
|
||||||
const defaultOpts = {
|
const defaultOpts = {
|
||||||
url: '/api/bigcommerce/cart',
|
url: '/api/bigcommerce/cart',
|
||||||
|
method: 'GET',
|
||||||
}
|
}
|
||||||
|
|
||||||
export type { Cart }
|
export type { Cart }
|
||||||
@ -14,12 +15,7 @@ export const fetcher: HookFetcher<Cart | null, CartInput> = (
|
|||||||
{ cartId },
|
{ cartId },
|
||||||
fetch
|
fetch
|
||||||
) => {
|
) => {
|
||||||
return cartId
|
return cartId ? fetch({ ...defaultOpts, ...options }) : null
|
||||||
? fetch({
|
|
||||||
url: options?.url,
|
|
||||||
query: options?.query,
|
|
||||||
})
|
|
||||||
: null
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function extendHook(
|
export function extendHook(
|
||||||
|
@ -19,8 +19,8 @@ export const fetcher: HookFetcher<Cart | null, RemoveItemBody> = (
|
|||||||
fetch
|
fetch
|
||||||
) => {
|
) => {
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url ?? defaultOpts.url,
|
...defaultOpts,
|
||||||
method: options?.method ?? defaultOpts.method,
|
...options,
|
||||||
body: { itemId },
|
body: { itemId },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -28,8 +28,8 @@ export const fetcher: HookFetcher<Cart | null, UpdateItemBody> = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url ?? defaultOpts.url,
|
...defaultOpts,
|
||||||
method: options?.method ?? defaultOpts.method,
|
...options,
|
||||||
body: { itemId, item },
|
body: { itemId, item },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
24
lib/bigcommerce/schema.d.ts
vendored
24
lib/bigcommerce/schema.d.ts
vendored
@ -1653,6 +1653,30 @@ export enum CurrencyCode {
|
|||||||
Zwr = 'ZWR',
|
Zwr = 'ZWR',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type GetLoggedInCustomerQueryVariables = Exact<{ [key: string]: never }>
|
||||||
|
|
||||||
|
export type GetLoggedInCustomerQuery = { __typename?: 'Query' } & {
|
||||||
|
customer?: Maybe<
|
||||||
|
{ __typename?: 'Customer' } & Pick<
|
||||||
|
Customer,
|
||||||
|
| 'entityId'
|
||||||
|
| 'firstName'
|
||||||
|
| 'lastName'
|
||||||
|
| 'email'
|
||||||
|
| 'company'
|
||||||
|
| 'customerGroupId'
|
||||||
|
| 'notes'
|
||||||
|
| 'phone'
|
||||||
|
| 'addressCount'
|
||||||
|
| 'attributeCount'
|
||||||
|
> & {
|
||||||
|
storeCredit: Array<
|
||||||
|
{ __typename?: 'Money' } & Pick<Money, 'value' | 'currencyCode'>
|
||||||
|
>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
}
|
||||||
|
|
||||||
export type CategoryTreeItemFragment = {
|
export type CategoryTreeItemFragment = {
|
||||||
__typename?: 'CategoryTreeItem'
|
__typename?: 'CategoryTreeItem'
|
||||||
} & Pick<
|
} & Pick<
|
||||||
|
41
lib/bigcommerce/use-customer.tsx
Normal file
41
lib/bigcommerce/use-customer.tsx
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
import { ConfigInterface } from 'swr'
|
||||||
|
import { HookFetcher } from '@lib/commerce/utils/types'
|
||||||
|
import useCommerceCustomer, { CustomerInput } from '@lib/commerce/use-customer'
|
||||||
|
import type { Customer, CustomerData } from './api/customers'
|
||||||
|
|
||||||
|
const defaultOpts = {
|
||||||
|
url: '/api/bigcommerce/customer',
|
||||||
|
method: 'GET',
|
||||||
|
}
|
||||||
|
|
||||||
|
export type { Customer }
|
||||||
|
|
||||||
|
export const fetcher: HookFetcher<CustomerData | null, CustomerInput> = (
|
||||||
|
options,
|
||||||
|
{ cartId },
|
||||||
|
fetch
|
||||||
|
) => {
|
||||||
|
return cartId ? fetch({ ...defaultOpts, ...options }) : null
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extendHook(
|
||||||
|
customFetcher: typeof fetcher,
|
||||||
|
swrOptions?: ConfigInterface
|
||||||
|
) {
|
||||||
|
const useCustomer = () => {
|
||||||
|
const cart = useCommerceCustomer<CustomerData | null>(
|
||||||
|
defaultOpts,
|
||||||
|
[],
|
||||||
|
customFetcher,
|
||||||
|
{ revalidateOnFocus: false, ...swrOptions }
|
||||||
|
)
|
||||||
|
|
||||||
|
return cart
|
||||||
|
}
|
||||||
|
|
||||||
|
useCustomer.extend = extendHook
|
||||||
|
|
||||||
|
return useCustomer
|
||||||
|
}
|
||||||
|
|
||||||
|
export default extendHook(fetcher)
|
49
lib/bigcommerce/use-login.tsx
Normal file
49
lib/bigcommerce/use-login.tsx
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
import { useCallback } from 'react'
|
||||||
|
import { HookFetcher } from '@lib/commerce/utils/types'
|
||||||
|
import useCommerceLogin from '@lib/commerce/use-login'
|
||||||
|
import type { LoginBody } from './api/customers/login'
|
||||||
|
|
||||||
|
const defaultOpts = {
|
||||||
|
url: '/api/bigcommerce/customers/login',
|
||||||
|
method: 'POST',
|
||||||
|
}
|
||||||
|
|
||||||
|
export type LoginInput = LoginBody
|
||||||
|
|
||||||
|
export const fetcher: HookFetcher<null, LoginBody> = (
|
||||||
|
options,
|
||||||
|
{ email, password },
|
||||||
|
fetch
|
||||||
|
) => {
|
||||||
|
if (!(email && password)) {
|
||||||
|
throw new Error(
|
||||||
|
'A first name, last name, email and password are required to login'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fetch({
|
||||||
|
...defaultOpts,
|
||||||
|
...options,
|
||||||
|
body: { email, password },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extendHook(customFetcher: typeof fetcher) {
|
||||||
|
const useLogin = () => {
|
||||||
|
const fn = useCommerceLogin<null, LoginInput>(defaultOpts, customFetcher)
|
||||||
|
|
||||||
|
return useCallback(
|
||||||
|
async function login(input: LoginInput) {
|
||||||
|
const data = await fn(input)
|
||||||
|
return data
|
||||||
|
},
|
||||||
|
[fn]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
useLogin.extend = extendHook
|
||||||
|
|
||||||
|
return useLogin
|
||||||
|
}
|
||||||
|
|
||||||
|
export default extendHook(fetcher)
|
49
lib/bigcommerce/use-signup.tsx
Normal file
49
lib/bigcommerce/use-signup.tsx
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
import { useCallback } from 'react'
|
||||||
|
import { HookFetcher } from '@lib/commerce/utils/types'
|
||||||
|
import useCommerceSignup from '@lib/commerce/use-signup'
|
||||||
|
import type { SignupBody } from './api/customers/signup'
|
||||||
|
|
||||||
|
const defaultOpts = {
|
||||||
|
url: '/api/bigcommerce/customers/signup',
|
||||||
|
method: 'POST',
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SignupInput = SignupBody
|
||||||
|
|
||||||
|
export const fetcher: HookFetcher<null, SignupBody> = (
|
||||||
|
options,
|
||||||
|
{ firstName, lastName, email, password },
|
||||||
|
fetch
|
||||||
|
) => {
|
||||||
|
if (!(firstName && lastName && email && password)) {
|
||||||
|
throw new Error(
|
||||||
|
'A first name, last name, email and password are required to signup'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fetch({
|
||||||
|
...defaultOpts,
|
||||||
|
...options,
|
||||||
|
body: { firstName, lastName, email, password },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extendHook(customFetcher: typeof fetcher) {
|
||||||
|
const useSignup = () => {
|
||||||
|
const fn = useCommerceSignup<null, SignupInput>(defaultOpts, customFetcher)
|
||||||
|
|
||||||
|
return useCallback(
|
||||||
|
async function signup(input: SignupInput) {
|
||||||
|
const data = await fn(input)
|
||||||
|
return data
|
||||||
|
},
|
||||||
|
[fn]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
useSignup.extend = extendHook
|
||||||
|
|
||||||
|
return useSignup
|
||||||
|
}
|
||||||
|
|
||||||
|
export default extendHook(fetcher)
|
@ -17,8 +17,8 @@ export const fetcher: HookFetcher<Wishlist, AddItemBody> = (
|
|||||||
fetch
|
fetch
|
||||||
) => {
|
) => {
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url ?? defaultOpts.url,
|
...defaultOpts,
|
||||||
method: options?.method ?? defaultOpts.method,
|
...options,
|
||||||
body: { wishlistId, item },
|
body: { wishlistId, item },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -19,8 +19,8 @@ export const fetcher: HookFetcher<Wishlist | null, RemoveItemBody> = (
|
|||||||
fetch
|
fetch
|
||||||
) => {
|
) => {
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url ?? defaultOpts.url,
|
...defaultOpts,
|
||||||
method: options?.method ?? defaultOpts.method,
|
...options,
|
||||||
body: { wishlistId, itemId },
|
body: { wishlistId, itemId },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@ import type { Wishlist } from '../api/wishlist'
|
|||||||
|
|
||||||
const defaultOpts = {
|
const defaultOpts = {
|
||||||
url: '/api/bigcommerce/wishlists',
|
url: '/api/bigcommerce/wishlists',
|
||||||
|
method: 'GET',
|
||||||
}
|
}
|
||||||
|
|
||||||
export type { Wishlist }
|
export type { Wishlist }
|
||||||
@ -18,7 +19,8 @@ export const fetcher: HookFetcher<Wishlist | null, WishlistInput> = (
|
|||||||
fetch
|
fetch
|
||||||
) => {
|
) => {
|
||||||
return fetch({
|
return fetch({
|
||||||
url: options?.url,
|
...defaultOpts,
|
||||||
|
...options,
|
||||||
body: { wishlistId },
|
body: { wishlistId },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -14,8 +14,4 @@ export interface CommerceAPIFetchOptions<V> {
|
|||||||
preview?: boolean
|
preview?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: define interfaces for all the available operations
|
// TODO: define interfaces for all the available operations and API endpoints
|
||||||
|
|
||||||
// export interface CommerceAPI {
|
|
||||||
// getAllProducts(options?: { query: string }): Promise<any>;
|
|
||||||
// }
|
|
||||||
|
30
lib/commerce/use-customer.tsx
Normal file
30
lib/commerce/use-customer.tsx
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import { responseInterface, ConfigInterface } from 'swr'
|
||||||
|
import Cookies from 'js-cookie'
|
||||||
|
import type { HookInput, HookFetcher, HookFetcherOptions } from './utils/types'
|
||||||
|
import useData from './utils/use-data'
|
||||||
|
import { useCommerce } from '.'
|
||||||
|
|
||||||
|
export type CustomerResponse<T> = responseInterface<T, Error>
|
||||||
|
|
||||||
|
export type CustomerInput = {
|
||||||
|
cartId: string | undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function useCustomer<T>(
|
||||||
|
options: HookFetcherOptions,
|
||||||
|
input: HookInput,
|
||||||
|
fetcherFn: HookFetcher<T | null, CustomerInput>,
|
||||||
|
swrOptions?: ConfigInterface<T | null>
|
||||||
|
) {
|
||||||
|
// TODO: Replace this with the login cookie
|
||||||
|
const { cartCookie } = useCommerce()
|
||||||
|
|
||||||
|
const fetcher: typeof fetcherFn = (options, input, fetch) => {
|
||||||
|
input.cartId = Cookies.get(cartCookie)
|
||||||
|
return fetcherFn(options, input, fetch)
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = useData(options, input, fetcher, swrOptions)
|
||||||
|
|
||||||
|
return response as CustomerResponse<T>
|
||||||
|
}
|
5
lib/commerce/use-login.tsx
Normal file
5
lib/commerce/use-login.tsx
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import useAction from './utils/use-action'
|
||||||
|
|
||||||
|
const useLogin = useAction
|
||||||
|
|
||||||
|
export default useLogin
|
5
lib/commerce/use-signup.tsx
Normal file
5
lib/commerce/use-signup.tsx
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import useAction from './utils/use-action'
|
||||||
|
|
||||||
|
const useSignup = useAction
|
||||||
|
|
||||||
|
export default useSignup
|
@ -9,9 +9,7 @@ export default function useAction<T, Input>(
|
|||||||
const { fetcherRef } = useCommerce()
|
const { fetcherRef } = useCommerce()
|
||||||
|
|
||||||
return useCallback(
|
return useCallback(
|
||||||
function addItem(input: Input) {
|
(input: Input) => fetcher(options, input, fetcherRef.current),
|
||||||
return fetcher(options, input, fetcherRef.current)
|
|
||||||
},
|
|
||||||
[fetcher]
|
[fetcher]
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -9,9 +9,15 @@ export default function useData<T, Input = any>(
|
|||||||
swrOptions?: ConfigInterface<T>
|
swrOptions?: ConfigInterface<T>
|
||||||
) {
|
) {
|
||||||
const { fetcherRef } = useCommerce()
|
const { fetcherRef } = useCommerce()
|
||||||
const fetcher = (url?: string, query?: string, ...args: any[]) => {
|
const fetcher = (
|
||||||
|
url?: string,
|
||||||
|
query?: string,
|
||||||
|
method?: string,
|
||||||
|
...args: any[]
|
||||||
|
) => {
|
||||||
return fetcherFn(
|
return fetcherFn(
|
||||||
{ url, query },
|
{ url, query, method },
|
||||||
|
// Transform the input array into an object
|
||||||
args.reduce((obj, val, i) => {
|
args.reduce((obj, val, i) => {
|
||||||
obj[input[i][0]!] = val
|
obj[input[i][0]!] = val
|
||||||
return obj
|
return obj
|
||||||
@ -23,7 +29,9 @@ export default function useData<T, Input = any>(
|
|||||||
const response = useSWR(
|
const response = useSWR(
|
||||||
() => {
|
() => {
|
||||||
const opts = typeof options === 'function' ? options() : options
|
const opts = typeof options === 'function' ? options() : options
|
||||||
return opts ? [opts.url, opts.query, ...input.map((e) => e[1])] : null
|
return opts
|
||||||
|
? [opts.url, opts.query, opts.method, ...input.map((e) => e[1])]
|
||||||
|
: null
|
||||||
},
|
},
|
||||||
fetcher,
|
fetcher,
|
||||||
swrOptions
|
swrOptions
|
||||||
|
3
pages/api/bigcommerce/customers/login.ts
Normal file
3
pages/api/bigcommerce/customers/login.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
import loginApi from '@lib/bigcommerce/api/customers/login'
|
||||||
|
|
||||||
|
export default loginApi()
|
3
pages/api/bigcommerce/customers/signup.ts
Normal file
3
pages/api/bigcommerce/customers/signup.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
import signupApi from '@lib/bigcommerce/api/customers/signup'
|
||||||
|
|
||||||
|
export default signupApi()
|
Loading…
x
Reference in New Issue
Block a user