@visulima/crud 1.0.11 → 1.0.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/base-crud-handler.ts","../../src/handler/create.ts","../../src/handler/delete.ts","../../src/handler/list.ts","../../src/handler/read.ts","../../src/handler/update.ts","../../src/query-parser.ts","../../src/utils/format-resource-id.ts","../../src/utils/get-resource-name-from-url.ts","../../src/utils/get-route-type.ts","../../src/utils/validate-adapter-methods.ts","../../src/next/api/edge/index.ts","../../src/next/api/node/index.ts"],"names":["createHttpError","ApiError","createHandler","adapter","query","resourceName","request","create_default","deleteHandler","resourceId","delete_default","paginate","listHandler","pagination","isPaginated","paginationOptions","resources","page","total","list_default","readHandler","resource","read_default","updateHandler","update_default","set","parse","parseRecursive","select","selectFields","field","parseWhere","where","whereObject","parsed","key","parseOrderBy","orderBy","orderByObject","parseQuery","queryString","parsedQuery","query_parser_default","formatResourceId","format_resource_id_default","ensureCamelCase","string_","getResourceNameFromUrl","url","models","realPath","modelName","name","routeName","camelCaseModel","match","getRouteType","method","entityMatcher","simpleMatcher","pathMatch","get_route_type_default","adapterMethods","validateAdapterMethods","validate_adapter_methods_default","baseHandler","responseExecutor","finalExecutor","options","error_","error","config","routeNames","modelRoutes","responseOrContext","routeType","modelConfig","get_accessible_routes_default","resourceIdFormatted","parameters","responseConfig","base_crud_handler_default","handler","_","edge_default","response","node_default"],"mappings":"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","sourcesContent":["import type { HttpError } from \"http-errors\";\nimport createHttpError from \"http-errors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { ApiError } from \"next/dist/server/api-utils\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport createHandler from \"./handler/create\";\nimport deleteHandler from \"./handler/delete\";\nimport listHandler from \"./handler/list\";\nimport readHandler from \"./handler/read\";\nimport updateHandler from \"./handler/update\";\nimport parseQuery from \"./query-parser\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, HandlerParameters, ParsedQueryParameters,\n} from \"./types.d\";\nimport { RouteType } from \"./types.d\";\nimport formatResourceId from \"./utils/format-resource-id\";\nimport getAccessibleRoutes from \"./utils/get-accessible-routes\";\nimport { getResourceNameFromUrl } from \"./utils/get-resource-name-from-url\";\nimport getRouteType from \"./utils/get-route-type\";\nimport validateAdapterMethods from \"./utils/validate-adapter-methods\";\n\ntype ResponseConfig = { status: number; data: any };\n\nasync function baseHandler<R extends Request, Context, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: Context, responseConfig: ResponseConfig) => Promise<Response>,\n finalExecutor: (responseOrContext: Context) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>>;\n\nasync function baseHandler<R extends IncomingMessage, RResponse extends ServerResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<void>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>>;\n\n// eslint-disable-next-line sonarjs/cognitive-complexity,max-len\nasync function baseHandler<R extends { url: string; method: string }, RResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<RResponse>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>> {\n try {\n validateAdapterMethods(adapter);\n } catch (error_: any) {\n const error = error_ as HttpError;\n\n throw new ApiError(error.statusCode, error.message);\n }\n\n await adapter.init?.();\n\n const config = {\n formatResourceId,\n pagination: {\n perPage: 20,\n },\n ...options,\n };\n\n const routeNames = await adapter.mapModelsToRouteNames?.();\n const modelRoutes: { [key in M]?: string } = {};\n\n adapter.getModels().forEach((modelName) => {\n modelRoutes[modelName as M] = config.models?.[modelName as M]?.name ?? routeNames?.[modelName] ?? modelName;\n });\n\n return async (request, responseOrContext) => {\n const { resourceName, modelName } = getResourceNameFromUrl(request.url, modelRoutes as { [key in M]: string });\n\n if (!resourceName) {\n if (process.env.NODE_ENV === \"development\") {\n const mappedModels = await adapter.mapModelsToRouteNames?.();\n\n if (typeof mappedModels === \"object\") {\n throw createHttpError(404, `Resource not found, possible models: ${Object.values(mappedModels).join(\", \")}`);\n }\n }\n\n throw createHttpError(404, `Resource not found: ${request.url}`);\n }\n\n const { routeType, resourceId } = getRouteType(request.method, request.url, resourceName);\n\n if (routeType === null) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n const modelConfig = options?.models?.[modelName as M];\n\n const accessibleRoutes = getAccessibleRoutes(modelConfig?.only, modelConfig?.exclude, options?.exposeStrategy ?? \"all\");\n\n if (!accessibleRoutes.includes(routeType)) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n try {\n const resourceIdFormatted = modelConfig?.formatResourceId?.(resourceId as string) ?? config.formatResourceId(resourceId as string);\n\n await adapter.connect?.();\n\n const parsedQuery = parseQuery(request.url);\n const parameters: HandlerParameters<T, Q> = {\n adapter,\n query: adapter.parseQuery(modelName as M, parsedQuery),\n resourceName: modelName,\n };\n\n try {\n let responseConfig: ResponseConfig;\n\n switch (routeType) {\n case RouteType.READ_ONE: {\n responseConfig = await (config.handlers?.get ?? readHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.READ_ALL: {\n responseConfig = await (config.handlers?.list ?? listHandler)<T, Q>({\n ...parameters,\n query: {\n ...parameters.query,\n page: parsedQuery.page ? Number(parsedQuery.page) : undefined,\n limit: parsedQuery.limit ? Number(parsedQuery.limit) : undefined,\n },\n pagination: config.pagination,\n });\n break;\n }\n case RouteType.CREATE: {\n responseConfig = await (config.handlers?.create ?? createHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Record<string, any> },\n });\n break;\n }\n case RouteType.UPDATE: {\n responseConfig = await (config.handlers?.update ?? updateHandler)<T, Q, R>({\n ...parameters,\n resourceId: resourceIdFormatted,\n request: request as R & { body: Partial<T> },\n });\n break;\n }\n case RouteType.DELETE: {\n responseConfig = await (config.handlers?.delete ?? deleteHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n default: {\n responseConfig = {\n status: 404,\n data: \"Method not found\",\n };\n }\n }\n\n await responseExecutor(responseOrContext, responseConfig);\n } catch (error: any) {\n if (adapter.handleError && !(error instanceof ApiError)) {\n adapter.handleError(error);\n } else {\n throw error;\n }\n }\n } finally {\n await adapter.disconnect?.();\n\n await finalExecutor(responseOrContext);\n }\n };\n}\n\nexport default baseHandler;\n","import type { HandlerParameters } from \"../types.d\";\n\nconst createHandler: Handler = async ({\n adapter, query, resourceName, request,\n}) => {\n const resources = await adapter.create(resourceName, request.body, query);\n\n return {\n data: resources,\n status: 201,\n };\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: HandlerParameters<T, Q> & { request: Request & { body: Record<string, any> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default createHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst deleteHandler: Handler = async ({\n adapter, query, resourceName, resourceId,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const deletedResource = await adapter.delete(resourceName, resourceId, query);\n\n return {\n data: deletedResource,\n status: 200,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\nexport default deleteHandler;\n","import { paginate } from \"@visulima/pagination\";\n\nimport type { HandlerParameters, PaginationConfig, ParsedQueryParameters } from \"../types.d\";\n\ntype PaginationOptions = {\n page: number;\n perPage: number;\n};\n\nconst listHandler: Handler = async ({\n adapter, query, resourceName, pagination,\n}) => {\n let isPaginated = false;\n let paginationOptions: PaginationOptions | undefined;\n\n if (query.page !== undefined) {\n if (query.page <= 0) {\n throw new Error(\"page query must be a strictly positive number\");\n }\n\n paginationOptions = {\n page: query.page,\n perPage: query.limit ?? pagination.perPage,\n };\n }\n\n if (paginationOptions) {\n isPaginated = true;\n\n // eslint-disable-next-line no-param-reassign\n query.skip = (paginationOptions.page - 1) * paginationOptions.perPage;\n // eslint-disable-next-line no-param-reassign\n query.limit = paginationOptions.perPage;\n }\n\n const resources = await adapter.getAll(resourceName, query);\n\n if (isPaginated) {\n const { page, total } = await adapter.getPaginationData(resourceName, query);\n\n const paginator = paginate(page, (paginationOptions as PaginationOptions).perPage, total, resources);\n\n return {\n data: paginator.toJSON(),\n status: 200,\n };\n }\n\n return {\n data: resources,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q extends ParsedQueryParameters>(\n parameters: HandlerParameters<T, Q> & { pagination: PaginationConfig },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default listHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst readHandler: Handler = async ({\n adapter, query, resourceName, resourceId,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource !== \"object\") {\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n }\n\n return {\n data: resource,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default readHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst updateHandler: Handler = async ({\n adapter, query, resourceName, resourceId, request,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const updatedResource = await adapter.update(resourceName, resourceId, request.body, query);\n\n return {\n status: 201,\n data: updatedResource,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: UniqueResourceHandlerParameters<T, Q> & { request: Request & { body: Partial<T> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default updateHandler;\n","import set from \"lodash.set\";\nimport { parse } from \"node:url\";\n\nimport type {\n OrderByField, ParsedQueryParameters, RecursiveField, WhereField,\n} from \"./types.d\";\n\nconst parseRecursive = (select: string): RecursiveField => {\n const selectFields: RecursiveField = {};\n\n const fields = select.split(\",\");\n\n fields.forEach((field) => {\n set(selectFields, field, true);\n });\n\n return selectFields;\n};\n\nconst parseWhere = (where: string): WhereField => {\n const whereObject = JSON.parse(where);\n const parsed: WhereField = {};\n\n Object.keys(whereObject).forEach((key) => {\n set(parsed, key, whereObject[key]);\n });\n\n return parsed;\n};\n\nconst parseOrderBy = (orderBy: string): OrderByField => {\n const parsed: OrderByField = {};\n const orderByObject = JSON.parse(orderBy);\n\n if (Object.keys(orderByObject).length > 0) {\n const key = Object.keys(orderByObject)[0] as string;\n\n if (orderByObject[key as keyof typeof orderByObject] === \"$asc\" || orderByObject[key as keyof typeof orderByObject] === \"$desc\") {\n parsed[key] = orderByObject[key as keyof typeof orderByObject];\n }\n }\n\n if (Object.keys(parsed).length !== 1) {\n throw new Error(\"orderBy needs to be an object with exactly 1 property with either $asc or $desc value\");\n }\n\n return parsed;\n};\n\n// eslint-disable-next-line sonarjs/cognitive-complexity\nconst parseQuery = (queryString?: string): ParsedQueryParameters => {\n if (queryString) {\n const { query } = parse(queryString, true);\n const parsedQuery: ParsedQueryParameters = {};\n\n if (query[\"select\"]) {\n parsedQuery.select = parseRecursive(query[\"select\"] as string);\n }\n\n if (query[\"include\"]) {\n parsedQuery.include = parseRecursive(query[\"include\"] as string);\n }\n\n if (query[\"where\"]) {\n parsedQuery.where = parseWhere(query[\"where\"] as string);\n }\n\n if (query[\"orderBy\"]) {\n parsedQuery.orderBy = parseOrderBy(query[\"orderBy\"] as string);\n }\n\n if (query[\"limit\"] !== undefined) {\n parsedQuery.limit = Number.isFinite(+query[\"limit\"]) ? +query[\"limit\"] : undefined;\n }\n if (query[\"skip\"] !== undefined) {\n parsedQuery.skip = Number.isFinite(+query[\"skip\"]) ? +query[\"skip\"] : undefined;\n }\n\n if (query[\"distinct\"]) {\n parsedQuery.distinct = query[\"distinct\"] as string;\n }\n\n if (query[\"page\"]) {\n parsedQuery.page = Number.isFinite(+query[\"page\"]) ? +query[\"page\"] : undefined;\n }\n\n return {\n originalQuery: query,\n ...parsedQuery,\n };\n }\n\n return {};\n};\n\nexport default parseQuery;\n","const formatResourceId = (resourceId: string): number | string => (Number.isSafeInteger(+resourceId) ? +resourceId : resourceId);\n\nexport default formatResourceId;\n","export const ensureCamelCase = (string_: string): string => `${string_.charAt(0).toLowerCase()}${string_.slice(1)}`;\n\nexport const getResourceNameFromUrl = <M extends string = string>(url: string, models: { [key in M]: string }): {\n modelName: string,\n resourceName: string,\n} => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n const modelName = (Object.keys(models) as M[]).find((name) => {\n const routeName = models[name];\n const camelCaseModel = ensureCamelCase(routeName);\n\n // eslint-disable-next-line @rushstack/security/no-unsafe-regexp\n return new RegExp(`(${routeName}|${camelCaseModel}$)|(${routeName}|${camelCaseModel}/)`, \"g\").test(realPath);\n });\n\n if (modelName === undefined) {\n throw new Error(`Couldn't find model name for url ${url}`);\n }\n\n return {\n modelName,\n resourceName: models[modelName],\n };\n};\n","import { match } from \"path-to-regexp\";\n\nimport { RouteType } from \"../types.d\";\n\ntype PathMatch = { id: string };\n\nconst getRouteType: (\n method: string,\n url: string,\n resourceName: string,\n) => GetRouteType = (method, url, resourceName) => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n if (!realPath.includes(`/${resourceName}`)) {\n throw new Error(`invalid resource name '${resourceName}' for route '${realPath}'`);\n }\n\n const entityMatcher = match<PathMatch>([`/(.*)/${resourceName}`, `/(.*)/${resourceName}/:id`], { decode: decodeURIComponent });\n const simpleMatcher = match(`/(.*)/${resourceName}`, {\n decode: decodeURIComponent,\n });\n\n switch (method) {\n case \"GET\": {\n const pathMatch = entityMatcher(realPath);\n\n // If we got a /something after the resource name, we are reading 1 entity\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.READ_ONE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: RouteType.READ_ALL,\n };\n }\n case \"POST\": {\n const pathMatch = simpleMatcher(realPath);\n\n if (pathMatch) {\n return {\n routeType: RouteType.CREATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"PUT\":\n case \"PATCH\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.UPDATE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"DELETE\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.DELETE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: null,\n };\n }\n default: {\n return {\n routeType: null,\n };\n }\n }\n};\n\nexport type GetRouteType = {\n routeType: RouteType | null;\n resourceId?: string;\n};\n\nexport default getRouteType;\n","import createHttpError from \"http-errors\";\n\nimport type { Adapter } from \"../types.d\";\n\nconst adapterMethods = [\"create\", \"delete\", \"getAll\", \"getOne\", \"parseQuery\", \"update\", \"getPaginationData\", \"getModels\"];\n\nconst validateAdapterMethods = <T, Q>(adapter: Adapter<T, Q>): void => {\n adapterMethods.forEach((method) => {\n if (!adapter[method as keyof Adapter<T, Q>]) {\n throw createHttpError(500, `Adapter must implement the \"${method}\" method.`);\n }\n });\n};\n\nexport default validateAdapterMethods;\n","import baseHandler from \"../../../base-crud-handler\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters,\n} from \"../../../types.d\";\n\nasync function handler<T, R extends Request, Context, Q extends ParsedQueryParameters = any, M extends string = string>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>> {\n return baseHandler<R, Context, T, Q, M>(\n async (_, responseConfig) => new Response(JSON.stringify(responseConfig.data), {\n status: responseConfig.status,\n headers: {\n \"content-type\": \"application/json; charset=utf-8\",\n },\n }),\n async () => {},\n adapter,\n options,\n );\n}\n\nexport default handler;\n","import type { NextApiRequest, NextApiResponse } from \"next\";\n\nimport baseHandler from \"../../../base-crud-handler\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters,\n} from \"../../../types.d\";\n\nasync function handler<\n T,\n Q extends ParsedQueryParameters = any,\n R extends NextApiRequest = NextApiRequest,\n Response extends NextApiResponse = NextApiResponse,\n M extends string = string,\n>(adapter: Adapter<T, Q>, options?: HandlerOptions<M>): Promise<ExecuteHandler<R, Response>> {\n return baseHandler<R, Response, T, Q, M>(\n async (response, responseConfig) => {\n response.status(responseConfig.status).send(responseConfig.data);\n },\n async (response) => {\n response.end();\n },\n adapter,\n options,\n );\n}\n\nexport default handler;\n"]}
1
+ {"version":3,"sources":["../../src/base-crud-handler.ts","../../src/handler/create.ts","../../src/handler/delete.ts","../../src/handler/list.ts","../../src/handler/read.ts","../../src/handler/update.ts","../../src/query-parser.ts","../../src/utils/format-resource-id.ts","../../src/utils/get-resource-name-from-url.ts","../../src/utils/get-route-type.ts","../../src/utils/validate-adapter-methods.ts","../../src/next/api/edge/index.ts","../../src/next/api/node/index.ts"],"names":["createHttpError","ApiError","createHandler","adapter","query","request","resourceName","create_default","deleteHandler","resourceId","delete_default","paginate","listHandler","pagination","isPaginated","paginationOptions","resources","page","total","list_default","readHandler","resource","read_default","updateHandler","update_default","set","parse","parseRecursive","select","selectFields","field","parseWhere","where","whereObject","parsed","key","parseOrderBy","orderBy","orderByObject","parseQuery","queryString","parsedQuery","query_parser_default","formatResourceId","format_resource_id_default","ensureCamelCase","string_","getResourceNameFromUrl","url","models","realPath","modelName","name","routeName","camelCaseModel","match","getRouteType","method","entityMatcher","simpleMatcher","pathMatch","get_route_type_default","adapterMethods","validateAdapterMethods","validate_adapter_methods_default","baseHandler","responseExecutor","finalExecutor","options","error_","error","config","routeNames","modelRoutes","responseOrContext","routeType","modelConfig","get_accessible_routes_default","resourceIdFormatted","parameters","responseConfig","base_crud_handler_default","handler","_","edge_default","response","node_default"],"mappings":"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","sourcesContent":["import type { HttpError } from \"http-errors\";\nimport createHttpError from \"http-errors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { ApiError } from \"next/dist/server/api-utils\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport createHandler from \"./handler/create\";\nimport deleteHandler from \"./handler/delete\";\nimport listHandler from \"./handler/list\";\nimport readHandler from \"./handler/read\";\nimport updateHandler from \"./handler/update\";\nimport parseQuery from \"./query-parser\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, HandlerParameters, ParsedQueryParameters } from \"./types.d\";\nimport { RouteType } from \"./types.d\";\nimport formatResourceId from \"./utils/format-resource-id\";\nimport getAccessibleRoutes from \"./utils/get-accessible-routes\";\nimport { getResourceNameFromUrl } from \"./utils/get-resource-name-from-url\";\nimport getRouteType from \"./utils/get-route-type\";\nimport validateAdapterMethods from \"./utils/validate-adapter-methods\";\n\ninterface ResponseConfig {\n data: any;\n status: number;\n}\n\nasync function baseHandler<R extends Request, Context, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: Context, responseConfig: ResponseConfig) => Promise<Response>,\n finalExecutor: (responseOrContext: Context) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>>;\n\nasync function baseHandler<R extends IncomingMessage, RResponse extends ServerResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<void>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>>;\n\n// eslint-disable-next-line sonarjs/cognitive-complexity,func-style\nasync function baseHandler<R extends { method: string; url: string }, RResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<RResponse>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>> {\n try {\n validateAdapterMethods(adapter);\n } catch (error_: any) {\n const error = error_ as HttpError;\n\n throw new ApiError(error.statusCode, error.message);\n }\n\n await adapter.init?.();\n\n const config = {\n formatResourceId,\n pagination: {\n perPage: 20,\n },\n ...options,\n };\n\n const routeNames = await adapter.mapModelsToRouteNames?.();\n const modelRoutes: { [key in M]?: string } = {};\n\n adapter.getModels().forEach((modelName) => {\n modelRoutes[modelName as M] = config.models?.[modelName as M]?.name ?? routeNames?.[modelName] ?? modelName;\n });\n\n return async (request, responseOrContext) => {\n const { modelName, resourceName } = getResourceNameFromUrl(request.url, modelRoutes as { [key in M]: string });\n\n if (!resourceName) {\n if (process.env.NODE_ENV === \"development\") {\n const mappedModels = await adapter.mapModelsToRouteNames?.();\n\n if (typeof mappedModels === \"object\") {\n throw createHttpError(404, `Resource not found, possible models: ${Object.values(mappedModels).join(\", \")}`);\n }\n }\n\n throw createHttpError(404, `Resource not found: ${request.url}`);\n }\n\n const { resourceId, routeType } = getRouteType(request.method, request.url, resourceName);\n\n if (routeType === null) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n const modelConfig = options?.models?.[modelName as M];\n\n const accessibleRoutes = getAccessibleRoutes(modelConfig?.only, modelConfig?.exclude, options?.exposeStrategy ?? \"all\");\n\n if (!accessibleRoutes.includes(routeType)) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n try {\n const resourceIdFormatted = modelConfig?.formatResourceId?.(resourceId as string) ?? config.formatResourceId(resourceId as string);\n\n await adapter.connect?.();\n\n const parsedQuery = parseQuery(request.url);\n const parameters: HandlerParameters<T, Q> = {\n adapter,\n query: adapter.parseQuery(modelName as M, parsedQuery),\n resourceName: modelName,\n };\n\n try {\n let responseConfig: ResponseConfig;\n\n switch (routeType) {\n case RouteType.READ_ONE: {\n responseConfig = await (config.handlers?.get ?? readHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.READ_ALL: {\n responseConfig = await (config.handlers?.list ?? listHandler)<T, Q>({\n ...parameters,\n pagination: config.pagination,\n query: {\n ...parameters.query,\n limit: parsedQuery.limit ? Number(parsedQuery.limit) : undefined,\n page: parsedQuery.page ? Number(parsedQuery.page) : undefined,\n },\n });\n break;\n }\n case RouteType.CREATE: {\n responseConfig = await (config.handlers?.create ?? createHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Record<string, any> },\n });\n break;\n }\n case RouteType.UPDATE: {\n responseConfig = await (config.handlers?.update ?? updateHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Partial<T> },\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.DELETE: {\n responseConfig = await (config.handlers?.delete ?? deleteHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n default: {\n responseConfig = {\n data: \"Method not found\",\n status: 404,\n };\n }\n }\n\n await responseExecutor(responseOrContext, responseConfig);\n } catch (error: any) {\n if (adapter.handleError && !(error instanceof ApiError)) {\n adapter.handleError(error);\n } else {\n throw error;\n }\n }\n } finally {\n await adapter.disconnect?.();\n\n await finalExecutor(responseOrContext);\n }\n };\n}\n\nexport default baseHandler;\n","import type { HandlerParameters } from \"../types.d\";\n\nconst createHandler: Handler = async ({ adapter, query, request, resourceName }) => {\n const resources = await adapter.create(resourceName, request.body, query);\n\n return {\n data: resources,\n status: 201,\n };\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: HandlerParameters<T, Q> & { request: Request & { body: Record<string, any> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default createHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst deleteHandler: Handler = async ({ adapter, query, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const deletedResource = await adapter.delete(resourceName, resourceId, query);\n\n return {\n data: deletedResource,\n status: 200,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\nexport default deleteHandler;\n","import { paginate } from \"@visulima/pagination\";\n\nimport type { HandlerParameters, PaginationConfig, ParsedQueryParameters } from \"../types.d\";\n\ninterface PaginationOptions {\n page: number;\n perPage: number;\n}\n\nconst listHandler: Handler = async ({ adapter, pagination, query, resourceName }) => {\n let isPaginated = false;\n let paginationOptions: PaginationOptions | undefined;\n\n if (query.page !== undefined) {\n if (query.page <= 0) {\n throw new Error(\"page query must be a strictly positive number\");\n }\n\n paginationOptions = {\n page: query.page,\n perPage: query.limit ?? pagination.perPage,\n };\n }\n\n if (paginationOptions) {\n isPaginated = true;\n\n // eslint-disable-next-line no-param-reassign\n query.skip = (paginationOptions.page - 1) * paginationOptions.perPage;\n // eslint-disable-next-line no-param-reassign\n query.limit = paginationOptions.perPage;\n }\n\n const resources = await adapter.getAll(resourceName, query);\n\n if (isPaginated) {\n const { page, total } = await adapter.getPaginationData(resourceName, query);\n\n const paginator = paginate(page, (paginationOptions as PaginationOptions).perPage, total, resources);\n\n return {\n data: paginator.toJSON(),\n status: 200,\n };\n }\n\n return {\n data: resources,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q extends ParsedQueryParameters>(\n parameters: HandlerParameters<T, Q> & { pagination: PaginationConfig },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default listHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst readHandler: Handler = async ({ adapter, query, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource !== \"object\") {\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n }\n\n return {\n data: resource,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default readHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst updateHandler: Handler = async ({ adapter, query, request, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const updatedResource = await adapter.update(resourceName, resourceId, request.body, query);\n\n return {\n data: updatedResource,\n status: 201,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: UniqueResourceHandlerParameters<T, Q> & { request: Request & { body: Partial<T> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default updateHandler;\n","// eslint-disable-next-line no-restricted-imports\nimport set from \"lodash.set\";\nimport { parse } from \"node:url\";\n\nimport type { OrderByField, ParsedQueryParameters, RecursiveField, WhereField } from \"./types.d\";\n\nconst parseRecursive = (select: string): RecursiveField => {\n const selectFields: RecursiveField = {};\n\n const fields = select.split(\",\");\n\n fields.forEach((field) => {\n set(selectFields, field, true);\n });\n\n return selectFields;\n};\n\nconst parseWhere = (where: string): WhereField => {\n const whereObject = JSON.parse(where);\n const parsed: WhereField = {};\n\n Object.keys(whereObject).forEach((key) => {\n set(parsed, key, whereObject[key]);\n });\n\n return parsed;\n};\n\nconst parseOrderBy = (orderBy: string): OrderByField => {\n const parsed: OrderByField = {};\n const orderByObject = JSON.parse(orderBy);\n\n if (Object.keys(orderByObject).length > 0) {\n const key = Object.keys(orderByObject)[0] as string;\n\n if (orderByObject[key as keyof typeof orderByObject] === \"$asc\" || orderByObject[key as keyof typeof orderByObject] === \"$desc\") {\n parsed[key] = orderByObject[key as keyof typeof orderByObject];\n }\n }\n\n if (Object.keys(parsed).length !== 1) {\n throw new Error(\"orderBy needs to be an object with exactly 1 property with either $asc or $desc value\");\n }\n\n return parsed;\n};\n\n// eslint-disable-next-line sonarjs/cognitive-complexity\nconst parseQuery = (queryString?: string): ParsedQueryParameters => {\n if (queryString) {\n const { query } = parse(queryString, true);\n const parsedQuery: ParsedQueryParameters = {};\n\n if (query[\"select\"]) {\n parsedQuery.select = parseRecursive(query[\"select\"] as string);\n }\n\n if (query[\"include\"]) {\n parsedQuery.include = parseRecursive(query[\"include\"] as string);\n }\n\n if (query[\"where\"]) {\n parsedQuery.where = parseWhere(query[\"where\"] as string);\n }\n\n if (query[\"orderBy\"]) {\n parsedQuery.orderBy = parseOrderBy(query[\"orderBy\"] as string);\n }\n\n if (query[\"limit\"] !== undefined) {\n parsedQuery.limit = Number.isFinite(+query[\"limit\"]) ? +query[\"limit\"] : undefined;\n }\n if (query[\"skip\"] !== undefined) {\n parsedQuery.skip = Number.isFinite(+query[\"skip\"]) ? +query[\"skip\"] : undefined;\n }\n\n if (query[\"distinct\"]) {\n parsedQuery.distinct = query[\"distinct\"] as string;\n }\n\n if (query[\"page\"]) {\n parsedQuery.page = Number.isFinite(+query[\"page\"]) ? +query[\"page\"] : undefined;\n }\n\n return {\n originalQuery: query,\n ...parsedQuery,\n };\n }\n\n return {};\n};\n\nexport default parseQuery;\n","const formatResourceId = (resourceId: string): number | string => (Number.isSafeInteger(+resourceId) ? +resourceId : resourceId);\n\nexport default formatResourceId;\n","export const ensureCamelCase = (string_: string): string => `${string_.charAt(0).toLowerCase()}${string_.slice(1)}`;\n\nexport const getResourceNameFromUrl = <M extends string = string>(\n url: string,\n models: { [key in M]: string },\n): {\n modelName: string;\n resourceName: string;\n} => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n const modelName = (Object.keys(models) as M[]).find((name) => {\n const routeName = models[name];\n const camelCaseModel = ensureCamelCase(routeName);\n\n // eslint-disable-next-line @rushstack/security/no-unsafe-regexp,security/detect-non-literal-regexp\n return new RegExp(`(${routeName}|${camelCaseModel}$)|(${routeName}|${camelCaseModel}/)`, \"g\").test(realPath);\n });\n\n if (modelName === undefined) {\n throw new Error(`Couldn't find model name for url ${url}`);\n }\n\n return {\n modelName,\n resourceName: models[modelName],\n };\n};\n","import { match } from \"path-to-regexp\";\n\nimport { RouteType } from \"../types.d\";\n\ninterface PathMatch {\n id: string;\n}\n\nconst getRouteType: (method: string, url: string, resourceName: string) => GetRouteType = (method, url, resourceName) => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n if (!realPath.includes(`/${resourceName}`)) {\n throw new Error(`invalid resource name '${resourceName}' for route '${realPath}'`);\n }\n\n const entityMatcher = match<PathMatch>([`/(.*)/${resourceName}`, `/(.*)/${resourceName}/:id`], { decode: decodeURIComponent });\n const simpleMatcher = match(`/(.*)/${resourceName}`, {\n decode: decodeURIComponent,\n });\n\n switch (method) {\n case \"GET\": {\n const pathMatch = entityMatcher(realPath);\n\n // If we got a /something after the resource name, we are reading 1 entity\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.READ_ONE,\n };\n }\n\n return {\n routeType: RouteType.READ_ALL,\n };\n }\n case \"POST\": {\n const pathMatch = simpleMatcher(realPath);\n\n if (pathMatch) {\n return {\n routeType: RouteType.CREATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"PUT\":\n case \"PATCH\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.UPDATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"DELETE\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.DELETE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n default: {\n return {\n routeType: null,\n };\n }\n }\n};\n\nexport interface GetRouteType {\n resourceId?: string;\n routeType: RouteType | null;\n}\n\nexport default getRouteType;\n","import createHttpError from \"http-errors\";\n\nimport type { Adapter } from \"../types.d\";\n\nconst adapterMethods = [\"create\", \"delete\", \"getAll\", \"getOne\", \"parseQuery\", \"update\", \"getPaginationData\", \"getModels\"];\n\nconst validateAdapterMethods = <T, Q>(adapter: Adapter<T, Q>): void => {\n adapterMethods.forEach((method) => {\n if (!adapter[method as keyof Adapter<T, Q>]) {\n throw createHttpError(500, `Adapter must implement the \"${method}\" method.`);\n }\n });\n};\n\nexport default validateAdapterMethods;\n","import baseHandler from \"../../../base-crud-handler\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters } from \"../../../types.d\";\n\nconst handler = async <T, R extends Request, Context, Q extends ParsedQueryParameters = any, M extends string = string>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>> =>\n await baseHandler<R, Context, T, Q, M>(\n async (_, responseConfig) =>\n new Response(JSON.stringify(responseConfig.data), {\n headers: {\n \"content-type\": \"application/json; charset=utf-8\",\n },\n status: responseConfig.status,\n }),\n async () => {},\n adapter,\n options,\n );\n\nexport default handler;\n","import type { NextApiRequest, NextApiResponse } from \"next\";\n\nimport baseHandler from \"../../../base-crud-handler\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters } from \"../../../types.d\";\n\nconst handler = async <\n T,\n Q extends ParsedQueryParameters = any,\n R extends NextApiRequest = NextApiRequest,\n Response extends NextApiResponse = NextApiResponse,\n M extends string = string,\n>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Response>> =>\n await baseHandler<R, Response, T, Q, M>(\n async (response, responseConfig) => {\n response.status(responseConfig.status).send(responseConfig.data);\n },\n async (response) => {\n response.end();\n },\n adapter,\n options,\n );\n\nexport default handler;\n"]}
@@ -1,12 +1,12 @@
1
- import { c } from '../chunk-JENMZJBI.mjs';
2
- import P from 'http-errors';
1
+ import { b as b$1 } from '../chunk-54HHLITU.mjs';
2
+ import R from 'http-errors';
3
3
  import { ApiError } from 'next/dist/server/api-utils';
