Files
payload/packages/drizzle/src/updateMany.ts
Alessio Gravili 9c88af4b20 refactor(drizzle): replace query chaining with dynamic query building (#11923)
This replaces usage of our `chainMethods` helper to dynamically chain
queries with [drizzle dynamic query
building](https://orm.drizzle.team/docs/dynamic-query-building).

This is more type-safe, more readable and requires less code
2025-03-31 20:37:45 +00:00

110 lines
2.7 KiB
TypeScript

import type { LibSQLDatabase } from 'drizzle-orm/libsql'
import type { UpdateMany } from 'payload'
import toSnakeCase from 'to-snake-case'
import type { DrizzleAdapter } from './types.js'
import buildQuery from './queries/buildQuery.js'
import { selectDistinct } from './queries/selectDistinct.js'
import { upsertRow } from './upsertRow/index.js'
import { getTransaction } from './utilities/getTransaction.js'
export const updateMany: UpdateMany = async function updateMany(
this: DrizzleAdapter,
{
collection: collectionSlug,
data,
joins: joinQuery,
limit,
locale,
req,
returning,
select,
sort: sortArg,
where: whereToUse,
},
) {
const db = await getTransaction(this, req)
const collection = this.payload.collections[collectionSlug].config
const tableName = this.tableNameMap.get(toSnakeCase(collection.slug))
const sort = sortArg !== undefined && sortArg !== null ? sortArg : collection.defaultSort
const { joins, orderBy, selectFields, where } = buildQuery({
adapter: this,
fields: collection.flattenedFields,
locale,
sort,
tableName,
where: whereToUse,
})
let idsToUpdate: (number | string)[] = []
const selectDistinctResult = await selectDistinct({
adapter: this,
db,
joins,
query: ({ query }) =>
orderBy ? query.orderBy(() => orderBy.map(({ column, order }) => order(column))) : query,
selectFields,
tableName,
where,
})
if (selectDistinctResult?.[0]?.id) {
idsToUpdate = selectDistinctResult?.map((doc) => doc.id)
} else if (whereToUse && !joins.length) {
// If id wasn't passed but `where` without any joins, retrieve it with findFirst
const _db = db as LibSQLDatabase
const table = this.tables[tableName]
let query = _db.select({ id: table.id }).from(table).where(where).$dynamic()
if (typeof limit === 'number' && limit > 0) {
query = query.limit(limit)
}
if (orderBy) {
query = query.orderBy(() => orderBy.map(({ column, order }) => order(column)))
}
const docsToUpdate = await query
idsToUpdate = docsToUpdate?.map((doc) => doc.id)
}
if (!idsToUpdate.length) {
return []
}
const results = []
// TODO: We need to batch this to reduce the amount of db calls. This can get very slow if we are updating a lot of rows.
for (const idToUpdate of idsToUpdate) {
const result = await upsertRow({
id: idToUpdate,
adapter: this,
data,
db,
fields: collection.flattenedFields,
ignoreResult: returning === false,
joinQuery,
operation: 'update',
req,
select,
tableName,
})
results.push(result)
}
if (returning === false) {
return null
}
return results
}