velocious 1.0.156 → 1.0.158

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.
Files changed (53) hide show
  1. package/README.md +10 -1
  2. package/build/src/configuration-types.d.ts +7 -0
  3. package/build/src/configuration-types.d.ts.map +1 -1
  4. package/build/src/configuration-types.js +3 -1
  5. package/build/src/configuration.d.ts +6 -1
  6. package/build/src/configuration.d.ts.map +1 -1
  7. package/build/src/configuration.js +12 -2
  8. package/build/src/database/drivers/base-column.d.ts +8 -0
  9. package/build/src/database/drivers/base-column.d.ts.map +1 -1
  10. package/build/src/database/drivers/base-column.js +20 -1
  11. package/build/src/database/drivers/base.d.ts +4 -0
  12. package/build/src/database/drivers/base.d.ts.map +1 -1
  13. package/build/src/database/drivers/base.js +8 -1
  14. package/build/src/database/drivers/mssql/index.d.ts.map +1 -1
  15. package/build/src/database/drivers/mssql/index.js +13 -11
  16. package/build/src/database/drivers/mysql/column.d.ts +1 -0
  17. package/build/src/database/drivers/mysql/column.d.ts.map +1 -1
  18. package/build/src/database/drivers/mysql/column.js +12 -6
  19. package/build/src/database/drivers/pgsql/column.d.ts +1 -0
  20. package/build/src/database/drivers/pgsql/column.d.ts.map +1 -1
  21. package/build/src/database/drivers/pgsql/column.js +7 -1
  22. package/build/src/database/drivers/pgsql/index.d.ts.map +1 -1
  23. package/build/src/database/drivers/pgsql/index.js +11 -1
  24. package/build/src/database/drivers/pgsql/table.d.ts.map +1 -1
  25. package/build/src/database/drivers/pgsql/table.js +3 -2
  26. package/build/src/database/drivers/sqlite/sql/alter-table.d.ts.map +1 -1
  27. package/build/src/database/drivers/sqlite/sql/alter-table.js +9 -2
  28. package/build/src/database/query/alter-table-base.d.ts.map +1 -1
  29. package/build/src/database/query/alter-table-base.js +13 -3
  30. package/build/src/database/query/create-table-base.d.ts.map +1 -1
  31. package/build/src/database/query/create-table-base.js +9 -1
  32. package/build/src/database/query/model-class-query.d.ts +5 -0
  33. package/build/src/database/query/model-class-query.d.ts.map +1 -1
  34. package/build/src/database/query/model-class-query.js +122 -1
  35. package/build/src/database/query/where-hash.d.ts.map +1 -1
  36. package/build/src/database/query/where-hash.js +7 -2
  37. package/build/src/database/query/where-model-class-hash.d.ts +71 -0
  38. package/build/src/database/query/where-model-class-hash.d.ts.map +1 -0
  39. package/build/src/database/query/where-model-class-hash.js +195 -0
  40. package/build/src/database/record/index.d.ts +10 -0
  41. package/build/src/database/record/index.d.ts.map +1 -1
  42. package/build/src/database/record/index.js +52 -10
  43. package/build/src/database/table-data/table-column.d.ts +23 -0
  44. package/build/src/database/table-data/table-column.d.ts.map +1 -1
  45. package/build/src/database/table-data/table-column.js +32 -2
  46. package/build/src/testing/test.d.ts +5 -0
  47. package/build/src/testing/test.d.ts.map +1 -1
  48. package/build/src/testing/test.js +32 -6
  49. package/build/src/utils/timezone-context.d.ts +20 -0
  50. package/build/src/utils/timezone-context.d.ts.map +1 -0
  51. package/build/src/utils/timezone-context.js +53 -0
  52. package/build/tsconfig.tsbuildinfo +1 -1
  53. package/package.json +1 -1
@@ -1,10 +1,12 @@
1
1
  // @ts-check
2
2
  import { incorporate } from "incorporator";
3
3
  import * as inflection from "inflection";
4
+ import { isPlainObject } from "is-plain-object";
4
5
  import { Logger } from "../../logger.js";
5
6
  import Preloader from "./preloader.js";
6
7
  import DatabaseQuery from "./index.js";
7
8
  import RecordNotFoundError from "../record/record-not-found-error.js";
9
+ import WhereModelClassHash from "./where-model-class-hash.js";
8
10
  /**
9
11
  * @template {typeof import("../record/index.js").default} MC
10
12
  */
@@ -240,5 +242,124 @@ export default class VelociousDatabaseQueryModelClassQuery extends DatabaseQuery
240
242
  }
241
243
  return rows.map((row) => columnNames.map((columnName) => row[columnName]));
242
244
  }
245
+ /**
246
+ * @param {import("./index.js").WhereArgumentType} where - Where.
247
+ * @returns {this} This query instance
248
+ */
249
+ where(where) {
250
+ if (typeof where == "string") {
251
+ return super.where(where);
252
+ }
253
+ if (isPlainObject(where)) {
254
+ const { resolvedHash, fallbackHash } = splitWhereHash({ hash: where, modelClass: this.getModelClass() });
255
+ const joinObject = buildJoinObjectFromWhereHash({ hash: where, modelClass: this.getModelClass() });
256
+ if (Object.keys(joinObject).length > 0) {
257
+ this.joins(joinObject);
258
+ }
259
+ if (Object.keys(resolvedHash).length > 0) {
260
+ const qualifyBaseTable = Object.keys(joinObject).length > 0;
261
+ this._wheres.push(new WhereModelClassHash({
262
+ hash: resolvedHash,
263
+ modelClass: this.getModelClass(),
264
+ qualifyBaseTable,
265
+ query: this
266
+ }));
267
+ }
268
+ if (Object.keys(fallbackHash).length > 0) {
269
+ super.where(fallbackHash);
270
+ }
271
+ return this;
272
+ }
273
+ throw new Error(`Invalid type of where: ${typeof where} (${where.constructor.name})`);
274
+ }
275
+ }
276
+ /**
277
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
278
+ * @param {string} relationshipName - Relationship name.
279
+ * @returns {import("../record/relationships/base.js").default | undefined} - The relationship.
280
+ */
281
+ function getRelationshipByName(modelClass, relationshipName) {
282
+ return modelClass.getRelationshipsMap()[relationshipName];
283
+ }
284
+ /**
285
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
286
+ * @param {string} key - Attribute or column name.
287
+ * @returns {string | undefined} - The resolved column name.
288
+ */
289
+ function resolveColumnName(modelClass, key) {
290
+ const attributeMap = modelClass.getAttributeNameToColumnNameMap();
291
+ const columnName = attributeMap[key];
292
+ if (columnName)
293
+ return columnName;
294
+ return undefined;
295
+ }
296
+ /**
297
+ * @param {object} args - Options.
298
+ * @param {Record<string, any>} args.hash - Where hash.
299
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
300
+ * @returns {{resolvedHash: Record<string, any>, fallbackHash: Record<string, any>}} - Split hashes.
301
+ */
302
+ function splitWhereHash({ hash, modelClass }) {
303
+ /** @type {Record<string, any>} */
304
+ const resolvedHash = {};
305
+ /** @type {Record<string, any>} */
306
+ const fallbackHash = {};
307
+ for (const key in hash) {
308
+ const value = hash[key];
309
+ const isNested = isPlainObject(value);
310
+ if (isNested) {
311
+ const relationship = getRelationshipByName(modelClass, key);
312
+ if (relationship) {
313
+ const targetModelClass = relationship.getTargetModelClass();
314
+ const nestedResult = splitWhereHash({ hash: value, modelClass: targetModelClass });
315
+ const nestedResolvedKeys = Object.keys(nestedResult.resolvedHash);
316
+ const nestedFallbackKeys = Object.keys(nestedResult.fallbackHash);
317
+ if (nestedResolvedKeys.length > 0) {
318
+ resolvedHash[key] = nestedResult.resolvedHash;
319
+ }
320
+ if (nestedFallbackKeys.length > 0) {
321
+ const tableName = targetModelClass.tableName();
322
+ if (!fallbackHash[tableName])
323
+ fallbackHash[tableName] = {};
324
+ Object.assign(fallbackHash[tableName], nestedResult.fallbackHash);
325
+ }
326
+ }
327
+ else {
328
+ fallbackHash[key] = value;
329
+ }
330
+ }
331
+ else {
332
+ const columnName = resolveColumnName(modelClass, key);
333
+ if (columnName) {
334
+ resolvedHash[key] = value;
335
+ }
336
+ else {
337
+ fallbackHash[key] = value;
338
+ }
339
+ }
340
+ }
341
+ return { resolvedHash, fallbackHash };
342
+ }
343
+ /**
344
+ * @param {object} args - Options.
345
+ * @param {Record<string, any>} args.hash - Where hash.
346
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
347
+ * @returns {Record<string, any>} - Join object.
348
+ */
349
+ function buildJoinObjectFromWhereHash({ hash, modelClass }) {
350
+ /** @type {Record<string, any>} */
351
+ const joinObject = {};
352
+ for (const key in hash) {
353
+ const value = hash[key];
354
+ if (!isPlainObject(value))
355
+ continue;
356
+ const relationship = getRelationshipByName(modelClass, key);
357
+ if (!relationship)
358
+ continue;
359
+ const targetModelClass = relationship.getTargetModelClass();
360
+ const nestedJoinObject = buildJoinObjectFromWhereHash({ hash: value, modelClass: targetModelClass });
361
+ joinObject[key] = Object.keys(nestedJoinObject).length > 0 ? nestedJoinObject : true;
362
+ }
363
+ return joinObject;
243
364
  }
