Files
payload/src/collections/operations/find.ts
Alessio Gravili 9467074fb9 chore: update 2.0 branch from master (#3207)
Co-authored-by: Jacob Fletcher <jacobsfletch@gmail.com>
Co-authored-by: Alessio Gravili <alessio@gravili.de>
Co-authored-by: PatrikKozak <patrik@trbl.design>
Co-authored-by: Lucas Blancas <lablancas@gmail.com>
Co-authored-by: Stef Gootzen <37367280+stefgootzen@users.noreply.github.com>
Co-authored-by: Jarrod Flesch <30633324+JarrodMFlesch@users.noreply.github.com>
Co-authored-by: Jessica Chowdhury <67977755+JessChowdhury@users.noreply.github.com>
Co-authored-by: PatrikKozak <35232443+PatrikKozak@users.noreply.github.com>
Co-authored-by: Greg Willard <Wickett06@gmail.com>
Co-authored-by: James Mikrut <james@payloadcms.com>
Co-authored-by: Dan Ribbens <dan.ribbens@gmail.com>
Co-authored-by: Elliot DeNolf <denolfe@gmail.com>
fix: WhereBuilder component does not accept all valid Where queries (#3087)
fix: passes in height to resizeOptions upload option to allow height resize (#3171)
2023-08-22 16:04:50 -04:00

262 lines
6.6 KiB
TypeScript

import { Where } from '../../types';
import { PayloadRequest } from '../../express/types';
import executeAccess from '../../auth/executeAccess';
import { Collection, TypeWithID } from '../config/types';
import type { PaginatedDocs } from '../../database/types';
import { AccessResult } from '../../config/types';
import { afterRead } from '../../fields/hooks/afterRead';
import { validateQueryPaths } from '../../database/queryValidation/validateQueryPaths';
import { appendVersionToQueryKey } from '../../versions/drafts/appendVersionToQueryKey';
import { buildVersionCollectionFields } from '../../versions/buildCollectionFields';
import { combineQueries } from '../../database/combineQueries';
import { initTransaction } from '../../utilities/initTransaction';
import { killTransaction } from '../../utilities/killTransaction';
import { buildAfterOperation } from './utils';
export type Arguments = {
collection: Collection
where?: Where
page?: number
limit?: number
sort?: string
depth?: number
currentDepth?: number
req?: PayloadRequest
overrideAccess?: boolean
disableErrors?: boolean
pagination?: boolean
showHiddenFields?: boolean
draft?: boolean
}
async function find<T extends TypeWithID & Record<string, unknown>>(
incomingArgs: Arguments,
): Promise<PaginatedDocs<T>> {
let args = incomingArgs;
// /////////////////////////////////////
// beforeOperation - Collection
// /////////////////////////////////////
await args.collection.config.hooks.beforeOperation.reduce(async (priorHook, hook) => {
await priorHook;
args = (await hook({
args,
operation: 'read',
context: args.req.context,
})) || args;
}, Promise.resolve());
const {
where,
page,
limit,
depth,
currentDepth,
draft: draftsEnabled,
collection,
collection: {
config: collectionConfig,
},
sort,
req,
req: {
locale,
payload,
},
overrideAccess,
disableErrors,
showHiddenFields,
pagination = true,
} = args;
try {
const shouldCommit = await initTransaction(req);
// /////////////////////////////////////
// beforeOperation - Collection
// /////////////////////////////////////
await args.collection.config.hooks.beforeOperation.reduce(async (priorHook, hook) => {
await priorHook;
args = (await hook({
args,
operation: 'read',
context: req.context,
})) || args;
}, Promise.resolve());
// /////////////////////////////////////
// Access
// /////////////////////////////////////
let accessResult: AccessResult;
if (!overrideAccess) {
accessResult = await executeAccess({ req, disableErrors }, collectionConfig.access.read);
// If errors are disabled, and access returns false, return empty results
if (accessResult === false) {
return {
docs: [],
totalDocs: 0,
totalPages: 1,
page: 1,
pagingCounter: 1,
hasPrevPage: false,
hasNextPage: false,
prevPage: null,
nextPage: null,
limit,
};
}
}
// /////////////////////////////////////
// Find
// /////////////////////////////////////
const usePagination = pagination && limit !== 0;
const sanitizedLimit = limit ?? (usePagination ? 10 : 0);
const sanitizedPage = page || 1;
let result: PaginatedDocs<T>;
let fullWhere = combineQueries(where, accessResult);
if (collectionConfig.versions?.drafts && draftsEnabled) {
fullWhere = appendVersionToQueryKey(fullWhere);
await validateQueryPaths({
collectionConfig: collection.config,
where: fullWhere,
req,
overrideAccess,
versionFields: buildVersionCollectionFields(collection.config),
});
result = await payload.db.queryDrafts<T>({
collection: collectionConfig.slug,
where: fullWhere,
page: sanitizedPage,
limit: sanitizedLimit,
sort,
pagination: usePagination,
locale,
req,
});
} else {
await validateQueryPaths({
collectionConfig,
where,
req,
overrideAccess,
});
result = await payload.db.find<T>({
collection: collectionConfig.slug,
where: fullWhere,
page: sanitizedPage,
limit: sanitizedLimit,
sort,
locale,
pagination,
req,
});
}
// /////////////////////////////////////
// beforeRead - Collection
// /////////////////////////////////////
result = {
...result,
docs: await Promise.all(result.docs.map(async (doc) => {
let docRef = doc;
await collectionConfig.hooks.beforeRead.reduce(async (priorHook, hook) => {
await priorHook;
docRef = await hook({
req,
query: fullWhere,
doc: docRef,
context: req.context,
}) || docRef;
}, Promise.resolve());
return docRef;
})),
};
// /////////////////////////////////////
// afterRead - Fields
// /////////////////////////////////////
result = {
...result,
docs: await Promise.all(result.docs.map(async (doc) => afterRead<T>({
depth,
currentDepth,
doc,
entityConfig: collectionConfig,
overrideAccess,
req,
showHiddenFields,
findMany: true,
context: req.context,
}))),
};
// /////////////////////////////////////
// afterRead - Collection
// /////////////////////////////////////
result = {
...result,
docs: await Promise.all(result.docs.map(async (doc) => {
let docRef = doc;
await collectionConfig.hooks.afterRead.reduce(async (priorHook, hook) => {
await priorHook;
docRef = await hook({
req,
query: fullWhere,
doc: docRef,
findMany: true,
context: req.context,
}) || doc;
}, Promise.resolve());
return docRef;
})),
};
// /////////////////////////////////////
// afterOperation - Collection
// /////////////////////////////////////
result = await buildAfterOperation<T>({
operation: 'find',
args,
result,
});
// /////////////////////////////////////
// Return results
// /////////////////////////////////////
if (shouldCommit) await payload.db.commitTransaction(req.transactionID);
return result;
} catch (error: unknown) {
await killTransaction(req);
throw error;
}
}
export default find;