nest-multi-storage 1.0.0

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 (54) hide show
  1. package/README.md +236 -0
  2. package/dist/drivers/azure-blob.driver.d.ts +27 -0
  3. package/dist/drivers/azure-blob.driver.d.ts.map +1 -0
  4. package/dist/drivers/azure-blob.driver.js +153 -0
  5. package/dist/drivers/azure-blob.driver.js.map +1 -0
  6. package/dist/drivers/index.d.ts +4 -0
  7. package/dist/drivers/index.d.ts.map +1 -0
  8. package/dist/drivers/index.js +20 -0
  9. package/dist/drivers/index.js.map +1 -0
  10. package/dist/drivers/local.driver.d.ts +23 -0
  11. package/dist/drivers/local.driver.d.ts.map +1 -0
  12. package/dist/drivers/local.driver.js +165 -0
  13. package/dist/drivers/local.driver.js.map +1 -0
  14. package/dist/drivers/s3.driver.d.ts +24 -0
  15. package/dist/drivers/s3.driver.d.ts.map +1 -0
  16. package/dist/drivers/s3.driver.js +178 -0
  17. package/dist/drivers/s3.driver.js.map +1 -0
  18. package/dist/index.d.ts +7 -0
  19. package/dist/index.d.ts.map +1 -0
  20. package/dist/index.js +23 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/interfaces/index.d.ts +3 -0
  23. package/dist/interfaces/index.d.ts.map +1 -0
  24. package/dist/interfaces/index.js +19 -0
  25. package/dist/interfaces/index.js.map +1 -0
  26. package/dist/interfaces/storage-driver.interface.d.ts +31 -0
  27. package/dist/interfaces/storage-driver.interface.d.ts.map +1 -0
  28. package/dist/interfaces/storage-driver.interface.js +3 -0
  29. package/dist/interfaces/storage-driver.interface.js.map +1 -0
  30. package/dist/interfaces/storage-options.interface.d.ts +42 -0
  31. package/dist/interfaces/storage-options.interface.d.ts.map +1 -0
  32. package/dist/interfaces/storage-options.interface.js +3 -0
  33. package/dist/interfaces/storage-options.interface.js.map +1 -0
  34. package/dist/storage.constants.d.ts +4 -0
  35. package/dist/storage.constants.d.ts.map +1 -0
  36. package/dist/storage.constants.js +7 -0
  37. package/dist/storage.constants.js.map +1 -0
  38. package/dist/storage.module.d.ts +9 -0
  39. package/dist/storage.module.d.ts.map +1 -0
  40. package/dist/storage.module.js +106 -0
  41. package/dist/storage.module.js.map +1 -0
  42. package/dist/storage.service.d.ts +26 -0
  43. package/dist/storage.service.d.ts.map +1 -0
  44. package/dist/storage.service.js +89 -0
  45. package/dist/storage.service.js.map +1 -0
  46. package/dist/utils/index.d.ts +2 -0
  47. package/dist/utils/index.d.ts.map +1 -0
  48. package/dist/utils/index.js +18 -0
  49. package/dist/utils/index.js.map +1 -0
  50. package/dist/utils/path.utils.d.ts +7 -0
  51. package/dist/utils/path.utils.d.ts.map +1 -0
  52. package/dist/utils/path.utils.js +61 -0
  53. package/dist/utils/path.utils.js.map +1 -0
  54. package/package.json +63 -0
