From 0dcbaa9edccc46e42eb9483d02a8521e1e897aec Mon Sep 17 00:00:00 2001 From: adamcikado Date: Sat, 27 Jan 2024 10:29:40 +0100 Subject: [PATCH] fix: compare DateTime in newUpIfMissing --- src/orm/base_model/index.ts | 30 +++++++++++++----- src/utils/index.ts | 19 +++++++++++- test/orm/base_model.spec.ts | 62 +++++++++++++++++++++++++++++++++++++ 3 files changed, 102 insertions(+), 9 deletions(-) diff --git a/src/orm/base_model/index.ts b/src/orm/base_model/index.ts index 1efbce2d..6f681610 100644 --- a/src/orm/base_model/index.ts +++ b/src/orm/base_model/index.ts @@ -63,6 +63,7 @@ import { ensureRelation, managedTransaction, normalizeCherryPickObject, + transformDateValue, } from '../../utils/index.js' const MANY_RELATIONS = ['hasMany', 'manyToMany', 'hasManyThrough'] @@ -207,9 +208,18 @@ class BaseModelImpl implements LucidRow { * array */ return rowObjects.map((rowObject: any) => { - const existingRow = existingRows.find((one: any) => { - /* eslint-disable-next-line eqeqeq */ - return keys.every((key) => one[key] == rowObject[key]) + const existingRow = existingRows.find((row: any) => { + return keys.every((key) => { + const objectValue = rowObject[key] + const rowValue = row[key] + + if (DateTime.isDateTime(rowValue) && DateTime.isDateTime(objectValue)) { + return rowValue.equals(objectValue) + } + + /* eslint-disable-next-line eqeqeq */ + return rowValue == objectValue + }) }) /** @@ -852,6 +862,8 @@ class BaseModelImpl implements LucidRow { payload: any, options?: ModelAssignOptions ): Promise { + const client = this.$adapter.modelConstructorClient(this as LucidModel, options) + uniqueKeys = Array.isArray(uniqueKeys) ? uniqueKeys : [uniqueKeys] const uniquenessPair: { key: string; value: string[] }[] = uniqueKeys.map( (uniqueKey: string) => { @@ -861,7 +873,7 @@ class BaseModelImpl implements LucidRow { throw new Exception( `Value for the "${uniqueKey}" is null or undefined inside "fetchOrNewUpMany" payload` ) - }), + }).map((value) => transformDateValue(value, client.dialect)), } } ) @@ -896,6 +908,8 @@ class BaseModelImpl implements LucidRow { payload: any, options?: ModelAssignOptions ): Promise { + const client = this.$adapter.modelConstructorClient(this as LucidModel, options) + uniqueKeys = Array.isArray(uniqueKeys) ? uniqueKeys : [uniqueKeys] const uniquenessPair: { key: string; value: string[] }[] = uniqueKeys.map( (uniqueKey: string) => { @@ -905,7 +919,7 @@ class BaseModelImpl implements LucidRow { throw new Exception( `Value for the "${uniqueKey}" is null or undefined inside "fetchOrCreateMany" payload` ) - }), + }).map((value) => transformDateValue(value, client.dialect)), } } ) @@ -960,6 +974,8 @@ class BaseModelImpl implements LucidRow { payload: any, options?: ModelAssignOptions ): Promise { + const client = this.$adapter.modelConstructorClient(this as LucidModel, options) + uniqueKeys = Array.isArray(uniqueKeys) ? uniqueKeys : [uniqueKeys] const uniquenessPair: { key: string; value: string[] }[] = uniqueKeys.map( (uniqueKey: string) => { @@ -969,13 +985,11 @@ class BaseModelImpl implements LucidRow { throw new Exception( `Value for the "${uniqueKey}" is null or undefined inside "updateOrCreateMany" payload` ) - }), + }).map((value) => transformDateValue(value, client.dialect)), } } ) - const client = this.$adapter.modelConstructorClient(this as LucidModel, options) - return managedTransaction(client, async (trx) => { /** * Find existing rows diff --git a/src/utils/index.ts b/src/utils/index.ts index 2af3ab10..afab52f0 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -12,9 +12,15 @@ import { join, extname } from 'node:path' import { Exception, fsReadAll, isScriptFile } from '@poppinss/utils' import { RelationshipsContract } from '../types/relations.js' import { LucidRow, ModelObject, CherryPickFields } from '../types/model.js' -import { FileNode, QueryClientContract, TransactionClientContract } from '../types/database.js' +import { + DialectContract, + FileNode, + QueryClientContract, + TransactionClientContract, +} from '../types/database.js' import { fileURLToPath, pathToFileURL } from 'node:url' import * as errors from '../errors.js' +import { DateTime } from 'luxon' /** * Ensure that relation is defined @@ -53,6 +59,17 @@ export function collectValues(payload: any[], key: string, missingCallback: () = }) } +/** + * Transform value if it is an instance of DateTime, so it can be processed by query builder + */ +export function transformDateValue(value: any, dialect: DialectContract) { + if (DateTime.isDateTime(value)) { + return value.toFormat(dialect.dateTimeFormat) + } + + return value +} + /** * Raises exception when a relationship `booted` property is false. */ diff --git a/test/orm/base_model.spec.ts b/test/orm/base_model.spec.ts index 205c4b1b..3ceaf54c 100644 --- a/test/orm/base_model.spec.ts +++ b/test/orm/base_model.spec.ts @@ -4888,6 +4888,68 @@ test.group('Base Model | fetch', (group) => { assert.lengthOf(usersList, 1) assert.equal(usersList[0].points, 2) }) + + test('updateOrCreateMany should work with DateTime', async ({ fs, assert }) => { + const app = new AppFactory().create(fs.baseUrl, () => {}) + await app.init() + const db = getDb() + const adapter = ormAdapter(db) + + const BaseModel = getBaseModel(adapter) + + class User extends BaseModel { + @column({ isPrimary: true }) + declare id: number + + @column() + declare username: string + + @column() + declare email: string + + @column.dateTime() + declare createdAt: DateTime + } + + const createdAt1 = DateTime.now().minus({ days: 2 }).startOf('second') + const createdAt2 = DateTime.now().minus({ days: 1 }).startOf('second') + + await User.createMany([ + { + username: 'virk1', + email: 'virk+1@adonisjs.com', + createdAt: createdAt1, + }, + { + username: 'virk2', + email: 'virk+2@adonisjs.com', + createdAt: createdAt2, + }, + ]) + + const users = await User.updateOrCreateMany('createdAt', [ + { + username: 'virk3', + email: 'virk+3@adonisjs.com', + createdAt: createdAt1, + }, + { + username: 'nikk', + email: 'nikk@adonisjs.com', + createdAt: DateTime.now(), + }, + ]) + + assert.lengthOf(users, 2) + assert.isTrue(users[0].$isPersisted) + assert.isFalse(users[0].$isLocal) + + assert.isTrue(users[1].$isPersisted) + assert.isTrue(users[1].$isLocal) + + const usersList = await db.query().from('users') + assert.lengthOf(usersList, 3) + }) }) test.group('Base Model | hooks', (group) => {