4
4
  import { paginate } from '@visulima/pagination';
5
5
  import M from 'lodash.set';
6
6
  import { parse } from 'url';
7
7
  import { match } from 'path-to-regexp';
8
8
 
9
- var v=async({adapter:s,query:e,resourceName:t,request:r})=>({data:await s.create(t,r.body,e),status:201}),T=v;var U=async({adapter:s,query:e,resourceName:t,resourceId:r})=>{if(typeof await s.getOne(t,r,e)=="object")return {data:await s.delete(t,r,e),status:200};throw P(404,`${t} ${r} not found`)},H=U;var D=async({adapter:s,query:e,resourceName:t,pagination:r})=>{let a=!1,i;if(e.page!==void 0){if(e.page<=0)throw new Error("page query must be a strictly positive number");i={page:e.page,perPage:e.limit??r.perPage};}i&&(a=!0,e.skip=(i.page-1)*i.perPage,e.limit=i.perPage);let n=await s.getAll(t,e);if(a){let{page:o,total:p}=await s.getPaginationData(t,e);return {data:paginate(o,i.perPage,p,n).toJSON(),status:200}}return {data:n,status:200}},E=D;var L=async({adapter:s,query:e,resourceName:t,resourceId:r})=>{let a=await s.getOne(t,r,e);if(typeof a!="object")throw P(404,`${t} ${r} not found`);return {data:a,status:200}},b=L;var q=async({adapter:s,query:e,resourceName:t,resourceId:r,request:a})=>{if(typeof await s.getOne(t,r,e)=="object")return {status:201,data:await s.update(t,r,a.body,e)};throw P(404,`${t} ${r} not found`)},w=q;var Q=s=>{let e={};return s.split(",").forEach(r=>{M(e,r,!0);}),e},J=s=>{let e=JSON.parse(s),t={};return Object.keys(e).forEach(r=>{M(t,r,e[r]);}),t},W=s=>{let e={},t=JSON.parse(s);if(Object.keys(t).length>0){let r=Object.keys(t)[0];(t[r]==="$asc"||t[r]==="$desc")&&(e[r]=t[r]);}if(Object.keys(e).length!==1)throw new Error("orderBy needs to be an object with exactly 1 property with either $asc or $desc value");return e},G=s=>{if(s){let{query:e}=parse(s,!0),t={};return e.select&&(t.select=Q(e.select)),e.include&&(t.include=Q(e.include)),e.where&&(t.where=J(e.where)),e.orderBy&&(t.orderBy=W(e.orderBy)),e.limit!==void 0&&(t.limit=Number.isFinite(+e.limit)?+e.limit:void 0),e.skip!==void 0&&(t.skip=Number.isFinite(+e.skip)?+e.skip:void 0),e.distinct&&(t.distinct=e.distinct),e.page&&(t.page=Number.isFinite(+e.page)?+e.page:void 0),{originalQuery:e,...t}}return {}},O=G;var V=s=>Number.isSafeInteger(+s)?+s:s,A=V;var z=s=>`${s.charAt(0).toLowerCase()}${s.slice(1)}`,C=(s,e)=>{let t=s.split("?")[0];if(t===void 0)throw new TypeError("Path is undefined");let r=Object.keys(e).find(a=>{let i=e[a],n=z(i);return new RegExp(`(${i}|${n}$)|(${i}|${n}/)`,"g").test(t)});if(r===void 0)throw new Error(`Couldn't find model name for url ${s}`);return {modelName:r,resourceName:e[r]}};var K=(s,e,t)=>{let r=e.split("?")[0];if(r===void 0)throw new TypeError("Path is undefined");if(!r.includes(`/${t}`))throw new Error(`invalid resource name '${t}' for route '${r}'`);let a=match([`/(.*)/${t}`,`/(.*)/${t}/:id`],{decode:decodeURIComponent}),i=match(`/(.*)/${t}`,{decode:decodeURIComponent});switch(s){case"GET":{let n=a(r);return typeof n=="object"&&n.params.id?{routeType:"READ_ONE",resourceId:n.params.id}:{routeType:"READ_ALL"}}case"POST":return i(r)?{routeType:"CREATE"}:{routeType:null};case"PUT":case"PATCH":{let n=a(r);return typeof n=="object"&&n.params.id?{routeType:"UPDATE",resourceId:n.params.id}:{routeType:null}}case"DELETE":{let n=a(r);return typeof n=="object"&&n.params.id?{routeType:"DELETE",resourceId:n.params.id}:{routeType:null}}default:return {routeType:null}}},k=K;var Y=["create","delete","getAll","getOne","parseQuery","update","getPaginationData","getModels"],Z=s=>{Y.forEach(e=>{if(!s[e])throw P(500,`Adapter must implement the "${e}" method.`)});},j=Z;async function ee(s,e,t,r){try{j(t);}catch(o){let p=o;throw new ApiError(p.statusCode,p.message)}await t.init?.();let a={formatResourceId:A,pagination:{perPage:20},...r},i=await t.mapModelsToRouteNames?.(),n={};return t.getModels().forEach(o=>{n[o]=a.models?.[o]?.name??i?.[o]??o;}),async(o,p)=>{let{resourceName:f,modelName:y}=C(o.url,n);if(!f)throw P(404,`Resource not found: ${o.url}`);let{routeType:g,resourceId:h}=k(o.method,o.url,f);if(g===null)throw P(404,`Route not found: ${o.url}`);let R=r?.models?.[y];if(!c(R?.only,R?.exclude,r?.exposeStrategy??"all").includes(g))throw P(404,`Route not found: ${o.url}`);try{let l=R?.formatResourceId?.(h)??a.formatResourceId(h);await t.connect?.();let c=O(o.url),u={adapter:t,query:t.parseQuery(y,c),resourceName:y};try{let d;switch(g){case"READ_ONE":{d=await(a.handlers?.get??b)({...u,resourceId:l});break}case"READ_ALL":{d=await(a.handlers?.list??E)({...u,query:{...u.query,page:c.page?Number(c.page):void 0,limit:c.limit?Number(c.limit):void 0},pagination:a.pagination});break}case"CREATE":{d=await(a.handlers?.create??T)({...u,request:o});break}case"UPDATE":{d=await(a.handlers?.update??w)({...u,resourceId:l,request:o});break}case"DELETE":{d=await(a.handlers?.delete??H)({...u,resourceId:l});break}default:d={status:404,data:"Method not found"};}await s(p,d);}catch(d){if(t.handleError&&!(d instanceof ApiError))t.handleError(d);else throw d}}finally{await t.disconnect?.(),await e(p);}}}var m=ee;async function te(s,e){return m(async(t,r)=>new Response(JSON.stringify(r.data),{status:r.status,headers:{"content-type":"application/json; charset=utf-8"}}),async()=>{},s,e)}var re=te;async function se(s,e){return m(async(t,r)=>{t.status(r.status).send(r.data);},async t=>{t.end();},s,e)}var ae=se;
9
+ var v=async({adapter:s,query:t,request:e,resourceName:r})=>({data:await s.create(r,e.body,t),status:201}),T=v;var U=async({adapter:s,query:t,resourceId:e,resourceName:r})=>{if(typeof await s.getOne(r,e,t)=="object")return {data:await s.delete(r,e,t),status:200};throw R(404,`${r} ${e} not found`)},H=U;var D=async({adapter:s,pagination:t,query:e,resourceName:r})=>{let a=!1,i;if(e.page!==void 0){if(e.page<=0)throw new Error("page query must be a strictly positive number");i={page:e.page,perPage:e.limit??t.perPage};}i&&(a=!0,e.skip=(i.page-1)*i.perPage,e.limit=i.perPage);let n=await s.getAll(r,e);if(a){let{page:o,total:p}=await s.getPaginationData(r,e);return {data:paginate(o,i.perPage,p,n).toJSON(),status:200}}return {data:n,status:200}},E=D;var L=async({adapter:s,query:t,resourceId:e,resourceName:r})=>{let a=await s.getOne(r,e,t);if(typeof a!="object")throw R(404,`${r} ${e} not found`);return {data:a,status:200}},b=L;var q=async({adapter:s,query:t,request:e,resourceId:r,resourceName:a})=>{if(typeof await s.getOne(a,r,t)=="object")return {data:await s.update(a,r,e.body,t),status:201};throw R(404,`${a} ${r} not found`)},w=q;var Q=s=>{let t={};return s.split(",").forEach(r=>{M(t,r,!0);}),t},J=s=>{let t=JSON.parse(s),e={};return Object.keys(t).forEach(r=>{M(e,r,t[r]);}),e},W=s=>{let t={},e=JSON.parse(s);if(Object.keys(e).length>0){let r=Object.keys(e)[0];(e[r]==="$asc"||e[r]==="$desc")&&(t[r]=e[r]);}if(Object.keys(t).length!==1)throw new Error("orderBy needs to be an object with exactly 1 property with either $asc or $desc value");return t},G=s=>{if(s){let{query:t}=parse(s,!0),e={};return t.select&&(e.select=Q(t.select)),t.include&&(e.include=Q(t.include)),t.where&&(e.where=J(t.where)),t.orderBy&&(e.orderBy=W(t.orderBy)),t.limit!==void 0&&(e.limit=Number.isFinite(+t.limit)?+t.limit:void 0),t.skip!==void 0&&(e.skip=Number.isFinite(+t.skip)?+t.skip:void 0),t.distinct&&(e.distinct=t.distinct),t.page&&(e.page=Number.isFinite(+t.page)?+t.page:void 0),{originalQuery:t,...e}}return {}},O=G;var V=s=>Number.isSafeInteger(+s)?+s:s,A=V;var z=s=>`${s.charAt(0).toLowerCase()}${s.slice(1)}`,C=(s,t)=>{let e=s.split("?")[0];if(e===void 0)throw new TypeError("Path is undefined");let r=Object.keys(t).find(a=>{let i=t[a],n=z(i);return new RegExp(`(${i}|${n}$)|(${i}|${n}/)`,"g").test(e)});if(r===void 0)throw new Error(`Couldn't find model name for url ${s}`);return {modelName:r,resourceName:t[r]}};var K=(s,t,e)=>{let r=t.split("?")[0];if(r===void 0)throw new TypeError("Path is undefined");if(!r.includes(`/${e}`))throw new Error(`invalid resource name '${e}' for route '${r}'`);let a=match([`/(.*)/${e}`,`/(.*)/${e}/:id`],{decode:decodeURIComponent}),i=match(`/(.*)/${e}`,{decode:decodeURIComponent});switch(s){case"GET":{let n=a(r);return typeof n=="object"&&n.params.id?{resourceId:n.params.id,routeType:"READ_ONE"}:{routeType:"READ_ALL"}}case"POST":return i(r)?{routeType:"CREATE"}:{routeType:null};case"PUT":case"PATCH":{let n=a(r);return typeof n=="object"&&n.params.id?{resourceId:n.params.id,routeType:"UPDATE"}:{routeType:null}}case"DELETE":{let n=a(r);return typeof n=="object"&&n.params.id?{resourceId:n.params.id,routeType:"DELETE"}:{routeType:null}}default:return {routeType:null}}},k=K;var Y=["create","delete","getAll","getOne","parseQuery","update","getPaginationData","getModels"],Z=s=>{Y.forEach(t=>{if(!s[t])throw R(500,`Adapter must implement the "${t}" method.`)});},j=Z;async function ee(s,t,e,r){try{j(e);}catch(o){let p=o;throw new ApiError(p.statusCode,p.message)}await e.init?.();let a={formatResourceId:A,pagination:{perPage:20},...r},i=await e.mapModelsToRouteNames?.(),n={};return e.getModels().forEach(o=>{n[o]=a.models?.[o]?.name??i?.[o]??o;}),async(o,p)=>{let{modelName:l,resourceName:P}=C(o.url,n);if(!P)throw R(404,`Resource not found: ${o.url}`);let{resourceId:h,routeType:y}=k(o.method,o.url,P);if(y===null)throw R(404,`Route not found: ${o.url}`);let g=r?.models?.[l];if(!b$1(g?.only,g?.exclude,r?.exposeStrategy??"all").includes(y))throw R(404,`Route not found: ${o.url}`);try{let m=g?.formatResourceId?.(h)??a.formatResourceId(h);await e.connect?.();let c=O(o.url),u={adapter:e,query:e.parseQuery(l,c),resourceName:l};try{let d;switch(y){case"READ_ONE":{d=await(a.handlers?.get??b)({...u,resourceId:m});break}case"READ_ALL":{d=await(a.handlers?.list??E)({...u,pagination:a.pagination,query:{...u.query,limit:c.limit?Number(c.limit):void 0,page:c.page?Number(c.page):void 0}});break}case"CREATE":{d=await(a.handlers?.create??T)({...u,request:o});break}case"UPDATE":{d=await(a.handlers?.update??w)({...u,request:o,resourceId:m});break}case"DELETE":{d=await(a.handlers?.delete??H)({...u,resourceId:m});break}default:d={data:"Method not found",status:404};}await s(p,d);}catch(d){if(e.handleError&&!(d instanceof ApiError))e.handleError(d);else throw d}}finally{await e.disconnect?.(),await t(p);}}}var f=ee;var te=async(s,t)=>await f(async(e,r)=>new Response(JSON.stringify(r.data),{headers:{"content-type":"application/json; charset=utf-8"},status:r.status}),async()=>{},s,t),re=te;var se=async(s,t)=>await f(async(e,r)=>{e.status(r.status).send(r.data);},async e=>{e.end();},s,t),ae=se;
10
10
 
