velocious 1.0.239 → 1.0.240

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.
@@ -12,12 +12,14 @@
12
12
  * @param {boolean} [args.allowPathInput] - Whether `{path: ...}` input is allowed.
13
13
  * @param {string[]} [args.allowedPathPrefixes] - Optional allowlist for path input.
14
14
  * @param {string} [args.defaultFilename] - Optional default filename.
15
+ * @param {import("../../../environment-handlers/base.js").default} [args.environmentHandler] - Optional environment handler for Node-only file operations.
15
16
  * @returns {Promise<NormalizedAttachmentInput>} - Normalized attachment input.
16
17
  */
17
18
  export default function normalizeRecordAttachmentInput(input: unknown, args?: {
18
19
  allowPathInput?: boolean;
19
20
  allowedPathPrefixes?: string[];
20
21
  defaultFilename?: string;
22
+ environmentHandler?: import("../../../environment-handlers/base.js").default;
21
23
  }): Promise<NormalizedAttachmentInput>;
22
24
  export type NormalizedAttachmentInput = {
23
25
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"normalize-input.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/normalize-input.js"],"names":[],"mappings":"AA6FA;;;;;;;GAOG;AAEH;;;;;;;GAOG;AACH,8DAPW,OAAO,SAEf;IAAuB,cAAc,GAA7B,OAAO;IACS,mBAAmB,GAAnC,MAAM,EAAE;IACM,eAAe,GAA7B,MAAM;CACd,GAAU,OAAO,CAAC,yBAAyB,CAAC,CAkE9C;;;;;cA/Ea,MAAM;;;;mBACN,MAAM;;;;mBACN,MAAM;;;;iBACN,MAAM,GAAG,IAAI;;;;cACb,MAAM"}
1
+ {"version":3,"file":"normalize-input.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/normalize-input.js"],"names":[],"mappings":"AA2FA;;;;;;;GAOG;AAEH;;;;;;;;GAQG;AACH,8DARW,OAAO,SAEf;IAAuB,cAAc,GAA7B,OAAO;IACS,mBAAmB,GAAnC,MAAM,EAAE;IACM,eAAe,GAA7B,MAAM;IACyD,kBAAkB,GAAjF,OAAO,uCAAuC,EAAE,OAAO;CAC/D,GAAU,OAAO,CAAC,yBAAyB,CAAC,CAwE9C;;;;;cAtFa,MAAM;;;;mBACN,MAAM;;;;mBACN,MAAM;;;;iBACN,MAAM,GAAG,IAAI;;;;cACb,MAAM"}
@@ -1,7 +1,17 @@
1
1
  // @ts-check
2
2
  import UploadedFile from "../../../http-server/client/uploaded-file/uploaded-file.js";
3
- import fs from "fs/promises";
4
- import path from "path";
3
+ /**
4
+ * @param {string} value - Path-like value.
5
+ * @returns {string} - Basename-like filename.
6
+ */
7
+ function baseName(value) {
8
+ const withoutTrailingSeparators = value.replace(/[\\/]+$/, "");
9
+ if (!withoutTrailingSeparators)
10
+ return "";
11
+ const normalized = withoutTrailingSeparators.replaceAll("\\", "/");
12
+ const parts = normalized.split("/");
13
+ return parts[parts.length - 1] || "";
14
+ }
5
15
  /**
6
16
  * @param {unknown} value - Candidate value.
7
17
  * @returns {value is Record<string, any>} - Whether value is a plain object.
@@ -33,25 +43,6 @@ function isArrayBuffer(value) {
33
43
  function isArrayBufferLike(value) {
34
44
  return Boolean(value && typeof value === "object" && typeof /** @type {any} */ (value).arrayBuffer === "function");
35
45
  }
36
- /**
37
- * @param {string} filePath - File path.
38
- * @param {string[]} allowedPathPrefixes - Allowed path prefixes.
39
- * @returns {boolean} - Whether path is allowed.
40
- */
41
- function pathWithinAllowedPrefixes(filePath, allowedPathPrefixes) {
42
- const resolvedPath = path.resolve(filePath);
43
- return allowedPathPrefixes.some((allowedPrefix) => {
44
- const resolvedPrefix = path.resolve(allowedPrefix);
45
- const relativePath = path.relative(resolvedPrefix, resolvedPath);
46
- if (!relativePath)
47
- return true;
48
- if (relativePath.startsWith(".."))
49
- return false;
50
- if (path.isAbsolute(relativePath))
51
- return false;
52
- return true;
53
- });
54
- }
55
46
  /**
56
47
  * @param {Uint8Array | Buffer | ArrayBuffer | string} value - Value.
57
48
  * @returns {Buffer} - Buffer value.
@@ -69,15 +60,19 @@ function toBuffer(value) {
69
60
  }
70
61
  /**
71
62
  * @param {UploadedFile} uploadedFile - Uploaded file.
63
+ * @param {import("../../../environment-handlers/base.js").default | undefined} environmentHandler - Environment handler.
72
64
  * @returns {Promise<Buffer>} - File content buffer.
73
65
  */
