@bgord/bun 1.2.6 → 1.2.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (79) hide show
  1. package/dist/content-hash-noop.adapter.d.ts +7 -0
  2. package/dist/content-hash-noop.adapter.d.ts.map +1 -0
  3. package/dist/content-hash-noop.adapter.js +6 -0
  4. package/dist/content-hash-noop.adapter.js.map +1 -0
  5. package/dist/content-hash-sha256-bun.adapter.d.ts +7 -0
  6. package/dist/content-hash-sha256-bun.adapter.d.ts.map +1 -0
  7. package/dist/content-hash-sha256-bun.adapter.js +8 -0
  8. package/dist/content-hash-sha256-bun.adapter.js.map +1 -0
  9. package/dist/content-hash.port.d.ts +7 -0
  10. package/dist/content-hash.port.d.ts.map +1 -0
  11. package/dist/content-hash.port.js +2 -0
  12. package/dist/content-hash.port.js.map +1 -0
  13. package/dist/decorators.service.d.ts +0 -3
  14. package/dist/decorators.service.d.ts.map +1 -1
  15. package/dist/decorators.service.js +0 -13
  16. package/dist/decorators.service.js.map +1 -1
  17. package/dist/file-draft-zip.service.d.ts +2 -1
  18. package/dist/file-draft-zip.service.d.ts.map +1 -1
  19. package/dist/file-draft-zip.service.js +2 -2
  20. package/dist/file-draft-zip.service.js.map +1 -1
  21. package/dist/file-draft.service.d.ts +3 -3
  22. package/dist/file-draft.service.d.ts.map +1 -1
  23. package/dist/file-draft.service.js +1 -1
  24. package/dist/file-draft.service.js.map +1 -1
  25. package/dist/file-hash-noop.adapter.js +1 -1
  26. package/dist/file-hash-noop.adapter.js.map +1 -1
  27. package/dist/file-uploader.middleware.d.ts +1 -1
  28. package/dist/file-uploader.middleware.d.ts.map +1 -1
  29. package/dist/file-uploader.middleware.js +2 -2
  30. package/dist/file-uploader.middleware.js.map +1 -1
  31. package/dist/index.d.ts +6 -0
  32. package/dist/index.d.ts.map +1 -1
  33. package/dist/index.js +6 -0
  34. package/dist/index.js.map +1 -1
  35. package/dist/markdown-generator-marked.adapter.d.ts +7 -0
  36. package/dist/markdown-generator-marked.adapter.d.ts.map +1 -0
  37. package/dist/markdown-generator-marked.adapter.js +12 -0
  38. package/dist/markdown-generator-marked.adapter.js.map +1 -0
  39. package/dist/markdown-generator-noop.adapter.d.ts +5 -0
  40. package/dist/markdown-generator-noop.adapter.d.ts.map +1 -0
  41. package/dist/markdown-generator-noop.adapter.js +6 -0
  42. package/dist/markdown-generator-noop.adapter.js.map +1 -0
  43. package/dist/markdown-generator.port.d.ts +4 -0
  44. package/dist/markdown-generator.port.d.ts.map +1 -0
  45. package/dist/markdown-generator.port.js +3 -0
  46. package/dist/markdown-generator.port.js.map +1 -0
  47. package/dist/prerequisites/dns.d.ts +16 -0
  48. package/dist/prerequisites/dns.d.ts.map +1 -0
  49. package/dist/prerequisites/dns.js +30 -0
  50. package/dist/prerequisites/dns.js.map +1 -0
  51. package/dist/prerequisites/index.d.ts +1 -0
  52. package/dist/prerequisites/index.d.ts.map +1 -1
  53. package/dist/prerequisites/index.js +1 -0
  54. package/dist/prerequisites/index.js.map +1 -1
  55. package/dist/prerequisites/ssl-certificate-expiry.d.ts +2 -2
  56. package/dist/prerequisites/ssl-certificate-expiry.d.ts.map +1 -1
  57. package/dist/prerequisites/ssl-certificate-expiry.js +3 -3
  58. package/dist/prerequisites/ssl-certificate-expiry.js.map +1 -1
  59. package/dist/remote-file-storage-noop.adapter.js +1 -1
  60. package/dist/remote-file-storage-noop.adapter.js.map +1 -1
  61. package/dist/tsconfig.tsbuildinfo +1 -1
  62. package/package.json +5 -3
  63. package/readme.md +7 -0
  64. package/src/content-hash-noop.adapter.ts +7 -0
  65. package/src/content-hash-sha256-bun.adapter.ts +10 -0
  66. package/src/content-hash.port.ts +5 -0
  67. package/src/decorators.service.ts +0 -15
  68. package/src/file-draft-zip.service.ts +3 -3
  69. package/src/file-draft.service.ts +2 -2
  70. package/src/file-hash-noop.adapter.ts +1 -1
  71. package/src/file-uploader.middleware.ts +3 -5
  72. package/src/index.ts +6 -0
  73. package/src/markdown-generator-marked.adapter.ts +15 -0
  74. package/src/markdown-generator-noop.adapter.ts +7 -0
  75. package/src/markdown-generator.port.ts +3 -0
  76. package/src/prerequisites/dns.ts +36 -0
  77. package/src/prerequisites/index.ts +1 -0
  78. package/src/prerequisites/ssl-certificate-expiry.ts +4 -4
  79. package/src/remote-file-storage-noop.adapter.ts +1 -1
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@bgord/bun",
3
- "version": "1.2.6",
3
+ "version": "1.2.8",
4
4
  "license": "MIT",