11
11
  export { re as edgeHandler, ae as nodeHandler };
12
12
  //# sourceMappingURL=out.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/base-crud-handler.ts","../../src/handler/create.ts","../../src/handler/delete.ts","../../src/handler/list.ts","../../src/handler/read.ts","../../src/handler/update.ts","../../src/query-parser.ts","../../src/utils/format-resource-id.ts","../../src/utils/get-resource-name-from-url.ts","../../src/utils/get-route-type.ts","../../src/utils/validate-adapter-methods.ts","../../src/next/api/edge/index.ts","../../src/next/api/node/index.ts"],"names":["createHttpError","ApiError","createHandler","adapter","query","resourceName","request","create_default","deleteHandler","resourceId","delete_default","paginate","listHandler","pagination","isPaginated","paginationOptions","resources","page","total","list_default","readHandler","resource","read_default","updateHandler","update_default","set","parse","parseRecursive","select","selectFields","field","parseWhere","where","whereObject","parsed","key","parseOrderBy","orderBy","orderByObject","parseQuery","queryString","parsedQuery","query_parser_default","formatResourceId","format_resource_id_default","ensureCamelCase","string_","getResourceNameFromUrl","url","models","realPath","modelName","name","routeName","camelCaseModel","match","getRouteType","method","entityMatcher","simpleMatcher","pathMatch","get_route_type_default","adapterMethods","validateAdapterMethods","validate_adapter_methods_default","baseHandler","responseExecutor","finalExecutor","options","error_","error","config","routeNames","modelRoutes","responseOrContext","routeType","modelConfig","get_accessible_routes_default","resourceIdFormatted","parameters","responseConfig","base_crud_handler_default","handler","_","edge_default","response","node_default"],"mappings":"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","sourcesContent":["import type { HttpError } from \"http-errors\";\nimport createHttpError from \"http-errors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { ApiError } from \"next/dist/server/api-utils\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport createHandler from \"./handler/create\";\nimport deleteHandler from \"./handler/delete\";\nimport listHandler from \"./handler/list\";\nimport readHandler from \"./handler/read\";\nimport updateHandler from \"./handler/update\";\nimport parseQuery from \"./query-parser\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, HandlerParameters, ParsedQueryParameters,\n} from \"./types.d\";\nimport { RouteType } from \"./types.d\";\nimport formatResourceId from \"./utils/format-resource-id\";\nimport getAccessibleRoutes from \"./utils/get-accessible-routes\";\nimport { getResourceNameFromUrl } from \"./utils/get-resource-name-from-url\";\nimport getRouteType from \"./utils/get-route-type\";\nimport validateAdapterMethods from \"./utils/validate-adapter-methods\";\n\ntype ResponseConfig = { status: number; data: any };\n\nasync function baseHandler<R extends Request, Context, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: Context, responseConfig: ResponseConfig) => Promise<Response>,\n finalExecutor: (responseOrContext: Context) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>>;\n\nasync function baseHandler<R extends IncomingMessage, RResponse extends ServerResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<void>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>>;\n\n// eslint-disable-next-line sonarjs/cognitive-complexity,max-len\nasync function baseHandler<R extends { url: string; method: string }, RResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<RResponse>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>> {\n try {\n validateAdapterMethods(adapter);\n } catch (error_: any) {\n const error = error_ as HttpError;\n\n throw new ApiError(error.statusCode, error.message);\n }\n\n await adapter.init?.();\n\n const config = {\n formatResourceId,\n pagination: {\n perPage: 20,\n },\n ...options,\n };\n\n const routeNames = await adapter.mapModelsToRouteNames?.();\n const modelRoutes: { [key in M]?: string } = {};\n\n adapter.getModels().forEach((modelName) => {\n modelRoutes[modelName as M] = config.models?.[modelName as M]?.name ?? routeNames?.[modelName] ?? modelName;\n });\n\n return async (request, responseOrContext) => {\n const { resourceName, modelName } = getResourceNameFromUrl(request.url, modelRoutes as { [key in M]: string });\n\n if (!resourceName) {\n if (process.env.NODE_ENV === \"development\") {\n const mappedModels = await adapter.mapModelsToRouteNames?.();\n\n if (typeof mappedModels === \"object\") {\n throw createHttpError(404, `Resource not found, possible models: ${Object.values(mappedModels).join(\", \")}`);\n }\n }\n\n throw createHttpError(404, `Resource not found: ${request.url}`);\n }\n\n const { routeType, resourceId } = getRouteType(request.method, request.url, resourceName);\n\n if (routeType === null) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n const modelConfig = options?.models?.[modelName as M];\n\n const accessibleRoutes = getAccessibleRoutes(modelConfig?.only, modelConfig?.exclude, options?.exposeStrategy ?? \"all\");\n\n if (!accessibleRoutes.includes(routeType)) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n try {\n const resourceIdFormatted = modelConfig?.formatResourceId?.(resourceId as string) ?? config.formatResourceId(resourceId as string);\n\n await adapter.connect?.();\n\n const parsedQuery = parseQuery(request.url);\n const parameters: HandlerParameters<T, Q> = {\n adapter,\n query: adapter.parseQuery(modelName as M, parsedQuery),\n resourceName: modelName,\n };\n\n try {\n let responseConfig: ResponseConfig;\n\n switch (routeType) {\n case RouteType.READ_ONE: {\n responseConfig = await (config.handlers?.get ?? readHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.READ_ALL: {\n responseConfig = await (config.handlers?.list ?? listHandler)<T, Q>({\n ...parameters,\n query: {\n ...parameters.query,\n page: parsedQuery.page ? Number(parsedQuery.page) : undefined,\n limit: parsedQuery.limit ? Number(parsedQuery.limit) : undefined,\n },\n pagination: config.pagination,\n });\n break;\n }\n case RouteType.CREATE: {\n responseConfig = await (config.handlers?.create ?? createHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Record<string, any> },\n });\n break;\n }\n case RouteType.UPDATE: {\n responseConfig = await (config.handlers?.update ?? updateHandler)<T, Q, R>({\n ...parameters,\n resourceId: resourceIdFormatted,\n request: request as R & { body: Partial<T> },\n });\n break;\n }\n case RouteType.DELETE: {\n responseConfig = await (config.handlers?.delete ?? deleteHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n default: {\n responseConfig = {\n status: 404,\n data: \"Method not found\",\n };\n }\n }\n\n await responseExecutor(responseOrContext, responseConfig);\n } catch (error: any) {\n if (adapter.handleError && !(error instanceof ApiError)) {\n adapter.handleError(error);\n } else {\n throw error;\n }\n }\n } finally {\n await adapter.disconnect?.();\n\n await finalExecutor(responseOrContext);\n }\n };\n}\n\nexport default baseHandler;\n","import type { HandlerParameters } from \"../types.d\";\n\nconst createHandler: Handler = async ({\n adapter, query, resourceName, request,\n}) => {\n const resources = await adapter.create(resourceName, request.body, query);\n\n return {\n data: resources,\n status: 201,\n };\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: HandlerParameters<T, Q> & { request: Request & { body: Record<string, any> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default createHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst deleteHandler: Handler = async ({\n adapter, query, resourceName, resourceId,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const deletedResource = await adapter.delete(resourceName, resourceId, query);\n\n return {\n data: deletedResource,\n status: 200,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\nexport default deleteHandler;\n","import { paginate } from \"@visulima/pagination\";\n\nimport type { HandlerParameters, PaginationConfig, ParsedQueryParameters } from \"../types.d\";\n\ntype PaginationOptions = {\n page: number;\n perPage: number;\n};\n\nconst listHandler: Handler = async ({\n adapter, query, resourceName, pagination,\n}) => {\n let isPaginated = false;\n let paginationOptions: PaginationOptions | undefined;\n\n if (query.page !== undefined) {\n if (query.page <= 0) {\n throw new Error(\"page query must be a strictly positive number\");\n }\n\n paginationOptions = {\n page: query.page,\n perPage: query.limit ?? pagination.perPage,\n };\n }\n\n if (paginationOptions) {\n isPaginated = true;\n\n // eslint-disable-next-line no-param-reassign\n query.skip = (paginationOptions.page - 1) * paginationOptions.perPage;\n // eslint-disable-next-line no-param-reassign\n query.limit = paginationOptions.perPage;\n }\n\n const resources = await adapter.getAll(resourceName, query);\n\n if (isPaginated) {\n const { page, total } = await adapter.getPaginationData(resourceName, query);\n\n const paginator = paginate(page, (paginationOptions as PaginationOptions).perPage, total, resources);\n\n return {\n data: paginator.toJSON(),\n status: 200,\n };\n }\n\n return {\n data: resources,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q extends ParsedQueryParameters>(\n parameters: HandlerParameters<T, Q> & { pagination: PaginationConfig },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default listHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst readHandler: Handler = async ({\n adapter, query, resourceName, resourceId,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource !== \"object\") {\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n }\n\n return {\n data: resource,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default readHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst updateHandler: Handler = async ({\n adapter, query, resourceName, resourceId, request,\n}) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const updatedResource = await adapter.update(resourceName, resourceId, request.body, query);\n\n return {\n status: 201,\n data: updatedResource,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: UniqueResourceHandlerParameters<T, Q> & { request: Request & { body: Partial<T> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default updateHandler;\n","import set from \"lodash.set\";\nimport { parse } from \"node:url\";\n\nimport type {\n OrderByField, ParsedQueryParameters, RecursiveField, WhereField,\n} from \"./types.d\";\n\nconst parseRecursive = (select: string): RecursiveField => {\n const selectFields: RecursiveField = {};\n\n const fields = select.split(\",\");\n\n fields.forEach((field) => {\n set(selectFields, field, true);\n });\n\n return selectFields;\n};\n\nconst parseWhere = (where: string): WhereField => {\n const whereObject = JSON.parse(where);\n const parsed: WhereField = {};\n\n Object.keys(whereObject).forEach((key) => {\n set(parsed, key, whereObject[key]);\n });\n\n return parsed;\n};\n\nconst parseOrderBy = (orderBy: string): OrderByField => {\n const parsed: OrderByField = {};\n const orderByObject = JSON.parse(orderBy);\n\n if (Object.keys(orderByObject).length > 0) {\n const key = Object.keys(orderByObject)[0] as string;\n\n if (orderByObject[key as keyof typeof orderByObject] === \"$asc\" || orderByObject[key as keyof typeof orderByObject] === \"$desc\") {\n parsed[key] = orderByObject[key as keyof typeof orderByObject];\n }\n }\n\n if (Object.keys(parsed).length !== 1) {\n throw new Error(\"orderBy needs to be an object with exactly 1 property with either $asc or $desc value\");\n }\n\n return parsed;\n};\n\n// eslint-disable-next-line sonarjs/cognitive-complexity\nconst parseQuery = (queryString?: string): ParsedQueryParameters => {\n if (queryString) {\n const { query } = parse(queryString, true);\n const parsedQuery: ParsedQueryParameters = {};\n\n if (query[\"select\"]) {\n parsedQuery.select = parseRecursive(query[\"select\"] as string);\n }\n\n if (query[\"include\"]) {\n parsedQuery.include = parseRecursive(query[\"include\"] as string);\n }\n\n if (query[\"where\"]) {\n parsedQuery.where = parseWhere(query[\"where\"] as string);\n }\n\n if (query[\"orderBy\"]) {\n parsedQuery.orderBy = parseOrderBy(query[\"orderBy\"] as string);\n }\n\n if (query[\"limit\"] !== undefined) {\n parsedQuery.limit = Number.isFinite(+query[\"limit\"]) ? +query[\"limit\"] : undefined;\n }\n if (query[\"skip\"] !== undefined) {\n parsedQuery.skip = Number.isFinite(+query[\"skip\"]) ? +query[\"skip\"] : undefined;\n }\n\n if (query[\"distinct\"]) {\n parsedQuery.distinct = query[\"distinct\"] as string;\n }\n\n if (query[\"page\"]) {\n parsedQuery.page = Number.isFinite(+query[\"page\"]) ? +query[\"page\"] : undefined;\n }\n\n return {\n originalQuery: query,\n ...parsedQuery,\n };\n }\n\n return {};\n};\n\nexport default parseQuery;\n","const formatResourceId = (resourceId: string): number | string => (Number.isSafeInteger(+resourceId) ? +resourceId : resourceId);\n\nexport default formatResourceId;\n","export const ensureCamelCase = (string_: string): string => `${string_.charAt(0).toLowerCase()}${string_.slice(1)}`;\n\nexport const getResourceNameFromUrl = <M extends string = string>(url: string, models: { [key in M]: string }): {\n modelName: string,\n resourceName: string,\n} => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n const modelName = (Object.keys(models) as M[]).find((name) => {\n const routeName = models[name];\n const camelCaseModel = ensureCamelCase(routeName);\n\n // eslint-disable-next-line @rushstack/security/no-unsafe-regexp\n return new RegExp(`(${routeName}|${camelCaseModel}$)|(${routeName}|${camelCaseModel}/)`, \"g\").test(realPath);\n });\n\n if (modelName === undefined) {\n throw new Error(`Couldn't find model name for url ${url}`);\n }\n\n return {\n modelName,\n resourceName: models[modelName],\n };\n};\n","import { match } from \"path-to-regexp\";\n\nimport { RouteType } from \"../types.d\";\n\ntype PathMatch = { id: string };\n\nconst getRouteType: (\n method: string,\n url: string,\n resourceName: string,\n) => GetRouteType = (method, url, resourceName) => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n if (!realPath.includes(`/${resourceName}`)) {\n throw new Error(`invalid resource name '${resourceName}' for route '${realPath}'`);\n }\n\n const entityMatcher = match<PathMatch>([`/(.*)/${resourceName}`, `/(.*)/${resourceName}/:id`], { decode: decodeURIComponent });\n const simpleMatcher = match(`/(.*)/${resourceName}`, {\n decode: decodeURIComponent,\n });\n\n switch (method) {\n case \"GET\": {\n const pathMatch = entityMatcher(realPath);\n\n // If we got a /something after the resource name, we are reading 1 entity\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.READ_ONE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: RouteType.READ_ALL,\n };\n }\n case \"POST\": {\n const pathMatch = simpleMatcher(realPath);\n\n if (pathMatch) {\n return {\n routeType: RouteType.CREATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"PUT\":\n case \"PATCH\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.UPDATE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"DELETE\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n routeType: RouteType.DELETE,\n resourceId: pathMatch.params.id,\n };\n }\n\n return {\n routeType: null,\n };\n }\n default: {\n return {\n routeType: null,\n };\n }\n }\n};\n\nexport type GetRouteType = {\n routeType: RouteType | null;\n resourceId?: string;\n};\n\nexport default getRouteType;\n","import createHttpError from \"http-errors\";\n\nimport type { Adapter } from \"../types.d\";\n\nconst adapterMethods = [\"create\", \"delete\", \"getAll\", \"getOne\", \"parseQuery\", \"update\", \"getPaginationData\", \"getModels\"];\n\nconst validateAdapterMethods = <T, Q>(adapter: Adapter<T, Q>): void => {\n adapterMethods.forEach((method) => {\n if (!adapter[method as keyof Adapter<T, Q>]) {\n throw createHttpError(500, `Adapter must implement the \"${method}\" method.`);\n }\n });\n};\n\nexport default validateAdapterMethods;\n","import baseHandler from \"../../../base-crud-handler\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters,\n} from \"../../../types.d\";\n\nasync function handler<T, R extends Request, Context, Q extends ParsedQueryParameters = any, M extends string = string>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>> {\n return baseHandler<R, Context, T, Q, M>(\n async (_, responseConfig) => new Response(JSON.stringify(responseConfig.data), {\n status: responseConfig.status,\n headers: {\n \"content-type\": \"application/json; charset=utf-8\",\n },\n }),\n async () => {},\n adapter,\n options,\n );\n}\n\nexport default handler;\n","import type { NextApiRequest, NextApiResponse } from \"next\";\n\nimport baseHandler from \"../../../base-crud-handler\";\nimport type {\n Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters,\n} from \"../../../types.d\";\n\nasync function handler<\n T,\n Q extends ParsedQueryParameters = any,\n R extends NextApiRequest = NextApiRequest,\n Response extends NextApiResponse = NextApiResponse,\n M extends string = string,\n>(adapter: Adapter<T, Q>, options?: HandlerOptions<M>): Promise<ExecuteHandler<R, Response>> {\n return baseHandler<R, Response, T, Q, M>(\n async (response, responseConfig) => {\n response.status(responseConfig.status).send(responseConfig.data);\n },\n async (response) => {\n response.end();\n },\n adapter,\n options,\n );\n}\n\nexport default handler;\n"]}
