177 lines
5.8 KiB
TypeScript
177 lines
5.8 KiB
TypeScript
/* eslint-disable no-await-in-loop */
|
|
import type { SQL } from 'drizzle-orm'
|
|
import type { Field, Operator, Where } from 'payload/types'
|
|
|
|
import { and, ilike, isNotNull, sql } from 'drizzle-orm'
|
|
// import createJSONQuery from 'mongo-query-to-postgres-jsonb'
|
|
import { QueryError } from 'payload/errors'
|
|
import { validOperators } from 'payload/types'
|
|
|
|
import type { GenericColumn, PostgresAdapter } from '../types'
|
|
import type { BuildQueryJoinAliases, BuildQueryJoins } from './buildQuery'
|
|
|
|
import { buildAndOrConditions } from './buildAndOrConditions'
|
|
import { convertPathToJSONQuery } from './convertPathToJSONQuery'
|
|
import { createJSONQuery } from './createJSONQuery'
|
|
// import convertJSONQuery from './convertJSONQuery'
|
|
import { getTableColumnFromPath } from './getTableColumnFromPath'
|
|
import { operatorMap } from './operatorMap'
|
|
import { sanitizeQueryValue } from './sanitizeQueryValue'
|
|
|
|
type Args = {
|
|
adapter: PostgresAdapter
|
|
fields: Field[]
|
|
joinAliases: BuildQueryJoinAliases
|
|
joins: BuildQueryJoins
|
|
locale: string
|
|
selectFields: Record<string, GenericColumn>
|
|
tableName: string
|
|
where: Where
|
|
}
|
|
|
|
export async function parseParams({
|
|
adapter,
|
|
fields,
|
|
joinAliases,
|
|
joins,
|
|
locale,
|
|
selectFields,
|
|
tableName,
|
|
where,
|
|
}: Args): Promise<SQL> {
|
|
let result: SQL
|
|
const constraints: SQL[] = []
|
|
|
|
if (typeof where === 'object' && Object.keys(where).length > 0) {
|
|
// We need to determine if the whereKey is an AND, OR, or a schema path
|
|
for (const relationOrPath of Object.keys(where)) {
|
|
if (relationOrPath) {
|
|
const condition = where[relationOrPath]
|
|
let conditionOperator: 'and' | 'or'
|
|
if (relationOrPath.toLowerCase() === 'and') {
|
|
conditionOperator = 'and'
|
|
} else if (relationOrPath.toLowerCase() === 'or') {
|
|
conditionOperator = 'or'
|
|
}
|
|
if (Array.isArray(condition)) {
|
|
const builtConditions = await buildAndOrConditions({
|
|
adapter,
|
|
fields,
|
|
joinAliases,
|
|
joins,
|
|
locale,
|
|
selectFields,
|
|
tableName,
|
|
where: condition,
|
|
})
|
|
if (builtConditions.length > 0) {
|
|
if (result) {
|
|
result = operatorMap[conditionOperator](result, ...builtConditions)
|
|
} else {
|
|
result = operatorMap[conditionOperator](...builtConditions)
|
|
}
|
|
}
|
|
} else {
|
|
// It's a path - and there can be multiple comparisons on a single path.
|
|
// For example - title like 'test' and title not equal to 'tester'
|
|
// So we need to loop on keys again here to handle each operator independently
|
|
const pathOperators = where[relationOrPath]
|
|
if (typeof pathOperators === 'object') {
|
|
for (const operator of Object.keys(pathOperators)) {
|
|
if (validOperators.includes(operator as Operator)) {
|
|
const {
|
|
columnName,
|
|
constraints: queryConstraints,
|
|
field,
|
|
getNotNullColumnByValue,
|
|
rawColumn,
|
|
table,
|
|
} = getTableColumnFromPath({
|
|
adapter,
|
|
collectionPath: relationOrPath,
|
|
fields,
|
|
joinAliases,
|
|
joins,
|
|
locale,
|
|
pathSegments: relationOrPath.replace(/__/g, '.').split('.'),
|
|
selectFields,
|
|
tableName,
|
|
})
|
|
|
|
const val = where[relationOrPath][operator]
|
|
|
|
queryConstraints.forEach(({ columnName: col, table: constraintTable, value }) => {
|
|
constraints.push(operatorMap.equals(constraintTable[col], value))
|
|
})
|
|
|
|
if (['json', 'richText'].includes(field.type)) {
|
|
const pathSegments = relationOrPath.split('.').slice(1)
|
|
pathSegments.unshift(table[columnName].name)
|
|
|
|
if (field.type === 'richText') {
|
|
const jsonQuery = createJSONQuery({
|
|
operator,
|
|
pathSegments,
|
|
treatAsArray: ['children'],
|
|
treatRootAsArray: true,
|
|
value: val,
|
|
})
|
|
|
|
// constraints.push(sql.raw(jsonQuery))
|
|
}
|
|
|
|
if (field.type === 'json') {
|
|
const jsonQuery = convertPathToJSONQuery(jsonPath)
|
|
constraints.push(sql.raw(`${table[columnName].name}${jsonQuery} = '%${val}%'`))
|
|
}
|
|
|
|
break
|
|
}
|
|
|
|
if (getNotNullColumnByValue) {
|
|
const columnName = getNotNullColumnByValue(val)
|
|
if (columnName) {
|
|
constraints.push(isNotNull(table[columnName]))
|
|
} else {
|
|
throw new QueryError([{ path: relationOrPath }])
|
|
}
|
|
break
|
|
}
|
|
|
|
if (operator === 'like') {
|
|
constraints.push(
|
|
and(...val.split(' ').map((word) => ilike(table[columnName], `%${word}%`))),
|
|
)
|
|
break
|
|
}
|
|
|
|
const { operator: queryOperator, value: queryValue } = sanitizeQueryValue({
|
|
field,
|
|
operator,
|
|
val,
|
|
})
|
|
|
|
constraints.push(
|
|
operatorMap[queryOperator](rawColumn || table[columnName], queryValue),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (constraints.length > 0) {
|
|
if (result) {
|
|
result = and(result, ...constraints)
|
|
} else {
|
|
result = and(...constraints)
|
|
}
|
|
}
|
|
if (constraints.length === 1 && !result) {
|
|
;[result] = constraints
|
|
}
|
|
|
|
return result
|
|
}
|