5
5
  "type": "module",
6
6
  "author": "Bartosz Gordon",
@@ -29,7 +29,7 @@
29
29
  "@types/yazl": "3.3.0",
30
30
  "cspell": "9.4.0",
31
31
  "knip": "5.71.0",
32
- "lefthook": "2.0.6",
32
+ "lefthook": "2.0.7",
33
33
  "only-allow": "1.2.2",
34
34
  "shellcheck": "4.1.0",
35
35
  "typescript": "5.9.3",
@@ -37,14 +37,16 @@
37
37
  },
38
38
  "dependencies": {
39
39
  "@axiomhq/winston": "1.3.1",
40
- "@bgord/tools": "1.1.9",
40
+ "@bgord/tools": "1.1.13",
41
41
  "@hono/ua-blocker": "0.1.20",
42
42
  "better-auth": "1.4.5",
43
43
  "croner": "9.1.0",
44
44
  "csv": "6.4.1",
45
45
  "hcaptcha": "0.2.0",
46
46
  "hono": "4.10.7",
47
+ "isomorphic-dompurify": "2.33.0",
47
48
  "lodash": "4.17.21",
49
+ "marked": "17.0.1",
48
50
  "node-cache": "5.1.2",
49
51
  "nodemailer": "7.0.11",
50
52
  "sharp": "0.34.5",
package/readme.md CHANGED
@@ -50,6 +50,9 @@ src/
50
50
  ├── command-envelope.ts
51
51
  ├── command-logger.service.ts
52
52
  ├── command.types.ts
53
+ ├── content-hash-noop.adapter.ts
54
+ ├── content-hash-sha256-bun.adapter.ts
55
+ ├── content-hash.port.ts
53
56
  ├── context.middleware.ts
54
57
  ├── correlation-id.vo.ts
55
58
  ├── correlation-storage.service.ts
@@ -148,6 +151,9 @@ src/
148
151
  ├── mailer-smtp.adapter.ts
149
152
  ├── mailer.port.ts
150
153
  ├── mailer.vo.ts
154
+ ├── markdown-generator-marked.adapter.ts
155
+ ├── markdown-generator-noop.adapter.ts
156
+ ├── markdown-generator.port.ts
151
157
  ├── memory-consumption.service.ts
152
158
  ├── modules
153
159
  │   ├── history
@@ -194,6 +200,7 @@ src/
194
200
  │   ├── clock-drift.ts
195
201
  │   ├── dependency-vulnerabilities.ts
196
202
  │   ├── directory.ts
203
+ │   ├── dns.ts
197
204
  │   ├── external-api.ts
198
205
  │   ├── jobs.ts
199
206
  │   ├── log-file.ts
@@ -0,0 +1,7 @@
1
+ import type { ContentHashPort } from "./content-hash.port";
2
+
3
+ export class ContentHashNoopAdapter implements ContentHashPort {
4
+ async hash(_content: string) {
5
+ return { etag: "noop" };
6
+ }
7
+ }
@@ -0,0 +1,10 @@
1
+ import type { ContentHashPort } from "./content-hash.port";
2
+
3
+ export class ContentHashSha256BunAdapter implements ContentHashPort {
4
+ async hash(content: string) {
5
+ const digest = await crypto.subtle.digest("SHA-256", new TextEncoder().encode(content));
6
+ const etag = Buffer.from(digest).toString("hex");
7
+
8
+ return { etag };
9
+ }
10
+ }
@@ -0,0 +1,5 @@
1
+ export type ContentHashResult = { etag: string };
2
+
3
+ export interface ContentHashPort {
4
+ hash(content: string): Promise<ContentHashResult>;
5
+ }
@@ -1,8 +1,6 @@
1
1
  import * as tools from "@bgord/tools";
2
2
  import type { LoggerPort } from "./logger.port";
3
3
 
4
- export class DecoratorTimeoutError extends Error {}
5
-
6
4
  export class Decorators {
7
5
  private readonly rounding = new tools.RoundToDecimal(2);
8
6
 
@@ -63,17 +61,4 @@ export class Decorators {
63
61
  };
64
62
  };
65
63
  }