1
+ {"version":3,"sources":["../../src/base-crud-handler.ts","../../src/handler/create.ts","../../src/handler/delete.ts","../../src/handler/list.ts","../../src/handler/read.ts","../../src/handler/update.ts","../../src/query-parser.ts","../../src/utils/format-resource-id.ts","../../src/utils/get-resource-name-from-url.ts","../../src/utils/get-route-type.ts","../../src/utils/validate-adapter-methods.ts","../../src/next/api/edge/index.ts","../../src/next/api/node/index.ts"],"names":["createHttpError","ApiError","createHandler","adapter","query","request","resourceName","create_default","deleteHandler","resourceId","delete_default","paginate","listHandler","pagination","isPaginated","paginationOptions","resources","page","total","list_default","readHandler","resource","read_default","updateHandler","update_default","set","parse","parseRecursive","select","selectFields","field","parseWhere","where","whereObject","parsed","key","parseOrderBy","orderBy","orderByObject","parseQuery","queryString","parsedQuery","query_parser_default","formatResourceId","format_resource_id_default","ensureCamelCase","string_","getResourceNameFromUrl","url","models","realPath","modelName","name","routeName","camelCaseModel","match","getRouteType","method","entityMatcher","simpleMatcher","pathMatch","get_route_type_default","adapterMethods","validateAdapterMethods","validate_adapter_methods_default","baseHandler","responseExecutor","finalExecutor","options","error_","error","config","routeNames","modelRoutes","responseOrContext","routeType","modelConfig","get_accessible_routes_default","resourceIdFormatted","parameters","responseConfig","base_crud_handler_default","handler","_","edge_default","response","node_default"],"mappings":"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","sourcesContent":["import type { HttpError } from \"http-errors\";\nimport createHttpError from \"http-errors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { ApiError } from \"next/dist/server/api-utils\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport createHandler from \"./handler/create\";\nimport deleteHandler from \"./handler/delete\";\nimport listHandler from \"./handler/list\";\nimport readHandler from \"./handler/read\";\nimport updateHandler from \"./handler/update\";\nimport parseQuery from \"./query-parser\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, HandlerParameters, ParsedQueryParameters } from \"./types.d\";\nimport { RouteType } from \"./types.d\";\nimport formatResourceId from \"./utils/format-resource-id\";\nimport getAccessibleRoutes from \"./utils/get-accessible-routes\";\nimport { getResourceNameFromUrl } from \"./utils/get-resource-name-from-url\";\nimport getRouteType from \"./utils/get-route-type\";\nimport validateAdapterMethods from \"./utils/validate-adapter-methods\";\n\ninterface ResponseConfig {\n data: any;\n status: number;\n}\n\nasync function baseHandler<R extends Request, Context, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: Context, responseConfig: ResponseConfig) => Promise<Response>,\n finalExecutor: (responseOrContext: Context) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>>;\n\nasync function baseHandler<R extends IncomingMessage, RResponse extends ServerResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<void>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>>;\n\n// eslint-disable-next-line sonarjs/cognitive-complexity,func-style\nasync function baseHandler<R extends { method: string; url: string }, RResponse, T, Q extends ParsedQueryParameters = any, M extends string = string>(\n responseExecutor: (responseOrContext: RResponse, responseConfig: ResponseConfig) => Promise<RResponse>,\n finalExecutor: (responseOrContext: RResponse) => Promise<void>,\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, RResponse>> {\n try {\n validateAdapterMethods(adapter);\n } catch (error_: any) {\n const error = error_ as HttpError;\n\n throw new ApiError(error.statusCode, error.message);\n }\n\n await adapter.init?.();\n\n const config = {\n formatResourceId,\n pagination: {\n perPage: 20,\n },\n ...options,\n };\n\n const routeNames = await adapter.mapModelsToRouteNames?.();\n const modelRoutes: { [key in M]?: string } = {};\n\n adapter.getModels().forEach((modelName) => {\n modelRoutes[modelName as M] = config.models?.[modelName as M]?.name ?? routeNames?.[modelName] ?? modelName;\n });\n\n return async (request, responseOrContext) => {\n const { modelName, resourceName } = getResourceNameFromUrl(request.url, modelRoutes as { [key in M]: string });\n\n if (!resourceName) {\n if (process.env.NODE_ENV === \"development\") {\n const mappedModels = await adapter.mapModelsToRouteNames?.();\n\n if (typeof mappedModels === \"object\") {\n throw createHttpError(404, `Resource not found, possible models: ${Object.values(mappedModels).join(\", \")}`);\n }\n }\n\n throw createHttpError(404, `Resource not found: ${request.url}`);\n }\n\n const { resourceId, routeType } = getRouteType(request.method, request.url, resourceName);\n\n if (routeType === null) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n const modelConfig = options?.models?.[modelName as M];\n\n const accessibleRoutes = getAccessibleRoutes(modelConfig?.only, modelConfig?.exclude, options?.exposeStrategy ?? \"all\");\n\n if (!accessibleRoutes.includes(routeType)) {\n throw createHttpError(404, `Route not found: ${request.url}`);\n }\n\n try {\n const resourceIdFormatted = modelConfig?.formatResourceId?.(resourceId as string) ?? config.formatResourceId(resourceId as string);\n\n await adapter.connect?.();\n\n const parsedQuery = parseQuery(request.url);\n const parameters: HandlerParameters<T, Q> = {\n adapter,\n query: adapter.parseQuery(modelName as M, parsedQuery),\n resourceName: modelName,\n };\n\n try {\n let responseConfig: ResponseConfig;\n\n switch (routeType) {\n case RouteType.READ_ONE: {\n responseConfig = await (config.handlers?.get ?? readHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.READ_ALL: {\n responseConfig = await (config.handlers?.list ?? listHandler)<T, Q>({\n ...parameters,\n pagination: config.pagination,\n query: {\n ...parameters.query,\n limit: parsedQuery.limit ? Number(parsedQuery.limit) : undefined,\n page: parsedQuery.page ? Number(parsedQuery.page) : undefined,\n },\n });\n break;\n }\n case RouteType.CREATE: {\n responseConfig = await (config.handlers?.create ?? createHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Record<string, any> },\n });\n break;\n }\n case RouteType.UPDATE: {\n responseConfig = await (config.handlers?.update ?? updateHandler)<T, Q, R>({\n ...parameters,\n request: request as R & { body: Partial<T> },\n resourceId: resourceIdFormatted,\n });\n break;\n }\n case RouteType.DELETE: {\n responseConfig = await (config.handlers?.delete ?? deleteHandler)<T, Q>({\n ...parameters,\n resourceId: resourceIdFormatted,\n });\n break;\n }\n default: {\n responseConfig = {\n data: \"Method not found\",\n status: 404,\n };\n }\n }\n\n await responseExecutor(responseOrContext, responseConfig);\n } catch (error: any) {\n if (adapter.handleError && !(error instanceof ApiError)) {\n adapter.handleError(error);\n } else {\n throw error;\n }\n }\n } finally {\n await adapter.disconnect?.();\n\n await finalExecutor(responseOrContext);\n }\n };\n}\n\nexport default baseHandler;\n","import type { HandlerParameters } from \"../types.d\";\n\nconst createHandler: Handler = async ({ adapter, query, request, resourceName }) => {\n const resources = await adapter.create(resourceName, request.body, query);\n\n return {\n data: resources,\n status: 201,\n };\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: HandlerParameters<T, Q> & { request: Request & { body: Record<string, any> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default createHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst deleteHandler: Handler = async ({ adapter, query, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const deletedResource = await adapter.delete(resourceName, resourceId, query);\n\n return {\n data: deletedResource,\n status: 200,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\nexport default deleteHandler;\n","import { paginate } from \"@visulima/pagination\";\n\nimport type { HandlerParameters, PaginationConfig, ParsedQueryParameters } from \"../types.d\";\n\ninterface PaginationOptions {\n page: number;\n perPage: number;\n}\n\nconst listHandler: Handler = async ({ adapter, pagination, query, resourceName }) => {\n let isPaginated = false;\n let paginationOptions: PaginationOptions | undefined;\n\n if (query.page !== undefined) {\n if (query.page <= 0) {\n throw new Error(\"page query must be a strictly positive number\");\n }\n\n paginationOptions = {\n page: query.page,\n perPage: query.limit ?? pagination.perPage,\n };\n }\n\n if (paginationOptions) {\n isPaginated = true;\n\n // eslint-disable-next-line no-param-reassign\n query.skip = (paginationOptions.page - 1) * paginationOptions.perPage;\n // eslint-disable-next-line no-param-reassign\n query.limit = paginationOptions.perPage;\n }\n\n const resources = await adapter.getAll(resourceName, query);\n\n if (isPaginated) {\n const { page, total } = await adapter.getPaginationData(resourceName, query);\n\n const paginator = paginate(page, (paginationOptions as PaginationOptions).perPage, total, resources);\n\n return {\n data: paginator.toJSON(),\n status: 200,\n };\n }\n\n return {\n data: resources,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q extends ParsedQueryParameters>(\n parameters: HandlerParameters<T, Q> & { pagination: PaginationConfig },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default listHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst readHandler: Handler = async ({ adapter, query, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource !== \"object\") {\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n }\n\n return {\n data: resource,\n status: 200,\n };\n};\n\nexport type Handler = <T, Q>(\n parameters: UniqueResourceHandlerParameters<T, Q>,\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default readHandler;\n","import createHttpError from \"http-errors\";\n\nimport type { UniqueResourceHandlerParameters } from \"../types.d\";\n\nconst updateHandler: Handler = async ({ adapter, query, request, resourceId, resourceName }) => {\n const resource = await adapter.getOne(resourceName, resourceId, query);\n\n if (typeof resource === \"object\") {\n const updatedResource = await adapter.update(resourceName, resourceId, request.body, query);\n\n return {\n data: updatedResource,\n status: 201,\n };\n }\n\n throw createHttpError(404, `${resourceName} ${resourceId} not found`);\n};\n\nexport type Handler = <T, Q, Request>(\n parameters: UniqueResourceHandlerParameters<T, Q> & { request: Request & { body: Partial<T> } },\n) => Promise<{\n data: any;\n status: number;\n}>;\n\nexport default updateHandler;\n","// eslint-disable-next-line no-restricted-imports\nimport set from \"lodash.set\";\nimport { parse } from \"node:url\";\n\nimport type { OrderByField, ParsedQueryParameters, RecursiveField, WhereField } from \"./types.d\";\n\nconst parseRecursive = (select: string): RecursiveField => {\n const selectFields: RecursiveField = {};\n\n const fields = select.split(\",\");\n\n fields.forEach((field) => {\n set(selectFields, field, true);\n });\n\n return selectFields;\n};\n\nconst parseWhere = (where: string): WhereField => {\n const whereObject = JSON.parse(where);\n const parsed: WhereField = {};\n\n Object.keys(whereObject).forEach((key) => {\n set(parsed, key, whereObject[key]);\n });\n\n return parsed;\n};\n\nconst parseOrderBy = (orderBy: string): OrderByField => {\n const parsed: OrderByField = {};\n const orderByObject = JSON.parse(orderBy);\n\n if (Object.keys(orderByObject).length > 0) {\n const key = Object.keys(orderByObject)[0] as string;\n\n if (orderByObject[key as keyof typeof orderByObject] === \"$asc\" || orderByObject[key as keyof typeof orderByObject] === \"$desc\") {\n parsed[key] = orderByObject[key as keyof typeof orderByObject];\n }\n }\n\n if (Object.keys(parsed).length !== 1) {\n throw new Error(\"orderBy needs to be an object with exactly 1 property with either $asc or $desc value\");\n }\n\n return parsed;\n};\n\n// eslint-disable-next-line sonarjs/cognitive-complexity\nconst parseQuery = (queryString?: string): ParsedQueryParameters => {\n if (queryString) {\n const { query } = parse(queryString, true);\n const parsedQuery: ParsedQueryParameters = {};\n\n if (query[\"select\"]) {\n parsedQuery.select = parseRecursive(query[\"select\"] as string);\n }\n\n if (query[\"include\"]) {\n parsedQuery.include = parseRecursive(query[\"include\"] as string);\n }\n\n if (query[\"where\"]) {\n parsedQuery.where = parseWhere(query[\"where\"] as string);\n }\n\n if (query[\"orderBy\"]) {\n parsedQuery.orderBy = parseOrderBy(query[\"orderBy\"] as string);\n }\n\n if (query[\"limit\"] !== undefined) {\n parsedQuery.limit = Number.isFinite(+query[\"limit\"]) ? +query[\"limit\"] : undefined;\n }\n if (query[\"skip\"] !== undefined) {\n parsedQuery.skip = Number.isFinite(+query[\"skip\"]) ? +query[\"skip\"] : undefined;\n }\n\n if (query[\"distinct\"]) {\n parsedQuery.distinct = query[\"distinct\"] as string;\n }\n\n if (query[\"page\"]) {\n parsedQuery.page = Number.isFinite(+query[\"page\"]) ? +query[\"page\"] : undefined;\n }\n\n return {\n originalQuery: query,\n ...parsedQuery,\n };\n }\n\n return {};\n};\n\nexport default parseQuery;\n","const formatResourceId = (resourceId: string): number | string => (Number.isSafeInteger(+resourceId) ? +resourceId : resourceId);\n\nexport default formatResourceId;\n","export const ensureCamelCase = (string_: string): string => `${string_.charAt(0).toLowerCase()}${string_.slice(1)}`;\n\nexport const getResourceNameFromUrl = <M extends string = string>(\n url: string,\n models: { [key in M]: string },\n): {\n modelName: string;\n resourceName: string;\n} => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n const modelName = (Object.keys(models) as M[]).find((name) => {\n const routeName = models[name];\n const camelCaseModel = ensureCamelCase(routeName);\n\n // eslint-disable-next-line @rushstack/security/no-unsafe-regexp,security/detect-non-literal-regexp\n return new RegExp(`(${routeName}|${camelCaseModel}$)|(${routeName}|${camelCaseModel}/)`, \"g\").test(realPath);\n });\n\n if (modelName === undefined) {\n throw new Error(`Couldn't find model name for url ${url}`);\n }\n\n return {\n modelName,\n resourceName: models[modelName],\n };\n};\n","import { match } from \"path-to-regexp\";\n\nimport { RouteType } from \"../types.d\";\n\ninterface PathMatch {\n id: string;\n}\n\nconst getRouteType: (method: string, url: string, resourceName: string) => GetRouteType = (method, url, resourceName) => {\n // Exclude the query params from the path\n const realPath = url.split(\"?\")[0];\n\n if (realPath === undefined) {\n throw new TypeError(\"Path is undefined\");\n }\n\n if (!realPath.includes(`/${resourceName}`)) {\n throw new Error(`invalid resource name '${resourceName}' for route '${realPath}'`);\n }\n\n const entityMatcher = match<PathMatch>([`/(.*)/${resourceName}`, `/(.*)/${resourceName}/:id`], { decode: decodeURIComponent });\n const simpleMatcher = match(`/(.*)/${resourceName}`, {\n decode: decodeURIComponent,\n });\n\n switch (method) {\n case \"GET\": {\n const pathMatch = entityMatcher(realPath);\n\n // If we got a /something after the resource name, we are reading 1 entity\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.READ_ONE,\n };\n }\n\n return {\n routeType: RouteType.READ_ALL,\n };\n }\n case \"POST\": {\n const pathMatch = simpleMatcher(realPath);\n\n if (pathMatch) {\n return {\n routeType: RouteType.CREATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"PUT\":\n case \"PATCH\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.UPDATE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n case \"DELETE\": {\n const pathMatch = entityMatcher(realPath);\n\n if (typeof pathMatch === \"object\" && pathMatch.params.id) {\n return {\n resourceId: pathMatch.params.id,\n routeType: RouteType.DELETE,\n };\n }\n\n return {\n routeType: null,\n };\n }\n default: {\n return {\n routeType: null,\n };\n }\n }\n};\n\nexport interface GetRouteType {\n resourceId?: string;\n routeType: RouteType | null;\n}\n\nexport default getRouteType;\n","import createHttpError from \"http-errors\";\n\nimport type { Adapter } from \"../types.d\";\n\nconst adapterMethods = [\"create\", \"delete\", \"getAll\", \"getOne\", \"parseQuery\", \"update\", \"getPaginationData\", \"getModels\"];\n\nconst validateAdapterMethods = <T, Q>(adapter: Adapter<T, Q>): void => {\n adapterMethods.forEach((method) => {\n if (!adapter[method as keyof Adapter<T, Q>]) {\n throw createHttpError(500, `Adapter must implement the \"${method}\" method.`);\n }\n });\n};\n\nexport default validateAdapterMethods;\n","import baseHandler from \"../../../base-crud-handler\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters } from \"../../../types.d\";\n\nconst handler = async <T, R extends Request, Context, Q extends ParsedQueryParameters = any, M extends string = string>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Context>> =>\n await baseHandler<R, Context, T, Q, M>(\n async (_, responseConfig) =>\n new Response(JSON.stringify(responseConfig.data), {\n headers: {\n \"content-type\": \"application/json; charset=utf-8\",\n },\n status: responseConfig.status,\n }),\n async () => {},\n adapter,\n options,\n );\n\nexport default handler;\n","import type { NextApiRequest, NextApiResponse } from \"next\";\n\nimport baseHandler from \"../../../base-crud-handler\";\nimport type { Adapter, ExecuteHandler, HandlerOptions, ParsedQueryParameters } from \"../../../types.d\";\n\nconst handler = async <\n T,\n Q extends ParsedQueryParameters = any,\n R extends NextApiRequest = NextApiRequest,\n Response extends NextApiResponse = NextApiResponse,\n M extends string = string,\n>(\n adapter: Adapter<T, Q>,\n options?: HandlerOptions<M>,\n): Promise<ExecuteHandler<R, Response>> =>\n await baseHandler<R, Response, T, Q, M>(\n async (response, responseConfig) => {\n response.status(responseConfig.status).send(responseConfig.data);\n },\n async (response) => {\n response.end();\n },\n adapter,\n options,\n );\n\nexport default handler;\n"]}
@@ -33,42 +33,43 @@ type Handler = <T, Q, Request>(parameters: UniqueResourceHandlerParameters<T, Q>
33
33
  status: number;
34
34
  }>;