244
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"model-class-query.js","sourceRoot":"","sources":["../../../../src/database/query/model-class-query.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {incorporate} from \"incorporator\"\nimport * as inflection from \"inflection\"\nimport {Logger} from \"../../logger.js\"\nimport Preloader from \"./preloader.js\"\nimport DatabaseQuery from \"./index.js\"\nimport RecordNotFoundError from \"../record/record-not-found-error.js\"\n\n/**\n * @template {typeof import(\"../record/index.js\").default} MC\n */\n/**\n * @template {typeof import(\"../record/index.js\").default} MC\n * @typedef {import(\"./index.js\").QueryArgsType & {modelClass: MC}} ModelClassQueryArgsType\n */\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../record/index.js\").default} MC\n */\nexport default class VelociousDatabaseQueryModelClassQuery extends DatabaseQuery {\n  /** @param {ModelClassQueryArgsType<MC>} args - Query constructor arguments. */\n  constructor(args) {\n    const {modelClass} = args\n\n    if (!modelClass) throw new Error(`No modelClass given in ${Object.keys(args).join(\", \")}`)\n\n    super(args)\n    this.logger = new Logger(this)\n\n    /** @type {MC} */\n    this.modelClass = modelClass\n  }\n\n  /** @returns {this} - The clone.  */\n  clone() {\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (new VelociousDatabaseQueryModelClassQuery({\n      driver: this._driverFn,\n      froms: [...this._froms],\n      handler: this.handler.clone(),\n      groups: [...this._groups],\n      joins: [...this._joins],\n      limit: this._limit,\n      modelClass: this.modelClass,\n      offset: this._offset,\n      orders: [...this._orders],\n      page: this._page,\n      perPage: this._perPage,\n      preload: {...this._preload},\n      distinct: this._distinct,\n      selects: [...this._selects],\n      wheres: [...this._wheres]\n    }))\n\n    // @ts-expect-error\n    return newQuery\n  }\n\n  /** @returns {Promise<number>} - Resolves with the count.  */\n  async count() {\n    // Generate count SQL\n    const primaryKey = `${this.driver.quoteTable(this.getModelClass().tableName())}.${this.driver.quoteColumn(this.getModelClass().primaryKey())}`\n    const distinctPrefix = this._distinct ? \"DISTINCT \" : \"\"\n    let sql = `COUNT(${distinctPrefix}${primaryKey})`\n\n    if (this.driver.getType() == \"pgsql\") sql += \"::int\"\n\n    sql += \" AS count\"\n\n\n    // Clone query and execute count\n    const countQuery = this.clone()\n\n    countQuery._distinct = false\n    countQuery._selects = []\n    countQuery.select(sql)\n\n    const results = /** @type {{count: number}[]} */ (await countQuery._executeQuery())\n\n    // The query isn't grouped and a single result has been given\n    if (results.length == 1) {\n      return results[0].count\n    }\n\n    // The query may be grouped and a lot of different counts a given\n    let countResult = 0\n\n    for (const result of results) {\n      if (!(\"count\" in result)) {\n        throw new Error(\"Invalid count result\")\n      }\n\n      countResult += result.count\n    }\n\n    return countResult\n  }\n\n  /** @returns {MC} - The model class.  */\n  getModelClass() {\n    if (!this.modelClass) throw new Error(\"modelClass not set\")\n\n    return this.modelClass\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async destroyAll() {\n    const records = await this.toArray()\n\n    for (const record of records) {\n      await record.destroy()\n    }\n  }\n\n  /**\n   * @param {number|string} recordId - Record id.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the find.\n   */\n  async find(recordId) {\n    /** @type {{[key: string]: number | string}} */\n    const conditions = {}\n\n    conditions[this.getModelClass().primaryKey()] = recordId\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(conditions)\n\n    const record = (await newQuery.first())\n\n    if (!record) {\n      throw new RecordNotFoundError(`Couldn't find ${this.getModelClass().name} with '${this.getModelClass().primaryKey()}'=${recordId}`)\n    }\n\n    return record\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @returns {Promise<InstanceType<MC> | null>} - Resolves with the by.\n   */\n  async findBy(conditions) {\n    /** @type {{[key: string]: number | string}} */\n    const newConditions = {}\n\n    for (const key in conditions) {\n      const keyUnderscore = inflection.underscore(key)\n\n      newConditions[keyUnderscore] = conditions[key]\n    }\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(newConditions)\n\n    return await newQuery.first()\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @param {function(InstanceType<MC>) : void} [callback] - Callback function.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the or create by.\n   */\n  async findOrCreateBy(conditions, callback) {\n    const record = await this.findOrInitializeBy(conditions, callback)\n\n    if (record.isNewRecord()) {\n      await record.save()\n    }\n\n    return record\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the by or fail.\n   */\n  async findByOrFail(conditions) {\n    /** @type {{[key: string]: number | string}} */\n    const newConditions = {}\n\n    for (const key in conditions) {\n      const keyUnderscore = inflection.underscore(key)\n\n      newConditions[keyUnderscore] = conditions[key]\n    }\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(newConditions)\n\n    const model = await newQuery.first()\n\n    if (!model) {\n      throw new Error(\"Record not found\")\n    }\n\n    return model\n  }\n\n  /**\n   * @param {object} conditions - Conditions.\n   * @param {function(InstanceType<MC>) : void} [callback] - Callback function.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the or initialize by.\n   */\n  async findOrInitializeBy(conditions, callback) {\n    const record = await this.findBy(conditions)\n\n    if (record) return record\n\n    const ModelClass = this.getModelClass()\n    const newRecord = /** @type {InstanceType<MC>} */ (new ModelClass(conditions))\n\n    if (callback) {\n      callback(newRecord)\n    }\n\n    return newRecord\n  }\n\n  /** @returns {Promise<InstanceType<MC> | undefined>} - Resolves with the first.  */\n  async first() {\n    const newQuery = this.clone().limit(1).reorder(`${this.driver.quoteTable(this.getModelClass().tableName())}.${this.driver.quoteColumn(this.getModelClass().orderableColumn())}`)\n    const results = await newQuery.toArray()\n\n    return results[0]\n  }\n\n  /** @returns {Promise<InstanceType<MC> | undefined>} - Resolves with the last.  */\n  async last() {\n    const primaryKey = this.getModelClass().primaryKey()\n    const tableName = this.getModelClass().tableName()\n    const results = await this.clone().reorder(`${this.driver.quoteTable(tableName)}.${this.driver.quoteColumn(primaryKey)} DESC`).limit(1).toArray()\n\n    return results[0]\n  }\n\n  /**\n   * @param {import(\"./index.js\").NestedPreloadRecord} data - Data payload.\n   * @returns {this} - The preload.\n   */\n  preload(data) {\n    incorporate(this._preload, data)\n    return this\n  }\n\n  /**\n   * Converts query results to array of model instances\n   * @returns {Promise<Array<InstanceType<MC>>>} - Resolves with the array.\n   */\n  async toArray() {\n    const models = []\n    const results = await this.results()\n\n    for (const result of results) {\n      const ModelClass = this.getModelClass()\n      const model = /** @type {InstanceType<MC>} */ (new ModelClass())\n\n      model.loadExistingRecord(result)\n      models.push(model)\n    }\n\n    if (Object.keys(this._preload).length > 0 && models.length > 0) {\n      const preloader = new Preloader({\n        modelClass: this.modelClass,\n        models,\n        preload: this._preload\n      })\n\n      await preloader.run()\n    }\n\n    return models\n  }\n\n  /**\n   * Plucks one or more columns directly from the database without instantiating models.\n   * @param {...string|string[]} columns - Column names.\n   * @returns {Promise<any[]>} - Resolves with the pluck.\n   */\n  async pluck(...columns) {\n    const flatColumns = columns.flat()\n\n    if (flatColumns.length === 0) throw new Error(\"No columns given to pluck\")\n\n    const modelClass = this.getModelClass()\n    const tableName = modelClass.tableName()\n    const attributeMap = modelClass.getAttributeNameToColumnNameMap()\n    const columnNames = flatColumns.map((column) => attributeMap[column] || column)\n\n    const query = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    query._preload = {}\n    query._selects = []\n\n    columnNames.forEach((columnName) => {\n      const selectSql = `${this.driver.quoteTable(tableName)}.${this.driver.quoteColumn(columnName)}`\n\n      query.select(selectSql)\n    })\n\n    const rows = await query._executeQuery()\n\n    if (columnNames.length === 1) {\n      const [columnName] = columnNames\n      return rows.map((row) => row[columnName])\n    }\n\n    return rows.map((row) => columnNames.map((columnName) => row[columnName]))\n  }\n}\n\n"]}
365
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"model-class-query.js","sourceRoot":"","sources":["../../../../src/database/query/model-class-query.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {incorporate} from \"incorporator\"\nimport * as inflection from \"inflection\"\nimport {isPlainObject} from \"is-plain-object\"\nimport {Logger} from \"../../logger.js\"\nimport Preloader from \"./preloader.js\"\nimport DatabaseQuery from \"./index.js\"\nimport RecordNotFoundError from \"../record/record-not-found-error.js\"\nimport WhereModelClassHash from \"./where-model-class-hash.js\"\n\n/**\n * @template {typeof import(\"../record/index.js\").default} MC\n */\n/**\n * @template {typeof import(\"../record/index.js\").default} MC\n * @typedef {import(\"./index.js\").QueryArgsType & {modelClass: MC}} ModelClassQueryArgsType\n */\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../record/index.js\").default} MC\n */\nexport default class VelociousDatabaseQueryModelClassQuery extends DatabaseQuery {\n  /** @param {ModelClassQueryArgsType<MC>} args - Query constructor arguments. */\n  constructor(args) {\n    const {modelClass} = args\n\n    if (!modelClass) throw new Error(`No modelClass given in ${Object.keys(args).join(\", \")}`)\n\n    super(args)\n    this.logger = new Logger(this)\n\n    /** @type {MC} */\n    this.modelClass = modelClass\n  }\n\n  /** @returns {this} - The clone.  */\n  clone() {\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (new VelociousDatabaseQueryModelClassQuery({\n      driver: this._driverFn,\n      froms: [...this._froms],\n      handler: this.handler.clone(),\n      groups: [...this._groups],\n      joins: [...this._joins],\n      limit: this._limit,\n      modelClass: this.modelClass,\n      offset: this._offset,\n      orders: [...this._orders],\n      page: this._page,\n      perPage: this._perPage,\n      preload: {...this._preload},\n      distinct: this._distinct,\n      selects: [...this._selects],\n      wheres: [...this._wheres]\n    }))\n\n    // @ts-expect-error\n    return newQuery\n  }\n\n  /** @returns {Promise<number>} - Resolves with the count.  */\n  async count() {\n    // Generate count SQL\n    const primaryKey = `${this.driver.quoteTable(this.getModelClass().tableName())}.${this.driver.quoteColumn(this.getModelClass().primaryKey())}`\n    const distinctPrefix = this._distinct ? \"DISTINCT \" : \"\"\n    let sql = `COUNT(${distinctPrefix}${primaryKey})`\n\n    if (this.driver.getType() == \"pgsql\") sql += \"::int\"\n\n    sql += \" AS count\"\n\n\n    // Clone query and execute count\n    const countQuery = this.clone()\n\n    countQuery._distinct = false\n    countQuery._selects = []\n    countQuery.select(sql)\n\n    const results = /** @type {{count: number}[]} */ (await countQuery._executeQuery())\n\n    // The query isn't grouped and a single result has been given\n    if (results.length == 1) {\n      return results[0].count\n    }\n\n    // The query may be grouped and a lot of different counts a given\n    let countResult = 0\n\n    for (const result of results) {\n      if (!(\"count\" in result)) {\n        throw new Error(\"Invalid count result\")\n      }\n\n      countResult += result.count\n    }\n\n    return countResult\n  }\n\n  /** @returns {MC} - The model class.  */\n  getModelClass() {\n    if (!this.modelClass) throw new Error(\"modelClass not set\")\n\n    return this.modelClass\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async destroyAll() {\n    const records = await this.toArray()\n\n    for (const record of records) {\n      await record.destroy()\n    }\n  }\n\n  /**\n   * @param {number|string} recordId - Record id.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the find.\n   */\n  async find(recordId) {\n    /** @type {{[key: string]: number | string}} */\n    const conditions = {}\n\n    conditions[this.getModelClass().primaryKey()] = recordId\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(conditions)\n\n    const record = (await newQuery.first())\n\n    if (!record) {\n      throw new RecordNotFoundError(`Couldn't find ${this.getModelClass().name} with '${this.getModelClass().primaryKey()}'=${recordId}`)\n    }\n\n    return record\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @returns {Promise<InstanceType<MC> | null>} - Resolves with the by.\n   */\n  async findBy(conditions) {\n    /** @type {{[key: string]: number | string}} */\n    const newConditions = {}\n\n    for (const key in conditions) {\n      const keyUnderscore = inflection.underscore(key)\n\n      newConditions[keyUnderscore] = conditions[key]\n    }\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(newConditions)\n\n    return await newQuery.first()\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @param {function(InstanceType<MC>) : void} [callback] - Callback function.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the or create by.\n   */\n  async findOrCreateBy(conditions, callback) {\n    const record = await this.findOrInitializeBy(conditions, callback)\n\n    if (record.isNewRecord()) {\n      await record.save()\n    }\n\n    return record\n  }\n\n  /**\n   * @param {{[key: string]: string | number}} conditions - Conditions hash keyed by attribute name.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the by or fail.\n   */\n  async findByOrFail(conditions) {\n    /** @type {{[key: string]: number | string}} */\n    const newConditions = {}\n\n    for (const key in conditions) {\n      const keyUnderscore = inflection.underscore(key)\n\n      newConditions[keyUnderscore] = conditions[key]\n    }\n\n    const newQuery = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    newQuery.where(newConditions)\n\n    const model = await newQuery.first()\n\n    if (!model) {\n      throw new Error(\"Record not found\")\n    }\n\n    return model\n  }\n\n  /**\n   * @param {object} conditions - Conditions.\n   * @param {function(InstanceType<MC>) : void} [callback] - Callback function.\n   * @returns {Promise<InstanceType<MC>>} - Resolves with the or initialize by.\n   */\n  async findOrInitializeBy(conditions, callback) {\n    const record = await this.findBy(conditions)\n\n    if (record) return record\n\n    const ModelClass = this.getModelClass()\n    const newRecord = /** @type {InstanceType<MC>} */ (new ModelClass(conditions))\n\n    if (callback) {\n      callback(newRecord)\n    }\n\n    return newRecord\n  }\n\n  /** @returns {Promise<InstanceType<MC> | undefined>} - Resolves with the first.  */\n  async first() {\n    const newQuery = this.clone().limit(1).reorder(`${this.driver.quoteTable(this.getModelClass().tableName())}.${this.driver.quoteColumn(this.getModelClass().orderableColumn())}`)\n    const results = await newQuery.toArray()\n\n    return results[0]\n  }\n\n  /** @returns {Promise<InstanceType<MC> | undefined>} - Resolves with the last.  */\n  async last() {\n    const primaryKey = this.getModelClass().primaryKey()\n    const tableName = this.getModelClass().tableName()\n    const results = await this.clone().reorder(`${this.driver.quoteTable(tableName)}.${this.driver.quoteColumn(primaryKey)} DESC`).limit(1).toArray()\n\n    return results[0]\n  }\n\n  /**\n   * @param {import(\"./index.js\").NestedPreloadRecord} data - Data payload.\n   * @returns {this} - The preload.\n   */\n  preload(data) {\n    incorporate(this._preload, data)\n    return this\n  }\n\n  /**\n   * Converts query results to array of model instances\n   * @returns {Promise<Array<InstanceType<MC>>>} - Resolves with the array.\n   */\n  async toArray() {\n    const models = []\n    const results = await this.results()\n\n    for (const result of results) {\n      const ModelClass = this.getModelClass()\n      const model = /** @type {InstanceType<MC>} */ (new ModelClass())\n\n      model.loadExistingRecord(result)\n      models.push(model)\n    }\n\n    if (Object.keys(this._preload).length > 0 && models.length > 0) {\n      const preloader = new Preloader({\n        modelClass: this.modelClass,\n        models,\n        preload: this._preload\n      })\n\n      await preloader.run()\n    }\n\n    return models\n  }\n\n  /**\n   * Plucks one or more columns directly from the database without instantiating models.\n   * @param {...string|string[]} columns - Column names.\n   * @returns {Promise<any[]>} - Resolves with the pluck.\n   */\n  async pluck(...columns) {\n    const flatColumns = columns.flat()\n\n    if (flatColumns.length === 0) throw new Error(\"No columns given to pluck\")\n\n    const modelClass = this.getModelClass()\n    const tableName = modelClass.tableName()\n    const attributeMap = modelClass.getAttributeNameToColumnNameMap()\n    const columnNames = flatColumns.map((column) => attributeMap[column] || column)\n\n    const query = /** @type {VelociousDatabaseQueryModelClassQuery<MC>} */ (this.clone())\n\n    query._preload = {}\n    query._selects = []\n\n    columnNames.forEach((columnName) => {\n      const selectSql = `${this.driver.quoteTable(tableName)}.${this.driver.quoteColumn(columnName)}`\n\n      query.select(selectSql)\n    })\n\n    const rows = await query._executeQuery()\n\n    if (columnNames.length === 1) {\n      const [columnName] = columnNames\n      return rows.map((row) => row[columnName])\n    }\n\n    return rows.map((row) => columnNames.map((columnName) => row[columnName]))\n  }\n\n  /**\n   * @param {import(\"./index.js\").WhereArgumentType} where - Where.\n   * @returns {this} This query instance\n   */\n  where(where) {\n    if (typeof where == \"string\") {\n      return super.where(where)\n    }\n\n    if (isPlainObject(where)) {\n      const {resolvedHash, fallbackHash} = splitWhereHash({hash: where, modelClass: this.getModelClass()})\n      const joinObject = buildJoinObjectFromWhereHash({hash: where, modelClass: this.getModelClass()})\n\n      if (Object.keys(joinObject).length > 0) {\n        this.joins(joinObject)\n      }\n\n      if (Object.keys(resolvedHash).length > 0) {\n        const qualifyBaseTable = Object.keys(joinObject).length > 0\n        this._wheres.push(new WhereModelClassHash({\n          hash: resolvedHash,\n          modelClass: this.getModelClass(),\n          qualifyBaseTable,\n          query: this\n        }))\n      }\n\n      if (Object.keys(fallbackHash).length > 0) {\n        super.where(fallbackHash)\n      }\n\n      return this\n    }\n\n    throw new Error(`Invalid type of where: ${typeof where} (${where.constructor.name})`)\n  }\n}\n\n/**\n * @param {typeof import(\"../record/index.js\").default} modelClass - Model class.\n * @param {string} relationshipName - Relationship name.\n * @returns {import(\"../record/relationships/base.js\").default | undefined} - The relationship.\n */\nfunction getRelationshipByName(modelClass, relationshipName) {\n  return modelClass.getRelationshipsMap()[relationshipName]\n}\n\n/**\n * @param {typeof import(\"../record/index.js\").default} modelClass - Model class.\n * @param {string} key - Attribute or column name.\n * @returns {string | undefined} - The resolved column name.\n */\nfunction resolveColumnName(modelClass, key) {\n  const attributeMap = modelClass.getAttributeNameToColumnNameMap()\n  const columnName = attributeMap[key]\n\n  if (columnName) return columnName\n\n  return undefined\n}\n\n/**\n * @param {object} args - Options.\n * @param {Record<string, any>} args.hash - Where hash.\n * @param {typeof import(\"../record/index.js\").default} args.modelClass - Model class.\n * @returns {{resolvedHash: Record<string, any>, fallbackHash: Record<string, any>}} - Split hashes.\n */\nfunction splitWhereHash({hash, modelClass}) {\n  /** @type {Record<string, any>} */\n  const resolvedHash = {}\n  /** @type {Record<string, any>} */\n  const fallbackHash = {}\n\n  for (const key in hash) {\n    const value = hash[key]\n    const isNested = isPlainObject(value)\n\n    if (isNested) {\n      const relationship = getRelationshipByName(modelClass, key)\n\n      if (relationship) {\n        const targetModelClass = relationship.getTargetModelClass()\n        const nestedResult = splitWhereHash({hash: value, modelClass: targetModelClass})\n        const nestedResolvedKeys = Object.keys(nestedResult.resolvedHash)\n        const nestedFallbackKeys = Object.keys(nestedResult.fallbackHash)\n\n        if (nestedResolvedKeys.length > 0) {\n          resolvedHash[key] = nestedResult.resolvedHash\n        }\n\n        if (nestedFallbackKeys.length > 0) {\n          const tableName = targetModelClass.tableName()\n\n          if (!fallbackHash[tableName]) fallbackHash[tableName] = {}\n          Object.assign(fallbackHash[tableName], nestedResult.fallbackHash)\n        }\n      } else {\n        fallbackHash[key] = value\n      }\n    } else {\n      const columnName = resolveColumnName(modelClass, key)\n\n      if (columnName) {\n        resolvedHash[key] = value\n      } else {\n        fallbackHash[key] = value\n      }\n    }\n  }\n\n  return {resolvedHash, fallbackHash}\n}\n\n/**\n * @param {object} args - Options.\n * @param {Record<string, any>} args.hash - Where hash.\n * @param {typeof import(\"../record/index.js\").default} args.modelClass - Model class.\n * @returns {Record<string, any>} - Join object.\n */\nfunction buildJoinObjectFromWhereHash({hash, modelClass}) {\n  /** @type {Record<string, any>} */\n  const joinObject = {}\n\n  for (const key in hash) {\n    const value = hash[key]\n\n    if (!isPlainObject(value)) continue\n\n    const relationship = getRelationshipByName(modelClass, key)\n\n    if (!relationship) continue\n\n    const targetModelClass = relationship.getTargetModelClass()\n    const nestedJoinObject = buildJoinObjectFromWhereHash({hash: value, modelClass: targetModelClass})\n\n    joinObject[key] = Object.keys(nestedJoinObject).length > 0 ? nestedJoinObject : true\n  }\n\n  return joinObject\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"where-hash.d.ts","sourceRoot":"","sources":["../../../../src/database/query/where-hash.js"],"names":[],"mappings":"AAIA;;GAEG;AAEH;IACE;;;OAGG;IACH,mBAHW,OAAO,YAAY,EAAE,OAAO,QAC5B,SAAS,EAMnB;IAFC,gBAAgB;IAgBlB;;;;;OAKG;IACH,wBALW,SAAS,cACT,MAAM,UACN,MAAM,GACJ,MAAM,CAiClB;CACF;wBAhEY;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,GAAG,KAAK,CAAC,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,CAAC,GAAG,SAAS,CAAA;CAAC;sBAH9F,iBAAiB"}
1
+ {"version":3,"file":"where-hash.d.ts","sourceRoot":"","sources":["../../../../src/database/query/where-hash.js"],"names":[],"mappings":"AAIA;;GAEG;AAEH;IACE;;;OAGG;IACH,mBAHW,OAAO,YAAY,EAAE,OAAO,QAC5B,SAAS,EAMnB;IAFC,gBAAgB;IAgBlB;;;;;OAKG;IACH,wBALW,SAAS,cACT,MAAM,UACN,MAAM,GACJ,MAAM,CAoClB;CACF;wBAnEY;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,GAAG,KAAK,CAAC,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,CAAC,GAAG,SAAS,CAAA;CAAC;sBAH9F,iBAAiB"}
@@ -33,7 +33,12 @@ export default class VelociousDatabaseQueryWhereHash extends WhereBase {
33
33
  let sql = "";
34
34
  for (const whereKey in hash) {
35
35
  const whereValue = hash[whereKey];
36
- if (!Array.isArray(whereValue) && whereValue !== null && typeof whereValue == "object") {
36
+ if (Array.isArray(whereValue) && whereValue.length === 0) {
37
+ if (index > 0)
38
+ sql += " AND ";
39
+ sql += "1=0";
40
+ }
41
+ else if (!Array.isArray(whereValue) && whereValue !== null && typeof whereValue == "object") {
37
42
  sql += this._whereSQLFromHash(whereValue, whereKey, index);
38
43
  }
39
44
  else {
@@ -58,4 +63,4 @@ export default class VelociousDatabaseQueryWhereHash extends WhereBase {
58
63
  return sql;
59
64
  }
60
65
  }
61
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2hlcmUtaGFzaC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9kYXRhYmFzZS9xdWVyeS93aGVyZS1oYXNoLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVk7QUFFWixPQUFPLFNBQVMsTUFBTSxpQkFBaUIsQ0FBQTtBQUV2Qzs7R0FFRztBQUVILE1BQU0sQ0FBQyxPQUFPLE9BQU8sK0JBQWdDLFNBQVEsU0FBUztJQUNwRTs7O09BR0c7SUFDSCxZQUFZLEtBQUssRUFBRSxJQUFJO1FBQ3JCLEtBQUssRUFBRSxDQUFBO1FBQ1AsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUE7UUFDaEIsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUE7SUFDcEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSztRQUNILElBQUksR0FBRyxHQUFHLEdBQUcsQ0FBQTtRQUViLEdBQUcsSUFBSSxJQUFJLENBQUMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFBO1FBQ3hDLEdBQUcsSUFBSSxHQUFHLENBQUE7UUFFVixPQUFPLEdBQUcsQ0FBQTtJQUNaLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNILGlCQUFpQixDQUFDLElBQUksRUFBRSxTQUFTLEVBQUUsS0FBSyxHQUFHLENBQUM7UUFDMUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFBO1FBQ2pDLElBQUksR0FBRyxHQUFHLEVBQUUsQ0FBQTtRQUVaLEtBQUssTUFBTSxRQUFRLElBQUksSUFBSSxFQUFFLENBQUM7WUFDNUIsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO1lBRWpDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxJQUFJLFVBQVUsS0FBSyxJQUFJLElBQUksT0FBTyxVQUFVLElBQUksUUFBUSxFQUFFLENBQUM7Z0JBQ3ZGLEdBQUcsSUFBSSxJQUFJLENBQUMsaUJBQWlCLENBQUMsVUFBVSxFQUFFLFFBQVEsRUFBRSxLQUFLLENBQUMsQ0FBQTtZQUM1RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sSUFBSSxLQUFLLEdBQUcsQ0FBQztvQkFBRSxHQUFHLElBQUksT0FBTyxDQUFBO2dCQUU3QixJQUFJLFNBQVMsRUFBRSxDQUFDO29CQUNkLEdBQUcsSUFBSSxHQUFHLE9BQU8sQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQTtnQkFDaEQsQ0FBQztnQkFFRCxHQUFHLElBQUksR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUE7Z0JBRTdDLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDO29CQUM5QixHQUFHLElBQUksUUFBUSxVQUFVLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUE7Z0JBQzlFLENBQUM7cUJBQU0sSUFBSSxVQUFVLEtBQUssSUFBSSxFQUFFLENBQUM7b0JBQy9CLEdBQUcsSUFBSSxVQUFVLENBQUE7Z0JBQ25CLENBQUM7cUJBQU0sQ0FBQztvQkFDTixHQUFHLElBQUksTUFBTSxPQUFPLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxFQUFFLENBQUE7Z0JBQzFDLENBQUM7WUFDSCxDQUFDO1lBRUQsS0FBSyxFQUFFLENBQUE7UUFDVCxDQUFDO1FBRUQsT0FBTyxHQUFHLENBQUE7SUFDWixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IFdoZXJlQmFzZSBmcm9tIFwiLi93aGVyZS1iYXNlLmpzXCJcblxuLyoqXG4gKiBAdHlwZWRlZiB7e1trZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsIHwgQXJyYXk8c3RyaW5nIHwgbnVtYmVyIHwgYm9vbGVhbiB8IG51bGw+IHwgV2hlcmVIYXNofX0gV2hlcmVIYXNoXG4gKi9cblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgVmVsb2Npb3VzRGF0YWJhc2VRdWVyeVdoZXJlSGFzaCBleHRlbmRzIFdoZXJlQmFzZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4vaW5kZXguanNcIikuZGVmYXVsdH0gcXVlcnkgLSBRdWVyeSBpbnN0YW5jZS5cbiAgICogQHBhcmFtIHtXaGVyZUhhc2h9IGhhc2ggLSBIYXNoLlxuICAgKi9cbiAgY29uc3RydWN0b3IocXVlcnksIGhhc2gpIHtcbiAgICBzdXBlcigpXG4gICAgdGhpcy5oYXNoID0gaGFzaFxuICAgIHRoaXMucXVlcnkgPSBxdWVyeVxuICB9XG5cbiAgLyoqXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IC0gU1FMIHN0cmluZy5cbiAgICovXG4gIHRvU3FsKCkge1xuICAgIGxldCBzcWwgPSBcIihcIlxuXG4gICAgc3FsICs9IHRoaXMuX3doZXJlU1FMRnJvbUhhc2godGhpcy5oYXNoKVxuICAgIHNxbCArPSBcIilcIlxuXG4gICAgcmV0dXJuIHNxbFxuICB9XG5cbiAgLyoqXG4gICAqIEBwYXJhbSB7V2hlcmVIYXNofSBoYXNoIC0gSGFzaC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IFt0YWJsZU5hbWVdIC0gVGFibGUgbmFtZS5cbiAgICogQHBhcmFtIHtudW1iZXJ9IGluZGV4IC0gSW5kZXggdmFsdWUuXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IC0gU1FMIHN0cmluZy5cbiAgICovXG4gIF93aGVyZVNRTEZyb21IYXNoKGhhc2gsIHRhYmxlTmFtZSwgaW5kZXggPSAwKSB7XG4gICAgY29uc3Qgb3B0aW9ucyA9IHRoaXMuZ2V0T3B0aW9ucygpXG4gICAgbGV0IHNxbCA9IFwiXCJcblxuICAgIGZvciAoY29uc3Qgd2hlcmVLZXkgaW4gaGFzaCkge1xuICAgICAgY29uc3Qgd2hlcmVWYWx1ZSA9IGhhc2hbd2hlcmVLZXldXG5cbiAgICAgIGlmICghQXJyYXkuaXNBcnJheSh3aGVyZVZhbHVlKSAmJiB3aGVyZVZhbHVlICE9PSBudWxsICYmIHR5cGVvZiB3aGVyZVZhbHVlID09IFwib2JqZWN0XCIpIHtcbiAgICAgICAgc3FsICs9IHRoaXMuX3doZXJlU1FMRnJvbUhhc2god2hlcmVWYWx1ZSwgd2hlcmVLZXksIGluZGV4KVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgaWYgKGluZGV4ID4gMCkgc3FsICs9IFwiIEFORCBcIlxuXG4gICAgICAgIGlmICh0YWJsZU5hbWUpIHtcbiAgICAgICAgICBzcWwgKz0gYCR7b3B0aW9ucy5xdW90ZVRhYmxlTmFtZSh0YWJsZU5hbWUpfS5gXG4gICAgICAgIH1cblxuICAgICAgICBzcWwgKz0gYCR7b3B0aW9ucy5xdW90ZUNvbHVtbk5hbWUod2hlcmVLZXkpfWBcblxuICAgICAgICBpZiAoQXJyYXkuaXNBcnJheSh3aGVyZVZhbHVlKSkge1xuICAgICAgICAgIHNxbCArPSBgIElOICgke3doZXJlVmFsdWUubWFwKCh2YWx1ZSkgPT4gb3B0aW9ucy5xdW90ZSh2YWx1ZSkpLmpvaW4oXCIsIFwiKX0pYFxuICAgICAgICB9IGVsc2UgaWYgKHdoZXJlVmFsdWUgPT09IG51bGwpIHtcbiAgICAgICAgICBzcWwgKz0gXCIgSVMgTlVMTFwiXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgc3FsICs9IGAgPSAke29wdGlvbnMucXVvdGUod2hlcmVWYWx1ZSl9YFxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGluZGV4KytcbiAgICB9XG5cbiAgICByZXR1cm4gc3FsXG4gIH1cbn1cbiJdfQ==
66
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2hlcmUtaGFzaC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9kYXRhYmFzZS9xdWVyeS93aGVyZS1oYXNoLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVk7QUFFWixPQUFPLFNBQVMsTUFBTSxpQkFBaUIsQ0FBQTtBQUV2Qzs7R0FFRztBQUVILE1BQU0sQ0FBQyxPQUFPLE9BQU8sK0JBQWdDLFNBQVEsU0FBUztJQUNwRTs7O09BR0c7SUFDSCxZQUFZLEtBQUssRUFBRSxJQUFJO1FBQ3JCLEtBQUssRUFBRSxDQUFBO1FBQ1AsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUE7UUFDaEIsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUE7SUFDcEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSztRQUNILElBQUksR0FBRyxHQUFHLEdBQUcsQ0FBQTtRQUViLEdBQUcsSUFBSSxJQUFJLENBQUMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFBO1FBQ3hDLEdBQUcsSUFBSSxHQUFHLENBQUE7UUFFVixPQUFPLEdBQUcsQ0FBQTtJQUNaLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNILGlCQUFpQixDQUFDLElBQUksRUFBRSxTQUFTLEVBQUUsS0FBSyxHQUFHLENBQUM7UUFDMUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFBO1FBQ2pDLElBQUksR0FBRyxHQUFHLEVBQUUsQ0FBQTtRQUVaLEtBQUssTUFBTSxRQUFRLElBQUksSUFBSSxFQUFFLENBQUM7WUFDNUIsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO1lBRWpDLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsSUFBSSxVQUFVLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO2dCQUN6RCxJQUFJLEtBQUssR0FBRyxDQUFDO29CQUFFLEdBQUcsSUFBSSxPQUFPLENBQUE7Z0JBQzdCLEdBQUcsSUFBSSxLQUFLLENBQUE7WUFDZCxDQUFDO2lCQUFNLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxJQUFJLFVBQVUsS0FBSyxJQUFJLElBQUksT0FBTyxVQUFVLElBQUksUUFBUSxFQUFFLENBQUM7Z0JBQzlGLEdBQUcsSUFBSSxJQUFJLENBQUMsaUJBQWlCLENBQUMsVUFBVSxFQUFFLFFBQVEsRUFBRSxLQUFLLENBQUMsQ0FBQTtZQUM1RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sSUFBSSxLQUFLLEdBQUcsQ0FBQztvQkFBRSxHQUFHLElBQUksT0FBTyxDQUFBO2dCQUU3QixJQUFJLFNBQVMsRUFBRSxDQUFDO29CQUNkLEdBQUcsSUFBSSxHQUFHLE9BQU8sQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQTtnQkFDaEQsQ0FBQztnQkFFRCxHQUFHLElBQUksR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUE7Z0JBRTdDLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDO29CQUM5QixHQUFHLElBQUksUUFBUSxVQUFVLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUE7Z0JBQzlFLENBQUM7cUJBQU0sSUFBSSxVQUFVLEtBQUssSUFBSSxFQUFFLENBQUM7b0JBQy9CLEdBQUcsSUFBSSxVQUFVLENBQUE7Z0JBQ25CLENBQUM7cUJBQU0sQ0FBQztvQkFDTixHQUFHLElBQUksTUFBTSxPQUFPLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxFQUFFLENBQUE7Z0JBQzFDLENBQUM7WUFDSCxDQUFDO1lBRUQsS0FBSyxFQUFFLENBQUE7UUFDVCxDQUFDO1FBRUQsT0FBTyxHQUFHLENBQUE7SUFDWixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IFdoZXJlQmFzZSBmcm9tIFwiLi93aGVyZS1iYXNlLmpzXCJcblxuLyoqXG4gKiBAdHlwZWRlZiB7e1trZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB8IGJvb2xlYW4gfCBudWxsIHwgQXJyYXk8c3RyaW5nIHwgbnVtYmVyIHwgYm9vbGVhbiB8IG51bGw+IHwgV2hlcmVIYXNofX0gV2hlcmVIYXNoXG4gKi9cblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgVmVsb2Npb3VzRGF0YWJhc2VRdWVyeVdoZXJlSGFzaCBleHRlbmRzIFdoZXJlQmFzZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4vaW5kZXguanNcIikuZGVmYXVsdH0gcXVlcnkgLSBRdWVyeSBpbnN0YW5jZS5cbiAgICogQHBhcmFtIHtXaGVyZUhhc2h9IGhhc2ggLSBIYXNoLlxuICAgKi9cbiAgY29uc3RydWN0b3IocXVlcnksIGhhc2gpIHtcbiAgICBzdXBlcigpXG4gICAgdGhpcy5oYXNoID0gaGFzaFxuICAgIHRoaXMucXVlcnkgPSBxdWVyeVxuICB9XG5cbiAgLyoqXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IC0gU1FMIHN0cmluZy5cbiAgICovXG4gIHRvU3FsKCkge1xuICAgIGxldCBzcWwgPSBcIihcIlxuXG4gICAgc3FsICs9IHRoaXMuX3doZXJlU1FMRnJvbUhhc2godGhpcy5oYXNoKVxuICAgIHNxbCArPSBcIilcIlxuXG4gICAgcmV0dXJuIHNxbFxuICB9XG5cbiAgLyoqXG4gICAqIEBwYXJhbSB7V2hlcmVIYXNofSBoYXNoIC0gSGFzaC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IFt0YWJsZU5hbWVdIC0gVGFibGUgbmFtZS5cbiAgICogQHBhcmFtIHtudW1iZXJ9IGluZGV4IC0gSW5kZXggdmFsdWUuXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IC0gU1FMIHN0cmluZy5cbiAgICovXG4gIF93aGVyZVNRTEZyb21IYXNoKGhhc2gsIHRhYmxlTmFtZSwgaW5kZXggPSAwKSB7XG4gICAgY29uc3Qgb3B0aW9ucyA9IHRoaXMuZ2V0T3B0aW9ucygpXG4gICAgbGV0IHNxbCA9IFwiXCJcblxuICAgIGZvciAoY29uc3Qgd2hlcmVLZXkgaW4gaGFzaCkge1xuICAgICAgY29uc3Qgd2hlcmVWYWx1ZSA9IGhhc2hbd2hlcmVLZXldXG5cbiAgICAgIGlmIChBcnJheS5pc0FycmF5KHdoZXJlVmFsdWUpICYmIHdoZXJlVmFsdWUubGVuZ3RoID09PSAwKSB7XG4gICAgICAgIGlmIChpbmRleCA+IDApIHNxbCArPSBcIiBBTkQgXCJcbiAgICAgICAgc3FsICs9IFwiMT0wXCJcbiAgICAgIH0gZWxzZSBpZiAoIUFycmF5LmlzQXJyYXkod2hlcmVWYWx1ZSkgJiYgd2hlcmVWYWx1ZSAhPT0gbnVsbCAmJiB0eXBlb2Ygd2hlcmVWYWx1ZSA9PSBcIm9iamVjdFwiKSB7XG4gICAgICAgIHNxbCArPSB0aGlzLl93aGVyZVNRTEZyb21IYXNoKHdoZXJlVmFsdWUsIHdoZXJlS2V5LCBpbmRleClcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGlmIChpbmRleCA+IDApIHNxbCArPSBcIiBBTkQgXCJcblxuICAgICAgICBpZiAodGFibGVOYW1lKSB7XG4gICAgICAgICAgc3FsICs9IGAke29wdGlvbnMucXVvdGVUYWJsZU5hbWUodGFibGVOYW1lKX0uYFxuICAgICAgICB9XG5cbiAgICAgICAgc3FsICs9IGAke29wdGlvbnMucXVvdGVDb2x1bW5OYW1lKHdoZXJlS2V5KX1gXG5cbiAgICAgICAgaWYgKEFycmF5LmlzQXJyYXkod2hlcmVWYWx1ZSkpIHtcbiAgICAgICAgICBzcWwgKz0gYCBJTiAoJHt3aGVyZVZhbHVlLm1hcCgodmFsdWUpID0+IG9wdGlvbnMucXVvdGUodmFsdWUpKS5qb2luKFwiLCBcIil9KWBcbiAgICAgICAgfSBlbHNlIGlmICh3aGVyZVZhbHVlID09PSBudWxsKSB7XG4gICAgICAgICAgc3FsICs9IFwiIElTIE5VTExcIlxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHNxbCArPSBgID0gJHtvcHRpb25zLnF1b3RlKHdoZXJlVmFsdWUpfWBcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpbmRleCsrXG4gICAgfVxuXG4gICAgcmV0dXJuIHNxbFxuICB9XG59XG4iXX0=
@@ -0,0 +1,71 @@
1
+ export default class VelociousDatabaseQueryWhereModelClassHash extends WhereBase {
2
+ /**
3
+ * @param {object} args - Options object.
4
+ * @param {import("./index.js").default} args.query - Query instance.
5
+ * @param {WhereHash} args.hash - Hash.
6
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
7
+ * @param {boolean} [args.qualifyBaseTable] - Whether to qualify base table columns.
8
+ */
9
+ constructor({ query, hash, modelClass, qualifyBaseTable }: {
10
+ query: import("./index.js").default;
11
+ hash: WhereHash;
12
+ modelClass: typeof import("../record/index.js").default;
13
+ qualifyBaseTable?: boolean;
14
+ });
15
+ hash: WhereHash;
16
+ modelClass: typeof import("../record/index.js").default;
17
+ qualifyBaseTable: boolean;
18
+ /**
19
+ * @returns {typeof import("../record/index.js").default} - The model class.
20
+ */
21
+ getModelClass(): typeof import("../record/index.js").default;
22
+ /**
23
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
24
+ * @param {string} key - Attribute or column name.
25
+ * @returns {string | undefined} - The resolved column name.
26
+ */
27
+ _resolveColumnName(modelClass: typeof import("../record/index.js").default, key: string): string | undefined;
28
+ /**
29
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
30
+ * @param {string} relationshipName - Relationship name.
31
+ * @returns {import("../record/relationships/base.js").default | undefined} - The relationship.
32
+ */
33
+ _getRelationship(modelClass: typeof import("../record/index.js").default, relationshipName: string): import("../record/relationships/base.js").default | undefined;
34
+ /**
35
+ * @param {object} args - Options object.
36
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
37
+ * @param {string} args.columnName - Column name.
38
+ * @param {any} args.value - Value to normalize.
39
+ * @returns {any} - Normalized value.
40
+ */
41
+ _normalizeSqliteBooleanValue({ modelClass, columnName, value }: {
42
+ modelClass: typeof import("../record/index.js").default;
43
+ columnName: string;
44
+ value: any;
45
+ }): any;
46
+ /**
47
+ * @param {object} args - Options object.
48
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
49
+ * @param {string} args.columnName - Column name.
50
+ * @param {any} args.value - Value to normalize.
51
+ * @returns {any} - Normalized value.
52
+ */
53
+ _normalizeValueForColumnType({ modelClass, columnName, value }: {
54
+ modelClass: typeof import("../record/index.js").default;
55
+ columnName: string;
56
+ value: any;
57
+ }): any;
58
+ /**
59
+ * @param {WhereHash} hash - Hash.
60
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
61
+ * @param {string} [tableName] - Table name.
62
+ * @param {number} index - Index value.
63
+ * @returns {string} - SQL string.
64
+ */
65
+ _whereSQLFromHash(hash: WhereHash, modelClass: typeof import("../record/index.js").default, tableName?: string, index?: number): string;
66
+ }
67
+ export type WhereHash = {
68
+ [key: string]: string | number | boolean | null | Array<string | number | boolean | null> | Record<string, any>;
69
+ };
70
+ import WhereBase from "./where-base.js";
71
+ //# sourceMappingURL=where-model-class-hash.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"where-model-class-hash.d.ts","sourceRoot":"","sources":["../../../../src/database/query/where-model-class-hash.js"],"names":[],"mappings":"AAWA;IACE;;;;;;OAMG;IACH,2DALG;QAA2C,KAAK,EAAxC,OAAO,YAAY,EAAE,OAAO;QACZ,IAAI,EAApB,SAAS;QACyC,UAAU,EAA5D,cAAc,oBAAoB,EAAE,OAAO;QAC5B,gBAAgB,GAA/B,OAAO;KACjB,EAOA;IAJC,gBAAgB;IAChB,wDAA4B;IAC5B,0BAAwC;IAI1C;;OAEG;IACH,iBAFa,cAAc,oBAAoB,EAAE,OAAO,CAMvD;IAgBD;;;;OAIG;IACH,+BAJW,cAAc,oBAAoB,EAAE,OAAO,OAC3C,MAAM,GACJ,MAAM,GAAG,SAAS,CAS9B;IAED;;;;OAIG;IACH,6BAJW,cAAc,oBAAoB,EAAE,OAAO,oBAC3C,MAAM,GACJ,OAAO,iCAAiC,EAAE,OAAO,GAAG,SAAS,CAIzE;IAED;;;;;;OAMG;IACH,gEALG;QAA0D,UAAU,EAA5D,cAAc,oBAAoB,EAAE,OAAO;QAC9B,UAAU,EAAvB,MAAM;QACI,KAAK,EAAf,GAAG;KACX,GAAU,GAAG,CAqBf;IAED;;;;;;OAMG;IACH,gEALG;QAA0D,UAAU,EAA5D,cAAc,oBAAoB,EAAE,OAAO;QAC9B,UAAU,EAAvB,MAAM;QACI,KAAK,EAAf,GAAG;KACX,GAAU,GAAG,CAkCf;IAED;;;;;;OAMG;IACH,wBANW,SAAS,cACT,cAAc,oBAAoB,EAAE,OAAO,cAC3C,MAAM,UACN,MAAM,GACJ,MAAM,CA4ElB;CACF;wBA1NY;IAAC,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,GAAG,KAAK,CAAC,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;CAAC;sBAHxG,iBAAiB"}
@@ -0,0 +1,195 @@
1
+ // @ts-check
2
+ import { isPlainObject } from "is-plain-object";
3
+ import WhereBase from "./where-base.js";
4
+ /**
5
+ * @typedef {{[key: string]: string | number | boolean | null | Array<string | number | boolean | null> | Record<string, any>}} WhereHash
6
+ */
7
+ const NO_MATCH = Symbol("no-match");
8
+ export default class VelociousDatabaseQueryWhereModelClassHash extends WhereBase {
9
+ /**
10
+ * @param {object} args - Options object.
11
+ * @param {import("./index.js").default} args.query - Query instance.
12
+ * @param {WhereHash} args.hash - Hash.
13
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
14
+ * @param {boolean} [args.qualifyBaseTable] - Whether to qualify base table columns.
15
+ */
16
+ constructor({ query, hash, modelClass, qualifyBaseTable = false }) {
17
+ super();
18
+ this.hash = hash;
19
+ this.modelClass = modelClass;
20
+ this.qualifyBaseTable = qualifyBaseTable;
21
+ this.query = query;
22
+ }
23
+ /**
24
+ * @returns {typeof import("../record/index.js").default} - The model class.
25
+ */
26
+ getModelClass() {
27
+ if (!this.modelClass)
28
+ throw new Error("modelClass not set");
29
+ return this.modelClass;
30
+ }
31
+ /**
32
+ * @returns {string} - SQL string.
33
+ */
34
+ toSql() {
35
+ let sql = "(";
36
+ const baseTableName = this.qualifyBaseTable ? this.getModelClass().tableName() : undefined;
37
+ sql += this._whereSQLFromHash(this.hash, this.getModelClass(), baseTableName);
38
+ sql += ")";
39
+ return sql;
40
+ }
41
+ /**
42
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
43
+ * @param {string} key - Attribute or column name.
44
+ * @returns {string | undefined} - The resolved column name.
45
+ */
46
+ _resolveColumnName(modelClass, key) {
47
+ const attributeMap = modelClass.getAttributeNameToColumnNameMap();
48
+ const columnName = attributeMap[key];
49
+ if (columnName)
50
+ return columnName;
51
+ return undefined;
52
+ }
53
+ /**
54
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
55
+ * @param {string} relationshipName - Relationship name.
56
+ * @returns {import("../record/relationships/base.js").default | undefined} - The relationship.
57
+ */
58
+ _getRelationship(modelClass, relationshipName) {
59
+ return modelClass.getRelationshipsMap()[relationshipName];
60
+ }
61
+ /**
62
+ * @param {object} args - Options object.
63
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
64
+ * @param {string} args.columnName - Column name.
65
+ * @param {any} args.value - Value to normalize.
66
+ * @returns {any} - Normalized value.
67
+ */
68
+ _normalizeSqliteBooleanValue({ modelClass, columnName, value }) {
69
+ if (modelClass.getDatabaseType() != "sqlite")
70
+ return value;
71
+ const columnType = modelClass.getColumnTypeByName(columnName);
72
+ if (!columnType || typeof columnType != "string")
73
+ return value;
74
+ if (columnType.toLowerCase() !== "boolean")
75
+ return value;
76
+ const normalize = (entry) => {
77
+ if (entry === true)
78
+ return 1;
79
+ if (entry === false)
80
+ return 0;
81
+ return entry;
82
+ };
83
+ if (Array.isArray(value)) {
84
+ return value.map((entry) => normalize(entry));
85
+ }
86
+ return normalize(value);
87
+ }
88
+ /**
89
+ * @param {object} args - Options object.
90
+ * @param {typeof import("../record/index.js").default} args.modelClass - Model class.
91
+ * @param {string} args.columnName - Column name.
92
+ * @param {any} args.value - Value to normalize.
93
+ * @returns {any} - Normalized value.
94
+ */
95
+ _normalizeValueForColumnType({ modelClass, columnName, value }) {
96
+ const columnType = modelClass.getColumnTypeByName(columnName);
97
+ if (!columnType || typeof columnType != "string")
98
+ return value;
99
+ const normalizedType = columnType.toLowerCase();
100
+ const stringTypes = new Set(["char", "varchar", "nvarchar", "string", "enum", "json", "jsonb", "citext", "binary", "varbinary"]);
101
+ const isUuidType = normalizedType.includes("uuid");
102
+ const shouldCoerceToString = normalizedType.includes("uuid") ||
103
+ normalizedType.includes("text") ||
104
+ stringTypes.has(normalizedType);
105
+ const normalize = (entry) => {
106
+ if (isUuidType && typeof entry === "number")
107
+ return NO_MATCH;
108
+ if (!shouldCoerceToString || typeof entry !== "number")
109
+ return entry;
110
+ return String(entry);
111
+ };
112
+ if (Array.isArray(value)) {
113
+ const normalized = value.map((entry) => normalize(entry)).filter((entry) => entry !== NO_MATCH);
114
+ if (isUuidType && normalized.length === 0)
115
+ return NO_MATCH;
116
+ return normalized;
117
+ }
118
+ const normalized = normalize(value);
119
+ if (normalized === NO_MATCH)
120
+ return NO_MATCH;
121
+ return normalized;
122
+ }
123
+ /**
124
+ * @param {WhereHash} hash - Hash.
125
+ * @param {typeof import("../record/index.js").default} modelClass - Model class.
126
+ * @param {string} [tableName] - Table name.
127
+ * @param {number} index - Index value.
128
+ * @returns {string} - SQL string.
129
+ */
130
+ _whereSQLFromHash(hash, modelClass, tableName, index = 0) {
131
+ const options = this.getOptions();
132
+ let sql = "";
133
+ for (const whereKey in hash) {
134
+ const whereValue = hash[whereKey];
135
+ if (Array.isArray(whereValue) && whereValue.length === 0) {
136
+ if (index > 0)
137
+ sql += " AND ";
138
+ sql += "1=0";
139
+ }
140
+ else if (isPlainObject(whereValue)) {
141
+ const relationship = this._getRelationship(modelClass, whereKey);
142
+ if (!relationship) {
143
+ throw new Error(`Unknown relationship "${whereKey}" for ${modelClass.name}`);
144
+ }
145
+ const targetModelClass = relationship.getTargetModelClass();
146
+ const nestedHash = /** @type {WhereHash} */ (whereValue);
147
+ sql += this._whereSQLFromHash(nestedHash, targetModelClass, targetModelClass.tableName(), index);
148
+ }
149
+ else {
150
+ if (index > 0)
151
+ sql += " AND ";
152
+ const columnName = this._resolveColumnName(modelClass, whereKey);
153
+ if (!columnName)
154
+ throw new Error(`Unknown attribute "${whereKey}" for ${modelClass.name}`);
155
+ const columnType = modelClass.getColumnTypeByName(columnName);
156
+ const normalizedValue = this._normalizeSqliteBooleanValue({
157
+ columnName,
158
+ modelClass,
159
+ value: whereValue
160
+ });
161
+ const typedValue = this._normalizeValueForColumnType({
162
+ columnName,
163
+ modelClass,
164
+ value: normalizedValue
165
+ });
166
+ if (typedValue === NO_MATCH) {
167
+ sql += "1=0";
168
+ index++;
169
+ continue;
170
+ }
171
+ let columnSql = `${options.quoteColumnName(columnName)}`;
172
+ if (tableName) {
173
+ columnSql = `${options.quoteTableName(tableName)}.${columnSql}`;
174
+ }
175
+ const driverType = this.getQuery().driver.getType();
176
+ if (driverType == "mssql" && typeof whereValue === "string" && columnType?.toLowerCase() == "text") {
177
+ columnSql = `CAST(${columnSql} AS NVARCHAR(MAX))`;
178
+ }
179
+ sql += columnSql;
180
+ if (Array.isArray(typedValue)) {
181
+ sql += ` IN (${typedValue.map((value) => options.quote(value)).join(", ")})`;
182
+ }
183
+ else if (typedValue === null) {
184
+ sql += " IS NULL";
185
+ }
186
+ else {
187
+ sql += ` = ${options.quote(typedValue)}`;
188
+ }
189
+ }
190
+ index++;
191
+ }
192
+ return sql;
193
+ }
194
+ }
195
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"where-model-class-hash.js","sourceRoot":"","sources":["../../../../src/database/query/where-model-class-hash.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,EAAC,aAAa,EAAC,MAAM,iBAAiB,CAAA;AAC7C,OAAO,SAAS,MAAM,iBAAiB,CAAA;AAEvC;;GAEG;AAEH,MAAM,QAAQ,GAAG,MAAM,CAAC,UAAU,CAAC,CAAA;AAEnC,MAAM,CAAC,OAAO,OAAO,yCAA0C,SAAQ,SAAS;IAC9E;;;;;;OAMG;IACH,YAAY,EAAC,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,gBAAgB,GAAG,KAAK,EAAC;QAC7D,KAAK,EAAE,CAAA;QACP,IAAI,CAAC,IAAI,GAAG,IAAI,CAAA;QAChB,IAAI,CAAC,UAAU,GAAG,UAAU,CAAA;QAC5B,IAAI,CAAC,gBAAgB,GAAG,gBAAgB,CAAA;QACxC,IAAI,CAAC,KAAK,GAAG,KAAK,CAAA;IACpB,CAAC;IAED;;OAEG;IACH,aAAa;QACX,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,MAAM,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAA;QAE3D,OAAO,IAAI,CAAC,UAAU,CAAA;IACxB,CAAC;IAED;;OAEG;IACH,KAAK;QACH,IAAI,GAAG,GAAG,GAAG,CAAA;QAEb,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,SAAS,CAAA;QAE1F,GAAG,IAAI,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,aAAa,EAAE,EAAE,aAAa,CAAC,CAAA;QAC7E,GAAG,IAAI,GAAG,CAAA;QAEV,OAAO,GAAG,CAAA;IACZ,CAAC;IAED;;;;OAIG;IACH,kBAAkB,CAAC,UAAU,EAAE,GAAG;QAChC,MAAM,YAAY,GAAG,UAAU,CAAC,+BAA+B,EAAE,CAAA;QACjE,MAAM,UAAU,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAEpC,IAAI,UAAU;YAAE,OAAO,UAAU,CAAA;QAEjC,OAAO,SAAS,CAAA;IAClB,CAAC;IAED;;;;OAIG;IACH,gBAAgB,CAAC,UAAU,EAAE,gBAAgB;QAC3C,OAAO,UAAU,CAAC,mBAAmB,EAAE,CAAC,gBAAgB,CAAC,CAAA;IAC3D,CAAC;IAED;;;;;;OAMG;IACH,4BAA4B,CAAC,EAAC,UAAU,EAAE,UAAU,EAAE,KAAK,EAAC;QAC1D,IAAI,UAAU,CAAC,eAAe,EAAE,IAAI,QAAQ;YAAE,OAAO,KAAK,CAAA;QAE1D,MAAM,UAAU,GAAG,UAAU,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAA;QAE7D,IAAI,CAAC,UAAU,IAAI,OAAO,UAAU,IAAI,QAAQ;YAAE,OAAO,KAAK,CAAA;QAC9D,IAAI,UAAU,CAAC,WAAW,EAAE,KAAK,SAAS;YAAE,OAAO,KAAK,CAAA;QAExD,MAAM,SAAS,GAAG,CAAC,KAAK,EAAE,EAAE;YAC1B,IAAI,KAAK,KAAK,IAAI;gBAAE,OAAO,CAAC,CAAA;YAC5B,IAAI,KAAK,KAAK,KAAK;gBAAE,OAAO,CAAC,CAAA;YAC7B,OAAO,KAAK,CAAA;QACd,CAAC,CAAA;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACzB,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAA;QAC/C,CAAC;QAED,OAAO,SAAS,CAAC,KAAK,CAAC,CAAA;IACzB,CAAC;IAED;;;;;;OAMG;IACH,4BAA4B,CAAC,EAAC,UAAU,EAAE,UAAU,EAAE,KAAK,EAAC;QAC1D,MAAM,UAAU,GAAG,UAAU,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAA;QAE7D,IAAI,CAAC,UAAU,IAAI,OAAO,UAAU,IAAI,QAAQ;YAAE,OAAO,KAAK,CAAA;QAE9D,MAAM,cAAc,GAAG,UAAU,CAAC,WAAW,EAAE,CAAA;QAC/C,MAAM,WAAW,GAAG,IAAI,GAAG,CAAC,CAAC,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAA;QAChI,MAAM,UAAU,GAAG,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;QAClD,MAAM,oBAAoB,GAAG,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC1D,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC/B,WAAW,CAAC,GAAG,CAAC,cAAc,CAAC,CAAA;QAEjC,MAAM,SAAS,GAAG,CAAC,KAAK,EAAE,EAAE;YAC1B,IAAI,UAAU,IAAI,OAAO,KAAK,KAAK,QAAQ;gBAAE,OAAO,QAAQ,CAAA;YAC5D,IAAI,CAAC,oBAAoB,IAAI,OAAO,KAAK,KAAK,QAAQ;gBAAE,OAAO,KAAK,CAAA;YAEpE,OAAO,MAAM,CAAC,KAAK,CAAC,CAAA;QACtB,CAAC,CAAA;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACzB,MAAM,UAAU,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,KAAK,QAAQ,CAAC,CAAA;YAE/F,IAAI,UAAU,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC;gBAAE,OAAO,QAAQ,CAAA;YAE1D,OAAO,UAAU,CAAA;QACnB,CAAC;QAED,MAAM,UAAU,GAAG,SAAS,CAAC,KAAK,CAAC,CAAA;QAEnC,IAAI,UAAU,KAAK,QAAQ;YAAE,OAAO,QAAQ,CAAA;QAE5C,OAAO,UAAU,CAAA;IACnB,CAAC;IAED;;;;;;OAMG;IACH,iBAAiB,CAAC,IAAI,EAAE,UAAU,EAAE,SAAS,EAAE,KAAK,GAAG,CAAC;QACtD,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAA;QACjC,IAAI,GAAG,GAAG,EAAE,CAAA;QAEZ,KAAK,MAAM,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC5B,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAA;YAEjC,IAAI,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACzD,IAAI,KAAK,GAAG,CAAC;oBAAE,GAAG,IAAI,OAAO,CAAA;gBAC7B,GAAG,IAAI,KAAK,CAAA;YACd,CAAC;iBAAM,IAAI,aAAa,CAAC,UAAU,CAAC,EAAE,CAAC;gBACrC,MAAM,YAAY,GAAG,IAAI,CAAC,gBAAgB,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAA;gBAEhE,IAAI,CAAC,YAAY,EAAE,CAAC;oBAClB,MAAM,IAAI,KAAK,CAAC,yBAAyB,QAAQ,SAAS,UAAU,CAAC,IAAI,EAAE,CAAC,CAAA;gBAC9E,CAAC;gBAED,MAAM,gBAAgB,GAAG,YAAY,CAAC,mBAAmB,EAAE,CAAA;gBAC3D,MAAM,UAAU,GAAG,wBAAwB,CAAC,CAAC,UAAU,CAAC,CAAA;gBAExD,GAAG,IAAI,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,gBAAgB,EAAE,gBAAgB,CAAC,SAAS,EAAE,EAAE,KAAK,CAAC,CAAA;YAClG,CAAC;iBAAM,CAAC;gBACN,IAAI,KAAK,GAAG,CAAC;oBAAE,GAAG,IAAI,OAAO,CAAA;gBAE7B,MAAM,UAAU,GAAG,IAAI,CAAC,kBAAkB,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAA;gBAEhE,IAAI,CAAC,UAAU;oBAAE,MAAM,IAAI,KAAK,CAAC,sBAAsB,QAAQ,SAAS,UAAU,CAAC,IAAI,EAAE,CAAC,CAAA;gBAE1F,MAAM,UAAU,GAAG,UAAU,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAA;gBAE7D,MAAM,eAAe,GAAG,IAAI,CAAC,4BAA4B,CAAC;oBACxD,UAAU;oBACV,UAAU;oBACV,KAAK,EAAE,UAAU;iBAClB,CAAC,CAAA;gBACF,MAAM,UAAU,GAAG,IAAI,CAAC,4BAA4B,CAAC;oBACnD,UAAU;oBACV,UAAU;oBACV,KAAK,EAAE,eAAe;iBACvB,CAAC,CAAA;gBAEF,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;oBAC5B,GAAG,IAAI,KAAK,CAAA;oBACZ,KAAK,EAAE,CAAA;oBACP,SAAQ;gBACV,CAAC;gBAED,IAAI,SAAS,GAAG,GAAG,OAAO,CAAC,eAAe,CAAC,UAAU,CAAC,EAAE,CAAA;gBAExD,IAAI,SAAS,EAAE,CAAC;oBACd,SAAS,GAAG,GAAG,OAAO,CAAC,cAAc,CAAC,SAAS,CAAC,IAAI,SAAS,EAAE,CAAA;gBACjE,CAAC;gBAED,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,CAAA;gBAEnD,IAAI,UAAU,IAAI,OAAO,IAAI,OAAO,UAAU,KAAK,QAAQ,IAAI,UAAU,EAAE,WAAW,EAAE,IAAI,MAAM,EAAE,CAAC;oBACnG,SAAS,GAAG,QAAQ,SAAS,oBAAoB,CAAA;gBACnD,CAAC;gBAED,GAAG,IAAI,SAAS,CAAA;gBAEhB,IAAI,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;oBAC9B,GAAG,IAAI,QAAQ,UAAU,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAA;gBAC9E,CAAC;qBAAM,IAAI,UAAU,KAAK,IAAI,EAAE,CAAC;oBAC/B,GAAG,IAAI,UAAU,CAAA;gBACnB,CAAC;qBAAM,CAAC;oBACN,GAAG,IAAI,MAAM,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE,CAAA;gBAC1C,CAAC;YACH,CAAC;YAED,KAAK,EAAE,CAAA;QACT,CAAC;QAED,OAAO,GAAG,CAAA;IACZ,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport {isPlainObject} from \"is-plain-object\"\nimport WhereBase from \"./where-base.js\"\n\n/**\n * @typedef {{[key: string]: string | number | boolean | null | Array<string | number | boolean | null> | Record<string, any>}} WhereHash\n */\n\nconst NO_MATCH = Symbol(\"no-match\")\n\nexport default class VelociousDatabaseQueryWhereModelClassHash extends WhereBase {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"./index.js\").default} args.query - Query instance.\n   * @param {WhereHash} args.hash - Hash.\n   * @param {typeof import(\"../record/index.js\").default} args.modelClass - Model class.\n   * @param {boolean} [args.qualifyBaseTable] - Whether to qualify base table columns.\n   */\n  constructor({query, hash, modelClass, qualifyBaseTable = false}) {\n    super()\n    this.hash = hash\n    this.modelClass = modelClass\n    this.qualifyBaseTable = qualifyBaseTable\n    this.query = query\n  }\n\n  /**\n   * @returns {typeof import(\"../record/index.js\").default} - The model class.\n   */\n  getModelClass() {\n    if (!this.modelClass) throw new Error(\"modelClass not set\")\n\n    return this.modelClass\n  }\n\n  /**\n   * @returns {string} - SQL string.\n   */\n  toSql() {\n    let sql = \"(\"\n\n    const baseTableName = this.qualifyBaseTable ? this.getModelClass().tableName() : undefined\n\n    sql += this._whereSQLFromHash(this.hash, this.getModelClass(), baseTableName)\n    sql += \")\"\n\n    return sql\n  }\n\n  /**\n   * @param {typeof import(\"../record/index.js\").default} modelClass - Model class.\n   * @param {string} key - Attribute or column name.\n   * @returns {string | undefined} - The resolved column name.\n   */\n  _resolveColumnName(modelClass, key) {\n    const attributeMap = modelClass.getAttributeNameToColumnNameMap()\n    const columnName = attributeMap[key]\n\n    if (columnName) return columnName\n\n    return undefined\n  }\n\n  /**\n   * @param {typeof import(\"../record/index.js\").default} modelClass - Model class.\n   * @param {string} relationshipName - Relationship name.\n   * @returns {import(\"../record/relationships/base.js\").default | undefined} - The relationship.\n   */\n  _getRelationship(modelClass, relationshipName) {\n    return modelClass.getRelationshipsMap()[relationshipName]\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {typeof import(\"../record/index.js\").default} args.modelClass - Model class.\n   * @param {string} args.columnName - Column name.\n   * @param {any} args.value - Value to normalize.\n   * @returns {any} - Normalized value.\n   */\n  _normalizeSqliteBooleanValue({modelClass, columnName, value}) {\n    if (modelClass.getDatabaseType() != \"sqlite\") return value\n\n    const columnType = modelClass.getColumnTypeByName(columnName)\n\n    if (!columnType || typeof columnType != \"string\") return value\n    if (columnType.toLowerCase() !== \"boolean\") return value\n\n    const normalize = (entry) => {\n      if (entry === true) return 1\n      if (entry === false) return 0\n      return entry\n    }\n\n    if (Array.isArray(value)) {\n      return value.map((entry) => normalize(entry))\n    }\n\n    return normalize(value)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {typeof import(\"../record/index.js\").default} args.modelClass - Model class.\n   * @param {string} args.columnName - Column name.\n   * @param {any} args.value - Value to normalize.\n   * @returns {any} - Normalized value.\n   */\n  _normalizeValueForColumnType({modelClass, columnName, value}) {\n    const columnType = modelClass.getColumnTypeByName(columnName)\n\n    if (!columnType || typeof columnType != \"string\") return value\n\n    const normalizedType = columnType.toLowerCase()\n    const stringTypes = new Set([\"char\", \"varchar\", \"nvarchar\", \"string\", \"enum\", \"json\", \"jsonb\", \"citext\", \"binary\", \"varbinary\"])\n    const isUuidType = normalizedType.includes(\"uuid\")\n    const shouldCoerceToString = normalizedType.includes(\"uuid\") ||\n      normalizedType.includes(\"text\") ||\n      stringTypes.has(normalizedType)\n\n    const normalize = (entry) => {\n      if (isUuidType && typeof entry === \"number\") return NO_MATCH\n      if (!shouldCoerceToString || typeof entry !== \"number\") return entry\n\n      return String(entry)\n    }\n\n    if (Array.isArray(value)) {\n      const normalized = value.map((entry) => normalize(entry)).filter((entry) => entry !== NO_MATCH)\n\n      if (isUuidType && normalized.length === 0) return NO_MATCH\n\n      return normalized\n    }\n\n    const normalized = normalize(value)\n\n    if (normalized === NO_MATCH) return NO_MATCH\n\n    return normalized\n  }\n\n  /**\n   * @param {WhereHash} hash - Hash.\n   * @param {typeof import(\"../record/index.js\").default} modelClass - Model class.\n   * @param {string} [tableName] - Table name.\n   * @param {number} index - Index value.\n   * @returns {string} - SQL string.\n   */\n  _whereSQLFromHash(hash, modelClass, tableName, index = 0) {\n    const options = this.getOptions()\n    let sql = \"\"\n\n    for (const whereKey in hash) {\n      const whereValue = hash[whereKey]\n\n      if (Array.isArray(whereValue) && whereValue.length === 0) {\n        if (index > 0) sql += \" AND \"\n        sql += \"1=0\"\n      } else if (isPlainObject(whereValue)) {\n        const relationship = this._getRelationship(modelClass, whereKey)\n\n        if (!relationship) {\n          throw new Error(`Unknown relationship \"${whereKey}\" for ${modelClass.name}`)\n        }\n\n        const targetModelClass = relationship.getTargetModelClass()\n        const nestedHash = /** @type {WhereHash} */ (whereValue)\n\n        sql += this._whereSQLFromHash(nestedHash, targetModelClass, targetModelClass.tableName(), index)\n      } else {\n        if (index > 0) sql += \" AND \"\n\n        const columnName = this._resolveColumnName(modelClass, whereKey)\n\n        if (!columnName) throw new Error(`Unknown attribute \"${whereKey}\" for ${modelClass.name}`)\n\n        const columnType = modelClass.getColumnTypeByName(columnName)\n\n        const normalizedValue = this._normalizeSqliteBooleanValue({\n          columnName,\n          modelClass,\n          value: whereValue\n        })\n        const typedValue = this._normalizeValueForColumnType({\n          columnName,\n          modelClass,\n          value: normalizedValue\n        })\n\n        if (typedValue === NO_MATCH) {\n          sql += \"1=0\"\n          index++\n          continue\n        }\n\n        let columnSql = `${options.quoteColumnName(columnName)}`\n\n        if (tableName) {\n          columnSql = `${options.quoteTableName(tableName)}.${columnSql}`\n        }\n\n        const driverType = this.getQuery().driver.getType()\n\n        if (driverType == \"mssql\" && typeof whereValue === \"string\" && columnType?.toLowerCase() == \"text\") {\n          columnSql = `CAST(${columnSql} AS NVARCHAR(MAX))`\n        }\n\n        sql += columnSql\n\n        if (Array.isArray(typedValue)) {\n          sql += ` IN (${typedValue.map((value) => options.quote(value)).join(\", \")})`\n        } else if (typedValue === null) {\n          sql += \" IS NULL\"\n        } else {\n          sql += ` = ${options.quote(typedValue)}`\n        }\n      }\n\n      index++\n    }\n\n    return sql\n  }\n}\n"]}
@@ -537,9 +537,19 @@ declare class VelociousDatabaseRecord {
537
537
  * @returns {any} - The column.
538
538
  */
539
539
  readColumn(attributeName: string): any;
540
+ /**
541
+ * @param {any} value - Value from database.
542
+ * @returns {any} - Normalized value.
543
+ */
544
+ _normalizeDateValueForRead(value: any): any;
540
545
  _belongsToChanges(): Record<string, any>;
541
546
  /** @returns {Promise<void>} - Resolves when complete. */
542
547
  _createNewRecord(): Promise<void>;
548
+ /**
549
+ * @param {Record<string, any>} data - Column-keyed data.
550
+ * @returns {void} - No return value.
551
+ */
552
+ _normalizeDateValuesForWrite(data: Record<string, any>): void;
543
553
  /** @returns {Promise<void>} - Resolves when complete. */
544
554
  _updateRecordWithChanges(): Promise<void>;
545
555
  /** @returns {number|string} - The id. */