74
- async function uploadedFileBuffer(uploadedFile) {
66
+ async function uploadedFileBuffer(uploadedFile, environmentHandler) {
75
67
  const memoryBuffer = /** @type {{getBuffer?: () => Buffer}} */ (uploadedFile).getBuffer?.();
76
68
  if (Buffer.isBuffer(memoryBuffer))
77
69
  return memoryBuffer;
78
70
  const tempPath = /** @type {{getPath?: () => string}} */ (uploadedFile).getPath?.();
79
71
  if (typeof tempPath === "string" && tempPath.length > 0) {
80
- return await fs.readFile(tempPath);
72
+ if (!environmentHandler || typeof environmentHandler.readAttachmentInputFile !== "function") {
73
+ throw new Error("Attachment temp-path input is unsupported in this environment");
74
+ }
75
+ return await environmentHandler.readAttachmentInputFile(tempPath);
81
76
  }
82
77
  throw new Error("Unsupported uploaded file type");
83
78
  }
@@ -95,10 +90,12 @@ async function uploadedFileBuffer(uploadedFile) {
95
90
  * @param {boolean} [args.allowPathInput] - Whether `{path: ...}` input is allowed.
96
91
  * @param {string[]} [args.allowedPathPrefixes] - Optional allowlist for path input.
97
92
  * @param {string} [args.defaultFilename] - Optional default filename.
93
+ * @param {import("../../../environment-handlers/base.js").default} [args.environmentHandler] - Optional environment handler for Node-only file operations.
98
94
  * @returns {Promise<NormalizedAttachmentInput>} - Normalized attachment input.
99
95
  */
100
96
  export default async function normalizeRecordAttachmentInput(input, args = {}) {
101
97
  const defaultFilename = args.defaultFilename || "attachment.bin";
98
+ const environmentHandler = args.environmentHandler;
102
99
  /** @type {Buffer} */
103
100
  let buffer;
104
101
  /** @type {string | null} */
@@ -106,7 +103,7 @@ export default async function normalizeRecordAttachmentInput(input, args = {}) {
106
103
  /** @type {string | undefined} */
107
104
  let filename;
108
105
  if (input instanceof UploadedFile) {
109
- buffer = await uploadedFileBuffer(input);
106
+ buffer = await uploadedFileBuffer(input, environmentHandler);
110
107
  filename = input.filename();
111
108
  contentType = input.contentType() || null;
112
109
  }
@@ -114,15 +111,18 @@ export default async function normalizeRecordAttachmentInput(input, args = {}) {
114
111
  if (args.allowPathInput !== true) {
115
112
  throw new Error("Attachment path input is disabled");
116
113
  }
117
- const filePath = path.resolve(input.path);
114
+ if (!environmentHandler || typeof environmentHandler.resolveAttachmentInputPath !== "function") {
115
+ throw new Error("Attachment path input is unsupported in this environment");
116
+ }
118
117
  const allowedPathPrefixes = Array.isArray(args.allowedPathPrefixes)
119
118
  ? args.allowedPathPrefixes.filter((entry) => typeof entry === "string" && entry.length > 0)
120
119
  : [];
121
- if (allowedPathPrefixes.length > 0 && !pathWithinAllowedPrefixes(filePath, allowedPathPrefixes)) {
122
- throw new Error("Attachment path is outside allowed directories");
123
- }
124
- buffer = await fs.readFile(filePath);
125
- filename = typeof input.filename === "string" && input.filename.length > 0 ? input.filename : path.basename(filePath);
120
+ const { buffer: fileBuffer, filePath } = await environmentHandler.resolveAttachmentInputPath({
121
+ allowedPathPrefixes,
122
+ inputPath: input.path
123
+ });
124
+ buffer = fileBuffer;
125
+ filename = typeof input.filename === "string" && input.filename.length > 0 ? input.filename : baseName(filePath);
126
126
  contentType = typeof input.contentType === "string" && input.contentType.length > 0 ? input.contentType : null;
127
127
  }
128
128
  else if (isPlainObject(input) && typeof input.contentBase64 === "string") {
@@ -152,13 +152,15 @@ export default async function normalizeRecordAttachmentInput(input, args = {}) {
152
152
  else {
153
153
  throw new Error("Unsupported attachment input");
154
154
  }
155
- const normalizedFilename = typeof filename === "string" && filename.length > 0 ? path.basename(filename) : defaultFilename;
155
+ const normalizedFilename = typeof filename === "string" && filename.length > 0
156
+ ? baseName(filename)
157
+ : "";
156
158
  return {
157
159
  byteSize: buffer.length,
158
160
  contentBuffer: buffer,
159
161
  contentBase64: buffer.toString("base64"),
160
162
  contentType,
161
- filename: normalizedFilename
163
+ filename: normalizedFilename || defaultFilename
162
164
  };
163
165
  }
164
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"normalize-input.js","sourceRoot":"","sources":["../../../../../src/database/record/attachments/normalize-input.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport UploadedFile from \"../../../http-server/client/uploaded-file/uploaded-file.js\"\nimport fs from \"fs/promises\"\nimport path from \"path\"\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Record<string, any>} - Whether value is a plain object.\n */\nfunction isPlainObject(value) {\n  if (!value || typeof value !== \"object\" || Array.isArray(value)) return false\n\n  const prototype = Object.getPrototypeOf(value)\n\n  return prototype === Object.prototype || prototype === null\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Uint8Array} - Whether value is a byte array.\n */\nfunction isUint8Array(value) {\n  return value instanceof Uint8Array\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is ArrayBuffer} - Whether value is array buffer.\n */\nfunction isArrayBuffer(value) {\n  return value instanceof ArrayBuffer\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is {arrayBuffer: () => Promise<ArrayBuffer>}} - Whether value supports arrayBuffer().\n */\nfunction isArrayBufferLike(value) {\n  return Boolean(value && typeof value === \"object\" && typeof /** @type {any} */ (value).arrayBuffer === \"function\")\n}\n\n/**\n * @param {string} filePath - File path.\n * @param {string[]} allowedPathPrefixes - Allowed path prefixes.\n * @returns {boolean} - Whether path is allowed.\n */\nfunction pathWithinAllowedPrefixes(filePath, allowedPathPrefixes) {\n  const resolvedPath = path.resolve(filePath)\n\n  return allowedPathPrefixes.some((allowedPrefix) => {\n    const resolvedPrefix = path.resolve(allowedPrefix)\n    const relativePath = path.relative(resolvedPrefix, resolvedPath)\n\n    if (!relativePath) return true\n    if (relativePath.startsWith(\"..\")) return false\n    if (path.isAbsolute(relativePath)) return false\n\n    return true\n  })\n}\n\n/**\n * @param {Uint8Array | Buffer | ArrayBuffer | string} value - Value.\n * @returns {Buffer} - Buffer value.\n */\nfunction toBuffer(value) {\n  if (Buffer.isBuffer(value)) return value\n  if (typeof value === \"string\") return Buffer.from(value)\n  if (isArrayBuffer(value)) return Buffer.from(value)\n  if (isUint8Array(value)) return Buffer.from(value)\n\n  throw new Error(\"Unsupported attachment content type\")\n}\n\n/**\n * @param {UploadedFile} uploadedFile - Uploaded file.\n * @returns {Promise<Buffer>} - File content buffer.\n */\nasync function uploadedFileBuffer(uploadedFile) {\n  const memoryBuffer = /** @type {{getBuffer?: () => Buffer}} */ (uploadedFile).getBuffer?.()\n\n  if (Buffer.isBuffer(memoryBuffer)) return memoryBuffer\n\n  const tempPath = /** @type {{getPath?: () => string}} */ (uploadedFile).getPath?.()\n\n  if (typeof tempPath === \"string\" && tempPath.length > 0) {\n    return await fs.readFile(tempPath)\n  }\n\n  throw new Error(\"Unsupported uploaded file type\")\n}\n\n/**\n * @typedef {object} NormalizedAttachmentInput\n * @property {number} byteSize - File size in bytes.\n * @property {Buffer} contentBuffer - Raw content bytes.\n * @property {string} contentBase64 - Base64 encoded content.\n * @property {string | null} contentType - Content type.\n * @property {string} filename - Filename.\n */\n\n/**\n * @param {unknown} input - Attachment input.\n * @param {object} [args] - Options.\n * @param {boolean} [args.allowPathInput] - Whether `{path: ...}` input is allowed.\n * @param {string[]} [args.allowedPathPrefixes] - Optional allowlist for path input.\n * @param {string} [args.defaultFilename] - Optional default filename.\n * @returns {Promise<NormalizedAttachmentInput>} - Normalized attachment input.\n */\nexport default async function normalizeRecordAttachmentInput(input, args = {}) {\n  const defaultFilename = args.defaultFilename || \"attachment.bin\"\n  /** @type {Buffer} */\n  let buffer\n  /** @type {string | null} */\n  let contentType = null\n  /** @type {string | undefined} */\n  let filename\n\n  if (input instanceof UploadedFile) {\n    buffer = await uploadedFileBuffer(input)\n    filename = input.filename()\n    contentType = input.contentType() || null\n  } else if (isPlainObject(input) && typeof input.path === \"string\" && input.path.length > 0) {\n    if (args.allowPathInput !== true) {\n      throw new Error(\"Attachment path input is disabled\")\n    }\n\n    const filePath = path.resolve(input.path)\n    const allowedPathPrefixes = Array.isArray(args.allowedPathPrefixes)\n      ? args.allowedPathPrefixes.filter((entry) => typeof entry === \"string\" && entry.length > 0)\n      : []\n\n    if (allowedPathPrefixes.length > 0 && !pathWithinAllowedPrefixes(filePath, allowedPathPrefixes)) {\n      throw new Error(\"Attachment path is outside allowed directories\")\n    }\n\n    buffer = await fs.readFile(filePath)\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : path.basename(filePath)\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isPlainObject(input) && typeof input.contentBase64 === \"string\") {\n    buffer = Buffer.from(input.contentBase64, \"base64\")\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : defaultFilename\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isPlainObject(input) && \"content\" in input) {\n    buffer = toBuffer(input.content)\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : defaultFilename\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isArrayBufferLike(input)) {\n    const arrayBuffer = await input.arrayBuffer()\n\n    buffer = Buffer.from(arrayBuffer)\n    filename = typeof /** @type {any} */ (input).name === \"string\" && /** @type {any} */ (input).name.length > 0\n      ? /** @type {any} */ (input).name\n      : defaultFilename\n    contentType = typeof /** @type {any} */ (input).type === \"string\" && /** @type {any} */ (input).type.length > 0\n      ? /** @type {any} */ (input).type\n      : null\n  } else if (typeof input === \"string\" || Buffer.isBuffer(input) || isArrayBuffer(input) || isUint8Array(input)) {\n    buffer = toBuffer(input)\n    filename = defaultFilename\n  } else {\n    throw new Error(\"Unsupported attachment input\")\n  }\n\n  const normalizedFilename = typeof filename === \"string\" && filename.length > 0 ? path.basename(filename) : defaultFilename\n\n  return {\n    byteSize: buffer.length,\n    contentBuffer: buffer,\n    contentBase64: buffer.toString(\"base64\"),\n    contentType,\n    filename: normalizedFilename\n  }\n}\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"normalize-input.js","sourceRoot":"","sources":["../../../../../src/database/record/attachments/normalize-input.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport UploadedFile from \"../../../http-server/client/uploaded-file/uploaded-file.js\"\n\n/**\n * @param {string} value - Path-like value.\n * @returns {string} - Basename-like filename.\n */\nfunction baseName(value) {\n  const withoutTrailingSeparators = value.replace(/[\\\\/]+$/, \"\")\n\n  if (!withoutTrailingSeparators) return \"\"\n\n  const normalized = withoutTrailingSeparators.replaceAll(\"\\\\\", \"/\")\n  const parts = normalized.split(\"/\")\n\n  return parts[parts.length - 1] || \"\"\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Record<string, any>} - Whether value is a plain object.\n */\nfunction isPlainObject(value) {\n  if (!value || typeof value !== \"object\" || Array.isArray(value)) return false\n\n  const prototype = Object.getPrototypeOf(value)\n\n  return prototype === Object.prototype || prototype === null\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Uint8Array} - Whether value is a byte array.\n */\nfunction isUint8Array(value) {\n  return value instanceof Uint8Array\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is ArrayBuffer} - Whether value is array buffer.\n */\nfunction isArrayBuffer(value) {\n  return value instanceof ArrayBuffer\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is {arrayBuffer: () => Promise<ArrayBuffer>}} - Whether value supports arrayBuffer().\n */\nfunction isArrayBufferLike(value) {\n  return Boolean(value && typeof value === \"object\" && typeof /** @type {any} */ (value).arrayBuffer === \"function\")\n}\n\n/**\n * @param {Uint8Array | Buffer | ArrayBuffer | string} value - Value.\n * @returns {Buffer} - Buffer value.\n */\nfunction toBuffer(value) {\n  if (Buffer.isBuffer(value)) return value\n  if (typeof value === \"string\") return Buffer.from(value)\n  if (isArrayBuffer(value)) return Buffer.from(value)\n  if (isUint8Array(value)) return Buffer.from(value)\n\n  throw new Error(\"Unsupported attachment content type\")\n}\n\n/**\n * @param {UploadedFile} uploadedFile - Uploaded file.\n * @param {import(\"../../../environment-handlers/base.js\").default | undefined} environmentHandler - Environment handler.\n * @returns {Promise<Buffer>} - File content buffer.\n */\nasync function uploadedFileBuffer(uploadedFile, environmentHandler) {\n  const memoryBuffer = /** @type {{getBuffer?: () => Buffer}} */ (uploadedFile).getBuffer?.()\n\n  if (Buffer.isBuffer(memoryBuffer)) return memoryBuffer\n\n  const tempPath = /** @type {{getPath?: () => string}} */ (uploadedFile).getPath?.()\n\n  if (typeof tempPath === \"string\" && tempPath.length > 0) {\n    if (!environmentHandler || typeof environmentHandler.readAttachmentInputFile !== \"function\") {\n      throw new Error(\"Attachment temp-path input is unsupported in this environment\")\n    }\n\n    return await environmentHandler.readAttachmentInputFile(tempPath)\n  }\n\n  throw new Error(\"Unsupported uploaded file type\")\n}\n\n/**\n * @typedef {object} NormalizedAttachmentInput\n * @property {number} byteSize - File size in bytes.\n * @property {Buffer} contentBuffer - Raw content bytes.\n * @property {string} contentBase64 - Base64 encoded content.\n * @property {string | null} contentType - Content type.\n * @property {string} filename - Filename.\n */\n\n/**\n * @param {unknown} input - Attachment input.\n * @param {object} [args] - Options.\n * @param {boolean} [args.allowPathInput] - Whether `{path: ...}` input is allowed.\n * @param {string[]} [args.allowedPathPrefixes] - Optional allowlist for path input.\n * @param {string} [args.defaultFilename] - Optional default filename.\n * @param {import(\"../../../environment-handlers/base.js\").default} [args.environmentHandler] - Optional environment handler for Node-only file operations.\n * @returns {Promise<NormalizedAttachmentInput>} - Normalized attachment input.\n */\nexport default async function normalizeRecordAttachmentInput(input, args = {}) {\n  const defaultFilename = args.defaultFilename || \"attachment.bin\"\n  const environmentHandler = args.environmentHandler\n  /** @type {Buffer} */\n  let buffer\n  /** @type {string | null} */\n  let contentType = null\n  /** @type {string | undefined} */\n  let filename\n\n  if (input instanceof UploadedFile) {\n    buffer = await uploadedFileBuffer(input, environmentHandler)\n    filename = input.filename()\n    contentType = input.contentType() || null\n  } else if (isPlainObject(input) && typeof input.path === \"string\" && input.path.length > 0) {\n    if (args.allowPathInput !== true) {\n      throw new Error(\"Attachment path input is disabled\")\n    }\n\n    if (!environmentHandler || typeof environmentHandler.resolveAttachmentInputPath !== \"function\") {\n      throw new Error(\"Attachment path input is unsupported in this environment\")\n    }\n\n    const allowedPathPrefixes = Array.isArray(args.allowedPathPrefixes)\n      ? args.allowedPathPrefixes.filter((entry) => typeof entry === \"string\" && entry.length > 0)\n      : []\n    const {buffer: fileBuffer, filePath} = await environmentHandler.resolveAttachmentInputPath({\n      allowedPathPrefixes,\n      inputPath: input.path\n    })\n\n    buffer = fileBuffer\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : baseName(filePath)\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isPlainObject(input) && typeof input.contentBase64 === \"string\") {\n    buffer = Buffer.from(input.contentBase64, \"base64\")\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : defaultFilename\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isPlainObject(input) && \"content\" in input) {\n    buffer = toBuffer(input.content)\n    filename = typeof input.filename === \"string\" && input.filename.length > 0 ? input.filename : defaultFilename\n    contentType = typeof input.contentType === \"string\" && input.contentType.length > 0 ? input.contentType : null\n  } else if (isArrayBufferLike(input)) {\n    const arrayBuffer = await input.arrayBuffer()\n\n    buffer = Buffer.from(arrayBuffer)\n    filename = typeof /** @type {any} */ (input).name === \"string\" && /** @type {any} */ (input).name.length > 0\n      ? /** @type {any} */ (input).name\n      : defaultFilename\n    contentType = typeof /** @type {any} */ (input).type === \"string\" && /** @type {any} */ (input).type.length > 0\n      ? /** @type {any} */ (input).type\n      : null\n  } else if (typeof input === \"string\" || Buffer.isBuffer(input) || isArrayBuffer(input) || isUint8Array(input)) {\n    buffer = toBuffer(input)\n    filename = defaultFilename\n  } else {\n    throw new Error(\"Unsupported attachment input\")\n  }\n\n  const normalizedFilename = typeof filename === \"string\" && filename.length > 0\n    ? baseName(filename)\n    : \"\"\n\n  return {\n    byteSize: buffer.length,\n    contentBuffer: buffer,\n    contentBase64: buffer.toString(\"base64\"),\n    contentType,\n    filename: normalizedFilename || defaultFilename\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/attachments/store.js"],"names":[],"mappings":"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"}
@@ -1,31 +1,11 @@
1
1
  // @ts-check
2
2
  import UUID from "pure-uuid";
3
3
  import TableData from "../../table-data/index.js";
4
- import NativeAttachmentStorageDriver from "./storage-drivers/native.js";
5
4
  import normalizeRecordAttachmentInput from "./normalize-input.js";
6
- import S3AttachmentStorageDriver from "./storage-drivers/s3.js";
7
5
  const ATTACHMENTS_TABLE = "velocious_attachments";
8
6
  /** @typedef {new (...args: any[]) => Record<string, any>} AttachmentDriverConstructor */
9
7
  /** @type {WeakMap<import("../../../configuration.js").default, Map<string, RecordAttachmentsStore>>} */
10
8
  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
9
  /**
30
10
  * @returns {string} - Generated UUID v4 value.
31
11
  */
@@ -138,7 +118,8 @@ export default class RecordAttachmentsStore {
138
118
  : undefined;
139
119
  const normalizedInput = await normalizeRecordAttachmentInput(input, {
140
120
  allowPathInput,
141
- allowedPathPrefixes
121
+ allowedPathPrefixes,
122
+ environmentHandler: this.configuration.getEnvironmentHandler()
142
123
  });
143
124
  const attachmentDriver = await this.resolveAttachmentDriver({ model, name });
144
125
  const attachmentDriverName = this._attachmentDriverNameFor({ model, name });
@@ -344,10 +325,13 @@ export default class RecordAttachmentsStore {
344
325
  return /** @type {Record<string, any>} */ (this._attachmentDriversByName.get(driverName));
345
326
  }
346
327
  const attachmentConfiguration = this.configuration.getAttachmentsConfiguration?.() || {};
347
- const configuredDriver = attachmentConfiguration.drivers?.[driverName] || {};
328
+ const configuredDriver = attachmentConfiguration.drivers?.[driverName];
348
329
  /** @type {Record<string, any>} */
349
330
  let attachmentDriver;
350
- if (configuredDriver.instance && typeof configuredDriver.instance === "object") {
331
+ if (!configuredDriver) {
332
+ throw new Error(`No configured attachment storage driver named "${driverName}"`);
333
+ }
334
+ else if (configuredDriver.instance && typeof configuredDriver.instance === "object") {
351
335
  attachmentDriver = configuredDriver.instance;
352
336
  }
353
337
  else if (typeof configuredDriver.driverClass === "function") {
@@ -364,24 +348,8 @@ export default class RecordAttachmentsStore {
364
348
  options: configuredDriver
365
349
  });
366
350
  }
367
- else if (driverName === "filesystem") {
368
- const FilesystemAttachmentStorageDriver = await filesystemAttachmentStorageDriverClass();
369
- attachmentDriver = new FilesystemAttachmentStorageDriver({
370
- configuration: this.configuration,
371
- options: configuredDriver
372
- });
373
- }
374
- else if (driverName === "native") {
375
- attachmentDriver = new NativeAttachmentStorageDriver({
376
- name: driverName,
377
- options: configuredDriver
378
- });
379
- }
380
- else if (driverName === "s3") {
381
- attachmentDriver = new S3AttachmentStorageDriver({ options: configuredDriver });
382
- }
383
351
  else {
384
- throw new Error(`Unknown attachment storage driver: ${driverName}`);
352
+ throw new Error(`Attachment storage driver "${driverName}" must define instance, driverClass, or create`);
385
353
  }
386
354
  if (!attachmentDriver || typeof attachmentDriver.write !== "function" || typeof attachmentDriver.read !== "function") {
387
355
  throw new Error(`Attachment storage driver "${driverName}" must implement write/read`);
@@ -431,6 +399,8 @@ export default class RecordAttachmentsStore {
431
399
  _attachmentDriverNameFor({ model, name }) {
432
400
  const attachmentDefinition = model.getModelClass().getAttachmentByName(name);
433
401
  const configuredDriver = attachmentDefinition.driver;
402
+ const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {};
403
+ const defaultDriver = attachmentsConfiguration.defaultDriver;
434
404
  if (typeof configuredDriver === "string" && configuredDriver.length > 0) {
435
405
  return configuredDriver;
436
406
  }
@@ -444,12 +414,10 @@ export default class RecordAttachmentsStore {
444
414
  }
445
415
  return "custom";
446
416
  }
447
- const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {};
448
- const defaultDriver = attachmentsConfiguration.defaultDriver;
449
417
  if (typeof defaultDriver === "string" && defaultDriver.length > 0) {
450
418
  return defaultDriver;
451
419
  }
452
- return "filesystem";
420
+ throw new Error(`No attachment driver configured for ${model.getModelClass().name}#${name}`);
453
421
  }
454
422
  /**
455
423
  * @param {object} args - Options.
@@ -509,4 +477,4 @@ export default class RecordAttachmentsStore {
509
477
  return /** @type {T} */ (result);
510
478
  }
511
479
  }
512
- //# 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 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"]}
480
+ //# 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 UUID from \"pure-uuid\"\nimport TableData from \"../../table-data/index.js\"\nimport normalizeRecordAttachmentInput from \"./normalize-input.js\"\n\nconst ATTACHMENTS_TABLE = \"velocious_attachments\"\n\n/** @typedef {new (...args: any[]) => Record<string, any>} AttachmentDriverConstructor */\n/** @type {WeakMap<import(\"../../../configuration.js\").default, Map<string, RecordAttachmentsStore>>} */\nconst storesByConfiguration = new WeakMap()\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      environmentHandler: this.configuration.getEnvironmentHandler()\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) {\n      throw new Error(`No configured attachment storage driver named \"${driverName}\"`)\n    } else 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 {\n      throw new Error(`Attachment storage driver \"${driverName}\" must define instance, driverClass, or create`)\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    const attachmentsConfiguration = this.configuration.getAttachmentsConfiguration?.() || {}\n    const defaultDriver = attachmentsConfiguration.defaultDriver\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    if (typeof defaultDriver === \"string\" && defaultDriver.length > 0) {\n      return defaultDriver\n    }\n\n    throw new Error(`No attachment driver configured for ${model.getModelClass().name}#${name}`)\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    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"]}
@@ -195,6 +195,24 @@ export default class VelociousEnvironmentHandlerBase {
195
195
  */
196
196
  setConfiguration(newConfiguration: import("../configuration.js").default): void;
197
197
  configuration: import("../configuration.js").default;
198
+ /**
199
+ * @param {string} _filePath - File path.
200
+ * @returns {Promise<Buffer>} - File bytes.
201
+ */
202
+ readAttachmentInputFile(_filePath: string): Promise<Buffer>;
203
+ /**
204
+ * @param {object} _args - Args.
205
+ * @param {string[]} _args.allowedPathPrefixes - Allowed path prefixes.
206
+ * @param {string} _args.inputPath - Input path.
207
+ * @returns {Promise<{buffer: Buffer, filePath: string}>} - Resolved path and file bytes.
208
+ */
209
+ resolveAttachmentInputPath(_args: {
210
+ allowedPathPrefixes: string[];
211
+ inputPath: string;
212
+ }): Promise<{
213
+ buffer: Buffer;
214
+ filePath: string;
215
+ }>;
198
216
  /**
199
217
  * @returns {import("../configuration.js").default} - The configuration.
200
218
  */
@@ -1 +1 @@
1
- {"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../src/environment-handlers/base.js"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../src/environment-handlers/base.js"],"names":[],"mappings":"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"}
@@ -266,6 +266,22 @@ export default class VelociousEnvironmentHandlerBase {
266
266
  * @returns {void} - No return value.
267
267
  */
268
268
  setConfiguration(newConfiguration) { this.configuration = newConfiguration; }
269
+ /**
270
+ * @param {string} _filePath - File path.
271
+ * @returns {Promise<Buffer>} - File bytes.
272
+ */
273
+ async readAttachmentInputFile(_filePath) {
274
+ throw new Error("Attachment file reads are not supported in this environment");
275
+ }
276
+ /**
277
+ * @param {object} _args - Args.
278
+ * @param {string[]} _args.allowedPathPrefixes - Allowed path prefixes.
279
+ * @param {string} _args.inputPath - Input path.
280
+ * @returns {Promise<{buffer: Buffer, filePath: string}>} - Resolved path and file bytes.
281
+ */
282
+ async resolveAttachmentInputPath(_args) {
283
+ throw new Error("Attachment path input is not supported in this environment");
284
+ }
269
285
  /**
270
286
  * @returns {import("../configuration.js").default} - The configuration.
271
287
  */
@@ -307,4 +323,4 @@ export default class VelociousEnvironmentHandlerBase {
307
323
  return;
308
324
  }
309
325
  }
310
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../src/environment-handlers/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\n/**\n * @typedef {object} CommandFileObjectType\n * @property {string} name - Command name.\n * @property {string} file - Command file path.\n */\n\n/**\n * @typedef {object} MigrationObjectType\n * @property {number} date - Migration timestamp parsed from filename.\n * @property {string} [fullPath] - Absolute path to the migration file.\n * @property {string} migrationClassName - Exported migration class name.\n * @property {string} file - Migration filename.\n */\n\nexport default class VelociousEnvironmentHandlerBase {\n  /**\n   * @param {number} _offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @param {() => Promise<any>} callback - Callback to run.\n   * @returns {Promise<any>} - Result of the callback.\n   */\n  async runWithTimezoneOffset(_offsetMinutes, callback) {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    const previousOffsetMinutes = this.configuration._timezoneOffsetMinutes\n\n    this.configuration._timezoneOffsetMinutes = _offsetMinutes\n\n    try {\n      return await callback()\n    } finally {\n      this.configuration._timezoneOffsetMinutes = previousOffsetMinutes\n    }\n  }\n\n  /**\n   * @param {number} _offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @returns {void} - No return value.\n   */\n  setTimezoneOffset(_offsetMinutes) {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    this.configuration._timezoneOffsetMinutes = _offsetMinutes\n  }\n\n  /**\n   * @param {import(\"../configuration.js\").default | undefined} configuration - Configuration instance.\n   * @returns {number} - Offset in minutes.\n   */\n  getTimezoneOffsetMinutes(configuration) {\n    const activeConfiguration = configuration || this.configuration\n\n    if (!activeConfiguration) throw new Error(\"Configuration hasn't been set\")\n\n    if (typeof activeConfiguration._timezoneOffsetMinutes === \"number\") {\n      return activeConfiguration._timezoneOffsetMinutes\n    }\n\n    return activeConfiguration.getTimezoneOffsetMinutes()\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set for callback scope.\n   * @param {() => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async runWithAbility(ability, callback) {\n    this._currentAbility = ability\n\n    try {\n      return await callback()\n    } finally {\n      this._currentAbility = undefined\n    }\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set.\n   * @returns {void} - No return value.\n   */\n  setCurrentAbility(ability) {\n    this._currentAbility = ability\n  }\n\n  /**\n   * @returns {import(\"../authorization/ability.js\").default | undefined} - Current ability.\n   */\n  getCurrentAbility() {\n    return this._currentAbility\n  }\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateBaseModels(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateBaseModels not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateFrontendModels(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateFrontendModels not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsInit(command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsInit not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationGenerate(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsMigrationGenerate not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationDestroy(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsMigrationDestroy not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateModel(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateModel not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRoutes(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRoutes not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsConsole(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsConsole not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsServer(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsServer not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsTest(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsTest not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsMain(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsMain not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsWorker(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsWorker not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsRunner(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsRunner not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSchemaDump(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsDbSchemaDump not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSeed(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsDbSeed not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunner(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRunner not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunScript(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRunScript not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @returns {Promise<CommandFileObjectType[]>} - Resolves with the commands.\n   */\n  async findCommands() { throw new Error(\"findCommands not implemented\") }\n\n  /**\n   * @abstract\n   * @returns {Promise<Array<MigrationObjectType>>} - Resolves with the migrations.\n   */\n  async findMigrations() { throw new Error(\"findMigrations not implemneted\") }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @param {typeof import(\"../cli/base-command.js\").default} CommandClass - Command class.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async forwardCommand(command, CommandClass) {\n    const newCommand = new CommandClass({\n      args: command.args,\n      cli: command.cli\n    })\n\n    return await newCommand.execute()\n  }\n\n  /**\n   * @abstract\n   * @returns {Promise<string>} - Resolves with the velocious path.\n   */\n  getVelociousPath() { throw new Error(\"getVelociousPath not implemented\") }\n\n  /**\n   * @abstract\n   * @returns {Promise<import(\"../routes/index.js\").default>} - Resolves with the import application routes.\n   */\n  async importApplicationRoutes() { throw new Error(\"importApplicationRoutes not implemented\") }\n\n  /**\n   * @abstract\n   * @param {string[]} _testFiles - Test files.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  importTestFiles(_testFiles) { throw new Error(\"'importTestFiles' not implemented\") } // eslint-disable-line no-unused-vars\n\n  /**\n   * @abstract\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  importTestingConfigPath() { throw new Error(`'importTestingConfigPath' not implemented`) }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async afterMigrations(args) { // eslint-disable-line no-unused-vars\n    return\n  }\n\n  /**\n   * @abstract\n   * @param {object} args - Options object.\n   * @param {string[]} args.commandParts - Command parts.\n   * @returns {Promise<typeof import (\"../cli/base-command.js\").default>} - Resolves with the require command.\n   */\n  async requireCommand({commandParts}) { throw new Error(\"'requireCommand' not implemented\") } // eslint-disable-line no-unused-vars\n\n  /**\n   * @param {object} newArgs - New args.\n   * @returns {void} - No return value.\n   */\n  setArgs(newArgs) { this.args = newArgs }\n\n  /**\n   * @param {import(\"../configuration.js\").default} newConfiguration - New configuration.\n   * @returns {void} - No return value.\n   */\n  setConfiguration(newConfiguration) { this.configuration = newConfiguration }\n\n  /**\n   * @returns {import(\"../configuration.js\").default} - The configuration.\n   */\n  getConfiguration() {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    return this.configuration\n  }\n\n  /**\n   * @param {string[]} newProcessArgs - New process args.\n   * @returns {void} - No return value.\n   */\n  setProcessArgs(newProcessArgs) { this.processArgs = newProcessArgs }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {import(\"../configuration.js\").default} _args.configuration - Configuration instance.\n   * @returns {string | undefined} - The default log directory.\n   */\n  getDefaultLogDirectory(_args) { // eslint-disable-line no-unused-vars\n    return undefined\n  }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {import(\"../configuration.js\").default} _args.configuration - Configuration instance.\n   * @param {string | undefined} _args.directory - Directory path.\n   * @param {string} _args.environment - Environment.\n   * @returns {string | undefined} - The log file path.\n   */\n  getLogFilePath(_args) { // eslint-disable-line no-unused-vars\n    return undefined\n  }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {string} _args.filePath - File path.\n   * @param {string} _args.message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async writeLogToFile(_args) { // eslint-disable-line no-unused-vars\n    return\n  }\n}\n"]}
326
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../src/environment-handlers/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\n/**\n * @typedef {object} CommandFileObjectType\n * @property {string} name - Command name.\n * @property {string} file - Command file path.\n */\n\n/**\n * @typedef {object} MigrationObjectType\n * @property {number} date - Migration timestamp parsed from filename.\n * @property {string} [fullPath] - Absolute path to the migration file.\n * @property {string} migrationClassName - Exported migration class name.\n * @property {string} file - Migration filename.\n */\n\nexport default class VelociousEnvironmentHandlerBase {\n  /**\n   * @param {number} _offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @param {() => Promise<any>} callback - Callback to run.\n   * @returns {Promise<any>} - Result of the callback.\n   */\n  async runWithTimezoneOffset(_offsetMinutes, callback) {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    const previousOffsetMinutes = this.configuration._timezoneOffsetMinutes\n\n    this.configuration._timezoneOffsetMinutes = _offsetMinutes\n\n    try {\n      return await callback()\n    } finally {\n      this.configuration._timezoneOffsetMinutes = previousOffsetMinutes\n    }\n  }\n\n  /**\n   * @param {number} _offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @returns {void} - No return value.\n   */\n  setTimezoneOffset(_offsetMinutes) {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    this.configuration._timezoneOffsetMinutes = _offsetMinutes\n  }\n\n  /**\n   * @param {import(\"../configuration.js\").default | undefined} configuration - Configuration instance.\n   * @returns {number} - Offset in minutes.\n   */\n  getTimezoneOffsetMinutes(configuration) {\n    const activeConfiguration = configuration || this.configuration\n\n    if (!activeConfiguration) throw new Error(\"Configuration hasn't been set\")\n\n    if (typeof activeConfiguration._timezoneOffsetMinutes === \"number\") {\n      return activeConfiguration._timezoneOffsetMinutes\n    }\n\n    return activeConfiguration.getTimezoneOffsetMinutes()\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set for callback scope.\n   * @param {() => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async runWithAbility(ability, callback) {\n    this._currentAbility = ability\n\n    try {\n      return await callback()\n    } finally {\n      this._currentAbility = undefined\n    }\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set.\n   * @returns {void} - No return value.\n   */\n  setCurrentAbility(ability) {\n    this._currentAbility = ability\n  }\n\n  /**\n   * @returns {import(\"../authorization/ability.js\").default | undefined} - Current ability.\n   */\n  getCurrentAbility() {\n    return this._currentAbility\n  }\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateBaseModels(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateBaseModels not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateFrontendModels(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateFrontendModels not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsInit(command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsInit not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationGenerate(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsMigrationGenerate not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationDestroy(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsMigrationDestroy not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateModel(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsGenerateModel not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRoutes(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRoutes not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsConsole(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsConsole not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsServer(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsServer not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsTest(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsTest not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsMain(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsMain not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsWorker(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsWorker not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsRunner(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsBackgroundJobsRunner not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSchemaDump(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsDbSchemaDump not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSeed(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsDbSeed not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunner(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRunner not implemented\")\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} _command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunScript(_command) { // eslint-disable-line no-unused-vars\n    throw new Error(\"cliCommandsRunScript not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @returns {Promise<CommandFileObjectType[]>} - Resolves with the commands.\n   */\n  async findCommands() { throw new Error(\"findCommands not implemented\") }\n\n  /**\n   * @abstract\n   * @returns {Promise<Array<MigrationObjectType>>} - Resolves with the migrations.\n   */\n  async findMigrations() { throw new Error(\"findMigrations not implemneted\") }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @param {typeof import(\"../cli/base-command.js\").default} CommandClass - Command class.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async forwardCommand(command, CommandClass) {\n    const newCommand = new CommandClass({\n      args: command.args,\n      cli: command.cli\n    })\n\n    return await newCommand.execute()\n  }\n\n  /**\n   * @abstract\n   * @returns {Promise<string>} - Resolves with the velocious path.\n   */\n  getVelociousPath() { throw new Error(\"getVelociousPath not implemented\") }\n\n  /**\n   * @abstract\n   * @returns {Promise<import(\"../routes/index.js\").default>} - Resolves with the import application routes.\n   */\n  async importApplicationRoutes() { throw new Error(\"importApplicationRoutes not implemented\") }\n\n  /**\n   * @abstract\n   * @param {string[]} _testFiles - Test files.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  importTestFiles(_testFiles) { throw new Error(\"'importTestFiles' not implemented\") } // eslint-disable-line no-unused-vars\n\n  /**\n   * @abstract\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  importTestingConfigPath() { throw new Error(`'importTestingConfigPath' not implemented`) }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async afterMigrations(args) { // eslint-disable-line no-unused-vars\n    return\n  }\n\n  /**\n   * @abstract\n   * @param {object} args - Options object.\n   * @param {string[]} args.commandParts - Command parts.\n   * @returns {Promise<typeof import (\"../cli/base-command.js\").default>} - Resolves with the require command.\n   */\n  async requireCommand({commandParts}) { throw new Error(\"'requireCommand' not implemented\") } // eslint-disable-line no-unused-vars\n\n  /**\n   * @param {object} newArgs - New args.\n   * @returns {void} - No return value.\n   */\n  setArgs(newArgs) { this.args = newArgs }\n\n  /**\n   * @param {import(\"../configuration.js\").default} newConfiguration - New configuration.\n   * @returns {void} - No return value.\n   */\n  setConfiguration(newConfiguration) { this.configuration = newConfiguration }\n\n  /**\n   * @param {string} _filePath - File path.\n   * @returns {Promise<Buffer>} - File bytes.\n   */\n  async readAttachmentInputFile(_filePath) { // eslint-disable-line no-unused-vars\n    throw new Error(\"Attachment file reads are not supported in this environment\")\n  }\n\n  /**\n   * @param {object} _args - Args.\n   * @param {string[]} _args.allowedPathPrefixes - Allowed path prefixes.\n   * @param {string} _args.inputPath - Input path.\n   * @returns {Promise<{buffer: Buffer, filePath: string}>} - Resolved path and file bytes.\n   */\n  async resolveAttachmentInputPath(_args) { // eslint-disable-line no-unused-vars\n    throw new Error(\"Attachment path input is not supported in this environment\")\n  }\n\n  /**\n   * @returns {import(\"../configuration.js\").default} - The configuration.\n   */\n  getConfiguration() {\n    if (!this.configuration) throw new Error(\"Configuration hasn't been set\")\n\n    return this.configuration\n  }\n\n  /**\n   * @param {string[]} newProcessArgs - New process args.\n   * @returns {void} - No return value.\n   */\n  setProcessArgs(newProcessArgs) { this.processArgs = newProcessArgs }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {import(\"../configuration.js\").default} _args.configuration - Configuration instance.\n   * @returns {string | undefined} - The default log directory.\n   */\n  getDefaultLogDirectory(_args) { // eslint-disable-line no-unused-vars\n    return undefined\n  }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {import(\"../configuration.js\").default} _args.configuration - Configuration instance.\n   * @param {string | undefined} _args.directory - Directory path.\n   * @param {string} _args.environment - Environment.\n   * @returns {string | undefined} - The log file path.\n   */\n  getLogFilePath(_args) { // eslint-disable-line no-unused-vars\n    return undefined\n  }\n\n  /**\n   * @param {object} _args - Options object.\n   * @param {string} _args.filePath - File path.\n   * @param {string} _args.message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async writeLogToFile(_args) { // eslint-disable-line no-unused-vars\n    return\n  }\n}\n"]}
@@ -1,4 +1,3 @@
1
- /** @typedef {{ability?: import("../authorization/ability.js").default, offsetMinutes: number}} TimezoneStore */
2
1
  export default class VelociousEnvironmentHandlerNode extends Base {
3
2
  /** @type {import("node:async_hooks").AsyncLocalStorage<TimezoneStore> | undefined} */
4
3
  _timezoneAsyncLocalStorage: import("node:async_hooks").AsyncLocalStorage<TimezoneStore> | undefined;
@@ -1 +1 @@
1
- {"version":3,"file":"node.d.ts","sourceRoot":"","sources":["../../../src/environment-handlers/node.js"],"names":[],"mappings":"AA6BA,gHAAgH;AAEhH;IACE,sFAAsF;IACtF,4BADW,OAAO,kBAAkB,EAAE,iBAAiB,CAAC,aAAa,CAAC,GAAG,SAAS,CACU;IAE5F,sEAAsE;IACtE,qBADW,OAAO,WAAW,EAAE,qBAAqB,EAAE,GAAG,SAAS,CACnC;IAmI/B;;OAEG;IACH,uBAFa,OAAO,CAAC,KAAK,CAAC,OAAO,WAAW,EAAE,qBAAqB,CAAC,CAAC,CAcrE;IAED;;;OAGG;IACH,kCAHW,MAAM,GACJ,MAAM,CAuBlB;IAiNG,uBAA6D;IAqEjE;;;OAGG;IACH,2BAHW,MAAM,GACJ,OAAO,CAAC,OAAO,gCAAgC,EAAE,OAAO,CAAC,CAUrE;IAED,+BAKC;IA4BD;;;;OAIG;IACH,mCAHG;QAA4E,GAAG,EAAvE,MAAM,CAAC,MAAM,EAAE,OAAO,6BAA6B,EAAE,OAAO,CAAC;KACrE,GAAU,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAkB3C;CACF;4BA/gBa;IAAC,OAAO,CAAC,EAAE,OAAO,6BAA6B,EAAE,OAAO,CAAC;IAAC,aAAa,EAAE,MAAM,CAAA;CAAC;iBA3B7E,WAAW"}
1
+ {"version":3,"file":"node.d.ts","sourceRoot":"","sources":["../../../src/environment-handlers/node.js"],"names":[],"mappings":"AAmDA;IACE,sFAAsF;IACtF,4BADW,OAAO,kBAAkB,EAAE,iBAAiB,CAAC,aAAa,CAAC,GAAG,SAAS,CACU;IAE5F,sEAAsE;IACtE,qBADW,OAAO,WAAW,EAAE,qBAAqB,EAAE,GAAG,SAAS,CACnC;IAgK/B;;OAEG;IACH,uBAFa,OAAO,CAAC,KAAK,CAAC,OAAO,WAAW,EAAE,qBAAqB,CAAC,CAAC,CAcrE;IAED;;;OAGG;IACH,kCAHW,MAAM,GACJ,MAAM,CAuBlB;IAiNG,uBAA6D;IAqEjE;;;OAGG;IACH,2BAHW,MAAM,GACJ,OAAO,CAAC,OAAO,gCAAgC,EAAE,OAAO,CAAC,CAUrE;IAED,+BAKC;IA4BD;;;;OAIG;IACH,mCAHG;QAA4E,GAAG,EAAvE,MAAM,CAAC,MAAM,EAAE,OAAO,6BAA6B,EAAE,OAAO,CAAC;KACrE,GAAU,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAkB3C;CACF;4BAhkBa;IAAC,OAAO,CAAC,EAAE,OAAO,6BAA6B,EAAE,OAAO,CAAC;IAAC,aAAa,EAAE,MAAM,CAAA;CAAC;iBA3B7E,WAAW"}
@@ -26,6 +26,25 @@ import path from "path";
26
26
  import { AsyncLocalStorage as NodeAsyncLocalStorage } from "node:async_hooks";
27
27
  import toImportSpecifier from "../utils/to-import-specifier.js";
28
28
  /** @typedef {{ability?: import("../authorization/ability.js").default, offsetMinutes: number}} TimezoneStore */
29
+ /**
30
+ * @param {string} filePath - Input file path.
31
+ * @param {string[]} allowedPathPrefixes - Allowed path prefixes.
32
+ * @returns {boolean} - Whether input path is inside an allowed prefix.
33
+ */
34
+ function pathWithinAllowedPrefixes(filePath, allowedPathPrefixes) {
35
+ const resolvedPath = path.resolve(filePath);
36
+ return allowedPathPrefixes.some((allowedPrefix) => {
37
+ const resolvedPrefix = path.resolve(allowedPrefix);
38
+ const relativePath = path.relative(resolvedPrefix, resolvedPath);
39
+ if (!relativePath)
40
+ return true;
41
+ if (relativePath.startsWith(".."))
42
+ return false;
43
+ if (path.isAbsolute(relativePath))
44
+ return false;
45
+ return true;
46
+ });
47
+ }
29
48
  export default class VelociousEnvironmentHandlerNode extends Base {
30
49
  /** @type {import("node:async_hooks").AsyncLocalStorage<TimezoneStore> | undefined} */
31
50
  _timezoneAsyncLocalStorage = NodeAsyncLocalStorage ? new NodeAsyncLocalStorage() : undefined;
@@ -41,6 +60,30 @@ export default class VelociousEnvironmentHandlerNode extends Base {
41
60
  newConfiguration.addRouteResolverHook(frontendModelCommandRouteHook);
42
61
  }
43
62
  }
63
+ /**
64
+ * @param {string} filePath - File path.
65
+ * @returns {Promise<Buffer>} - File bytes.
66
+ */
67
+ async readAttachmentInputFile(filePath) {
68
+ return await fs.readFile(filePath);
69
+ }
70
+ /**
71
+ * @param {object} args - Args.
72
+ * @param {string[]} args.allowedPathPrefixes - Allowed path prefixes.
73
+ * @param {string} args.inputPath - Input path.
74
+ * @returns {Promise<{buffer: Buffer, filePath: string}>} - Resolved path and bytes.
75
+ */
76
+ async resolveAttachmentInputPath({ allowedPathPrefixes, inputPath }) {
77
+ const filePath = path.resolve(inputPath);
78
+ const prefixes = Array.isArray(allowedPathPrefixes)
79
+ ? allowedPathPrefixes.filter((entry) => typeof entry === "string" && entry.length > 0)
80
+ : [];
81
+ if (prefixes.length > 0 && !pathWithinAllowedPrefixes(filePath, prefixes)) {
82
+ throw new Error("Attachment path is outside allowed directories");
83
+ }
84
+ const buffer = await this.readAttachmentInputFile(filePath);
85
+ return { buffer, filePath };
86
+ }
44
87
  /**
45
88
  * @returns {Promise<Array<import("./base.js").CommandFileObjectType>>} - Resolves with the commands.
46
89
  */
@@ -468,4 +511,4 @@ export default class VelociousEnvironmentHandlerNode extends Base {
468
511
  return sqlByIdentifier;
469
512
  }
470
513
  }
471
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.js","sourceRoot":"","sources":["../../../src/environment-handlers/node.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Base from \"./base.js\"\nimport CliCommandsDestroyMigration from \"./node/cli/commands/destroy/migration.js\"\nimport CliCommandsInit from \"./node/cli/commands/init.js\"\nimport CliCommandsGenerateBaseModels from \"./node/cli/commands/generate/base-models.js\"\nimport CliCommandsGenerateFrontendModels from \"./node/cli/commands/generate/frontend-models.js\"\nimport CliCommandsGenerateMigration from \"./node/cli/commands/generate/migration.js\"\nimport CliCommandsGenerateModel from \"./node/cli/commands/generate/model.js\"\nimport CliCommandsRoutes from \"./node/cli/commands/routes.js\"\nimport CliCommandsServer from \"./node/cli/commands/server.js\"\nimport CliCommandsTest from \"./node/cli/commands/test.js\"\nimport CliCommandsBackgroundJobsMain from \"./node/cli/commands/background-jobs-main.js\"\nimport CliCommandsBackgroundJobsWorker from \"./node/cli/commands/background-jobs-worker.js\"\nimport CliCommandsBackgroundJobsRunner from \"./node/cli/commands/background-jobs-runner.js\"\nimport CliCommandsConsole from \"./node/cli/commands/console.js\"\nimport CliCommandsDbSchemaDump from \"./node/cli/commands/db/schema/dump.js\"\nimport CliCommandsDbSeed from \"./node/cli/commands/db/seed.js\"\nimport CliCommandsRunner from \"./node/cli/commands/runner.js\"\nimport CliCommandsRunScript from \"./node/cli/commands/run-script.js\"\nimport frontendModelCommandRouteHook from \"../routes/hooks/frontend-model-command-route-hook.js\"\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport path from \"path\"\nimport {AsyncLocalStorage as NodeAsyncLocalStorage} from \"node:async_hooks\"\nimport toImportSpecifier from \"../utils/to-import-specifier.js\"\n\n/** @typedef {{ability?: import(\"../authorization/ability.js\").default, offsetMinutes: number}} TimezoneStore */\n\nexport default class VelociousEnvironmentHandlerNode extends Base{\n  /** @type {import(\"node:async_hooks\").AsyncLocalStorage<TimezoneStore> | undefined} */\n  _timezoneAsyncLocalStorage = NodeAsyncLocalStorage ? new NodeAsyncLocalStorage() : undefined\n\n  /** @type {import(\"./base.js\").CommandFileObjectType[] | undefined} */\n  _findCommandsResult = undefined\n\n  /**\n   * @param {import(\"../configuration.js\").default} newConfiguration - New configuration.\n   * @returns {void} - No return value.\n   */\n  setConfiguration(newConfiguration) {\n    super.setConfiguration(newConfiguration)\n\n    if (!newConfiguration.getRouteResolverHooks().includes(frontendModelCommandRouteHook)) {\n      newConfiguration.addRouteResolverHook(frontendModelCommandRouteHook)\n    }\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").CommandFileObjectType>>} - Resolves with the commands.\n   */\n  async findCommands() {\n    this._findCommandsResult ||= await this._actualFindCommands()\n\n    if (!this._findCommandsResult) throw new Error(\"Could not get commands\")\n\n    return this._findCommandsResult\n  }\n\n  /**\n   * @param {number} offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @param {() => Promise<any>} callback - Callback to run.\n   * @returns {Promise<any>} - Result of the callback.\n   */\n  async runWithTimezoneOffset(offsetMinutes, callback) {\n    if (!this._timezoneAsyncLocalStorage) {\n      return await callback()\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    return await this._timezoneAsyncLocalStorage.run({\n      ability: existingStore?.ability,\n      offsetMinutes\n    }, callback)\n  }\n\n  /**\n   * @param {number} offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @returns {void} - No return value.\n   */\n  setTimezoneOffset(offsetMinutes) {\n    if (!this._timezoneAsyncLocalStorage) return\n\n    const store = this._timezoneAsyncLocalStorage.getStore()\n\n    if (store) {\n      store.offsetMinutes = offsetMinutes\n    } else {\n      const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n      this._timezoneAsyncLocalStorage.enterWith({\n        ability: existingStore?.ability,\n        offsetMinutes\n      })\n    }\n  }\n\n  /**\n   * @param {import(\"../configuration.js\").default | undefined} configuration - Configuration instance.\n   * @returns {number} - Offset in minutes.\n   */\n  getTimezoneOffsetMinutes(configuration) {\n    if (this._timezoneAsyncLocalStorage) {\n      const store = this._timezoneAsyncLocalStorage.getStore()\n\n      if (store && typeof store.offsetMinutes === \"number\") {\n        return store.offsetMinutes\n      }\n    }\n\n    return super.getTimezoneOffsetMinutes(configuration)\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set for callback scope.\n   * @param {() => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async runWithAbility(ability, callback) {\n    if (!this._timezoneAsyncLocalStorage) {\n      return await super.runWithAbility(ability, callback)\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    return await this._timezoneAsyncLocalStorage.run({\n      ability,\n      offsetMinutes: existingStore?.offsetMinutes ?? this.getTimezoneOffsetMinutes(this.getConfiguration())\n    }, callback)\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set.\n   * @returns {void} - No return value.\n   */\n  setCurrentAbility(ability) {\n    if (!this._timezoneAsyncLocalStorage) {\n      super.setCurrentAbility(ability)\n      return\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    if (existingStore) {\n      existingStore.ability = ability\n    } else {\n      this._timezoneAsyncLocalStorage.enterWith({\n        ability,\n        offsetMinutes: this.getTimezoneOffsetMinutes(this.getConfiguration())\n      })\n    }\n  }\n\n  /**\n   * @returns {import(\"../authorization/ability.js\").default | undefined} - Current ability.\n   */\n  getCurrentAbility() {\n    if (!this._timezoneAsyncLocalStorage) {\n      return super.getCurrentAbility()\n    }\n\n    return this._timezoneAsyncLocalStorage.getStore()?.ability\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").CommandFileObjectType>>} - Resolves with discovered command files.\n   */\n  async _actualFindCommands() {\n    const basePath = await this.getBasePath()\n    const commandFiles = fs.glob(`${basePath}/src/cli/commands/**/*.js`)\n    const commands = []\n\n    for await (const aFilePath of commandFiles) {\n      const commandName = this.commandNameFromFilePath(aFilePath)\n\n      commands.push({name: commandName, file: aFilePath})\n    }\n\n    return commands\n  }\n\n  /**\n   * @param {string} filePath - Full command file path.\n   * @returns {string} - Parsed command name.\n   */\n  commandNameFromFilePath(filePath) {\n    const aFilePathParts = filePath.split(/[\\\\/]/)\n    const commandPathLocation = aFilePathParts.indexOf(\"commands\")\n\n    if (commandPathLocation === -1) {\n      throw new Error(`Could not parse command file path: ${filePath}`)\n    }\n\n    const commandParts = aFilePathParts.slice(commandPathLocation + 1)\n    const lastPart = commandParts[commandParts.length - 1]\n    let name, paths\n\n    if (lastPart == \"index.js\") {\n      name = commandParts[commandParts.length - 2]\n      paths = commandParts.slice(0, -2)\n    } else {\n      name = lastPart.replace(\".js\", \"\")\n      paths = commandParts.slice(0, -1)\n    }\n\n    return `${paths.join(\":\")}${paths.length > 0 ? \":\" : \"\"}${name}`\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsInit(command) {\n    return await this.forwardCommand(command, CliCommandsInit)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationGenerate(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateMigration)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationDestroy(command) {\n    return await this.forwardCommand(command, CliCommandsDestroyMigration)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateBaseModels(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateBaseModels)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateFrontendModels(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateFrontendModels)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateModel(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateModel)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRoutes(command) {\n    return await this.forwardCommand(command, CliCommandsRoutes)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsConsole(command) {\n    return await this.forwardCommand(command, CliCommandsConsole)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsServer(command) {\n    return await this.forwardCommand(command, CliCommandsServer)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsTest(command) {\n    return await this.forwardCommand(command, CliCommandsTest)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsMain(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsMain)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsWorker(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsWorker)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsRunner(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsRunner)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSchemaDump(command) {\n    return await this.forwardCommand(command, CliCommandsDbSchemaDump)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSeed(command) {\n    return await this.forwardCommand(command, CliCommandsDbSeed)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunner(command) {\n    return await this.forwardCommand(command, CliCommandsRunner)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunScript(command) {\n    return await this.forwardCommand(command, CliCommandsRunScript)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string[]} args.commandParts - Command parts.\n   * @returns {Promise<typeof import (\"../cli/base-command.js\").default>} - Resolves with the require command.\n   */\n  async requireCommand({commandParts}) {\n    const commands = await this.findCommands()\n    const commandName = commandParts.join(\":\")\n    const command = commands.find((aCommand) => aCommand.name === commandName)\n\n    if (!command) {\n      const possibleCommands = commands.map(aCommand => aCommand.name)\n\n      throw new Error(`Unknown command: ${commandParts.join(\":\")} which should have been one of: ${possibleCommands.sort().join(\", \")}`)\n    }\n\n    const commandClassImport = await import(toImportSpecifier(command.file))\n    const CommandClass = commandClassImport.default\n\n    return CommandClass\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").MigrationObjectType>>} - Resolves with the migrations.\n   */\n  async findMigrations() {\n    const migrationsPath = `${this.getConfiguration().getDirectory()}/src/database/migrations`\n    const glob = await fs.glob(`${migrationsPath}/**/*.js`)\n    let files = []\n\n    for await (const fullPath of glob) {\n      const file = await path.basename(fullPath)\n\n      const match = file.match(/^(\\d{14})-(.+)\\.js$/)\n\n      if (!match) continue\n\n      const date = parseInt(match[1])\n      const migrationName = match[2]\n      const migrationClassName = inflection.camelize(migrationName.replaceAll(\"-\", \"_\"))\n\n      files.push({\n        file,\n        fullPath: `${migrationsPath}/${file}`,\n        date,\n        migrationClassName\n      })\n    }\n\n    files = files.sort((migration1, migration2) => migration1.date - migration2.date)\n\n    return files\n  }\n\n  /**\n   * @returns {Promise<import(\"../routes/index.js\").default>} - Resolves with the import application routes.\n   */\n  async importApplicationRoutes() {\n    const routesImport = await import(toImportSpecifier(`${this.getConfiguration().getDirectory()}/src/config/routes.js`))\n\n    return routesImport.default\n  }\n\n  /**\n   * @returns {Promise<string>} - Resolves with the velocious path.\n   */\n  async getVelociousPath() {\n    if (!this._velociousPath) {\n      const __filename = fileURLToPath(import.meta.url)\n      const __dirname = dirname(__filename)\n\n      this._velociousPath = await fs.realpath(`${__dirname}/../..`)\n    }\n\n    return this._velociousPath\n  }\n\n  /**\n   * @param {string[]} testFiles - Test files.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async importTestFiles(testFiles) {\n    for (const testFile of testFiles) {\n      await import(toImportSpecifier(testFile))\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @returns {string} - The default log directory.\n   */\n  getDefaultLogDirectory({configuration}) {\n    return path.join(configuration.getDirectory(), \"log\")\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string | undefined} args.directory - Directory path.\n   * @param {string} args.environment - Environment.\n   * @returns {string | undefined} - The log file path.\n   */\n  getLogFilePath({configuration, directory, environment}) {\n    const actualDirectory = directory || configuration?.getDirectory?.()\n\n    if (!actualDirectory) return undefined\n\n    return path.join(actualDirectory, `${environment}.log`)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.filePath - File path.\n   * @param {string} args.message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async writeLogToFile({filePath, message}) {\n    await fs.mkdir(path.dirname(filePath), {recursive: true})\n    await fs.appendFile(filePath, `${message}\\n`, \"utf8\")\n  }\n\n  async importTestingConfigPath() {\n    const testingConfigPath = this.getConfiguration().getTesting()\n\n    if (!testingConfigPath) return\n\n    const testingImport = await import(toImportSpecifier(testingConfigPath))\n    const testingDefault = testingImport.default\n\n    if (!testingDefault) throw new Error(\"Testing config must export a default function\")\n    if (typeof testingDefault !== \"function\") throw new Error(\"Testing config default export isn't a function\")\n\n    const result = await testingDefault()\n\n    if (typeof result === \"function\") {\n      await result()\n    }\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {Promise<import(\"../database/migration/index.js\").default>} - Resolves with the require migration.\n   */\n  async requireMigration(filePath) {\n    const migrationImport = await import(toImportSpecifier(filePath))\n    const migrationImportDefault = migrationImport.default\n\n    if (!migrationImportDefault) throw new Error(\"Migration file must export a default migration class\")\n    if (typeof migrationImportDefault !== \"function\") throw new Error(\"Migration default export isn't a function (should be a class which is a function in JS)\")\n\n    return migrationImportDefault\n  }\n\n  async getBasePath() {\n    const __filename = fileURLToPath(import.meta.url)\n    const basePath = await fs.realpath(`${dirname(__filename)}/../..`)\n\n    return basePath\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async afterMigrations({dbs}) {\n    const configuration = this.getConfiguration()\n\n    if (!configuration.shouldWriteStructureSql()) return\n\n    const dbDir = path.join(configuration.getDirectory(), \"db\")\n    const structureSqlByIdentifier = await this._structureSqlByIdentifier({dbs})\n\n    await fs.mkdir(dbDir, {recursive: true})\n\n    for (const identifier of Object.keys(structureSqlByIdentifier)) {\n      const structureSql = structureSqlByIdentifier[identifier]\n\n      if (!structureSql) continue\n\n      const filePath = path.join(dbDir, `structure-${identifier}.sql`)\n\n      await fs.writeFile(filePath, structureSql)\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<Record<string, string>>} - Resolves with SQL string.\n   */\n  async _structureSqlByIdentifier({dbs}) {\n    const sqlByIdentifier = /** @type {Record<string, string>} */ ({})\n\n    for (const identifier of Object.keys(dbs)) {\n      const db = dbs[identifier]\n\n      if (typeof db.structureSql !== \"function\") continue\n\n      const structureSql = await db.structureSql()\n\n      if (structureSql) {\n        sqlByIdentifier[identifier] = structureSql\n      }\n    }\n\n    return sqlByIdentifier\n  }\n}\n"]}
514
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.js","sourceRoot":"","sources":["../../../src/environment-handlers/node.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Base from \"./base.js\"\nimport CliCommandsDestroyMigration from \"./node/cli/commands/destroy/migration.js\"\nimport CliCommandsInit from \"./node/cli/commands/init.js\"\nimport CliCommandsGenerateBaseModels from \"./node/cli/commands/generate/base-models.js\"\nimport CliCommandsGenerateFrontendModels from \"./node/cli/commands/generate/frontend-models.js\"\nimport CliCommandsGenerateMigration from \"./node/cli/commands/generate/migration.js\"\nimport CliCommandsGenerateModel from \"./node/cli/commands/generate/model.js\"\nimport CliCommandsRoutes from \"./node/cli/commands/routes.js\"\nimport CliCommandsServer from \"./node/cli/commands/server.js\"\nimport CliCommandsTest from \"./node/cli/commands/test.js\"\nimport CliCommandsBackgroundJobsMain from \"./node/cli/commands/background-jobs-main.js\"\nimport CliCommandsBackgroundJobsWorker from \"./node/cli/commands/background-jobs-worker.js\"\nimport CliCommandsBackgroundJobsRunner from \"./node/cli/commands/background-jobs-runner.js\"\nimport CliCommandsConsole from \"./node/cli/commands/console.js\"\nimport CliCommandsDbSchemaDump from \"./node/cli/commands/db/schema/dump.js\"\nimport CliCommandsDbSeed from \"./node/cli/commands/db/seed.js\"\nimport CliCommandsRunner from \"./node/cli/commands/runner.js\"\nimport CliCommandsRunScript from \"./node/cli/commands/run-script.js\"\nimport frontendModelCommandRouteHook from \"../routes/hooks/frontend-model-command-route-hook.js\"\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport path from \"path\"\nimport {AsyncLocalStorage as NodeAsyncLocalStorage} from \"node:async_hooks\"\nimport toImportSpecifier from \"../utils/to-import-specifier.js\"\n\n/** @typedef {{ability?: import(\"../authorization/ability.js\").default, offsetMinutes: number}} TimezoneStore */\n\n/**\n * @param {string} filePath - Input file path.\n * @param {string[]} allowedPathPrefixes - Allowed path prefixes.\n * @returns {boolean} - Whether input path is inside an allowed prefix.\n */\nfunction pathWithinAllowedPrefixes(filePath, allowedPathPrefixes) {\n  const resolvedPath = path.resolve(filePath)\n\n  return allowedPathPrefixes.some((allowedPrefix) => {\n    const resolvedPrefix = path.resolve(allowedPrefix)\n    const relativePath = path.relative(resolvedPrefix, resolvedPath)\n\n    if (!relativePath) return true\n    if (relativePath.startsWith(\"..\")) return false\n    if (path.isAbsolute(relativePath)) return false\n\n    return true\n  })\n}\n\nexport default class VelociousEnvironmentHandlerNode extends Base{\n  /** @type {import(\"node:async_hooks\").AsyncLocalStorage<TimezoneStore> | undefined} */\n  _timezoneAsyncLocalStorage = NodeAsyncLocalStorage ? new NodeAsyncLocalStorage() : undefined\n\n  /** @type {import(\"./base.js\").CommandFileObjectType[] | undefined} */\n  _findCommandsResult = undefined\n\n  /**\n   * @param {import(\"../configuration.js\").default} newConfiguration - New configuration.\n   * @returns {void} - No return value.\n   */\n  setConfiguration(newConfiguration) {\n    super.setConfiguration(newConfiguration)\n\n    if (!newConfiguration.getRouteResolverHooks().includes(frontendModelCommandRouteHook)) {\n      newConfiguration.addRouteResolverHook(frontendModelCommandRouteHook)\n    }\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {Promise<Buffer>} - File bytes.\n   */\n  async readAttachmentInputFile(filePath) {\n    return await fs.readFile(filePath)\n  }\n\n  /**\n   * @param {object} args - Args.\n   * @param {string[]} args.allowedPathPrefixes - Allowed path prefixes.\n   * @param {string} args.inputPath - Input path.\n   * @returns {Promise<{buffer: Buffer, filePath: string}>} - Resolved path and bytes.\n   */\n  async resolveAttachmentInputPath({allowedPathPrefixes, inputPath}) {\n    const filePath = path.resolve(inputPath)\n    const prefixes = Array.isArray(allowedPathPrefixes)\n      ? allowedPathPrefixes.filter((entry) => typeof entry === \"string\" && entry.length > 0)\n      : []\n\n    if (prefixes.length > 0 && !pathWithinAllowedPrefixes(filePath, prefixes)) {\n      throw new Error(\"Attachment path is outside allowed directories\")\n    }\n\n    const buffer = await this.readAttachmentInputFile(filePath)\n\n    return {buffer, filePath}\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").CommandFileObjectType>>} - Resolves with the commands.\n   */\n  async findCommands() {\n    this._findCommandsResult ||= await this._actualFindCommands()\n\n    if (!this._findCommandsResult) throw new Error(\"Could not get commands\")\n\n    return this._findCommandsResult\n  }\n\n  /**\n   * @param {number} offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @param {() => Promise<any>} callback - Callback to run.\n   * @returns {Promise<any>} - Result of the callback.\n   */\n  async runWithTimezoneOffset(offsetMinutes, callback) {\n    if (!this._timezoneAsyncLocalStorage) {\n      return await callback()\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    return await this._timezoneAsyncLocalStorage.run({\n      ability: existingStore?.ability,\n      offsetMinutes\n    }, callback)\n  }\n\n  /**\n   * @param {number} offsetMinutes - Offset in minutes (Date#getTimezoneOffset).\n   * @returns {void} - No return value.\n   */\n  setTimezoneOffset(offsetMinutes) {\n    if (!this._timezoneAsyncLocalStorage) return\n\n    const store = this._timezoneAsyncLocalStorage.getStore()\n\n    if (store) {\n      store.offsetMinutes = offsetMinutes\n    } else {\n      const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n      this._timezoneAsyncLocalStorage.enterWith({\n        ability: existingStore?.ability,\n        offsetMinutes\n      })\n    }\n  }\n\n  /**\n   * @param {import(\"../configuration.js\").default | undefined} configuration - Configuration instance.\n   * @returns {number} - Offset in minutes.\n   */\n  getTimezoneOffsetMinutes(configuration) {\n    if (this._timezoneAsyncLocalStorage) {\n      const store = this._timezoneAsyncLocalStorage.getStore()\n\n      if (store && typeof store.offsetMinutes === \"number\") {\n        return store.offsetMinutes\n      }\n    }\n\n    return super.getTimezoneOffsetMinutes(configuration)\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set for callback scope.\n   * @param {() => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async runWithAbility(ability, callback) {\n    if (!this._timezoneAsyncLocalStorage) {\n      return await super.runWithAbility(ability, callback)\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    return await this._timezoneAsyncLocalStorage.run({\n      ability,\n      offsetMinutes: existingStore?.offsetMinutes ?? this.getTimezoneOffsetMinutes(this.getConfiguration())\n    }, callback)\n  }\n\n  /**\n   * @param {import(\"../authorization/ability.js\").default | undefined} ability - Ability to set.\n   * @returns {void} - No return value.\n   */\n  setCurrentAbility(ability) {\n    if (!this._timezoneAsyncLocalStorage) {\n      super.setCurrentAbility(ability)\n      return\n    }\n\n    const existingStore = this._timezoneAsyncLocalStorage.getStore()\n\n    if (existingStore) {\n      existingStore.ability = ability\n    } else {\n      this._timezoneAsyncLocalStorage.enterWith({\n        ability,\n        offsetMinutes: this.getTimezoneOffsetMinutes(this.getConfiguration())\n      })\n    }\n  }\n\n  /**\n   * @returns {import(\"../authorization/ability.js\").default | undefined} - Current ability.\n   */\n  getCurrentAbility() {\n    if (!this._timezoneAsyncLocalStorage) {\n      return super.getCurrentAbility()\n    }\n\n    return this._timezoneAsyncLocalStorage.getStore()?.ability\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").CommandFileObjectType>>} - Resolves with discovered command files.\n   */\n  async _actualFindCommands() {\n    const basePath = await this.getBasePath()\n    const commandFiles = fs.glob(`${basePath}/src/cli/commands/**/*.js`)\n    const commands = []\n\n    for await (const aFilePath of commandFiles) {\n      const commandName = this.commandNameFromFilePath(aFilePath)\n\n      commands.push({name: commandName, file: aFilePath})\n    }\n\n    return commands\n  }\n\n  /**\n   * @param {string} filePath - Full command file path.\n   * @returns {string} - Parsed command name.\n   */\n  commandNameFromFilePath(filePath) {\n    const aFilePathParts = filePath.split(/[\\\\/]/)\n    const commandPathLocation = aFilePathParts.indexOf(\"commands\")\n\n    if (commandPathLocation === -1) {\n      throw new Error(`Could not parse command file path: ${filePath}`)\n    }\n\n    const commandParts = aFilePathParts.slice(commandPathLocation + 1)\n    const lastPart = commandParts[commandParts.length - 1]\n    let name, paths\n\n    if (lastPart == \"index.js\") {\n      name = commandParts[commandParts.length - 2]\n      paths = commandParts.slice(0, -2)\n    } else {\n      name = lastPart.replace(\".js\", \"\")\n      paths = commandParts.slice(0, -1)\n    }\n\n    return `${paths.join(\":\")}${paths.length > 0 ? \":\" : \"\"}${name}`\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsInit(command) {\n    return await this.forwardCommand(command, CliCommandsInit)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationGenerate(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateMigration)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsMigrationDestroy(command) {\n    return await this.forwardCommand(command, CliCommandsDestroyMigration)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateBaseModels(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateBaseModels)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateFrontendModels(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateFrontendModels)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsGenerateModel(command) {\n    return await this.forwardCommand(command, CliCommandsGenerateModel)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRoutes(command) {\n    return await this.forwardCommand(command, CliCommandsRoutes)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsConsole(command) {\n    return await this.forwardCommand(command, CliCommandsConsole)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsServer(command) {\n    return await this.forwardCommand(command, CliCommandsServer)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsTest(command) {\n    return await this.forwardCommand(command, CliCommandsTest)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsMain(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsMain)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsWorker(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsWorker)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsBackgroundJobsRunner(command) {\n    return await this.forwardCommand(command, CliCommandsBackgroundJobsRunner)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSchemaDump(command) {\n    return await this.forwardCommand(command, CliCommandsDbSchemaDump)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsDbSeed(command) {\n    return await this.forwardCommand(command, CliCommandsDbSeed)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunner(command) {\n    return await this.forwardCommand(command, CliCommandsRunner)\n  }\n\n  /**\n   * @param {import(\"../cli/base-command.js\").default} command - Command.\n   * @returns {Promise<unknown>} - Resolves with the command result.\n   */\n  async cliCommandsRunScript(command) {\n    return await this.forwardCommand(command, CliCommandsRunScript)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string[]} args.commandParts - Command parts.\n   * @returns {Promise<typeof import (\"../cli/base-command.js\").default>} - Resolves with the require command.\n   */\n  async requireCommand({commandParts}) {\n    const commands = await this.findCommands()\n    const commandName = commandParts.join(\":\")\n    const command = commands.find((aCommand) => aCommand.name === commandName)\n\n    if (!command) {\n      const possibleCommands = commands.map(aCommand => aCommand.name)\n\n      throw new Error(`Unknown command: ${commandParts.join(\":\")} which should have been one of: ${possibleCommands.sort().join(\", \")}`)\n    }\n\n    const commandClassImport = await import(toImportSpecifier(command.file))\n    const CommandClass = commandClassImport.default\n\n    return CommandClass\n  }\n\n  /**\n   * @returns {Promise<Array<import(\"./base.js\").MigrationObjectType>>} - Resolves with the migrations.\n   */\n  async findMigrations() {\n    const migrationsPath = `${this.getConfiguration().getDirectory()}/src/database/migrations`\n    const glob = await fs.glob(`${migrationsPath}/**/*.js`)\n    let files = []\n\n    for await (const fullPath of glob) {\n      const file = await path.basename(fullPath)\n\n      const match = file.match(/^(\\d{14})-(.+)\\.js$/)\n\n      if (!match) continue\n\n      const date = parseInt(match[1])\n      const migrationName = match[2]\n      const migrationClassName = inflection.camelize(migrationName.replaceAll(\"-\", \"_\"))\n\n      files.push({\n        file,\n        fullPath: `${migrationsPath}/${file}`,\n        date,\n        migrationClassName\n      })\n    }\n\n    files = files.sort((migration1, migration2) => migration1.date - migration2.date)\n\n    return files\n  }\n\n  /**\n   * @returns {Promise<import(\"../routes/index.js\").default>} - Resolves with the import application routes.\n   */\n  async importApplicationRoutes() {\n    const routesImport = await import(toImportSpecifier(`${this.getConfiguration().getDirectory()}/src/config/routes.js`))\n\n    return routesImport.default\n  }\n\n  /**\n   * @returns {Promise<string>} - Resolves with the velocious path.\n   */\n  async getVelociousPath() {\n    if (!this._velociousPath) {\n      const __filename = fileURLToPath(import.meta.url)\n      const __dirname = dirname(__filename)\n\n      this._velociousPath = await fs.realpath(`${__dirname}/../..`)\n    }\n\n    return this._velociousPath\n  }\n\n  /**\n   * @param {string[]} testFiles - Test files.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async importTestFiles(testFiles) {\n    for (const testFile of testFiles) {\n      await import(toImportSpecifier(testFile))\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @returns {string} - The default log directory.\n   */\n  getDefaultLogDirectory({configuration}) {\n    return path.join(configuration.getDirectory(), \"log\")\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string | undefined} args.directory - Directory path.\n   * @param {string} args.environment - Environment.\n   * @returns {string | undefined} - The log file path.\n   */\n  getLogFilePath({configuration, directory, environment}) {\n    const actualDirectory = directory || configuration?.getDirectory?.()\n\n    if (!actualDirectory) return undefined\n\n    return path.join(actualDirectory, `${environment}.log`)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.filePath - File path.\n   * @param {string} args.message - Message text.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async writeLogToFile({filePath, message}) {\n    await fs.mkdir(path.dirname(filePath), {recursive: true})\n    await fs.appendFile(filePath, `${message}\\n`, \"utf8\")\n  }\n\n  async importTestingConfigPath() {\n    const testingConfigPath = this.getConfiguration().getTesting()\n\n    if (!testingConfigPath) return\n\n    const testingImport = await import(toImportSpecifier(testingConfigPath))\n    const testingDefault = testingImport.default\n\n    if (!testingDefault) throw new Error(\"Testing config must export a default function\")\n    if (typeof testingDefault !== \"function\") throw new Error(\"Testing config default export isn't a function\")\n\n    const result = await testingDefault()\n\n    if (typeof result === \"function\") {\n      await result()\n    }\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {Promise<import(\"../database/migration/index.js\").default>} - Resolves with the require migration.\n   */\n  async requireMigration(filePath) {\n    const migrationImport = await import(toImportSpecifier(filePath))\n    const migrationImportDefault = migrationImport.default\n\n    if (!migrationImportDefault) throw new Error(\"Migration file must export a default migration class\")\n    if (typeof migrationImportDefault !== \"function\") throw new Error(\"Migration default export isn't a function (should be a class which is a function in JS)\")\n\n    return migrationImportDefault\n  }\n\n  async getBasePath() {\n    const __filename = fileURLToPath(import.meta.url)\n    const basePath = await fs.realpath(`${dirname(__filename)}/../..`)\n\n    return basePath\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async afterMigrations({dbs}) {\n    const configuration = this.getConfiguration()\n\n    if (!configuration.shouldWriteStructureSql()) return\n\n    const dbDir = path.join(configuration.getDirectory(), \"db\")\n    const structureSqlByIdentifier = await this._structureSqlByIdentifier({dbs})\n\n    await fs.mkdir(dbDir, {recursive: true})\n\n    for (const identifier of Object.keys(structureSqlByIdentifier)) {\n      const structureSql = structureSqlByIdentifier[identifier]\n\n      if (!structureSql) continue\n\n      const filePath = path.join(dbDir, `structure-${identifier}.sql`)\n\n      await fs.writeFile(filePath, structureSql)\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {Record<string, import(\"../database/drivers/base.js\").default>} args.dbs - Dbs.\n   * @returns {Promise<Record<string, string>>} - Resolves with SQL string.\n   */\n  async _structureSqlByIdentifier({dbs}) {\n    const sqlByIdentifier = /** @type {Record<string, string>} */ ({})\n\n    for (const identifier of Object.keys(dbs)) {\n      const db = dbs[identifier]\n\n      if (typeof db.structureSql !== \"function\") continue\n\n      const structureSql = await db.structureSql()\n\n      if (structureSql) {\n        sqlByIdentifier[identifier] = structureSql\n      }\n    }\n\n    return sqlByIdentifier\n  }\n}\n"]}
package/package.json CHANGED
@@ -3,7 +3,7 @@
3
3
  "velocious": "build/bin/velocious.js"
4
4
  },
5
5
  "name": "velocious",
6
- "version": "1.0.239",
6
+ "version": "1.0.240",
7
7
  "main": "build/index.js",
8
8
  "types": "build/index.d.ts",
9
9
  "files": [