35
35
 
36
+ // eslint-disable-next-line no-shadow
36
37
  declare enum RouteType {
37
38
  CREATE = "CREATE",
39
+ DELETE = "DELETE",
38
40
  READ_ALL = "READ_ALL",
39
41
  READ_ONE = "READ_ONE",
40
42
  UPDATE = "UPDATE",
41
- DELETE = "DELETE",
42
43
  }
43
44
 
44
- type ModelOption = {
45
- name?: string
46
- only?: RouteType[]
47
- exclude?: RouteType[]
48
- formatResourceId?: (resourceId: string) => number | string
49
- };
45
+ interface ModelOption {
46
+ exclude?: RouteType[];
47
+ formatResourceId?: (resourceId: string) => number | string;
48
+ name?: string;
49
+ only?: RouteType[];
50
+ }
50
51
 
51
52
  type ModelsOptions<M extends string = string> = {
52
- [key in M]?: ModelOption
53
+ [key in M]?: ModelOption;
53
54
  };
54
55
 
55
- type HandlerOptions<M extends string = string> = {
56
- formatResourceId?: (resourceId: string) => number | string;
57
- models?: ModelsOptions<M>;
56
+ interface HandlerOptions<M extends string = string> {
58
57
  exposeStrategy?: "all" | "none";
59
- pagination?: PaginationConfig,
58
+ formatResourceId?: (resourceId: string) => number | string;
60
59
  handlers?: {
61
60
  create?: Handler$4;
62
61
  delete?: Handler$3;
63
62
  get?: Handler$1;
64
63
  list?: Handler$2;
65
64
  update?: Handler;
66
- },
67
- };
65
+ };
66
+ models?: ModelsOptions<M>;
67
+ pagination?: PaginationConfig;
68
+ }
68
69
 
