velocious 1.0.237 → 1.0.239
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.
|
@@ -111,9 +111,9 @@ export default class RecordAttachmentsStore {
|
|
|
111
111
|
}): Promise<void>;
|
|
112
112
|
/**
|
|
113
113
|
* @param {string} driverName - Driver name.
|
|
114
|
-
* @returns {Record<string, any
|
|
114
|
+
* @returns {Promise<Record<string, any>>} - Attachment storage driver instance.
|
|
115
115
|
*/
|
|
116
|
-
attachmentDriverByName(driverName: string): Record<string, any
|
|
116
|
+
attachmentDriverByName(driverName: string): Promise<Record<string, any>>;
|
|
117
117
|
/**
|
|
118
118
|
* @param {object} args - Options.
|
|
119
119
|
* @param {AttachmentDriverConstructor | Record<string, any>} args.driverReference - Driver class or instance.
|
|
@@ -141,13 +141,13 @@ export default class RecordAttachmentsStore {
|
|
|
141
141
|
* @param {import("../index.js").default} args.model - Model instance.
|
|
142
142
|
* @param {string} args.name - Attachment name.
|
|
143
143
|
* @param {Record<string, any>} [args.row] - Attachment row.
|
|
144
|
-
* @returns {Record<string, any
|
|
144
|
+
* @returns {Promise<Record<string, any>>} - Attachment storage driver instance.
|
|
145
145
|
*/
|
|
146
146
|
resolveAttachmentDriver({ model, name, row }: {
|
|
147
147
|
model: import("../index.js").default;
|
|
148
148
|
name: string;
|
|
149
149
|
row?: Record<string, any>;
|
|
150
|
-
}): Record<string, any
|
|
150
|
+
}): Promise<Record<string, any>>;
|
|
151
151
|
/**
|
|
152
152
|
* @param {object} args - Options.
|
|
153
153
|
* @param {import("../../../database/drivers/base.js").default} args.db - DB connection.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"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"}
|
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
// @ts-check
|
|
2
|
-
import
|
|
2
|
+
import UUID from "pure-uuid";
|
|
3
3
|
import TableData from "../../table-data/index.js";
|
|
4
|
-
import FilesystemAttachmentStorageDriver from "./storage-drivers/filesystem.js";
|
|
5
4
|
import NativeAttachmentStorageDriver from "./storage-drivers/native.js";
|
|
6
5
|
import normalizeRecordAttachmentInput from "./normalize-input.js";
|
|
7
6
|
import S3AttachmentStorageDriver from "./storage-drivers/s3.js";
|
|
@@ -9,6 +8,30 @@ const ATTACHMENTS_TABLE = "velocious_attachments";
|
|
|
9
8
|
/** @typedef {new (...args: any[]) => Record<string, any>} AttachmentDriverConstructor */
|
|
10
9
|
/** @type {WeakMap<import("../../../configuration.js").default, Map<string, RecordAttachmentsStore>>} */
|
|
11
10
|
const storesByConfiguration = new WeakMap();
|
|
11
|
+
/** @type {Promise<any> | undefined} */
|
|
12
|
+
let filesystemAttachmentStorageDriverClassPromise;
|
|
13
|
+
/**
|
|
14
|
+
* @param {string} specifier - Module specifier.
|
|
15
|
+
* @returns {Promise<any>} - Imported module.
|
|
16
|
+
*/
|
|
17
|
+
async function dynamicImport(specifier) {
|
|
18
|
+
const importer = /** @type {(moduleSpecifier: string) => Promise<any>} */ (new Function("moduleSpecifier", "return import(moduleSpecifier)"));
|
|
19
|
+
return await importer(specifier);
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* @returns {Promise<any>} - Filesystem driver class.
|
|
23
|
+
*/
|
|
24
|
+
async function filesystemAttachmentStorageDriverClass() {
|
|
25
|
+
filesystemAttachmentStorageDriverClassPromise ||= dynamicImport("./storage-drivers/filesystem.js")
|
|
26
|
+
.then((module) => module.default);
|
|
27
|
+
return await filesystemAttachmentStorageDriverClassPromise;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* @returns {string} - Generated UUID v4 value.
|
|
31
|
+
*/
|
|
32
|
+
function generateUUID() {
|
|
33
|
+
return new UUID(4).format();
|
|
34
|
+
}
|
|
12
35
|
/**
|
|
13
36
|
* @param {import("../index.js").default} model - Model instance.
|
|
14
37
|
* @returns {string} - Store key.
|
|
@@ -117,12 +140,12 @@ export default class RecordAttachmentsStore {
|
|
|
117
140
|
allowPathInput,
|
|
118
141
|
allowedPathPrefixes
|
|
119
142
|
});
|
|
120
|
-
const attachmentDriver = this.resolveAttachmentDriver({ model, name });
|
|
143
|
+
const attachmentDriver = await this.resolveAttachmentDriver({ model, name });
|
|
121
144
|
const attachmentDriverName = this._attachmentDriverNameFor({ model, name });
|
|
122
145
|
const now = Date.now();
|
|
123
146
|
const recordType = model.getModelClass().name;
|
|
124
147
|
const recordId = String(model.id());
|
|
125
|
-
const attachmentId =
|
|
148
|
+
const attachmentId = generateUUID();
|
|
126
149
|
const { storageKey } = await attachmentDriver.write({
|
|
127
150
|
attachmentId,
|
|
128
151
|
input: normalizedInput,
|
|
@@ -214,7 +237,7 @@ export default class RecordAttachmentsStore {
|
|
|
214
237
|
if (!storageKey) {
|
|
215
238
|
throw new Error(`Attachment row ${String(row.id)} is missing storage key`);
|
|
216
239
|
}
|
|
217
|
-
const attachmentDriver = this.resolveAttachmentDriver({ model, name, row });
|
|
240
|
+
const attachmentDriver = await this.resolveAttachmentDriver({ model, name, row });
|
|
218
241
|
return await attachmentDriver.read({
|
|
219
242
|
model,
|
|
220
243
|
name,
|
|
@@ -230,7 +253,7 @@ export default class RecordAttachmentsStore {
|
|
|
230
253
|
* @returns {Promise<string | null>} - Attachment URL.
|
|
231
254
|
*/
|
|
232
255
|
async attachmentRowUrl({ model, name, row }) {
|
|
233
|
-
const attachmentDriver = this.resolveAttachmentDriver({ model, name, row });
|
|
256
|
+
const attachmentDriver = await this.resolveAttachmentDriver({ model, name, row });
|
|
234
257
|
if (typeof attachmentDriver.url !== "function")
|
|
235
258
|
return null;
|
|
236
259
|
const storageKey = typeof row.storage_key === "string" && row.storage_key.length > 0
|
|
@@ -302,7 +325,7 @@ export default class RecordAttachmentsStore {
|
|
|
302
325
|
const storageKey = typeof row.storage_key === "string" && row.storage_key.length > 0 ? row.storage_key : null;
|
|
303
326
|
if (!storageKey)
|
|
304
327
|
return;
|
|
305
|
-
const attachmentDriver = this.resolveAttachmentDriver({ model, name, row });
|
|
328
|
+
const attachmentDriver = await this.resolveAttachmentDriver({ model, name, row });
|
|
306
329
|
if (typeof attachmentDriver.delete !== "function")
|
|
307
330
|
return;
|
|
308
331
|
await attachmentDriver.delete({
|
|
@@ -314,9 +337,9 @@ export default class RecordAttachmentsStore {
|
|
|
314
337
|
}
|
|
315
338
|
/**
|
|
316
339
|
* @param {string} driverName - Driver name.
|
|
317
|
-
* @returns {Record<string, any
|
|
340
|
+
* @returns {Promise<Record<string, any>>} - Attachment storage driver instance.
|
|
318
341
|
*/
|
|
319
|
-
attachmentDriverByName(driverName) {
|
|
342
|
+
async attachmentDriverByName(driverName) {
|
|
320
343
|
if (this._attachmentDriversByName.has(driverName)) {
|
|
321
344
|
return /** @type {Record<string, any>} */ (this._attachmentDriversByName.get(driverName));
|
|
322
345
|
}
|
|
@@ -342,6 +365,7 @@ export default class RecordAttachmentsStore {
|
|
|
342
365
|
});
|
|
343
366
|
}
|
|
344
367
|
else if (driverName === "filesystem") {
|
|
368
|
+
const FilesystemAttachmentStorageDriver = await filesystemAttachmentStorageDriverClass();
|
|
345
369
|
attachmentDriver = new FilesystemAttachmentStorageDriver({
|
|
346
370
|
configuration: this.configuration,
|
|
347
371
|
options: configuredDriver
|
|
@@ -432,9 +456,9 @@ export default class RecordAttachmentsStore {
|
|
|
432
456
|
* @param {import("../index.js").default} args.model - Model instance.
|
|
433
457
|
* @param {string} args.name - Attachment name.
|
|
434
458
|
* @param {Record<string, any>} [args.row] - Attachment row.
|
|
435
|
-
* @returns {Record<string, any
|
|
459
|
+
* @returns {Promise<Record<string, any>>} - Attachment storage driver instance.
|
|
436
460
|
*/
|
|
437
|
-
resolveAttachmentDriver({ model, name, row }) {
|
|
461
|
+
async resolveAttachmentDriver({ model, name, row }) {
|
|
438
462
|
const attachmentDefinition = model.getModelClass().getAttachmentByName(name);
|
|
439
463
|
const configuredDriver = attachmentDefinition.driver;
|
|
440
464
|
if (typeof configuredDriver === "function" || (configuredDriver && typeof configuredDriver === "object")) {
|
|
@@ -447,7 +471,7 @@ export default class RecordAttachmentsStore {
|
|
|
447
471
|
const fallbackDriverName = typeof row?.driver === "string" && row.driver.length > 0
|
|
448
472
|
? row.driver
|
|
449
473
|
: this._attachmentDriverNameFor({ model, name });
|
|
450
|
-
return this.attachmentDriverByName(fallbackDriverName);
|
|
474
|
+
return await this.attachmentDriverByName(fallbackDriverName);
|
|
451
475
|
}
|
|
452
476
|
/**
|
|
453
477
|
* @param {object} args - Options.
|
|
@@ -485,4 +509,4 @@ export default class RecordAttachmentsStore {
|
|
|
485
509
|
return /** @type {T} */ (result);
|
|
486
510
|
}
|
|
487
511
|
}
|
|
488
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {randomUUID} from \"crypto\"\nimport TableData from \"../../table-data/index.js\"\nimport FilesystemAttachmentStorageDriver from \"./storage-drivers/filesystem.js\"\nimport NativeAttachmentStorageDriver from \"./storage-drivers/native.js\"\nimport normalizeRecordAttachmentInput from \"./normalize-input.js\"\nimport S3AttachmentStorageDriver from \"./storage-drivers/s3.js\"\n\nconst ATTACHMENTS_TABLE = \"velocious_attachments\"\n\n/** @typedef {new (...args: any[]) => Record<string, any>} AttachmentDriverConstructor */\n\n/** @type {WeakMap<import(\"../../../configuration.js\").default, Map<string, RecordAttachmentsStore>>} */\nconst storesByConfiguration = new WeakMap()\n\n/**\n * @param {import(\"../index.js\").default} model - Model instance.\n * @returns {string} - Store key.\n */\nfunction storeKeyForModel(model) {\n  return `${model.getModelClass().getDatabaseIdentifier()}`\n}\n\n/**\n * @param {import(\"../index.js\").default} model - Model instance.\n * @returns {RecordAttachmentsStore} - Store instance.\n */\nexport function recordAttachmentsStoreForModel(model) {\n  const configuration = model._getConfiguration()\n  let storesByDatabaseIdentifier = storesByConfiguration.get(configuration)\n\n  if (!storesByDatabaseIdentifier) {\n    storesByDatabaseIdentifier = new Map()\n    storesByConfiguration.set(configuration, storesByDatabaseIdentifier)\n  }\n\n  const key = storeKeyForModel(model)\n  let store = storesByDatabaseIdentifier.get(key)\n\n  if (store) return store\n\n  store = new RecordAttachmentsStore({\n    configuration,\n    databaseIdentifier: model.getModelClass().getDatabaseIdentifier()\n  })\n\n  storesByDatabaseIdentifier.set(key, store)\n\n  return store\n}\n\n/**\n * Attachment persistence store.\n */\nexport default class RecordAttachmentsStore {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} args.databaseIdentifier - Database identifier.\n   */\n  constructor({configuration, databaseIdentifier}) {\n    this.configuration = configuration\n    this.databaseIdentifier = databaseIdentifier\n    this._readyPromise = null\n    this._driverColumnsAvailable = false\n    this._contentBase64Nullable = true\n    /** @type {Map<string, Record<string, any>>} */\n    this._attachmentDriversByName = new Map()\n    /** @type {Map<AttachmentDriverConstructor | Record<string, any>, Record<string, any>>} */\n    this._attachmentDriversByReference = new Map()\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when schema is ready.\n   */\n  async ensureReady() {\n    if (this._readyPromise) {\n      await this._readyPromise\n      return\n    }\n\n    this._readyPromise = (async () => {\n      await this._withDb(async (db) => {\n        if (await db.tableExists(ATTACHMENTS_TABLE)) {\n          await this.ensureAttachmentStoreSchema({db})\n          return\n        }\n\n        const table = new TableData(ATTACHMENTS_TABLE, {ifNotExists: true})\n\n        table.string(\"id\", {null: false, primaryKey: true})\n        table.string(\"record_type\", {null: false, index: true})\n        table.string(\"record_id\", {null: false, index: true})\n        table.string(\"name\", {null: false, index: true})\n        table.integer(\"position\", {null: false})\n        table.string(\"filename\", {null: false})\n        table.string(\"content_type\", {null: true})\n        table.bigint(\"byte_size\", {null: false})\n        table.string(\"driver\", {null: true})\n        table.string(\"storage_key\", {null: true})\n        table.text(\"content_base64\", {null: true})\n        table.bigint(\"created_at_ms\", {null: false})\n        table.bigint(\"updated_at_ms\", {null: false})\n\n        await db.createTable(table)\n        this._driverColumnsAvailable = true\n        this._contentBase64Nullable = true\n      })\n    })()\n\n    try {\n      await this._readyPromise\n    } finally {\n      this._readyPromise = null\n    }\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {unknown} args.input - Attachment input.\n   * @param {boolean} args.replace - Whether to replace existing attachments.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async attach({input, model, name, replace}) {\n    await this.ensureReady()\n    const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const allowPathInput = attachmentsConfiguration.allowPathInput === true\n    const allowedPathPrefixes = Array.isArray(attachmentsConfiguration.allowedPathPrefixes)\n      ? attachmentsConfiguration.allowedPathPrefixes\n      : undefined\n\n    const normalizedInput = await normalizeRecordAttachmentInput(input, {\n      allowPathInput,\n      allowedPathPrefixes\n    })\n    const attachmentDriver = this.resolveAttachmentDriver({model, name})\n    const attachmentDriverName = this._attachmentDriverNameFor({model, name})\n    const now = Date.now()\n    const recordType = model.getModelClass().name\n    const recordId = String(model.id())\n    const attachmentId = randomUUID()\n    const {storageKey} = await attachmentDriver.write({\n      attachmentId,\n      input: normalizedInput,\n      model,\n      name\n    })\n\n    await this._withDb(async (db) => {\n      if (replace) {\n        const existingRows = await db\n          .newQuery()\n          .from(ATTACHMENTS_TABLE)\n          .where({name, record_id: recordId, record_type: recordType})\n          .results()\n\n        for (const existingRow of existingRows) {\n          await this.deleteAttachmentRowStorage({model, name, row: existingRow})\n        }\n\n        await db.delete({\n          conditions: {name, record_id: recordId, record_type: recordType},\n          tableName: ATTACHMENTS_TABLE\n        })\n      }\n\n      const position = replace ? 0 : await this._nextPosition({db, name, recordId, recordType})\n      /** @type {Record<string, any>} */\n      const insertData = {\n        byte_size: normalizedInput.byteSize,\n        content_base64: this._contentBase64Nullable ? null : normalizedInput.contentBase64,\n        content_type: normalizedInput.contentType,\n        created_at_ms: now,\n        filename: normalizedInput.filename,\n        id: attachmentId,\n        name,\n        position,\n        record_id: recordId,\n        record_type: recordType,\n        updated_at_ms: now\n      }\n\n      if (this._driverColumnsAvailable) {\n        insertData.driver = attachmentDriverName\n        insertData.storage_key = storageKey\n      }\n\n      await db.insert({\n        data: insertData,\n        tableName: ATTACHMENTS_TABLE\n      })\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../database/drivers/base.js\").default} args.db - DB connection.\n   * @returns {Promise<void>} - Resolves when schema columns are ensured.\n   */\n  async ensureAttachmentStoreSchema({db}) {\n    const table = await db.getTableByNameOrFail(ATTACHMENTS_TABLE)\n    const columns = await table.getColumns()\n    const hasDriverColumn = columns.some((column) => column.getName() === \"driver\")\n    const hasStorageKeyColumn = columns.some((column) => column.getName() === \"storage_key\")\n    const contentBase64Column = columns.find((column) => column.getName() === \"content_base64\")\n    const alterTable = new TableData(ATTACHMENTS_TABLE)\n    let shouldAlter = false\n\n    if (!hasDriverColumn) {\n      alterTable.string(\"driver\", {null: true})\n      shouldAlter = true\n    }\n\n    if (!hasStorageKeyColumn) {\n      alterTable.string(\"storage_key\", {null: true})\n      shouldAlter = true\n    }\n\n    if (shouldAlter) {\n      const alterTableSQLs = await db.alterTableSQLs(alterTable)\n\n      for (const sql of alterTableSQLs) {\n        await db.query(sql)\n      }\n    }\n\n    this._driverColumnsAvailable = true\n    this._contentBase64Nullable = contentBase64Column ? contentBase64Column.getNull() : true\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<Buffer>} - Attachment bytes.\n   */\n  async readAttachmentRow({model, name, row}) {\n    if (typeof row.content_base64 === \"string\" && row.content_base64.length > 0) {\n      return Buffer.from(row.content_base64, \"base64\")\n    }\n\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0 ? row.storage_key : null\n\n    if (!storageKey) {\n      throw new Error(`Attachment row ${String(row.id)} is missing storage key`)\n    }\n\n    const attachmentDriver = this.resolveAttachmentDriver({model, name, row})\n\n    return await attachmentDriver.read({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<string | null>} - Attachment URL.\n   */\n  async attachmentRowUrl({model, name, row}) {\n    const attachmentDriver = this.resolveAttachmentDriver({model, name, row})\n\n    if (typeof attachmentDriver.url !== \"function\") return null\n\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0\n      ? row.storage_key\n      : (typeof row.id === \"string\" ? row.id : \"\")\n\n    if (!storageKey) return null\n\n    return await attachmentDriver.url({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {string} [args.id] - Optional attachment id.\n   * @returns {Promise<Record<string, any> | null>} - Attachment row.\n   */\n  async findOne({id, model, name}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const recordType = model.getModelClass().name\n      const recordId = String(model.id())\n      let query = db\n        .newQuery()\n        .from(ATTACHMENTS_TABLE)\n        .where({name, record_id: recordId, record_type: recordType})\n        .order(\"position ASC\")\n        .order(\"created_at_ms DESC\")\n        .limit(1)\n\n      if (id) {\n        query = query.where({id})\n      }\n\n      const rows = await query.results()\n\n      return rows[0] || null\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @returns {Promise<Array<Record<string, any>>>} - Attachment rows.\n   */\n  async findMany({model, name}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const recordType = model.getModelClass().name\n      const recordId = String(model.id())\n      const query = db\n        .newQuery()\n        .from(ATTACHMENTS_TABLE)\n        .where({name, record_id: recordId, record_type: recordType})\n        .order(\"position ASC\")\n        .order(\"created_at_ms ASC\")\n\n      return await query.results()\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<void>} - Resolves when row storage has been deleted.\n   */\n  async deleteAttachmentRowStorage({model, name, row}) {\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0 ? row.storage_key : null\n\n    if (!storageKey) return\n\n    const attachmentDriver = this.resolveAttachmentDriver({model, name, row})\n\n    if (typeof attachmentDriver.delete !== \"function\") return\n\n    await attachmentDriver.delete({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {string} driverName - Driver name.\n   * @returns {Record<string, any>} - Attachment storage driver instance.\n   */\n  attachmentDriverByName(driverName) {\n    if (this._attachmentDriversByName.has(driverName)) {\n      return /** @type {Record<string, any>} */ (this._attachmentDriversByName.get(driverName))\n    }\n\n    const attachmentConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const configuredDriver = attachmentConfiguration.drivers?.[driverName] || {}\n    /** @type {Record<string, any>} */\n    let attachmentDriver\n\n    if (configuredDriver.instance && typeof configuredDriver.instance === \"object\") {\n      attachmentDriver = configuredDriver.instance\n    } else if (typeof configuredDriver.driverClass === \"function\") {\n      attachmentDriver = new configuredDriver.driverClass({\n        configuration: this.configuration,\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (typeof configuredDriver.create === \"function\") {\n      attachmentDriver = configuredDriver.create({\n        configuration: this.configuration,\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (driverName === \"filesystem\") {\n      attachmentDriver = new FilesystemAttachmentStorageDriver({\n        configuration: this.configuration,\n        options: configuredDriver\n      })\n    } else if (driverName === \"native\") {\n      attachmentDriver = new NativeAttachmentStorageDriver({\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (driverName === \"s3\") {\n      attachmentDriver = new S3AttachmentStorageDriver({options: configuredDriver})\n    } else {\n      throw new Error(`Unknown attachment storage driver: ${driverName}`)\n    }\n\n    if (!attachmentDriver || typeof attachmentDriver.write !== \"function\" || typeof attachmentDriver.read !== \"function\") {\n      throw new Error(`Attachment storage driver \"${driverName}\" must implement write/read`)\n    }\n\n    this._attachmentDriversByName.set(driverName, attachmentDriver)\n\n    return attachmentDriver\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {AttachmentDriverConstructor | Record<string, any>} args.driverReference - Driver class or instance.\n   * @param {string} args.attachmentName - Attachment name.\n   * @param {typeof import(\"../index.js\").default} args.modelClass - Model class.\n   * @returns {Record<string, any>} - Attachment driver instance.\n   */\n  attachmentDriverByReference({attachmentName, driverReference, modelClass}) {\n    if (this._attachmentDriversByReference.has(driverReference)) {\n      return /** @type {Record<string, any>} */ (this._attachmentDriversByReference.get(driverReference))\n    }\n\n    /** @type {Record<string, any>} */\n    let attachmentDriver\n\n    if (typeof driverReference === \"function\") {\n      const DriverClass = /** @type {AttachmentDriverConstructor} */ (driverReference)\n\n      attachmentDriver = new DriverClass({\n        attachmentName,\n        configuration: this.configuration,\n        modelClass\n      })\n    } else if (driverReference && typeof driverReference === \"object\") {\n      attachmentDriver = driverReference\n    } else {\n      throw new Error(`Invalid attachment driver reference for ${modelClass.name}#${attachmentName}`)\n    }\n\n    if (typeof attachmentDriver.write !== \"function\" || typeof attachmentDriver.read !== \"function\") {\n      throw new Error(`Attachment driver for ${modelClass.name}#${attachmentName} must implement write/read`)\n    }\n\n    this._attachmentDriversByReference.set(driverReference, attachmentDriver)\n\n    return attachmentDriver\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @returns {string} - Attachment driver name.\n   */\n  _attachmentDriverNameFor({model, name}) {\n    const attachmentDefinition = model.getModelClass().getAttachmentByName(name)\n    const configuredDriver = attachmentDefinition.driver\n\n    if (typeof configuredDriver === \"string\" && configuredDriver.length > 0) {\n      return configuredDriver\n    }\n\n    if (typeof configuredDriver === \"function\") {\n      return configuredDriver.name || \"custom\"\n    }\n\n    if (configuredDriver && typeof configuredDriver === \"object\") {\n      const constructorName = configuredDriver.constructor?.name\n\n      if (typeof constructorName === \"string\" && constructorName.length > 0 && constructorName !== \"Object\") {\n        return constructorName\n      }\n\n      return \"custom\"\n    }\n\n    const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const defaultDriver = attachmentsConfiguration.defaultDriver\n\n    if (typeof defaultDriver === \"string\" && defaultDriver.length > 0) {\n      return defaultDriver\n    }\n\n    return \"filesystem\"\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} [args.row] - Attachment row.\n   * @returns {Record<string, any>} - Attachment storage driver instance.\n   */\n  resolveAttachmentDriver({model, name, row}) {\n    const attachmentDefinition = model.getModelClass().getAttachmentByName(name)\n    const configuredDriver = attachmentDefinition.driver\n\n    if (typeof configuredDriver === \"function\" || (configuredDriver && typeof configuredDriver === \"object\")) {\n      return this.attachmentDriverByReference({\n        attachmentName: name,\n        driverReference: configuredDriver,\n        modelClass: model.getModelClass()\n      })\n    }\n\n    const fallbackDriverName = typeof row?.driver === \"string\" && row.driver.length > 0\n      ? row.driver\n      : this._attachmentDriverNameFor({model, name})\n\n    return this.attachmentDriverByName(fallbackDriverName)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../database/drivers/base.js\").default} args.db - DB connection.\n   * @param {string} args.name - Attachment name.\n   * @param {string} args.recordId - Record id.\n   * @param {string} args.recordType - Record type.\n   * @returns {Promise<number>} - Next position.\n   */\n  async _nextPosition({db, name, recordId, recordType}) {\n    const query = db\n      .newQuery()\n      .from(ATTACHMENTS_TABLE)\n      .where({name, record_id: recordId, record_type: recordType})\n      .order(\"position DESC\")\n      .limit(1)\n    const rows = await query.results()\n    const current = Number(rows[0]?.position)\n\n    if (!Number.isFinite(current)) return 0\n\n    return current + 1\n  }\n\n  /**\n   * @template T\n   * @param {(db: import(\"../../../database/drivers/base.js\").default) => Promise<T>} callback - Callback.\n   * @returns {Promise<T>} - Callback result.\n   */\n  async _withDb(callback) {\n    const pool = this.configuration.getDatabasePool(this.databaseIdentifier)\n    /** @type {T | undefined} */\n    let result\n\n    await pool.withConnection(async (db) => {\n      result = await callback(db)\n    })\n\n    return /** @type {T} */ (result)\n  }\n}\n"]}
|
|
512
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,IAAI,MAAM,WAAW,CAAA;AAC5B,OAAO,SAAS,MAAM,2BAA2B,CAAA;AACjD,OAAO,6BAA6B,MAAM,6BAA6B,CAAA;AACvE,OAAO,8BAA8B,MAAM,sBAAsB,CAAA;AACjE,OAAO,yBAAyB,MAAM,yBAAyB,CAAA;AAE/D,MAAM,iBAAiB,GAAG,uBAAuB,CAAA;AAEjD,yFAAyF;AAEzF,wGAAwG;AACxG,MAAM,qBAAqB,GAAG,IAAI,OAAO,EAAE,CAAA;AAC3C,uCAAuC;AACvC,IAAI,6CAA6C,CAAA;AAEjD;;;GAGG;AACH,KAAK,UAAU,aAAa,CAAC,SAAS;IACpC,MAAM,QAAQ,GAAG,wDAAwD,CAAC,CACxE,IAAI,QAAQ,CAAC,iBAAiB,EAAE,gCAAgC,CAAC,CAClE,CAAA;IAED,OAAO,MAAM,QAAQ,CAAC,SAAS,CAAC,CAAA;AAClC,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,sCAAsC;IACnD,6CAA6C,KAAK,aAAa,CAAC,iCAAiC,CAAC;SAC/F,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,CAAA;IAEnC,OAAO,MAAM,6CAA6C,CAAA;AAC5D,CAAC;AAED;;GAEG;AACH,SAAS,YAAY;IACnB,OAAO,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAA;AAC7B,CAAC;AAED;;;GAGG;AACH,SAAS,gBAAgB,CAAC,KAAK;IAC7B,OAAO,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,qBAAqB,EAAE,EAAE,CAAA;AAC3D,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,8BAA8B,CAAC,KAAK;IAClD,MAAM,aAAa,GAAG,KAAK,CAAC,iBAAiB,EAAE,CAAA;IAC/C,IAAI,0BAA0B,GAAG,qBAAqB,CAAC,GAAG,CAAC,aAAa,CAAC,CAAA;IAEzE,IAAI,CAAC,0BAA0B,EAAE,CAAC;QAChC,0BAA0B,GAAG,IAAI,GAAG,EAAE,CAAA;QACtC,qBAAqB,CAAC,GAAG,CAAC,aAAa,EAAE,0BAA0B,CAAC,CAAA;IACtE,CAAC;IAED,MAAM,GAAG,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAA;IACnC,IAAI,KAAK,GAAG,0BAA0B,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;IAE/C,IAAI,KAAK;QAAE,OAAO,KAAK,CAAA;IAEvB,KAAK,GAAG,IAAI,sBAAsB,CAAC;QACjC,aAAa;QACb,kBAAkB,EAAE,KAAK,CAAC,aAAa,EAAE,CAAC,qBAAqB,EAAE;KAClE,CAAC,CAAA;IAEF,0BAA0B,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAA;IAE1C,OAAO,KAAK,CAAA;AACd,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,sBAAsB;IACzC;;;;OAIG;IACH,YAAY,EAAC,aAAa,EAAE,kBAAkB,EAAC;QAC7C,IAAI,CAAC,aAAa,GAAG,aAAa,CAAA;QAClC,IAAI,CAAC,kBAAkB,GAAG,kBAAkB,CAAA;QAC5C,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;QACzB,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAA;QACpC,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAA;QAClC,+CAA+C;QAC/C,IAAI,CAAC,wBAAwB,GAAG,IAAI,GAAG,EAAE,CAAA;QACzC,0FAA0F;QAC1F,IAAI,CAAC,6BAA6B,GAAG,IAAI,GAAG,EAAE,CAAA;IAChD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW;QACf,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,MAAM,IAAI,CAAC,aAAa,CAAA;YACxB,OAAM;QACR,CAAC;QAED,IAAI,CAAC,aAAa,GAAG,CAAC,KAAK,IAAI,EAAE;YAC/B,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;gBAC9B,IAAI,MAAM,EAAE,CAAC,WAAW,CAAC,iBAAiB,CAAC,EAAE,CAAC;oBAC5C,MAAM,IAAI,CAAC,2BAA2B,CAAC,EAAC,EAAE,EAAC,CAAC,CAAA;oBAC5C,OAAM;gBACR,CAAC;gBAED,MAAM,KAAK,GAAG,IAAI,SAAS,CAAC,iBAAiB,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,CAAC,CAAA;gBAEnE,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,EAAC,IAAI,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAC,CAAC,CAAA;gBACnD,KAAK,CAAC,MAAM,CAAC,aAAa,EAAE,EAAC,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;gBACvD,KAAK,CAAC,MAAM,CAAC,WAAW,EAAE,EAAC,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;gBACrD,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE,EAAC,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;gBAChD,KAAK,CAAC,OAAO,CAAC,UAAU,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC,CAAC,CAAA;gBACxC,KAAK,CAAC,MAAM,CAAC,UAAU,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC,CAAC,CAAA;gBACvC,KAAK,CAAC,MAAM,CAAC,cAAc,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;gBAC1C,KAAK,CAAC,MAAM,CAAC,WAAW,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC,CAAC,CAAA;gBACxC,KAAK,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;gBACpC,KAAK,CAAC,MAAM,CAAC,aAAa,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;gBACzC,KAAK,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;gBAC1C,KAAK,CAAC,MAAM,CAAC,eAAe,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC,CAAC,CAAA;gBAC5C,KAAK,CAAC,MAAM,CAAC,eAAe,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC,CAAC,CAAA;gBAE5C,MAAM,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,CAAA;gBAC3B,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAA;gBACnC,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAA;YACpC,CAAC,CAAC,CAAA;QACJ,CAAC,CAAC,EAAE,CAAA;QAEJ,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,aAAa,CAAA;QAC1B,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;QAC3B,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,MAAM,CAAC,EAAC,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,OAAO,EAAC;QACxC,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QACxB,MAAM,wBAAwB,GAAG,IAAI,CAAC,aAAa,CAAC,2BAA2B,EAAE,EAAE,IAAI,EAAE,CAAA;QACzF,MAAM,cAAc,GAAG,wBAAwB,CAAC,cAAc,KAAK,IAAI,CAAA;QACvE,MAAM,mBAAmB,GAAG,KAAK,CAAC,OAAO,CAAC,wBAAwB,CAAC,mBAAmB,CAAC;YACrF,CAAC,CAAC,wBAAwB,CAAC,mBAAmB;YAC9C,CAAC,CAAC,SAAS,CAAA;QAEb,MAAM,eAAe,GAAG,MAAM,8BAA8B,CAAC,KAAK,EAAE;YAClE,cAAc;YACd,mBAAmB;SACpB,CAAC,CAAA;QACF,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;QAC1E,MAAM,oBAAoB,GAAG,IAAI,CAAC,wBAAwB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;QACzE,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;QACtB,MAAM,UAAU,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,IAAI,CAAA;QAC7C,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAA;QACnC,MAAM,YAAY,GAAG,YAAY,EAAE,CAAA;QACnC,MAAM,EAAC,UAAU,EAAC,GAAG,MAAM,gBAAgB,CAAC,KAAK,CAAC;YAChD,YAAY;YACZ,KAAK,EAAE,eAAe;YACtB,KAAK;YACL,IAAI;SACL,CAAC,CAAA;QAEF,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YAC9B,IAAI,OAAO,EAAE,CAAC;gBACZ,MAAM,YAAY,GAAG,MAAM,EAAE;qBAC1B,QAAQ,EAAE;qBACV,IAAI,CAAC,iBAAiB,CAAC;qBACvB,KAAK,CAAC,EAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAC,CAAC;qBAC3D,OAAO,EAAE,CAAA;gBAEZ,KAAK,MAAM,WAAW,IAAI,YAAY,EAAE,CAAC;oBACvC,MAAM,IAAI,CAAC,0BAA0B,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,WAAW,EAAC,CAAC,CAAA;gBACxE,CAAC;gBAED,MAAM,EAAE,CAAC,MAAM,CAAC;oBACd,UAAU,EAAE,EAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAC;oBAChE,SAAS,EAAE,iBAAiB;iBAC7B,CAAC,CAAA;YACJ,CAAC;YAED,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC,aAAa,CAAC,EAAC,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,UAAU,EAAC,CAAC,CAAA;YACzF,kCAAkC;YAClC,MAAM,UAAU,GAAG;gBACjB,SAAS,EAAE,eAAe,CAAC,QAAQ;gBACnC,cAAc,EAAE,IAAI,CAAC,sBAAsB,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,eAAe,CAAC,aAAa;gBAClF,YAAY,EAAE,eAAe,CAAC,WAAW;gBACzC,aAAa,EAAE,GAAG;gBAClB,QAAQ,EAAE,eAAe,CAAC,QAAQ;gBAClC,EAAE,EAAE,YAAY;gBAChB,IAAI;gBACJ,QAAQ;gBACR,SAAS,EAAE,QAAQ;gBACnB,WAAW,EAAE,UAAU;gBACvB,aAAa,EAAE,GAAG;aACnB,CAAA;YAED,IAAI,IAAI,CAAC,uBAAuB,EAAE,CAAC;gBACjC,UAAU,CAAC,MAAM,GAAG,oBAAoB,CAAA;gBACxC,UAAU,CAAC,WAAW,GAAG,UAAU,CAAA;YACrC,CAAC;YAED,MAAM,EAAE,CAAC,MAAM,CAAC;gBACd,IAAI,EAAE,UAAU;gBAChB,SAAS,EAAE,iBAAiB;aAC7B,CAAC,CAAA;QACJ,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,2BAA2B,CAAC,EAAC,EAAE,EAAC;QACpC,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,oBAAoB,CAAC,iBAAiB,CAAC,CAAA;QAC9D,MAAM,OAAO,GAAG,MAAM,KAAK,CAAC,UAAU,EAAE,CAAA;QACxC,MAAM,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,QAAQ,CAAC,CAAA;QAC/E,MAAM,mBAAmB,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,aAAa,CAAC,CAAA;QACxF,MAAM,mBAAmB,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,gBAAgB,CAAC,CAAA;QAC3F,MAAM,UAAU,GAAG,IAAI,SAAS,CAAC,iBAAiB,CAAC,CAAA;QACnD,IAAI,WAAW,GAAG,KAAK,CAAA;QAEvB,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,UAAU,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;YACzC,WAAW,GAAG,IAAI,CAAA;QACpB,CAAC;QAED,IAAI,CAAC,mBAAmB,EAAE,CAAC;YACzB,UAAU,CAAC,MAAM,CAAC,aAAa,EAAE,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAAA;YAC9C,WAAW,GAAG,IAAI,CAAA;QACpB,CAAC;QAED,IAAI,WAAW,EAAE,CAAC;YAChB,MAAM,cAAc,GAAG,MAAM,EAAE,CAAC,cAAc,CAAC,UAAU,CAAC,CAAA;YAE1D,KAAK,MAAM,GAAG,IAAI,cAAc,EAAE,CAAC;gBACjC,MAAM,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;YACrB,CAAC;QACH,CAAC;QAED,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAA;QACnC,IAAI,CAAC,sBAAsB,GAAG,mBAAmB,CAAC,CAAC,CAAC,mBAAmB,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAAA;IAC1F,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,iBAAiB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC;QACxC,IAAI,OAAO,GAAG,CAAC,cAAc,KAAK,QAAQ,IAAI,GAAG,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5E,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,cAAc,EAAE,QAAQ,CAAC,CAAA;QAClD,CAAC;QAED,MAAM,UAAU,GAAG,OAAO,GAAG,CAAC,WAAW,KAAK,QAAQ,IAAI,GAAG,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAA;QAE7G,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,IAAI,KAAK,CAAC,kBAAkB,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,yBAAyB,CAAC,CAAA;QAC5E,CAAC;QAED,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC,CAAC,CAAA;QAE/E,OAAO,MAAM,gBAAgB,CAAC,IAAI,CAAC;YACjC,KAAK;YACL,IAAI;YACJ,GAAG;YACH,UAAU;SACX,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,gBAAgB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC;QACvC,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC,CAAC,CAAA;QAE/E,IAAI,OAAO,gBAAgB,CAAC,GAAG,KAAK,UAAU;YAAE,OAAO,IAAI,CAAA;QAE3D,MAAM,UAAU,GAAG,OAAO,GAAG,CAAC,WAAW,KAAK,QAAQ,IAAI,GAAG,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC;YAClF,CAAC,CAAC,GAAG,CAAC,WAAW;YACjB,CAAC,CAAC,CAAC,OAAO,GAAG,CAAC,EAAE,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAA;QAE9C,IAAI,CAAC,UAAU;YAAE,OAAO,IAAI,CAAA;QAE5B,OAAO,MAAM,gBAAgB,CAAC,GAAG,CAAC;YAChC,KAAK;YACL,IAAI;YACJ,GAAG;YACH,UAAU;SACX,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,OAAO,CAAC,EAAC,EAAE,EAAE,KAAK,EAAE,IAAI,EAAC;QAC7B,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QAExB,OAAO,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YACrC,MAAM,UAAU,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,IAAI,CAAA;YAC7C,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAA;YACnC,IAAI,KAAK,GAAG,EAAE;iBACX,QAAQ,EAAE;iBACV,IAAI,CAAC,iBAAiB,CAAC;iBACvB,KAAK,CAAC,EAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAC,CAAC;iBAC3D,KAAK,CAAC,cAAc,CAAC;iBACrB,KAAK,CAAC,oBAAoB,CAAC;iBAC3B,KAAK,CAAC,CAAC,CAAC,CAAA;YAEX,IAAI,EAAE,EAAE,CAAC;gBACP,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,EAAC,EAAE,EAAC,CAAC,CAAA;YAC3B,CAAC;YAED,MAAM,IAAI,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE,CAAA;YAElC,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,CAAA;QACxB,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,QAAQ,CAAC,EAAC,KAAK,EAAE,IAAI,EAAC;QAC1B,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QAExB,OAAO,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YACrC,MAAM,UAAU,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,IAAI,CAAA;YAC7C,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAA;YACnC,MAAM,KAAK,GAAG,EAAE;iBACb,QAAQ,EAAE;iBACV,IAAI,CAAC,iBAAiB,CAAC;iBACvB,KAAK,CAAC,EAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAC,CAAC;iBAC3D,KAAK,CAAC,cAAc,CAAC;iBACrB,KAAK,CAAC,mBAAmB,CAAC,CAAA;YAE7B,OAAO,MAAM,KAAK,CAAC,OAAO,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,0BAA0B,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC;QACjD,MAAM,UAAU,GAAG,OAAO,GAAG,CAAC,WAAW,KAAK,QAAQ,IAAI,GAAG,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAA;QAE7G,IAAI,CAAC,UAAU;YAAE,OAAM;QAEvB,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC,CAAC,CAAA;QAE/E,IAAI,OAAO,gBAAgB,CAAC,MAAM,KAAK,UAAU;YAAE,OAAM;QAEzD,MAAM,gBAAgB,CAAC,MAAM,CAAC;YAC5B,KAAK;YACL,IAAI;YACJ,GAAG;YACH,UAAU;SACX,CAAC,CAAA;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,sBAAsB,CAAC,UAAU;QACrC,IAAI,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC;YAClD,OAAO,kCAAkC,CAAC,CAAC,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAA;QAC3F,CAAC;QAED,MAAM,uBAAuB,GAAG,IAAI,CAAC,aAAa,CAAC,2BAA2B,EAAE,EAAE,IAAI,EAAE,CAAA;QACxF,MAAM,gBAAgB,GAAG,uBAAuB,CAAC,OAAO,EAAE,CAAC,UAAU,CAAC,IAAI,EAAE,CAAA;QAC5E,kCAAkC;QAClC,IAAI,gBAAgB,CAAA;QAEpB,IAAI,gBAAgB,CAAC,QAAQ,IAAI,OAAO,gBAAgB,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;YAC/E,gBAAgB,GAAG,gBAAgB,CAAC,QAAQ,CAAA;QAC9C,CAAC;aAAM,IAAI,OAAO,gBAAgB,CAAC,WAAW,KAAK,UAAU,EAAE,CAAC;YAC9D,gBAAgB,GAAG,IAAI,gBAAgB,CAAC,WAAW,CAAC;gBAClD,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,IAAI,EAAE,UAAU;gBAChB,OAAO,EAAE,gBAAgB;aAC1B,CAAC,CAAA;QACJ,CAAC;aAAM,IAAI,OAAO,gBAAgB,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;YACzD,gBAAgB,GAAG,gBAAgB,CAAC,MAAM,CAAC;gBACzC,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,IAAI,EAAE,UAAU;gBAChB,OAAO,EAAE,gBAAgB;aAC1B,CAAC,CAAA;QACJ,CAAC;aAAM,IAAI,UAAU,KAAK,YAAY,EAAE,CAAC;YACvC,MAAM,iCAAiC,GAAG,MAAM,sCAAsC,EAAE,CAAA;YAExF,gBAAgB,GAAG,IAAI,iCAAiC,CAAC;gBACvD,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,OAAO,EAAE,gBAAgB;aAC1B,CAAC,CAAA;QACJ,CAAC;aAAM,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;YACnC,gBAAgB,GAAG,IAAI,6BAA6B,CAAC;gBACnD,IAAI,EAAE,UAAU;gBAChB,OAAO,EAAE,gBAAgB;aAC1B,CAAC,CAAA;QACJ,CAAC;aAAM,IAAI,UAAU,KAAK,IAAI,EAAE,CAAC;YAC/B,gBAAgB,GAAG,IAAI,yBAAyB,CAAC,EAAC,OAAO,EAAE,gBAAgB,EAAC,CAAC,CAAA;QAC/E,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,sCAAsC,UAAU,EAAE,CAAC,CAAA;QACrE,CAAC;QAED,IAAI,CAAC,gBAAgB,IAAI,OAAO,gBAAgB,CAAC,KAAK,KAAK,UAAU,IAAI,OAAO,gBAAgB,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;YACrH,MAAM,IAAI,KAAK,CAAC,8BAA8B,UAAU,6BAA6B,CAAC,CAAA;QACxF,CAAC;QAED,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,UAAU,EAAE,gBAAgB,CAAC,CAAA;QAE/D,OAAO,gBAAgB,CAAA;IACzB,CAAC;IAED;;;;;;OAMG;IACH,2BAA2B,CAAC,EAAC,cAAc,EAAE,eAAe,EAAE,UAAU,EAAC;QACvE,IAAI,IAAI,CAAC,6BAA6B,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC;YAC5D,OAAO,kCAAkC,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC,CAAA;QACrG,CAAC;QAED,kCAAkC;QAClC,IAAI,gBAAgB,CAAA;QAEpB,IAAI,OAAO,eAAe,KAAK,UAAU,EAAE,CAAC;YAC1C,MAAM,WAAW,GAAG,0CAA0C,CAAC,CAAC,eAAe,CAAC,CAAA;YAEhF,gBAAgB,GAAG,IAAI,WAAW,CAAC;gBACjC,cAAc;gBACd,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,UAAU;aACX,CAAC,CAAA;QACJ,CAAC;aAAM,IAAI,eAAe,IAAI,OAAO,eAAe,KAAK,QAAQ,EAAE,CAAC;YAClE,gBAAgB,GAAG,eAAe,CAAA;QACpC,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,2CAA2C,UAAU,CAAC,IAAI,IAAI,cAAc,EAAE,CAAC,CAAA;QACjG,CAAC;QAED,IAAI,OAAO,gBAAgB,CAAC,KAAK,KAAK,UAAU,IAAI,OAAO,gBAAgB,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;YAChG,MAAM,IAAI,KAAK,CAAC,yBAAyB,UAAU,CAAC,IAAI,IAAI,cAAc,4BAA4B,CAAC,CAAA;QACzG,CAAC;QAED,IAAI,CAAC,6BAA6B,CAAC,GAAG,CAAC,eAAe,EAAE,gBAAgB,CAAC,CAAA;QAEzE,OAAO,gBAAgB,CAAA;IACzB,CAAC;IAED;;;;;OAKG;IACH,wBAAwB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAC;QACpC,MAAM,oBAAoB,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAA;QAC5E,MAAM,gBAAgB,GAAG,oBAAoB,CAAC,MAAM,CAAA;QAEpD,IAAI,OAAO,gBAAgB,KAAK,QAAQ,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxE,OAAO,gBAAgB,CAAA;QACzB,CAAC;QAED,IAAI,OAAO,gBAAgB,KAAK,UAAU,EAAE,CAAC;YAC3C,OAAO,gBAAgB,CAAC,IAAI,IAAI,QAAQ,CAAA;QAC1C,CAAC;QAED,IAAI,gBAAgB,IAAI,OAAO,gBAAgB,KAAK,QAAQ,EAAE,CAAC;YAC7D,MAAM,eAAe,GAAG,gBAAgB,CAAC,WAAW,EAAE,IAAI,CAAA;YAE1D,IAAI,OAAO,eAAe,KAAK,QAAQ,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,IAAI,eAAe,KAAK,QAAQ,EAAE,CAAC;gBACtG,OAAO,eAAe,CAAA;YACxB,CAAC;YAED,OAAO,QAAQ,CAAA;QACjB,CAAC;QAED,MAAM,wBAAwB,GAAG,IAAI,CAAC,aAAa,CAAC,2BAA2B,EAAE,EAAE,IAAI,EAAE,CAAA;QACzF,MAAM,aAAa,GAAG,wBAAwB,CAAC,aAAa,CAAA;QAE5D,IAAI,OAAO,aAAa,KAAK,QAAQ,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClE,OAAO,aAAa,CAAA;QACtB,CAAC;QAED,OAAO,YAAY,CAAA;IACrB,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,uBAAuB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAC;QAC9C,MAAM,oBAAoB,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAA;QAC5E,MAAM,gBAAgB,GAAG,oBAAoB,CAAC,MAAM,CAAA;QAEpD,IAAI,OAAO,gBAAgB,KAAK,UAAU,IAAI,CAAC,gBAAgB,IAAI,OAAO,gBAAgB,KAAK,QAAQ,CAAC,EAAE,CAAC;YACzG,OAAO,IAAI,CAAC,2BAA2B,CAAC;gBACtC,cAAc,EAAE,IAAI;gBACpB,eAAe,EAAE,gBAAgB;gBACjC,UAAU,EAAE,KAAK,CAAC,aAAa,EAAE;aAClC,CAAC,CAAA;QACJ,CAAC;QAED,MAAM,kBAAkB,GAAG,OAAO,GAAG,EAAE,MAAM,KAAK,QAAQ,IAAI,GAAG,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC;YACjF,CAAC,CAAC,GAAG,CAAC,MAAM;YACZ,CAAC,CAAC,IAAI,CAAC,wBAAwB,CAAC,EAAC,KAAK,EAAE,IAAI,EAAC,CAAC,CAAA;QAEhD,OAAO,MAAM,IAAI,CAAC,sBAAsB,CAAC,kBAAkB,CAAC,CAAA;IAC9D,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,aAAa,CAAC,EAAC,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,UAAU,EAAC;QAClD,MAAM,KAAK,GAAG,EAAE;aACb,QAAQ,EAAE;aACV,IAAI,CAAC,iBAAiB,CAAC;aACvB,KAAK,CAAC,EAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAC,CAAC;aAC3D,KAAK,CAAC,eAAe,CAAC;aACtB,KAAK,CAAC,CAAC,CAAC,CAAA;QACX,MAAM,IAAI,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE,CAAA;QAClC,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAA;QAEzC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,OAAO,CAAC,CAAA;QAEvC,OAAO,OAAO,GAAG,CAAC,CAAA;IACpB,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,OAAO,CAAC,QAAQ;QACpB,MAAM,IAAI,GAAG,IAAI,CAAC,aAAa,CAAC,eAAe,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAA;QACxE,4BAA4B;QAC5B,IAAI,MAAM,CAAA;QAEV,MAAM,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YACrC,MAAM,GAAG,MAAM,QAAQ,CAAC,EAAE,CAAC,CAAA;QAC7B,CAAC,CAAC,CAAA;QAEF,OAAO,gBAAgB,CAAC,CAAC,MAAM,CAAC,CAAA;IAClC,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport UUID from \"pure-uuid\"\nimport TableData from \"../../table-data/index.js\"\nimport NativeAttachmentStorageDriver from \"./storage-drivers/native.js\"\nimport normalizeRecordAttachmentInput from \"./normalize-input.js\"\nimport S3AttachmentStorageDriver from \"./storage-drivers/s3.js\"\n\nconst ATTACHMENTS_TABLE = \"velocious_attachments\"\n\n/** @typedef {new (...args: any[]) => Record<string, any>} AttachmentDriverConstructor */\n\n/** @type {WeakMap<import(\"../../../configuration.js\").default, Map<string, RecordAttachmentsStore>>} */\nconst storesByConfiguration = new WeakMap()\n/** @type {Promise<any> | undefined} */\nlet filesystemAttachmentStorageDriverClassPromise\n\n/**\n * @param {string} specifier - Module specifier.\n * @returns {Promise<any>} - Imported module.\n */\nasync function dynamicImport(specifier) {\n  const importer = /** @type {(moduleSpecifier: string) => Promise<any>} */ (\n    new Function(\"moduleSpecifier\", \"return import(moduleSpecifier)\")\n  )\n\n  return await importer(specifier)\n}\n\n/**\n * @returns {Promise<any>} - Filesystem driver class.\n */\nasync function filesystemAttachmentStorageDriverClass() {\n  filesystemAttachmentStorageDriverClassPromise ||= dynamicImport(\"./storage-drivers/filesystem.js\")\n    .then((module) => module.default)\n\n  return await filesystemAttachmentStorageDriverClassPromise\n}\n\n/**\n * @returns {string} - Generated UUID v4 value.\n */\nfunction generateUUID() {\n  return new UUID(4).format()\n}\n\n/**\n * @param {import(\"../index.js\").default} model - Model instance.\n * @returns {string} - Store key.\n */\nfunction storeKeyForModel(model) {\n  return `${model.getModelClass().getDatabaseIdentifier()}`\n}\n\n/**\n * @param {import(\"../index.js\").default} model - Model instance.\n * @returns {RecordAttachmentsStore} - Store instance.\n */\nexport function recordAttachmentsStoreForModel(model) {\n  const configuration = model._getConfiguration()\n  let storesByDatabaseIdentifier = storesByConfiguration.get(configuration)\n\n  if (!storesByDatabaseIdentifier) {\n    storesByDatabaseIdentifier = new Map()\n    storesByConfiguration.set(configuration, storesByDatabaseIdentifier)\n  }\n\n  const key = storeKeyForModel(model)\n  let store = storesByDatabaseIdentifier.get(key)\n\n  if (store) return store\n\n  store = new RecordAttachmentsStore({\n    configuration,\n    databaseIdentifier: model.getModelClass().getDatabaseIdentifier()\n  })\n\n  storesByDatabaseIdentifier.set(key, store)\n\n  return store\n}\n\n/**\n * Attachment persistence store.\n */\nexport default class RecordAttachmentsStore {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} args.databaseIdentifier - Database identifier.\n   */\n  constructor({configuration, databaseIdentifier}) {\n    this.configuration = configuration\n    this.databaseIdentifier = databaseIdentifier\n    this._readyPromise = null\n    this._driverColumnsAvailable = false\n    this._contentBase64Nullable = true\n    /** @type {Map<string, Record<string, any>>} */\n    this._attachmentDriversByName = new Map()\n    /** @type {Map<AttachmentDriverConstructor | Record<string, any>, Record<string, any>>} */\n    this._attachmentDriversByReference = new Map()\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when schema is ready.\n   */\n  async ensureReady() {\n    if (this._readyPromise) {\n      await this._readyPromise\n      return\n    }\n\n    this._readyPromise = (async () => {\n      await this._withDb(async (db) => {\n        if (await db.tableExists(ATTACHMENTS_TABLE)) {\n          await this.ensureAttachmentStoreSchema({db})\n          return\n        }\n\n        const table = new TableData(ATTACHMENTS_TABLE, {ifNotExists: true})\n\n        table.string(\"id\", {null: false, primaryKey: true})\n        table.string(\"record_type\", {null: false, index: true})\n        table.string(\"record_id\", {null: false, index: true})\n        table.string(\"name\", {null: false, index: true})\n        table.integer(\"position\", {null: false})\n        table.string(\"filename\", {null: false})\n        table.string(\"content_type\", {null: true})\n        table.bigint(\"byte_size\", {null: false})\n        table.string(\"driver\", {null: true})\n        table.string(\"storage_key\", {null: true})\n        table.text(\"content_base64\", {null: true})\n        table.bigint(\"created_at_ms\", {null: false})\n        table.bigint(\"updated_at_ms\", {null: false})\n\n        await db.createTable(table)\n        this._driverColumnsAvailable = true\n        this._contentBase64Nullable = true\n      })\n    })()\n\n    try {\n      await this._readyPromise\n    } finally {\n      this._readyPromise = null\n    }\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {unknown} args.input - Attachment input.\n   * @param {boolean} args.replace - Whether to replace existing attachments.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async attach({input, model, name, replace}) {\n    await this.ensureReady()\n    const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const allowPathInput = attachmentsConfiguration.allowPathInput === true\n    const allowedPathPrefixes = Array.isArray(attachmentsConfiguration.allowedPathPrefixes)\n      ? attachmentsConfiguration.allowedPathPrefixes\n      : undefined\n\n    const normalizedInput = await normalizeRecordAttachmentInput(input, {\n      allowPathInput,\n      allowedPathPrefixes\n    })\n    const attachmentDriver = await this.resolveAttachmentDriver({model, name})\n    const attachmentDriverName = this._attachmentDriverNameFor({model, name})\n    const now = Date.now()\n    const recordType = model.getModelClass().name\n    const recordId = String(model.id())\n    const attachmentId = generateUUID()\n    const {storageKey} = await attachmentDriver.write({\n      attachmentId,\n      input: normalizedInput,\n      model,\n      name\n    })\n\n    await this._withDb(async (db) => {\n      if (replace) {\n        const existingRows = await db\n          .newQuery()\n          .from(ATTACHMENTS_TABLE)\n          .where({name, record_id: recordId, record_type: recordType})\n          .results()\n\n        for (const existingRow of existingRows) {\n          await this.deleteAttachmentRowStorage({model, name, row: existingRow})\n        }\n\n        await db.delete({\n          conditions: {name, record_id: recordId, record_type: recordType},\n          tableName: ATTACHMENTS_TABLE\n        })\n      }\n\n      const position = replace ? 0 : await this._nextPosition({db, name, recordId, recordType})\n      /** @type {Record<string, any>} */\n      const insertData = {\n        byte_size: normalizedInput.byteSize,\n        content_base64: this._contentBase64Nullable ? null : normalizedInput.contentBase64,\n        content_type: normalizedInput.contentType,\n        created_at_ms: now,\n        filename: normalizedInput.filename,\n        id: attachmentId,\n        name,\n        position,\n        record_id: recordId,\n        record_type: recordType,\n        updated_at_ms: now\n      }\n\n      if (this._driverColumnsAvailable) {\n        insertData.driver = attachmentDriverName\n        insertData.storage_key = storageKey\n      }\n\n      await db.insert({\n        data: insertData,\n        tableName: ATTACHMENTS_TABLE\n      })\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../database/drivers/base.js\").default} args.db - DB connection.\n   * @returns {Promise<void>} - Resolves when schema columns are ensured.\n   */\n  async ensureAttachmentStoreSchema({db}) {\n    const table = await db.getTableByNameOrFail(ATTACHMENTS_TABLE)\n    const columns = await table.getColumns()\n    const hasDriverColumn = columns.some((column) => column.getName() === \"driver\")\n    const hasStorageKeyColumn = columns.some((column) => column.getName() === \"storage_key\")\n    const contentBase64Column = columns.find((column) => column.getName() === \"content_base64\")\n    const alterTable = new TableData(ATTACHMENTS_TABLE)\n    let shouldAlter = false\n\n    if (!hasDriverColumn) {\n      alterTable.string(\"driver\", {null: true})\n      shouldAlter = true\n    }\n\n    if (!hasStorageKeyColumn) {\n      alterTable.string(\"storage_key\", {null: true})\n      shouldAlter = true\n    }\n\n    if (shouldAlter) {\n      const alterTableSQLs = await db.alterTableSQLs(alterTable)\n\n      for (const sql of alterTableSQLs) {\n        await db.query(sql)\n      }\n    }\n\n    this._driverColumnsAvailable = true\n    this._contentBase64Nullable = contentBase64Column ? contentBase64Column.getNull() : true\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<Buffer>} - Attachment bytes.\n   */\n  async readAttachmentRow({model, name, row}) {\n    if (typeof row.content_base64 === \"string\" && row.content_base64.length > 0) {\n      return Buffer.from(row.content_base64, \"base64\")\n    }\n\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0 ? row.storage_key : null\n\n    if (!storageKey) {\n      throw new Error(`Attachment row ${String(row.id)} is missing storage key`)\n    }\n\n    const attachmentDriver = await this.resolveAttachmentDriver({model, name, row})\n\n    return await attachmentDriver.read({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<string | null>} - Attachment URL.\n   */\n  async attachmentRowUrl({model, name, row}) {\n    const attachmentDriver = await this.resolveAttachmentDriver({model, name, row})\n\n    if (typeof attachmentDriver.url !== \"function\") return null\n\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0\n      ? row.storage_key\n      : (typeof row.id === \"string\" ? row.id : \"\")\n\n    if (!storageKey) return null\n\n    return await attachmentDriver.url({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {string} [args.id] - Optional attachment id.\n   * @returns {Promise<Record<string, any> | null>} - Attachment row.\n   */\n  async findOne({id, model, name}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const recordType = model.getModelClass().name\n      const recordId = String(model.id())\n      let query = db\n        .newQuery()\n        .from(ATTACHMENTS_TABLE)\n        .where({name, record_id: recordId, record_type: recordType})\n        .order(\"position ASC\")\n        .order(\"created_at_ms DESC\")\n        .limit(1)\n\n      if (id) {\n        query = query.where({id})\n      }\n\n      const rows = await query.results()\n\n      return rows[0] || null\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @returns {Promise<Array<Record<string, any>>>} - Attachment rows.\n   */\n  async findMany({model, name}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const recordType = model.getModelClass().name\n      const recordId = String(model.id())\n      const query = db\n        .newQuery()\n        .from(ATTACHMENTS_TABLE)\n        .where({name, record_id: recordId, record_type: recordType})\n        .order(\"position ASC\")\n        .order(\"created_at_ms ASC\")\n\n      return await query.results()\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} args.row - Attachment row.\n   * @returns {Promise<void>} - Resolves when row storage has been deleted.\n   */\n  async deleteAttachmentRowStorage({model, name, row}) {\n    const storageKey = typeof row.storage_key === \"string\" && row.storage_key.length > 0 ? row.storage_key : null\n\n    if (!storageKey) return\n\n    const attachmentDriver = await this.resolveAttachmentDriver({model, name, row})\n\n    if (typeof attachmentDriver.delete !== \"function\") return\n\n    await attachmentDriver.delete({\n      model,\n      name,\n      row,\n      storageKey\n    })\n  }\n\n  /**\n   * @param {string} driverName - Driver name.\n   * @returns {Promise<Record<string, any>>} - Attachment storage driver instance.\n   */\n  async attachmentDriverByName(driverName) {\n    if (this._attachmentDriversByName.has(driverName)) {\n      return /** @type {Record<string, any>} */ (this._attachmentDriversByName.get(driverName))\n    }\n\n    const attachmentConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const configuredDriver = attachmentConfiguration.drivers?.[driverName] || {}\n    /** @type {Record<string, any>} */\n    let attachmentDriver\n\n    if (configuredDriver.instance && typeof configuredDriver.instance === \"object\") {\n      attachmentDriver = configuredDriver.instance\n    } else if (typeof configuredDriver.driverClass === \"function\") {\n      attachmentDriver = new configuredDriver.driverClass({\n        configuration: this.configuration,\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (typeof configuredDriver.create === \"function\") {\n      attachmentDriver = configuredDriver.create({\n        configuration: this.configuration,\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (driverName === \"filesystem\") {\n      const FilesystemAttachmentStorageDriver = await filesystemAttachmentStorageDriverClass()\n\n      attachmentDriver = new FilesystemAttachmentStorageDriver({\n        configuration: this.configuration,\n        options: configuredDriver\n      })\n    } else if (driverName === \"native\") {\n      attachmentDriver = new NativeAttachmentStorageDriver({\n        name: driverName,\n        options: configuredDriver\n      })\n    } else if (driverName === \"s3\") {\n      attachmentDriver = new S3AttachmentStorageDriver({options: configuredDriver})\n    } else {\n      throw new Error(`Unknown attachment storage driver: ${driverName}`)\n    }\n\n    if (!attachmentDriver || typeof attachmentDriver.write !== \"function\" || typeof attachmentDriver.read !== \"function\") {\n      throw new Error(`Attachment storage driver \"${driverName}\" must implement write/read`)\n    }\n\n    this._attachmentDriversByName.set(driverName, attachmentDriver)\n\n    return attachmentDriver\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {AttachmentDriverConstructor | Record<string, any>} args.driverReference - Driver class or instance.\n   * @param {string} args.attachmentName - Attachment name.\n   * @param {typeof import(\"../index.js\").default} args.modelClass - Model class.\n   * @returns {Record<string, any>} - Attachment driver instance.\n   */\n  attachmentDriverByReference({attachmentName, driverReference, modelClass}) {\n    if (this._attachmentDriversByReference.has(driverReference)) {\n      return /** @type {Record<string, any>} */ (this._attachmentDriversByReference.get(driverReference))\n    }\n\n    /** @type {Record<string, any>} */\n    let attachmentDriver\n\n    if (typeof driverReference === \"function\") {\n      const DriverClass = /** @type {AttachmentDriverConstructor} */ (driverReference)\n\n      attachmentDriver = new DriverClass({\n        attachmentName,\n        configuration: this.configuration,\n        modelClass\n      })\n    } else if (driverReference && typeof driverReference === \"object\") {\n      attachmentDriver = driverReference\n    } else {\n      throw new Error(`Invalid attachment driver reference for ${modelClass.name}#${attachmentName}`)\n    }\n\n    if (typeof attachmentDriver.write !== \"function\" || typeof attachmentDriver.read !== \"function\") {\n      throw new Error(`Attachment driver for ${modelClass.name}#${attachmentName} must implement write/read`)\n    }\n\n    this._attachmentDriversByReference.set(driverReference, attachmentDriver)\n\n    return attachmentDriver\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @returns {string} - Attachment driver name.\n   */\n  _attachmentDriverNameFor({model, name}) {\n    const attachmentDefinition = model.getModelClass().getAttachmentByName(name)\n    const configuredDriver = attachmentDefinition.driver\n\n    if (typeof configuredDriver === \"string\" && configuredDriver.length > 0) {\n      return configuredDriver\n    }\n\n    if (typeof configuredDriver === \"function\") {\n      return configuredDriver.name || \"custom\"\n    }\n\n    if (configuredDriver && typeof configuredDriver === \"object\") {\n      const constructorName = configuredDriver.constructor?.name\n\n      if (typeof constructorName === \"string\" && constructorName.length > 0 && constructorName !== \"Object\") {\n        return constructorName\n      }\n\n      return \"custom\"\n    }\n\n    const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const defaultDriver = attachmentsConfiguration.defaultDriver\n\n    if (typeof defaultDriver === \"string\" && defaultDriver.length > 0) {\n      return defaultDriver\n    }\n\n    return \"filesystem\"\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../index.js\").default} args.model - Model instance.\n   * @param {string} args.name - Attachment name.\n   * @param {Record<string, any>} [args.row] - Attachment row.\n   * @returns {Promise<Record<string, any>>} - Attachment storage driver instance.\n   */\n  async resolveAttachmentDriver({model, name, row}) {\n    const attachmentDefinition = model.getModelClass().getAttachmentByName(name)\n    const configuredDriver = attachmentDefinition.driver\n\n    if (typeof configuredDriver === \"function\" || (configuredDriver && typeof configuredDriver === \"object\")) {\n      return this.attachmentDriverByReference({\n        attachmentName: name,\n        driverReference: configuredDriver,\n        modelClass: model.getModelClass()\n      })\n    }\n\n    const fallbackDriverName = typeof row?.driver === \"string\" && row.driver.length > 0\n      ? row.driver\n      : this._attachmentDriverNameFor({model, name})\n\n    return await this.attachmentDriverByName(fallbackDriverName)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../../../database/drivers/base.js\").default} args.db - DB connection.\n   * @param {string} args.name - Attachment name.\n   * @param {string} args.recordId - Record id.\n   * @param {string} args.recordType - Record type.\n   * @returns {Promise<number>} - Next position.\n   */\n  async _nextPosition({db, name, recordId, recordType}) {\n    const query = db\n      .newQuery()\n      .from(ATTACHMENTS_TABLE)\n      .where({name, record_id: recordId, record_type: recordType})\n      .order(\"position DESC\")\n      .limit(1)\n    const rows = await query.results()\n    const current = Number(rows[0]?.position)\n\n    if (!Number.isFinite(current)) return 0\n\n    return current + 1\n  }\n\n  /**\n   * @template T\n   * @param {(db: import(\"../../../database/drivers/base.js\").default) => Promise<T>} callback - Callback.\n   * @returns {Promise<T>} - Callback result.\n   */\n  async _withDb(callback) {\n    const pool = this.configuration.getDatabasePool(this.databaseIdentifier)\n    /** @type {T | undefined} */\n    let result\n\n    await pool.withConnection(async (db) => {\n      result = await callback(db)\n    })\n\n    return /** @type {T} */ (result)\n  }\n}\n"]}
|