@opengis/fastify-table 2.2.3 → 2.2.5

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.
@@ -1,3 +1,3 @@
1
- declare function isFileExists(filePath: string, options?: Record<string, string>): Promise<any>;
1
+ declare function isFileExists(filePath: string, options?: Record<string, any>): Promise<any>;
2
2
  export default isFileExists;
3
3
  //# sourceMappingURL=isFileExists.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"isFileExists.d.ts","sourceRoot":"","sources":["../../../../server/plugins/file/isFileExists.ts"],"names":[],"mappings":"AAGA,iBAAe,YAAY,CACzB,QAAQ,EAAE,MAAM,EAChB,OAAO,GAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAM,gBAcrC;AAED,eAAe,YAAY,CAAC"}
1
+ {"version":3,"file":"isFileExists.d.ts","sourceRoot":"","sources":["../../../../server/plugins/file/isFileExists.ts"],"names":[],"mappings":"AAGA,iBAAe,YAAY,CACzB,QAAQ,EAAE,MAAM,EAChB,OAAO,GAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAM,gBAclC;AAED,eAAe,YAAY,CAAC"}
@@ -1,5 +1,6 @@
1
1
  import fs from "node:fs";
2
2
  export default function fsStorage(): {
3
+ name: string;
3
4
  deleteFile: (fp: string) => Promise<void>;
4
5
  downloadFile: (fp: string, options?: Record<string, any>) => Promise<NonSharedBuffer | fs.ReadStream | {
5
6
  original: string;
@@ -1 +1 @@
1
- {"version":3,"file":"fs.d.ts","sourceRoot":"","sources":["../../../../../server/plugins/file/providers/fs.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AAuGzB,MAAM,CAAC,OAAO,UAAU,SAAS;qBAtFG,MAAM;uBAsB7B,MAAM,YAAW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;;;;qBAyBpC,MAAM,QAAQ,GAAG;2BALX,GAAG,QAAO,GAAG;qBAgCI,MAAM;mBAlER,MAAM,gBAAgB,MAAM;4BAqEnB,GAAG;yBAGN,GAAG;wBAGJ,GAAG;EAezC"}
1
+ {"version":3,"file":"fs.d.ts","sourceRoot":"","sources":["../../../../../server/plugins/file/providers/fs.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AAuGzB,MAAM,CAAC,OAAO,UAAU,SAAS;;qBAtFG,MAAM;uBAsB7B,MAAM,YAAW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;;;;qBAyBpC,MAAM,QAAQ,GAAG;2BALX,GAAG,QAAO,GAAG;qBAgCI,MAAM;mBAlER,MAAM,gBAAgB,MAAM;4BAqEnB,GAAG;yBAGN,GAAG;wBAGJ,GAAG;EAgBzC"}
@@ -75,6 +75,7 @@ const getMDate = () => async (filepath) => new Date((await fsp.stat(getPath(file
75
75
  const readdir = () => async (filepath) => fsp.readdir(getPath(filepath) || "");
76
76
  export default function fsStorage() {
77
77
  return {
78
+ name: "fs",
78
79
  deleteFile: deleteFile(),
79
80
  downloadFile: downloadFile(),
80
81
  uploadFile: uploadFile(),
@@ -1 +1 @@
1
- {"version":3,"file":"uploadMultiPart.d.ts","sourceRoot":"","sources":["../../../../server/plugins/file/uploadMultiPart.ts"],"names":[],"mappings":"AAqBA,MAAM,WAAW,KAAK;IACpB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AA2CD,wBAAsB,eAAe,CACnC,GAAG,EAAE,GAAG,EACR,EACE,UAAU,EACV,GAAG,EACH,MAAM,EACN,gBAAwB,GACzB,GAAE;IACD,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,GAAG,CAAC,EAAE,OAAO,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,gBAAgB,CAAC,EAAE,OAAO,CAAC;CACvB,GACL,OAAO,CAAC,KAAK,CAAC,CAmGhB;AAED,eAAe,eAAe,CAAC"}
1
+ {"version":3,"file":"uploadMultiPart.d.ts","sourceRoot":"","sources":["../../../../server/plugins/file/uploadMultiPart.ts"],"names":[],"mappings":"AAqBA,MAAM,WAAW,KAAK;IACpB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AA+DD,wBAAsB,eAAe,CACnC,GAAG,EAAE,GAAG,EACR,EACE,UAAU,EACV,GAAG,EACH,MAAM,EACN,gBAAwB,GACzB,GAAE;IACD,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,GAAG,CAAC,EAAE,OAAO,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,gBAAgB,CAAC,EAAE,OAAO,CAAC;CACvB,GACL,OAAO,CAAC,KAAK,CAAC,CAoGhB;AAED,eAAe,eAAe,CAAC"}
@@ -1,6 +1,6 @@
1
1
  /* eslint-disable no-console */
2
2
  import path from "node:path";
3
- import { mkdir, writeFile } from "node:fs/promises";
3
+ import { mkdir, writeFile, readdir } from "node:fs/promises";
4
4
  import { randomUUID } from "node:crypto";
5
5
  import { imageSize } from "image-size";
6
6
  import { existsSync } from "node:fs";
@@ -36,6 +36,20 @@ async function writeFileToDisk(file, buffer) {
36
36
  await writeFile(file.filepath, buffer);
37
37
  return null;
38
38
  }
39
+ async function rotate(filename, dirpath) {
40
+ const { name, ext } = path.parse(filename);
41
+ // ? read whole dir once, rotate later - better works for large dirs, worse - for small ones
42
+ // ? in practice, latency is more consistent than w/ multiple existsSync calls
43
+ const files = new Set(await readdir(dirpath));
44
+ if (!files.has(filename)) {
45
+ return filename;
46
+ }
47
+ let idx = 1;
48
+ while (files.has(`${name}_${idx}${ext}`)) {
49
+ idx++;
50
+ }
51
+ return `${name}_${idx}${ext}`;
52
+ }
39
53
  export async function uploadMultiPart(req, { extensions, raw, subdir, originalFilename = false, } = {}) {
40
54
  const allowedExtensions = req.routeOptions?.url === "/file/upload-image/*" ? images : all;
41
55
  const parts = req.parts();
@@ -73,6 +87,7 @@ export async function uploadMultiPart(req, { extensions, raw, subdir, originalFi
73
87
  buffer: fileBuffer,
74
88
  filename: value.filename,
75
89
  extension: ext.substring(1),
90
+ mimetype: value.mimetype,
76
91
  fields: value.fields,
77
92
  };
78
93
  }
@@ -83,7 +98,7 @@ export async function uploadMultiPart(req, { extensions, raw, subdir, originalFi
83
98
  const reldirpath = path.join("/files", dir, yearMonthDay);
84
99
  const folder = path.join(rootDir, config.folder || "", reldirpath);
85
100
  const newFilename = originalFilename
86
- ? value.filename
101
+ ? await rotate(value.filename, folder)
87
102
  : `${randomUUID()}${ext}`;
88
103
  if (existsSync(path.join(folder, newFilename).replace(/\\/g, "/"))) {
89
104
  throw new BadRequestError("file with specified name already exists in directory");
@@ -17,10 +17,13 @@ import type { FastifyReply } from "fastify";
17
17
  * @returns {Object} headers Заголовки HTTP
18
18
  * @returns {String} pipe Шлях до файла для скачування або відображення
19
19
  */
20
- export default function getFile({ params, user }: {
20
+ export default function getFile({ params, query, user, }: {
21
21
  params: {
22
22
  "*": string;
23
23
  };
24
+ query: {
25
+ debug: string;
26
+ };
24
27
  user?: Record<string, any>;
25
28
  }, reply: FastifyReply): Promise<any>;
26
29
  //# sourceMappingURL=files.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"files.d.ts","sourceRoot":"","sources":["../../../../../server/routes/file/controllers/files.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAgB5C;;;;;;;;;;;;;;;;;GAiBG;AAEH,wBAA8B,OAAO,CACnC,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE;IAAE,MAAM,EAAE;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;CAAE,EACzE,KAAK,EAAE,YAAY,gBAgDpB"}
1
+ {"version":3,"file":"files.d.ts","sourceRoot":"","sources":["../../../../../server/routes/file/controllers/files.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAkB5C;;;;;;;;;;;;;;;;;GAiBG;AAEH,wBAA8B,OAAO,CACnC,EACE,MAAM,EACN,KAAK,EACL,IAAI,GACL,EAAE;IACD,MAAM,EAAE;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC;IACxB,KAAK,EAAE;QAAE,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC;IACzB,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;CAC5B,EACD,KAAK,EAAE,YAAY,gBA+DpB"}
@@ -2,8 +2,10 @@ import path from "node:path";
2
2
  import mime from "../../../plugins/file/providers/mime/index.js";
3
3
  // import isFileExists from "../../../plugins/file/isFileExists.js";
4
4
  import downloadFile from "../../../plugins/file/downloadFile.js";
5
+ import config from "../../../../config.js";
5
6
  import logger from "../../../plugins/logger/getLogger.js";
6
7
  import applyHook from "../../../plugins/hook/applyHook.js";
8
+ import providers from "../../../plugins/file/providers/index.js";
7
9
  const grayGif = Buffer.from("R0lGODdhAwADAIEAAICAgAAAAAAAAAAAACwAAAAAAwADAAAIBwABCBw4MCAAOw==", "base64");
8
10
  /**
9
11
  * Апі використовується для отримання різних файлів і можливістю змінювати їх
@@ -23,7 +25,7 @@ const grayGif = Buffer.from("R0lGODdhAwADAIEAAICAgAAAAAAAAAAAACwAAAAAAwADAAAIBwA
23
25
  * @returns {Object} headers Заголовки HTTP
24
26
  * @returns {String} pipe Шлях до файла для скачування або відображення
25
27
  */
26
- export default async function getFile({ params, user }, reply) {
28
+ export default async function getFile({ params, query, user, }, reply) {
27
29
  if (!params?.["*"]) {
28
30
  return reply.status(400).send({ error: "not enough params", code: 400 });
29
31
  }
@@ -43,6 +45,18 @@ export default async function getFile({ params, user }, reply) {
43
45
  if (hookData)
44
46
  return hookData;
45
47
  const filepath = path.join("files", relpath);
48
+ if ((config.local || user?.user_type?.includes?.("admin")) && query.debug) {
49
+ const fp = providers();
50
+ const existsS3 = fp.name === "s3"
51
+ ? await fp
52
+ .fileExists(filepath, { fallback: false })
53
+ .catch(() => false)
54
+ .then((el) => !!el)
55
+ : undefined;
56
+ const { fileExists } = providers({ provider: "fs" });
57
+ const existsFS = await fileExists(filepath);
58
+ return reply.status(200).send({ provider: fp.name, existsS3, existsFS });
59
+ }
46
60
  const fileStream = await downloadFile(filepath).catch();
47
61
  if (!fileStream) {
48
62
  // return reply.status(404).send({ error: "Файл не знайдено", code: 404 });
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@opengis/fastify-table",
3
- "version": "2.2.3",
3
+ "version": "2.2.5",
4
4
  "type": "module",
5
5
  "description": "core-plugins",
6
6
  "keywords": [