69
- type PaginationConfig = {
70
- perPage: number
71
- };
70
+ interface PaginationConfig {
71
+ perPage: number;
72
+ }
72
73
 
73
74
  interface HandlerParameters<T, Q> {
74
75
  adapter: Adapter<T, Q>;
@@ -79,84 +80,82 @@ interface HandlerParameters<T, Q> {
79
80
  interface UniqueResourceHandlerParameters<T, Q> {
80
81
  adapter: Adapter<T, Q>;
81
82
  query: Q;
82
- resourceName: string;
83
83
  resourceId: number | string;
84
+ resourceName: string;
84
85
  }
85
86
 
86
87
  interface Adapter<T, Q, M extends string = string> {
87
- models?: M[];
88
- init?: () => Promise<void>;
89
- parseQuery: (resourceName: M, query: ParsedQueryParameters) => Q;
90
- getAll: (resourceName: M, query: Q) => Promise<T[]>;
91
- getOne: (resourceName: M, resourceId: number | string, query: Q) => Promise<T>;
88
+ connect?: () => Promise<void>;
92
89
  create: (resourceName: M, data: any, query: Q) => Promise<T>;
93
- update: (resourceName: M, resourceId: number | string, data: any, query: Q) => Promise<T>;
94
90
  delete: (resourceName: M, resourceId: number | string, query: Q) => Promise<T>;
95
- getPaginationData: (resourceName: M, query: Q) => Promise<PaginationData>;
96
- getModels: () => M[];
97
- connect?: () => Promise<void>;
98
91
  disconnect?: () => Promise<void>;
92
+ getAll: (resourceName: M, query: Q) => Promise<T[]>;
93
+ getModels: () => M[];
94
+ getOne: (resourceName: M, resourceId: number | string, query: Q) => Promise<T>;
95
+ getPaginationData: (resourceName: M, query: Q) => Promise<PaginationData>;
99
96
  handleError?: (error: Error) => void;
97
+ init?: () => Promise<void>;
100
98
  mapModelsToRouteNames?: () => Promise<{ [key in M]?: string }>;
99
+ models?: M[];
100
+ parseQuery: (resourceName: M, query: ParsedQueryParameters) => Q;
101
+ update: (resourceName: M, resourceId: number | string, data: any, query: Q) => Promise<T>;
101
102
  }
102
103
 
103
- type PaginationData = {
104
- total: number
105
- pageCount: number
106
- page: number
107
- };
104
+ interface PaginationData {
105
+ page: number;
106
+ pageCount: number;
107
+ total: number;
108
+ }
108
109
 
109
- type RecursiveField = {
110
- [key: string]: TRecursiveField | boolean;
111
- };
110
+ type RecursiveField = Record<string, Record<string, boolean> | boolean>;
112
111
 
113
112
  type WhereOperator = "$cont" | "$ends" | "$eq" | "$gt" | "$gte" | "$in" | "$isnull" | "$lt" | "$lte" | "$neq" | "$notin" | "$starts";
114
113
 
115
114
  type SearchCondition = Date | boolean | number | string | null;
116
115
 
117
116
  type WhereCondition = {
118
- [key in TWhereOperator]?: TSearchCondition;
117
+ [key in WhereOperator]?: SearchCondition;
119
118
  };
120
119
 
121
- type Condition = {
122
- [key: string]: TCondition | TSearchCondition | TWhereCondition;
123
- };
120
+ // TODO: find the correct way to type this
121
+ // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
122
+ type Condition = Record<string, Condition | SearchCondition | WhereCondition>;
124
123
 
124
+ // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
125
125
  type WhereField = Condition & {
126
- $and?: TCondition | TCondition[];
127
- $or?: TCondition | TCondition[];
128
- $not?: TCondition | TCondition[];
126
+ // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
127
+ $and?: Condition | Condition[];
128
+ // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
129
+ $not?: Condition | Condition[];
130
+ // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents
131
+ $or?: Condition | Condition[];
129
132
  };
130
133
 
131
134
  type OrderByOperator = "$asc" | "$desc";
132
135
 
133
- type OrderByField = {
134
- [key: string]: TOrderByOperator;
135
- };
136
+ type OrderByField = Record<string, TOrderByOperator>;
136
137
 
137
138
  interface ParsedQueryParameters {
138
- select?: RecursiveField;
139
+ distinct?: string;
139
140
  include?: RecursiveField;
140
- where?: WhereField;
141
- orderBy?: OrderByField;
142
141
  limit?: number;
143
- skip?: number;
144
- distinct?: string;
142
+ orderBy?: OrderByField;
143
+ originalQuery?: Record<string, any>;
145
144
  page?: number;
146
- originalQuery?: {
147
- [key: string]: any;
148
- };
145
+ select?: RecursiveField;
146
+ skip?: number;
147
+ where?: WhereField;
149
148
  }
150
149
 
151
150
  type ExecuteHandler<Request, Response> = (request: Request, response: Response) => Promise<void>;
152
151
 
153
- type FakePrismaClient = {
154
- _dmmf?: any;
155
- _getDmmf?: () => any;
152
+ interface FakePrismaClient {
156
153
  $connect: () => void;
157
154
  $disconnect: () => Promise<void>;
155
+ [key: string]: any;
156
+ _dmmf?: any;
158
157
 
159
- [key: string]: any
160
- };
158
+ _getDmmf?: () => any;
159
+ }
161
160
 
162
- export { Adapter as A, Condition as C, ExecuteHandler as E, FakePrismaClient as F, HandlerParameters as H, ModelsOptions as M, OrderByField as O, PaginationData as P, RouteType as R, SearchCondition as S, UniqueResourceHandlerParameters as U, WhereField as W, ParsedQueryParameters as a, PaginationConfig as b, HandlerOptions as c, ModelOption as d, OrderByOperator as e, RecursiveField as f, WhereOperator as g, WhereCondition as h };
161
+ export { Adapter as A, Condition as C, ExecuteHandler as E, FakePrismaClient as F, HandlerOptions as H, ModelsOptions as M, OrderByField as O, PaginationData as P, RouteType as R, SearchCondition as S, UniqueResourceHandlerParameters as U, WhereCondition as W, ParsedQueryParameters as a, HandlerParameters as b, ModelOption as c, OrderByOperator as d, PaginationConfig as e, RecursiveField as f, WhereField as g, WhereOperator as h };
package/next/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@visulima/crud-next",
3
- "private": true,
4
3
  "version": "0.0.0",
4
+ "private": true,
5
5
  "description": "visulima crud next",
6
6
  "license": "MIT",
7
7
  "exports": {