package/README.md ADDED
@@ -0,0 +1,236 @@
1
+ # @c-nutthapol/nest-multi-storage
2
+
3
+ A flexible multi-driver storage module for NestJS that supports **Amazon S3**, **Azure Blob Storage**, and **Local File System** with a unified API.
4
+
5
+ ## Features
6
+
7
+ - 🔌 **Multiple Storage Drivers**: Support for Local, S3, and Azure Blob Storage
8
+ - 💾 **Multi-Disk Configuration**: Configure multiple disks with different drivers
9
+ - 🔄 **Unified API**: Same interface for all storage operations
10
+ - 🔐 **Signed URLs**: Generate temporary access URLs for S3 and Azure
11
+ - 📦 **Stream Support**: Handle large files with streaming
12
+ - 🧩 **NestJS Integration**: Dynamic module with `forRoot` and `forRootAsync`
13
+
14
+ ## Installation
15
+
16
+ ```bash
17
+ npm install @c-nutthapol/nest-multi-storage
18
+
19
+ # Peer dependencies
20
+ npm install @nestjs/common @nestjs/core
21
+ ```
22
+
23
+ ## Quick Start
24
+
25
+ ### 1. Import and Configure
26
+
27
+ ```typescript
28
+ // app.module.ts
29
+ import { Module } from '@nestjs/common';
30
+ import { StorageModule } from '@c-nutthapol/nest-multi-storage';
31
+
32
+ @Module({
33
+ imports: [
34
+ StorageModule.forRoot({
35
+ default: 'local',
36
+ disks: {
37
+ local: {
38
+ driver: 'local',
39
+ basePath: './storage',
40
+ baseUrl: 'http://localhost:3000/files',
41
+ },
42
+ s3: {
43
+ driver: 's3',
44
+ region: 'ap-southeast-1',
45
+ bucket: 'my-bucket',
46
+ credentials: {
47
+ accessKeyId: process.env.AWS_ACCESS_KEY_ID!,
48
+ secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY!,
49
+ },
50
+ },
51
+ azure: {
52
+ driver: 'azure',
53
+ connectionString: process.env.AZURE_STORAGE_CONNECTION_STRING!,
54
+ container: 'my-container',
55
+ },
56
+ },
57
+ }),
58
+ ],
59
+ })
60
+ export class AppModule {}
61
+ ```
62
+
63
+ ### 2. Use in Services
64
+
65
+ ```typescript
66
+ import { Injectable } from '@nestjs/common';
67
+ import { StorageService } from '@c-nutthapol/nest-multi-storage';
68
+
69
+ @Injectable()
70
+ export class FileService {
71
+ constructor(private storage: StorageService) {}
72
+
73
+ async uploadFile(file: Express.Multer.File) {
74
+ // Upload to default disk
75
+ const result = await this.storage.put(
76
+ `uploads/${file.originalname}`,
77
+ file.buffer,
78
+ { contentType: file.mimetype },
79
+ );
80
+
81
+ return { url: result.url };
82
+ }
83
+
84
+ async uploadToS3(file: Express.Multer.File) {
85
+ // Upload to specific disk
86
+ const result = await this.storage
87
+ .disk('s3')
88
+ .put(`uploads/${file.originalname}`, file.buffer);
89
+
90
+ // Get signed URL (valid for 1 hour)
91
+ const signedUrl = await this.storage
92
+ .disk('s3')
93
+ .signedUrl(`uploads/${file.originalname}`, 3600);
94
+
95
+ return { url: result.url, signedUrl };
96
+ }
97
+
98
+ async getFile(path: string) {
99
+ // Check if file exists
100
+ if (await this.storage.exists(path)) {
101
+ return this.storage.get(path);
102
+ }
103
+ throw new Error('File not found');
104
+ }
105
+
106
+ async deleteFile(path: string) {
107
+ return this.storage.delete(path);
108
+ }
109
+ }
110
+ ```
111
+
112
+ ## Async Configuration
113
+
114
+ ```typescript
115
+ import { ConfigModule, ConfigService } from '@nestjs/config';
116
+ import { StorageModule } from '@c-nutthapol/nest-multi-storage';
117
+
118
+ @Module({
119
+ imports: [
120
+ ConfigModule.forRoot(),
121
+ StorageModule.forRootAsync({
122
+ imports: [ConfigModule],
123
+ inject: [ConfigService],
124
+ useFactory: (config: ConfigService) => ({
125
+ default: config.get('STORAGE_DISK', 'local'),
126
+ disks: {
127
+ local: {
128
+ driver: 'local',
129
+ basePath: config.get('STORAGE_PATH', './storage'),
130
+ },
131
+ s3: {
132
+ driver: 's3',
133
+ region: config.get('AWS_REGION'),
134
+ bucket: config.get('AWS_BUCKET'),
135
+ credentials: {
136
+ accessKeyId: config.get('AWS_ACCESS_KEY_ID'),
137
+ secretAccessKey: config.get('AWS_SECRET_ACCESS_KEY'),
138
+ },
139
+ },
140
+ },
141
+ }),
142
+ }),
143
+ ],
144
+ })
145
+ export class AppModule {}
146
+ ```
147
+
148
+ ## API Reference
149
+
150
+ ### StorageService Methods
151
+
152
+ | Method | Description |
153
+ |--------|-------------|
154
+ | `disk(name)` | Switch to a specific disk |
155
+ | `put(path, content, options?)` | Store a file |
156
+ | `get(path)` | Get file content as Buffer |
157
+ | `getStream(path)` | Get file content as Readable stream |
158
+ | `delete(path)` | Delete a file |
159
+ | `deleteMany(paths)` | Delete multiple files |
160
+ | `exists(path)` | Check if file exists |
161
+ | `url(path)` | Get public URL |
162
+ | `signedUrl(path, expiresIn?)` | Get temporary signed URL |
163
+ | `copy(from, to)` | Copy a file |
164
+ | `move(from, to)` | Move a file |
165
+ | `stat(path)` | Get file statistics |
166
+ | `list(prefix?)` | List files |
167
+
168
+ ### Driver Configuration
169
+
170
+ #### Local Driver
171
+
172
+ ```typescript
173
+ {
174
+ driver: 'local',
175
+ basePath: './storage', // Required: Base directory for storage
176
+ baseUrl: 'http://...', // Optional: Public URL prefix
177
+ }
178
+ ```
179
+
180
+ #### S3 Driver
181
+
182
+ ```typescript
183
+ {
184
+ driver: 's3',
185
+ region: 'ap-southeast-1', // Required: AWS region
186
+ bucket: 'my-bucket', // Required: Bucket name
187
+ credentials: { // Optional: Use for explicit credentials
188
+ accessKeyId: '...',
189
+ secretAccessKey: '...',
190
+ },
191
+ endpoint: 'http://...', // Optional: Custom endpoint (MinIO, etc.)
192
+ forcePathStyle: false, // Optional: Use path-style URLs
193
+ cdnUrl: 'https://cdn...', // Optional: CDN URL for public files
194
+ signedUrlExpiry: 3600, // Optional: Default signed URL expiry (seconds)
195
+ }
196
+ ```
197
+
198
+ #### Azure Blob Storage Driver
199
+
200
+ ```typescript
201
+ {
202
+ driver: 'azure',
203
+ connectionString: '...', // Required: Azure connection string
204
+ container: 'my-container', // Required: Container name
205
+ accountName: 'account', // Optional: Explicit account name for SAS
206
+ sasUrlExpiry: 3600, // Optional: Default SAS URL expiry (seconds)
207
+ }
208
+ ```
209
+
210
+ ## Stream Support
211
+
212
+ ```typescript
213
+ import { createReadStream } from 'fs';
214
+
215
+ // Upload from stream
216
+ const fileStream = createReadStream('/path/to/large-file.zip');
217
+ await this.storage.put('backups/large-file.zip', fileStream);
218
+
219
+ // Download as stream
220
+ const downloadStream = await this.storage.getStream('backups/large-file.zip');
221
+ downloadStream.pipe(response); // Pipe to HTTP response
222
+ ```
223
+
224
+ ## Testing
225
+
226
+ ```bash
227
+ # Run tests
228
+ npm run test
229
+
230
+ # Run tests with coverage
231
+ npm run test:cov
232
+ ```
233
+
234
+ ## License
235
+
236
+ MIT
@@ -0,0 +1,27 @@
1
+ import { Readable } from 'stream';
2
+ import { StorageDriver, PutOptions, StorageResponse, FileStats } from '../interfaces/storage-driver.interface';
3
+ import { AzureDriverOptions } from '../interfaces/storage-options.interface';
4
+ export declare class AzureBlobDriver implements StorageDriver {
5
+ private readonly containerClient;
6
+ private readonly sasUrlExpiry;
7
+ private readonly connectionString;
8
+ private readonly containerName;
9
+ private readonly accountName?;
10
+ constructor(options: AzureDriverOptions);
11
+ private normalizeBlobName;
12
+ private parseAccountKey;
13
+ private parseAccountName;
14
+ put(path: string, content: Buffer | Readable, options?: PutOptions): Promise<StorageResponse>;
15
+ get(path: string): Promise<Buffer>;
16
+ getStream(path: string): Promise<Readable>;
17
+ delete(path: string): Promise<boolean>;
18
+ deleteMany(paths: string[]): Promise<boolean>;
19
+ exists(path: string): Promise<boolean>;
20
+ url(path: string): string;
21
+ signedUrl(path: string, expiresIn?: number): Promise<string>;
22
+ copy(from: string, to: string): Promise<StorageResponse>;
23
+ move(from: string, to: string): Promise<StorageResponse>;
24
+ stat(path: string): Promise<FileStats>;
25
+ list(prefix?: string): Promise<string[]>;
26
+ }
27
+ //# sourceMappingURL=azure-blob.driver.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"azure-blob.driver.d.ts","sourceRoot":"","sources":["../../src/drivers/azure-blob.driver.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AAQlC,OAAO,EACH,aAAa,EACb,UAAU,EACV,eAAe,EACf,SAAS,EACZ,MAAM,wCAAwC,CAAC;AAChD,OAAO,EAAE,kBAAkB,EAAE,MAAM,yCAAyC,CAAC;AAO7E,qBAAa,eAAgB,YAAW,aAAa;IACjD,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAkB;IAClD,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAS;IACtC,OAAO,CAAC,QAAQ,CAAC,gBAAgB,CAAS;IAC1C,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAS;IACvC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAS;gBAE1B,OAAO,EAAE,kBAAkB;IAYvC,OAAO,CAAC,iBAAiB;IAOzB,OAAO,CAAC,eAAe;IAQvB,OAAO,CAAC,gBAAgB;IAMlB,GAAG,CACL,IAAI,EAAE,MAAM,EACZ,OAAO,EAAE,MAAM,GAAG,QAAQ,EAC1B,OAAO,CAAC,EAAE,UAAU,GACrB,OAAO,CAAC,eAAe,CAAC;IA0BrB,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAelC,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IAQ1C,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAYtC,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,OAAO,CAAC;IAK7C,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAO5C,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAMnB,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IA8B5D,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC;IAexD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC;IAMxD,IAAI,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC;IActC,IAAI,CAAC,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;CAUjD"}
@@ -0,0 +1,153 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AzureBlobDriver = void 0;
4
+ const storage_blob_1 = require("@azure/storage-blob");
5
+ const storage_constants_1 = require("../storage.constants");
6
+ const path_utils_1 = require("../utils/path.utils");
7
+ class AzureBlobDriver {
8
+ constructor(options) {
9
+ const blobServiceClient = storage_blob_1.BlobServiceClient.fromConnectionString(options.connectionString);
10
+ this.containerClient = blobServiceClient.getContainerClient(options.container);
11
+ this.sasUrlExpiry = options.sasUrlExpiry ?? storage_constants_1.DEFAULT_SIGNED_URL_EXPIRY;
12
+ this.connectionString = options.connectionString;
13
+ this.containerName = options.container;
14
+ this.accountName = options.accountName;
15
+ }
16
+ normalizeBlobName(name) {
17
+ return (0, path_utils_1.trimLeadingSlash)(name);
18
+ }
19
+ parseAccountKey() {
20
+ const match = this.connectionString.match(/AccountKey=([^;]+)/);
21
+ return match?.[1];
22
+ }
23
+ parseAccountName() {
24
+ if (this.accountName)
25
+ return this.accountName;
26
+ const match = this.connectionString.match(/AccountName=([^;]+)/);
27
+ return match?.[1];
28
+ }
29
+ async put(path, content, options) {
30
+ const blobName = this.normalizeBlobName(path);
31
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
32
+ if (Buffer.isBuffer(content)) {
33
+ await blockBlobClient.uploadData(content, {
34
+ blobHTTPHeaders: {
35
+ blobContentType: options?.contentType,
36
+ },
37
+ metadata: options?.metadata,
38
+ });
39
+ }
40
+ else {
41
+ await blockBlobClient.uploadStream(content, undefined, undefined, {
42
+ blobHTTPHeaders: {
43
+ blobContentType: options?.contentType,
44
+ },
45
+ metadata: options?.metadata,
46
+ });
47
+ }
48
+ return {
49
+ path: blobName,
50
+ url: this.url(blobName),
51
+ };
52
+ }
53
+ async get(path) {
54
+ const blobName = this.normalizeBlobName(path);
55
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
56
+ const downloadResponse = await blockBlobClient.download();
57
+ const stream = downloadResponse.readableStreamBody;
58
+ return new Promise((resolve, reject) => {
59
+ const chunks = [];
60
+ stream.on('data', (chunk) => chunks.push(chunk));
61
+ stream.on('end', () => resolve(Buffer.concat(chunks)));
62
+ stream.on('error', reject);
63
+ });
64
+ }
65
+ async getStream(path) {
66
+ const blobName = this.normalizeBlobName(path);
67
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
68
+ const downloadResponse = await blockBlobClient.download();
69
+ return downloadResponse.readableStreamBody;
70
+ }
71
+ async delete(path) {
72
+ const blobName = this.normalizeBlobName(path);
73
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
74
+ try {
75
+ await blockBlobClient.delete();
76
+ return true;
77
+ }
78
+ catch {
79
+ return false;
80
+ }
81
+ }
82
+ async deleteMany(paths) {
83
+ const results = await Promise.all(paths.map((p) => this.delete(p)));
84
+ return results.every((r) => r);
85
+ }
86
+ async exists(path) {
87
+ const blobName = this.normalizeBlobName(path);
88
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
89
+ return blockBlobClient.exists();
90
+ }
91
+ url(path) {
92
+ const blobName = this.normalizeBlobName(path);
93
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
94
+ return blockBlobClient.url;
95
+ }
96
+ async signedUrl(path, expiresIn) {
97
+ const blobName = this.normalizeBlobName(path);
98
+ const expiry = expiresIn ?? this.sasUrlExpiry;
99
+ const accountName = this.parseAccountName();
100
+ const accountKey = this.parseAccountKey();
101
+ if (!accountName || !accountKey) {
102
+ return this.url(blobName);
103
+ }
104
+ const sharedKeyCredential = new storage_blob_1.StorageSharedKeyCredential(accountName, accountKey);
105
+ const startsOn = new Date();
106
+ const expiresOn = new Date(startsOn.getTime() + expiry * 1000);
107
+ const sasToken = (0, storage_blob_1.generateBlobSASQueryParameters)({
108
+ containerName: this.containerName,
109
+ blobName: blobName,
110
+ permissions: storage_blob_1.BlobSASPermissions.parse('r'),
111
+ startsOn,
112
+ expiresOn,
113
+ }, sharedKeyCredential).toString();
114
+ return `${this.url(blobName)}?${sasToken}`;
115
+ }
116
+ async copy(from, to) {
117
+ const fromBlobName = this.normalizeBlobName(from);
118
+ const toBlobName = this.normalizeBlobName(to);
119
+ const sourceClient = this.containerClient.getBlockBlobClient(fromBlobName);
120
+ const destClient = this.containerClient.getBlockBlobClient(toBlobName);
121
+ await destClient.beginCopyFromURL(sourceClient.url);
122
+ return {
123
+ path: toBlobName,
124
+ url: this.url(toBlobName),
125
+ };
126
+ }
127
+ async move(from, to) {
128
+ const result = await this.copy(from, to);
129
+ await this.delete(from);
130
+ return result;
131
+ }
132
+ async stat(path) {
133
+ const blobName = this.normalizeBlobName(path);
134
+ const blockBlobClient = this.containerClient.getBlockBlobClient(blobName);
135
+ const properties = await blockBlobClient.getProperties();
136
+ return {
137
+ size: properties.contentLength ?? 0,
138
+ lastModified: properties.lastModified ?? new Date(),
139
+ contentType: properties.contentType,
140
+ metadata: properties.metadata,
141
+ };
142
+ }
143
+ async list(prefix) {
144
+ const normalizedPrefix = prefix ? this.normalizeBlobName(prefix) : undefined;
145
+ const files = [];
146
+ for await (const blob of this.containerClient.listBlobsFlat({ prefix: normalizedPrefix })) {
147
+ files.push(blob.name);
148
+ }
149
+ return files;
150
+ }
151
+ }
152
+ exports.AzureBlobDriver = AzureBlobDriver;
153
+ //# sourceMappingURL=azure-blob.driver.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"azure-blob.driver.js","sourceRoot":"","sources":["../../src/drivers/azure-blob.driver.ts"],"names":[],"mappings":";;;AACA,sDAM6B;AAQ7B,4DAAiE;AACjE,oDAAuD;AAKvD,MAAa,eAAe;IAOxB,YAAY,OAA2B;QACnC,MAAM,iBAAiB,GAAG,gCAAiB,CAAC,oBAAoB,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;QAC3F,IAAI,CAAC,eAAe,GAAG,iBAAiB,CAAC,kBAAkB,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAC/E,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,IAAI,6CAAyB,CAAC;QACtE,IAAI,CAAC,gBAAgB,GAAG,OAAO,CAAC,gBAAgB,CAAC;QACjD,IAAI,CAAC,aAAa,GAAG,OAAO,CAAC,SAAS,CAAC;QACvC,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;IAC3C,CAAC;IAKO,iBAAiB,CAAC,IAAY;QAClC,OAAO,IAAA,6BAAgB,EAAC,IAAI,CAAC,CAAC;IAClC,CAAC;IAKO,eAAe;QACnB,MAAM,KAAK,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;QAChE,OAAO,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC;IACtB,CAAC;IAKO,gBAAgB;QACpB,IAAI,IAAI,CAAC,WAAW;YAAE,OAAO,IAAI,CAAC,WAAW,CAAC;QAC9C,MAAM,KAAK,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;QACjE,OAAO,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC;IACtB,CAAC;IAED,KAAK,CAAC,GAAG,CACL,IAAY,EACZ,OAA0B,EAC1B,OAAoB;QAEpB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,IAAI,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3B,MAAM,eAAe,CAAC,UAAU,CAAC,OAAO,EAAE;gBACtC,eAAe,EAAE;oBACb,eAAe,EAAE,OAAO,EAAE,WAAW;iBACxC;gBACD,QAAQ,EAAE,OAAO,EAAE,QAAQ;aAC9B,CAAC,CAAC;QACP,CAAC;aAAM,CAAC;YACJ,MAAM,eAAe,CAAC,YAAY,CAAC,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE;gBAC9D,eAAe,EAAE;oBACb,eAAe,EAAE,OAAO,EAAE,WAAW;iBACxC;gBACD,QAAQ,EAAE,OAAO,EAAE,QAAQ;aAC9B,CAAC,CAAC;QACP,CAAC;QAED,OAAO;YACH,IAAI,EAAE,QAAQ;YACd,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC;SAC1B,CAAC;IACN,CAAC;IAED,KAAK,CAAC,GAAG,CAAC,IAAY;QAClB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,MAAM,gBAAgB,GAAG,MAAM,eAAe,CAAC,QAAQ,EAAE,CAAC;QAC1D,MAAM,MAAM,GAAG,gBAAgB,CAAC,kBAA8B,CAAC;QAE/D,OAAO,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3C,MAAM,MAAM,GAAa,EAAE,CAAC;YAC5B,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;YACjD,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;YACvD,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;IACP,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,IAAY;QACxB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,MAAM,gBAAgB,GAAG,MAAM,eAAe,CAAC,QAAQ,EAAE,CAAC;QAC1D,OAAO,gBAAgB,CAAC,kBAA8B,CAAC;IAC3D,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,IAAY;QACrB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,IAAI,CAAC;YACD,MAAM,eAAe,CAAC,MAAM,EAAE,CAAC;YAC/B,OAAO,IAAI,CAAC;QAChB,CAAC;QAAC,MAAM,CAAC;YACL,OAAO,KAAK,CAAC;QACjB,CAAC;IACL,CAAC;IAED,KAAK,CAAC,UAAU,CAAC,KAAe;QAC5B,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpE,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;IACnC,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,IAAY;QACrB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,OAAO,eAAe,CAAC,MAAM,EAAE,CAAC;IACpC,CAAC;IAED,GAAG,CAAC,IAAY;QACZ,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAC1E,OAAO,eAAe,CAAC,GAAG,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,IAAY,EAAE,SAAkB;QAC5C,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,MAAM,GAAG,SAAS,IAAI,IAAI,CAAC,YAAY,CAAC;QAE9C,MAAM,WAAW,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC5C,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAE1C,IAAI,CAAC,WAAW,IAAI,CAAC,UAAU,EAAE,CAAC;YAE9B,OAAO,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC9B,CAAC;QAED,MAAM,mBAAmB,GAAG,IAAI,yCAA0B,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;QACpF,MAAM,QAAQ,GAAG,IAAI,IAAI,EAAE,CAAC;QAC5B,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,GAAG,MAAM,GAAG,IAAI,CAAC,CAAC;QAE/D,MAAM,QAAQ,GAAG,IAAA,6CAA8B,EAC3C;YACI,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,QAAQ,EAAE,QAAQ;YAClB,WAAW,EAAE,iCAAkB,CAAC,KAAK,CAAC,GAAG,CAAC;YAC1C,QAAQ;YACR,SAAS;SACZ,EACD,mBAAmB,CACtB,CAAC,QAAQ,EAAE,CAAC;QAEb,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,QAAQ,EAAE,CAAC;IAC/C,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,IAAY,EAAE,EAAU;QAC/B,MAAM,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAClD,MAAM,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC;QAE9C,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;QAC3E,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAEvE,MAAM,UAAU,CAAC,gBAAgB,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QAEpD,OAAO;YACH,IAAI,EAAE,UAAU;YAChB,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC;SAC5B,CAAC;IACN,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,IAAY,EAAE,EAAU;QAC/B,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QACzC,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACxB,OAAO,MAAM,CAAC;IAClB,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,IAAY;QACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAE1E,MAAM,UAAU,GAAG,MAAM,eAAe,CAAC,aAAa,EAAE,CAAC;QAEzD,OAAO;YACH,IAAI,EAAE,UAAU,CAAC,aAAa,IAAI,CAAC;YACnC,YAAY,EAAE,UAAU,CAAC,YAAY,IAAI,IAAI,IAAI,EAAE;YACnD,WAAW,EAAE,UAAU,CAAC,WAAW;YACnC,QAAQ,EAAE,UAAU,CAAC,QAAQ;SAChC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,MAAe;QACtB,MAAM,gBAAgB,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAC7E,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,IAAI,KAAK,EAAE,MAAM,IAAI,IAAI,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,EAAE,MAAM,EAAE,gBAAgB,EAAE,CAAC,EAAE,CAAC;YACxF,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1B,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;CACJ;AAtMD,0CAsMC"}
@@ -0,0 +1,4 @@
1
+ export * from './local.driver';
2
+ export * from './s3.driver';
3
+ export * from './azure-blob.driver';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/drivers/index.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC;AAC/B,cAAc,aAAa,CAAC;AAC5B,cAAc,qBAAqB,CAAC"}
@@ -0,0 +1,20 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./local.driver"), exports);
18
+ __exportStar(require("./s3.driver"), exports);
19
+ __exportStar(require("./azure-blob.driver"), exports);
20
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/drivers/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,iDAA+B;AAC/B,8CAA4B;AAC5B,sDAAoC"}
@@ -0,0 +1,23 @@
1
+ import { Readable } from 'stream';
2
+ import { StorageDriver, PutOptions, StorageResponse, FileStats } from '../interfaces/storage-driver.interface';
3
+ import { LocalDriverOptions } from '../interfaces/storage-options.interface';
4
+ export declare class LocalDriver implements StorageDriver {
5
+ private readonly basePath;
6
+ private readonly baseUrl?;
7
+ constructor(options: LocalDriverOptions);
8
+ private getFullPath;
9
+ private ensureDir;
10
+ put(filePath: string, content: Buffer | Readable, options?: PutOptions): Promise<StorageResponse>;
11
+ get(filePath: string): Promise<Buffer>;
12
+ getStream(filePath: string): Promise<Readable>;
13
+ delete(filePath: string): Promise<boolean>;
14
+ deleteMany(paths: string[]): Promise<boolean>;
15
+ exists(filePath: string): Promise<boolean>;
16
+ url(filePath: string): string;
17
+ signedUrl(filePath: string, _expiresIn?: number): Promise<string>;
18
+ copy(from: string, to: string): Promise<StorageResponse>;
19
+ move(from: string, to: string): Promise<StorageResponse>;
20
+ stat(filePath: string): Promise<FileStats>;
21
+ list(prefix?: string): Promise<string[]>;
22
+ }
23
+ //# sourceMappingURL=local.driver.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local.driver.d.ts","sourceRoot":"","sources":["../../src/drivers/local.driver.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AAIlC,OAAO,EACH,aAAa,EACb,UAAU,EACV,eAAe,EACf,SAAS,EACZ,MAAM,wCAAwC,CAAC;AAChD,OAAO,EAAE,kBAAkB,EAAE,MAAM,yCAAyC,CAAC;AAM7E,qBAAa,WAAY,YAAW,aAAa;IAC7C,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;IAClC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAS;gBAEtB,OAAO,EAAE,kBAAkB;IAQvC,OAAO,CAAC,WAAW;YAOL,SAAS;IAKjB,GAAG,CACL,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,GAAG,QAAQ,EAC1B,OAAO,CAAC,EAAE,UAAU,GACrB,OAAO,CAAC,eAAe,CAAC;IAsBrB,GAAG,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAKtC,SAAS,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IAK9C,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAU1C,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,OAAO,CAAC;IAK7C,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAUhD,GAAG,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM;IAOvB,SAAS,CAAC,QAAQ,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAKjE,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC;IAWxD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC;IAWxD,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC;IAS1C,IAAI,CAAC,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;CAwBjD"}
@@ -0,0 +1,165 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.LocalDriver = void 0;
37
+ const fs = __importStar(require("fs/promises"));
38
+ const fsSync = __importStar(require("fs"));
39
+ const path = __importStar(require("path"));
40
+ const path_utils_1 = require("../utils/path.utils");
41
+ class LocalDriver {
42
+ constructor(options) {
43
+ this.basePath = path.resolve(options.basePath);
44
+ this.baseUrl = options.baseUrl ? (0, path_utils_1.trimTrailingSlash)(options.baseUrl) : undefined;
45
+ }
46
+ getFullPath(filePath) {
47
+ return (0, path_utils_1.joinPath)(this.basePath, (0, path_utils_1.trimLeadingSlash)(filePath));
48
+ }
49
+ async ensureDir(filePath) {
50
+ const dir = (0, path_utils_1.dirname)(filePath);
51
+ await fs.mkdir(dir, { recursive: true });
52
+ }
53
+ async put(filePath, content, options) {
54
+ const fullPath = this.getFullPath(filePath);
55
+ await this.ensureDir(fullPath);
56
+ if (Buffer.isBuffer(content)) {
57
+ await fs.writeFile(fullPath, content);
58
+ }
59
+ else {
60
+ await new Promise((resolve, reject) => {
61
+ const writeStream = fsSync.createWriteStream(fullPath);
62
+ content.pipe(writeStream);
63
+ writeStream.on('finish', resolve);
64
+ writeStream.on('error', reject);
65
+ });
66
+ }
67
+ return {
68
+ path: filePath,
69
+ url: this.url(filePath),
70
+ };
71
+ }
72
+ async get(filePath) {
73
+ const fullPath = this.getFullPath(filePath);
74
+ return fs.readFile(fullPath);
75
+ }
76
+ async getStream(filePath) {
77
+ const fullPath = this.getFullPath(filePath);
78
+ return fsSync.createReadStream(fullPath);
79
+ }
80
+ async delete(filePath) {
81
+ const fullPath = this.getFullPath(filePath);
82
+ try {
83
+ await fs.unlink(fullPath);
84
+ return true;
85
+ }
86
+ catch {
87
+ return false;
88
+ }
89
+ }
90
+ async deleteMany(paths) {
91
+ const results = await Promise.all(paths.map((p) => this.delete(p)));
92
+ return results.every((r) => r);
93
+ }
94
+ async exists(filePath) {
95
+ const fullPath = this.getFullPath(filePath);
96
+ try {
97
+ await fs.access(fullPath);
98
+ return true;
99
+ }
100
+ catch {
101
+ return false;
102
+ }
103
+ }
104
+ url(filePath) {
105
+ if (this.baseUrl) {
106
+ return `${this.baseUrl}/${(0, path_utils_1.trimLeadingSlash)(filePath)}`;
107
+ }
108
+ return this.getFullPath(filePath);
109
+ }
110
+ async signedUrl(filePath, _expiresIn) {
111
+ return this.url(filePath);
112
+ }
113
+ async copy(from, to) {
114
+ const fromPath = this.getFullPath(from);
115
+ const toPath = this.getFullPath(to);
116
+ await this.ensureDir(toPath);
117
+ await fs.copyFile(fromPath, toPath);
118
+ return {
119
+ path: to,
120
+ url: this.url(to),
121
+ };
122
+ }
123
+ async move(from, to) {
124
+ const fromPath = this.getFullPath(from);
125
+ const toPath = this.getFullPath(to);
126
+ await this.ensureDir(toPath);
127
+ await fs.rename(fromPath, toPath);
128
+ return {
129
+ path: to,
130
+ url: this.url(to),
131
+ };
132
+ }
133
+ async stat(filePath) {
134
+ const fullPath = this.getFullPath(filePath);
135
+ const stats = await fs.stat(fullPath);
136
+ return {
137
+ size: stats.size,
138
+ lastModified: stats.mtime,
139
+ };
140
+ }
141
+ async list(prefix) {
142
+ const dir = prefix ? this.getFullPath(prefix) : this.basePath;
143
+ const files = [];
144
+ try {
145
+ const entries = await fs.readdir(dir, { withFileTypes: true });
146
+ for (const entry of entries) {
147
+ const entryPath = prefix
148
+ ? `${(0, path_utils_1.trimLeadingSlash)(prefix)}/${entry.name}`
149
+ : entry.name;
150
+ if (entry.isFile()) {
151
+ files.push(entryPath);
152
+ }
153
+ else if (entry.isDirectory()) {
154
+ const subFiles = await this.list(entryPath);
155
+ files.push(...subFiles);
156
+ }
157
+ }
158
+ }
159
+ catch {
160
+ }
161
+ return files;
162
+ }
163
+ }
164
+ exports.LocalDriver = LocalDriver;
165
+ //# sourceMappingURL=local.driver.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local.driver.js","sourceRoot":"","sources":["../../src/drivers/local.driver.ts"],"names":[],"mappings":"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"}