66
-
67
- timeout(ms: number) {
68
- return (_target: any, _key: string, descriptor: PropertyDescriptor) => {
69
- const original: (...args: unknown[]) => unknown = descriptor.value;
70
-
71
- descriptor.value = async function (...args: any[]) {
72
- return await Promise.race([
73
- original.apply(this, args),
74
- new Promise((_, reject) => setTimeout(() => reject(new DecoratorTimeoutError()), ms)),
75
- ]);
76
- };
77
- };
78
- }
79
64
  }
@@ -6,8 +6,8 @@ import { FileDraft } from "./file-draft.service";
6
6
  export class FileDraftZip extends FileDraft {
7
7
  private readonly parts: FileDraft[];
8
8
 
9
- constructor(config: { filename: string; parts: FileDraft[] }) {
10
- super({ filename: config.filename, mime: new tools.Mime("application/zip") });
9
+ constructor(config: { filename: tools.Filename; parts: FileDraft[] }) {
10
+ super({ filename: config.filename, mime: tools.MIMES.text });
11
11
  this.parts = config.parts;
12
12
  }
13
13
 
@@ -17,7 +17,7 @@ export class FileDraftZip extends FileDraft {
17
17
  const chunks: Buffer[] = [];
18
18
 
19
19
  for (const part of this.parts) {
20
- zip.addReadStream((await part.create()) as Readable, part.config.filename);
20
+ zip.addReadStream((await part.create()) as Readable, part.config.filename.get());
21
21
  }
22
22
  zip.end();
23
23
 
@@ -4,12 +4,12 @@ import type * as tools from "@bgord/tools";
4
4
  export type DraftBody = BodyInit | NodeJS.ReadableStream | ReadableStream;
5
5
 
6
6
  export abstract class FileDraft {
7
- constructor(readonly config: { filename: string; mime: tools.Mime }) {}
7
+ constructor(readonly config: { filename: tools.Filename; mime: tools.Mime }) {}
8
8
 
9
9
  getHeaders(): Headers {
10
10
  return new Headers({
11
11
  "Content-Type": this.config.mime.toString(),
12
- "Content-Disposition": `attachment; filename="${this.config.filename}"`,
12
+ "Content-Disposition": `attachment; filename="${this.config.filename.get()}"`,
13
13
  });
14
14
  }
15
15
 
@@ -7,7 +7,7 @@ export class FileHashNoopAdapter implements FileHashPort {
7
7
  etag: "noop",
8
8
  size: tools.Size.fromBytes(10),
9
9
  lastModified: tools.Timestamp.fromNumber(1000),
10
- mime: new tools.Mime("text/plain"),
10
+ mime: tools.MIMES.text,
11
11
  };
12
12
  }
13
13
  }
@@ -3,7 +3,7 @@ import { bodyLimit } from "hono/body-limit";
3
3
  import { createMiddleware } from "hono/factory";
4
4
  import { HTTPException } from "hono/http-exception";
5
5
 
6
- type FileUploaderConfigType = { mimeTypes: string[]; maxFilesSize: tools.Size };
6
+ type FileUploaderConfigType = { mimeTypes: tools.Mime[]; maxFilesSize: tools.Size };
7
7
 
8
8
  export const InvalidFileMimeTypeError = new HTTPException(400, { message: "invalid_file_mime_type_error" });
9
9
  export const FileTooBigError = new HTTPException(400, { message: "file_too_big_error" });
@@ -25,10 +25,8 @@ export class FileUploader {
25
25
 
26
26
  if (!(file instanceof File)) throw InvalidFileMimeTypeError;
27
27
 
28
- const contentType = new tools.Mime(file.type);
29
- const accepted = config.mimeTypes.some((acceptedMimeType) =>
30
- new tools.Mime(acceptedMimeType).isSatisfiedBy(contentType),
31
- );
28
+ const contentType = tools.Mime.fromString(file.type);
29
+ const accepted = config.mimeTypes.some((accepted) => accepted.isSatisfiedBy(contentType));
32
30
 
33
31
  if (!accepted) throw InvalidFileMimeTypeError;
34
32
  return next();
package/src/index.ts CHANGED
@@ -23,6 +23,9 @@ export * from "./clock-system.adapter";
23
23
  export * from "./command.types";
24
24
  export * from "./command-envelope";
25
25
  export * from "./command-logger.service";
26
+ export * from "./content-hash.port";
27
+ export * from "./content-hash-noop.adapter";
28
+ export * from "./content-hash-sha256-bun.adapter";
26
29
  export * from "./context.middleware";
27
30
  export * from "./correlation-id.vo";
28
31
  export * from "./correlation-storage.service";
@@ -124,6 +127,9 @@ export * from "./mailer.vo";
124
127
  export * from "./mailer-noop.adapter";
125
128
  export * from "./mailer-smtp.adapter";
126
129
  export * from "./mailer-smtp-with-logger.adapter";
130
+ export * from "./markdown-generator.port";
131
+ export * from "./markdown-generator-marked.adapter";
132
+ export * from "./markdown-generator-noop.adapter";
127
133
  export * from "./memory-consumption.service";
128
134
  export * as History from "./modules/history";
129
135
  export * as Preferences from "./modules/preferences";
@@ -0,0 +1,15 @@
1
+ import DOMPurify from "isomorphic-dompurify";
2
+ import { Marked } from "marked";
3
+ import type { MarkdownGeneratorPort } from "./markdown-generator.port";
4
+
5
+ export class MarkdownGeneratorMarkedAdapter implements MarkdownGeneratorPort {
6
+ private readonly instance: Marked;
7
+
8
+ constructor() {
9
+ this.instance = new Marked();
10
+ }
11
+
12
+ async generate(content: string) {
13
+ return DOMPurify.sanitize(await this.instance.parse(content));
14
+ }
15
+ }
@@ -0,0 +1,7 @@
1
+ import type { MarkdownGeneratorPort } from "./markdown-generator.port";
2
+
3
+ export class MarkdownGeneratorNoopAdapter implements MarkdownGeneratorPort {
4
+ async generate(_content: string) {
5
+ return "<p>noop</p>";
6
+ }
7
+ }
@@ -0,0 +1,3 @@
1
+ export abstract class MarkdownGeneratorPort {
2
+ abstract generate(input: string): Promise<string>;
3
+ }
@@ -0,0 +1,36 @@
1
+ import dns from "dns/promises";
2
+ import * as tools from "@bgord/tools";
3
+ import type { ClockPort } from "../clock.port";
4
+ import * as prereqs from "../prerequisites.service";
5
+ import { Timeout } from "../timeout.service";
6
+
7
+ export class PrerequisiteDNS implements prereqs.Prerequisite {
8
+ readonly kind = "dns";
9
+ readonly label: prereqs.PrerequisiteLabelType;
10
+ readonly enabled?: boolean = true;
11
+
12
+ private readonly hostname: string;
13
+ readonly timeout: tools.Duration;
14
+
15
+ constructor(config: prereqs.PrerequisiteConfigType & { hostname: string; timeout?: tools.Duration }) {
16
+ this.label = config.label;
17
+ this.enabled = config.enabled === undefined ? true : config.enabled;
18
+
19
+ this.hostname = config.hostname;
20
+ this.timeout = config.timeout ?? tools.Duration.Seconds(1);
21
+ }
22
+
23
+ async verify(clock: ClockPort): Promise<prereqs.VerifyOutcome> {
24
+ const stopwatch = new tools.Stopwatch(clock.now());
25
+
26
+ if (!this.enabled) return prereqs.Verification.undetermined(stopwatch.stop());
27
+
28
+ try {
29
+ await Timeout.run(dns.lookup(this.hostname), this.timeout);
30
+
31
+ return prereqs.Verification.success(stopwatch.stop());
32
+ } catch (error) {
33
+ return prereqs.Verification.failure(stopwatch.stop(), error as Error);
34
+ }
35
+ }
36
+ }
@@ -3,6 +3,7 @@ export * from "./bun";
3
3
  export * from "./clock-drift";
4
4
  export * from "./dependency-vulnerabilities";
5
5
  export * from "./directory";
6
+ export * from "./dns";
6
7
  export * from "./external-api";
7
8
  export * from "./jobs";
8
9
  export * from "./log-file";
@@ -8,13 +8,13 @@ export class PrerequisiteSSLCertificateExpiry implements prereqs.Prerequisite {
8
8
  readonly label: prereqs.PrerequisiteLabelType;
9
9
  readonly enabled?: boolean = true;
10
10
 
11
- private readonly host: string;
11
+ private readonly hostname: string;
12
12
  private readonly days: number;
13
13
  private readonly CertificateInspector: CertificateInspectorPort;
14
14
 
15
15
  constructor(
16
16
  config: prereqs.PrerequisiteConfigType & {
17
- host: string;
17
+ hostname: string;
18
18
  days: number;
19
19
  CertificateInspector: CertificateInspectorPort;
20
20
  },
@@ -22,7 +22,7 @@ export class PrerequisiteSSLCertificateExpiry implements prereqs.Prerequisite {
22
22
  this.label = config.label;
23
23
  this.enabled = config.enabled === undefined ? true : config.enabled;
24
24
 
25
- this.host = config.host;
25
+ this.hostname = config.hostname;
26
26
  this.days = config.days;
27
27
  this.CertificateInspector = config.CertificateInspector;
28
28
  }
@@ -32,7 +32,7 @@ export class PrerequisiteSSLCertificateExpiry implements prereqs.Prerequisite {
32
32
 
33
33
  if (!this.enabled) return prereqs.Verification.undetermined(stopwatch.stop());
34
34
 
35
- const result = await this.CertificateInspector.inspect(this.host);
35
+ const result = await this.CertificateInspector.inspect(this.hostname);
36
36
 
37
37
  if (!result.success)
38
38
  return prereqs.Verification.failure(stopwatch.stop(), { message: "Certificate unavailable" });
@@ -35,7 +35,7 @@ export class RemoteFileStorageNoopAdapter implements RemoteFileStoragePort {
35
35
  etag: "noop",
36
36
  size: tools.Size.fromBytes(10),
37
37
  lastModified: this.deps.Clock.now(),
38
- mime: new tools.Mime("text/plain"),
38
+ mime: tools.MIMES.text,
39
39
  };
40
40
  }
41
41