@ooneex/storage 1.1.3 → 1.1.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.
package/dist/index.d.ts CHANGED
@@ -28,10 +28,7 @@ interface IStorage {
28
28
  declare class BunnyStorage implements IStorage {
29
29
  private readonly env;
30
30
  private bucket;
31
- private readonly accessKey;
32
31
  private readonly storageZone;
33
- private readonly region;
34
- private readonly baseUrl;
35
32
  constructor(env: AppEnv);
36
33
  getBucket(): string;
37
34
  setBucket(name: string): this;
@@ -46,7 +43,8 @@ declare class BunnyStorage implements IStorage {
46
43
  getAsJson<T = unknown>(key: string): Promise<T>;
47
44
  getAsArrayBuffer(key: string): Promise<ArrayBuffer>;
48
45
  getAsStream(key: string): ReadableStream;
49
- private buildFileUrl;
46
+ private buildFilePath;
47
+ private toReadableStream;
50
48
  }
51
49
  import { AppEnv as AppEnv2 } from "@ooneex/app-env";
52
50
  import { S3Options as S3Options2 } from "bun";
@@ -110,6 +108,6 @@ declare class FilesystemStorage extends Storage {
110
108
  }
111
109
  import { Exception } from "@ooneex/exception";
112
110
  declare class StorageException extends Exception {
113
- constructor(message: string, data?: Record<string, unknown>);
111
+ constructor(message: string, key: string, data?: Record<string, unknown>);
114
112
  }
115
113
  export { decorator, StorageException, StorageClassType, Storage, PutDirOptionsType, IStorage, GetFileOptionsType, FilesystemStorage, CloudflareStorage, BunnyStorage };
package/dist/index.js CHANGED
@@ -1,4 +1,613 @@
1
1
  // @bun
2
- var h=function(t,e,r,i){var s=arguments.length,n=s<3?e:i===null?i=Object.getOwnPropertyDescriptor(e,r):i,c;if(typeof Reflect==="object"&&typeof Reflect.decorate==="function")n=Reflect.decorate(t,e,r,i);else for(var o=t.length-1;o>=0;o--)if(c=t[o])n=(s<3?c(n):s>3?c(e,r,n):c(e,r))||n;return s>3&&n&&Object.defineProperty(e,r,n),n},p=(t,e)=>(r,i)=>e(r,i,t),g=(t,e)=>{if(typeof Reflect==="object"&&typeof Reflect.metadata==="function")return Reflect.metadata(t,e)};import{readdir as U}from"fs/promises";import{basename as D,join as A}from"path";import{AppEnv as E}from"@ooneex/app-env";import{inject as G}from"@ooneex/container";import{container as k,EContainerScope as x,injectable as C}from"@ooneex/container";var u={storage:(t=x.Singleton)=>{return(e)=>{C()(e),k.add(e,t)}}};import{Exception as K}from"@ooneex/exception";import{HttpStatus as _}from"@ooneex/http-status";class a extends K{constructor(t,e={}){super(t,{status:_.Code.InternalServerError,data:e});this.name="StorageException"}}var L={de:"storage.bunnycdn.com",uk:"uk.storage.bunnycdn.com",ny:"ny.storage.bunnycdn.com",la:"la.storage.bunnycdn.com",sg:"sg.storage.bunnycdn.com",se:"se.storage.bunnycdn.com",br:"br.storage.bunnycdn.com",jh:"jh.storage.bunnycdn.com",syd:"syd.storage.bunnycdn.com"};class b{env;bucket="";accessKey;storageZone;region;baseUrl;constructor(t){this.env=t;let e=this.env.STORAGE_BUNNY_ACCESS_KEY,r=this.env.STORAGE_BUNNY_STORAGE_ZONE,i=this.env.STORAGE_BUNNY_REGION;if(!e)throw new a("Bunny access key is required. Please provide an access key either through the constructor options or set the STORAGE_BUNNY_ACCESS_KEY environment variable.");if(!r)throw new a("Bunny storage zone is required. Please provide a storage zone either through the constructor options or set the STORAGE_BUNNY_STORAGE_ZONE environment variable.");this.accessKey=e,this.storageZone=r,this.region=i??"de",this.baseUrl=`https://${L[this.region]}`}getBucket(){return this.bucket}setBucket(t){return this.bucket=t,this}async list(){let t=this.bucket?`${this.bucket}/`:"",e=`${this.baseUrl}/${this.storageZone}/${t}`,r=await fetch(e,{method:"GET",headers:{AccessKey:this.accessKey,accept:"application/json"}});if(!r.ok)throw new a(`Failed to list files: ${r.status} ${r.statusText}`,{status:r.status,path:t});return(await r.json()).filter((s)=>!s.IsDirectory).map((s)=>s.ObjectName)}async clearBucket(){let t=await this.list();return await Promise.all(t.map((e)=>this.delete(e))),this}async exists(t){let e=this.buildFileUrl(t);return(await fetch(e,{method:"HEAD",headers:{AccessKey:this.accessKey}})).ok}async delete(t){let e=this.buildFileUrl(t),r=await fetch(e,{method:"DELETE",headers:{AccessKey:this.accessKey}});if(!r.ok&&r.status!==404)throw new a(`Failed to delete file: ${r.status} ${r.statusText}`,{status:r.status,key:t})}async putFile(t,e){let r=Bun.file(e);return await this.put(t,r)}async putDir(t,e){let{path:r,filter:i}=e,s=await U(r,{withFileTypes:!0}),n=[];for(let o of s){let l=A(r,o.name),m=t?`${t}/${o.name}`:o.name;if(i&&!i.test(l))continue;if(o.isDirectory()){let d={path:l};if(i)d.filter=i;n.push(this.putDir(m,d))}else n.push(this.putFile(m,l))}return(await Promise.all(n)).reduce((o,l)=>o+l,0)}async put(t,e){let r=this.buildFileUrl(t),i,s;if(typeof e==="string")i=e,s=new TextEncoder().encode(e).length;else if(e instanceof ArrayBuffer)i=new Blob([e]),s=e.byteLength;else if(e instanceof SharedArrayBuffer){let c=new Uint8Array(e),o=new Uint8Array(c.length);o.set(c),i=new Blob([o]),s=e.byteLength}else if(ArrayBuffer.isView(e))i=new Blob([new Uint8Array(e.buffer,e.byteOffset,e.byteLength)]),s=e.byteLength;else if(e instanceof Blob)i=e,s=e.size;else if(e instanceof Request||e instanceof Response){let c=await e.arrayBuffer();i=new Blob([c]),s=c.byteLength}else if(typeof e==="object"&&e!==null&&"arrayBuffer"in e){let o=await e.arrayBuffer();i=new Blob([o]),s=o.byteLength}else throw new a("Unsupported content type for upload",{key:t});let n=await fetch(r,{method:"PUT",headers:{AccessKey:this.accessKey,"Content-Type":"application/octet-stream"},body:i});if(!n.ok)throw new a(`Failed to upload file: ${n.status} ${n.statusText}`,{status:n.status,key:t});return s}async getFile(t,e){let r=await this.getAsArrayBuffer(t),i=e.filename??D(t),s=A(e.outputDir,i);return await Bun.write(s,r)}async getAsJson(t){let e=this.buildFileUrl(t),r=await fetch(e,{method:"GET",headers:{AccessKey:this.accessKey}});if(!r.ok)throw new a(`Failed to get file as JSON: ${r.status} ${r.statusText}`,{status:r.status,key:t});return await r.json()}async getAsArrayBuffer(t){let e=this.buildFileUrl(t),r=await fetch(e,{method:"GET",headers:{AccessKey:this.accessKey}});if(!r.ok)throw new a(`Failed to get file as ArrayBuffer: ${r.status} ${r.statusText}`,{status:r.status,key:t});return await r.arrayBuffer()}getAsStream(t){let e=this.buildFileUrl(t),r=this.accessKey,{readable:i,writable:s}=new TransformStream;return fetch(e,{method:"GET",headers:{AccessKey:r}}).then((n)=>{if(!n.ok){s.abort(new a(`Failed to get file as stream: ${n.status} ${n.statusText}`,{status:n.status,key:t}));return}if(!n.body){s.abort(new a("Response body is null",{key:t}));return}n.body.pipeTo(s)}),i}buildFileUrl(t){let e=this.bucket?`${this.bucket}/${t}`:t;return`${this.baseUrl}/${this.storageZone}/${e}`}}b=h([u.storage(),p(0,G(E)),g("design:paramtypes",[typeof E==="undefined"?Object:E])],b);import{AppEnv as T}from"@ooneex/app-env";import{inject as j}from"@ooneex/container";import{readdir as N}from"fs/promises";import{basename as I,join as v}from"path";class y{client=null;getBucket(){return this.bucket}setBucket(t){return this.bucket=t,this.client=new Bun.S3Client(this.getOptions()),this}async list(){return(await this.getClient().list()).contents?.map((e)=>e.key)||[]}async clearBucket(){let t=this.getClient(),e=await this.list();return await Promise.all(e.map((r)=>t.delete(r))),this}async exists(t){return await this.getClient().exists(t)}async delete(t){await this.getClient().delete(t)}async putFile(t,e){let r=Bun.file(e);return await this.put(t,r)}async putDir(t,e){let{path:r,filter:i}=e,s=await N(r,{withFileTypes:!0}),n=[];for(let o of s){let l=v(r,o.name),m=t?`${t}/${o.name}`:o.name;if(i&&!i.test(l))continue;if(o.isDirectory()){let d={path:l};if(i)d.filter=i;n.push(this.putDir(m,d))}else n.push(this.putFile(m,l))}return(await Promise.all(n)).reduce((o,l)=>o+l,0)}async put(t,e){return await this.getS3File(t).write(e)}async getFile(t,e){let r=await this.getAsArrayBuffer(t),i=e.filename??I(t),s=v(e.outputDir,i);return await Bun.write(s,r)}async getAsJson(t){return await this.getS3File(t).json()}async getAsArrayBuffer(t){return await this.getS3File(t).arrayBuffer()}getAsStream(t){return this.getS3File(t).stream()}getClient(){if(!this.client)this.client=new Bun.S3Client(this.getOptions());return this.client}getS3File(t){return this.getClient().file(t)}}class w extends y{env;bucket;accessKey;secretKey;endpoint;region;constructor(t){super();this.env=t;let e=this.env.STORAGE_CLOUDFLARE_ACCESS_KEY,r=this.env.STORAGE_CLOUDFLARE_SECRET_KEY,i=this.env.STORAGE_CLOUDFLARE_ENDPOINT;if(!e)throw new a("Cloudflare access key is required. Please provide an access key either through the constructor options or set the STORAGE_CLOUDFLARE_ACCESS_KEY environment variable.");if(!r)throw new a("Cloudflare secret key is required. Please provide a secret key either through the constructor options or set the STORAGE_CLOUDFLARE_SECRET_KEY environment variable.");if(!i)throw new a("Cloudflare endpoint is required. Please provide an endpoint either through the constructor options or set the STORAGE_CLOUDFLARE_ENDPOINT environment variable.");this.accessKey=e,this.secretKey=r,this.endpoint=i,this.region=this.env.STORAGE_CLOUDFLARE_REGION||"EEUR"}getOptions(){return{accessKeyId:this.accessKey,secretAccessKey:this.secretKey,endpoint:this.endpoint,bucket:this.bucket,region:this.region}}}w=h([u.storage(),p(0,j(T)),g("design:paramtypes",[typeof T==="undefined"?Object:T])],w);import{existsSync as f,mkdirSync as O}from"fs";import{mkdir as F,readdir as R,rm as Y,rmdir as q}from"fs/promises";import{basename as Z,dirname as B,join as S}from"path";import{AppEnv as $}from"@ooneex/app-env";import{inject as J}from"@ooneex/container";class P extends y{env;bucket;storagePath;constructor(t){super();this.env=t;let e=this.env.FILESYSTEM_STORAGE_PATH;if(!e)throw new a("Base path is required. Please provide a base path either through the constructor options or set the FILESYSTEM_STORAGE_PATH environment variable.");this.storagePath=e;try{if(!f(e))O(e,{recursive:!0})}catch(r){throw new a(`Failed to create base storage directory at ${e}: ${r instanceof Error?r.message:String(r)}`)}}getOptions(){return{accessKeyId:"filesystem",secretAccessKey:"filesystem",endpoint:this.storagePath,bucket:this.bucket,region:"local"}}getBucketPath(){if(!this.bucket)throw new a("Bucket name is required. Please call setBucket() first.");return S(this.storagePath,this.bucket)}getFilePath(t){return S(this.getBucketPath(),t)}setBucket(t){this.bucket=t;let e=this.getBucketPath();try{if(!f(e))O(e,{recursive:!0})}catch(r){throw new a(`Failed to create bucket directory at ${e}: ${r instanceof Error?r.message:String(r)}`)}return this}async list(){let t=this.getBucketPath();if(!f(t))return[];try{return await this.listFilesRecursive(t,t)}catch(e){throw new a(`Failed to list files in bucket: ${e instanceof Error?e.message:String(e)}`)}}async listFilesRecursive(t,e){let r=await R(t,{withFileTypes:!0});return(await Promise.all(r.map(async(s)=>{let n=S(t,s.name);if(s.isDirectory())return this.listFilesRecursive(n,e);return[n.substring(e.length+1)]}))).flat()}async clearBucket(){let t=this.getBucketPath();if(!f(t))return this;try{await Y(t,{recursive:!0}),await F(t,{recursive:!0})}catch(e){throw new a(`Failed to clear bucket: ${e instanceof Error?e.message:String(e)}`)}return this}async exists(t){let e=this.getFilePath(t);return await Bun.file(e).exists()}async delete(t){let e=this.getFilePath(t),r=Bun.file(e);if(!await r.exists())return;try{await r.delete();let i=B(e),s=this.getBucketPath();while(i!==s&&i!==this.storagePath)try{if((await R(i)).length===0)await q(i),i=B(i);else break}catch{break}}catch(i){throw new a(`Failed to delete file ${t}: ${i instanceof Error?i.message:String(i)}`)}}async putFile(t,e){let r=Bun.file(e);return await this.put(t,r)}async put(t,e){let r=this.getFilePath(t),i=B(r);try{if(!f(i))await F(i,{recursive:!0})}catch(s){throw new a(`Failed to create directory ${i}: ${s instanceof Error?s.message:String(s)}`)}try{let s;if(typeof e==="string")s=await Bun.write(r,e);else if(e instanceof ArrayBuffer)s=await Bun.write(r,e);else if(e instanceof SharedArrayBuffer){let n=new ArrayBuffer(e.byteLength);new Uint8Array(n).set(new Uint8Array(e)),s=await Bun.write(r,n)}else if(e instanceof Request){let n=await e.arrayBuffer();s=await Bun.write(r,n)}else if(e instanceof Response){let n=await e.arrayBuffer();s=await Bun.write(r,n)}else if(e instanceof Blob)s=await Bun.write(r,e);else{let n=await e.arrayBuffer();s=await Bun.write(r,n)}return s}catch(s){throw new a(`Failed to write file ${t}: ${s instanceof Error?s.message:String(s)}`)}}async getFile(t,e){let r=this.getFilePath(t),i=Bun.file(r);if(!await i.exists())throw new a(`File ${t} does not exist`);let s=e.filename??Z(t),n=S(e.outputDir,s),c=B(n);try{if(!f(c))await F(c,{recursive:!0})}catch(o){throw new a(`Failed to create directory ${c}: ${o instanceof Error?o.message:String(o)}`)}try{return await Bun.write(n,i)}catch(o){throw new a(`Failed to save file ${t} to ${n}: ${o instanceof Error?o.message:String(o)}`)}}async getAsJson(t){let e=this.getFilePath(t),r=Bun.file(e);if(!await r.exists())throw new a(`File ${t} does not exist`);try{return await r.json()}catch(i){throw new a(`Failed to read file ${t} as JSON: ${i instanceof Error?i.message:String(i)}`)}}async getAsArrayBuffer(t){let e=this.getFilePath(t),r=Bun.file(e);if(!await r.exists())throw new a(`File ${t} does not exist`);try{return await r.arrayBuffer()}catch(i){throw new a(`Failed to read file ${t} as ArrayBuffer: ${i instanceof Error?i.message:String(i)}`)}}getAsStream(t){let e=this.getFilePath(t);if(!f(e))throw new a(`File ${t} does not exist`);let r=Bun.file(e);try{return r.stream()}catch(i){throw new a(`Failed to read file ${t} as stream: ${i instanceof Error?i.message:String(i)}`)}}}P=h([u.storage(),p(0,J($)),g("design:paramtypes",[typeof $==="undefined"?Object:$])],P);export{u as decorator,a as StorageException,y as Storage,P as FilesystemStorage,w as CloudflareStorage,b as BunnyStorage};
2
+ var __legacyDecorateClassTS = function(decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function")
5
+ r = Reflect.decorate(decorators, target, key, desc);
6
+ else
7
+ for (var i = decorators.length - 1;i >= 0; i--)
8
+ if (d = decorators[i])
9
+ r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
10
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
11
+ };
12
+ var __legacyDecorateParamTS = (index, decorator) => (target, key) => decorator(target, key, index);
13
+ var __legacyMetadataTS = (k, v) => {
14
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function")
15
+ return Reflect.metadata(k, v);
16
+ };
3
17
 
4
- //# debugId=E5B84E64D1323F2864756E2164756E21
18
+ // src/BunnyStorage.ts
19
+ import { readdir } from "fs/promises";
20
+ import { basename, join } from "path";
21
+ import * as BunnyStorageSDK from "@bunny.net/storage-sdk";
22
+ import { AppEnv } from "@ooneex/app-env";
23
+ import { inject } from "@ooneex/container";
24
+
25
+ // src/decorators.ts
26
+ import { container, EContainerScope } from "@ooneex/container";
27
+ var decorator = {
28
+ storage: (scope = EContainerScope.Singleton) => {
29
+ return (target) => {
30
+ container.add(target, scope);
31
+ };
32
+ }
33
+ };
34
+
35
+ // src/StorageException.ts
36
+ import { Exception } from "@ooneex/exception";
37
+ import { HttpStatus } from "@ooneex/http-status";
38
+
39
+ class StorageException extends Exception {
40
+ constructor(message, key, data = {}) {
41
+ super(message, {
42
+ key,
43
+ status: HttpStatus.Code.InternalServerError,
44
+ data
45
+ });
46
+ this.name = "StorageException";
47
+ }
48
+ }
49
+
50
+ // src/BunnyStorage.ts
51
+ var REGION_MAP = {
52
+ de: BunnyStorageSDK.regions.StorageRegion.Falkenstein,
53
+ uk: BunnyStorageSDK.regions.StorageRegion.London,
54
+ ny: BunnyStorageSDK.regions.StorageRegion.NewYork,
55
+ la: BunnyStorageSDK.regions.StorageRegion.LosAngeles,
56
+ sg: BunnyStorageSDK.regions.StorageRegion.Singapore,
57
+ se: BunnyStorageSDK.regions.StorageRegion.Stockholm,
58
+ br: BunnyStorageSDK.regions.StorageRegion.SaoPaulo,
59
+ jh: BunnyStorageSDK.regions.StorageRegion.Johannesburg,
60
+ syd: BunnyStorageSDK.regions.StorageRegion.Sydney
61
+ };
62
+
63
+ class BunnyStorage {
64
+ env;
65
+ bucket = "";
66
+ storageZone;
67
+ constructor(env) {
68
+ this.env = env;
69
+ const accessKey = this.env.STORAGE_BUNNY_ACCESS_KEY;
70
+ const storageZone = this.env.STORAGE_BUNNY_STORAGE_ZONE;
71
+ const region = this.env.STORAGE_BUNNY_REGION;
72
+ if (!accessKey) {
73
+ throw new StorageException("Bunny access key is required. Please provide an access key either through the constructor options or set the STORAGE_BUNNY_ACCESS_KEY environment variable.", "API_KEY_REQUIRED");
74
+ }
75
+ if (!storageZone) {
76
+ throw new StorageException("Bunny storage zone is required. Please provide a storage zone either through the constructor options or set the STORAGE_BUNNY_STORAGE_ZONE environment variable.", "STORAGE_ZONE_REQUIRED");
77
+ }
78
+ const sdkRegion = REGION_MAP[region ?? "de"];
79
+ this.storageZone = BunnyStorageSDK.zone.connect_with_accesskey(sdkRegion, storageZone, accessKey);
80
+ }
81
+ getBucket() {
82
+ return this.bucket;
83
+ }
84
+ setBucket(name) {
85
+ this.bucket = name;
86
+ return this;
87
+ }
88
+ async list() {
89
+ const path = this.bucket ? `/${this.bucket}/` : "/";
90
+ try {
91
+ const files = await BunnyStorageSDK.file.list(this.storageZone, path);
92
+ return files.filter((file2) => !file2.isDirectory).map((file2) => file2.objectName);
93
+ } catch (error) {
94
+ throw new StorageException(`Failed to list files: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_LIST_FAILED", {
95
+ path
96
+ });
97
+ }
98
+ }
99
+ async clearBucket() {
100
+ const keys = await this.list();
101
+ await Promise.all(keys.map((key) => this.delete(key)));
102
+ return this;
103
+ }
104
+ async exists(key) {
105
+ try {
106
+ await BunnyStorageSDK.file.get(this.storageZone, this.buildFilePath(key));
107
+ return true;
108
+ } catch {
109
+ return false;
110
+ }
111
+ }
112
+ async delete(key) {
113
+ try {
114
+ await BunnyStorageSDK.file.remove(this.storageZone, this.buildFilePath(key));
115
+ } catch {}
116
+ }
117
+ async putFile(key, localPath) {
118
+ const file2 = Bun.file(localPath);
119
+ return await this.put(key, file2);
120
+ }
121
+ async putDir(bucket, options) {
122
+ const { path, filter } = options;
123
+ const entries = await readdir(path, { withFileTypes: true });
124
+ const tasks = [];
125
+ for (const entry of entries) {
126
+ const entryLocalPath = join(path, entry.name);
127
+ const entryKey = bucket ? `${bucket}/${entry.name}` : entry.name;
128
+ if (filter && !filter.test(entryLocalPath)) {
129
+ continue;
130
+ }
131
+ if (entry.isDirectory()) {
132
+ const subOptions = { path: entryLocalPath };
133
+ if (filter) {
134
+ subOptions.filter = filter;
135
+ }
136
+ tasks.push(this.putDir(entryKey, subOptions));
137
+ } else {
138
+ tasks.push(this.putFile(entryKey, entryLocalPath));
139
+ }
140
+ }
141
+ const results = await Promise.all(tasks);
142
+ return results.reduce((sum, bytes) => sum + bytes, 0);
143
+ }
144
+ async put(key, content) {
145
+ const { stream, length } = await this.toReadableStream(content);
146
+ const filePath = this.buildFilePath(key);
147
+ try {
148
+ await BunnyStorageSDK.file.upload(this.storageZone, filePath, stream);
149
+ } catch (error) {
150
+ throw new StorageException(`Failed to upload file: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_UPLOAD_FAILED", {
151
+ key
152
+ });
153
+ }
154
+ return length;
155
+ }
156
+ async getFile(key, options) {
157
+ const arrayBuffer = await this.getAsArrayBuffer(key);
158
+ const filename = options.filename ?? basename(key);
159
+ const localPath = join(options.outputDir, filename);
160
+ return await Bun.write(localPath, arrayBuffer);
161
+ }
162
+ async getAsJson(key) {
163
+ try {
164
+ const { stream } = await BunnyStorageSDK.file.download(this.storageZone, this.buildFilePath(key));
165
+ return await new Response(stream).json();
166
+ } catch (error) {
167
+ throw new StorageException(`Failed to get file as JSON: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED", { key });
168
+ }
169
+ }
170
+ async getAsArrayBuffer(key) {
171
+ try {
172
+ const { stream } = await BunnyStorageSDK.file.download(this.storageZone, this.buildFilePath(key));
173
+ return await new Response(stream).arrayBuffer();
174
+ } catch (error) {
175
+ throw new StorageException(`Failed to get file as ArrayBuffer: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED", { key });
176
+ }
177
+ }
178
+ getAsStream(key) {
179
+ const filePath = this.buildFilePath(key);
180
+ const { readable, writable } = new TransformStream;
181
+ BunnyStorageSDK.file.download(this.storageZone, filePath).then(({ stream }) => {
182
+ stream.pipeTo(writable);
183
+ }).catch((error) => {
184
+ writable.abort(new StorageException(`Failed to get file as stream: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED", { key }));
185
+ });
186
+ return readable;
187
+ }
188
+ buildFilePath(key) {
189
+ return this.bucket ? `/${this.bucket}/${key}` : `/${key}`;
190
+ }
191
+ async toReadableStream(content) {
192
+ if (typeof content === "string") {
193
+ const encoded = new TextEncoder().encode(content);
194
+ return { stream: new Blob([encoded]).stream(), length: encoded.length };
195
+ }
196
+ if (content instanceof ArrayBuffer) {
197
+ return { stream: new Blob([content]).stream(), length: content.byteLength };
198
+ }
199
+ if (content instanceof SharedArrayBuffer) {
200
+ const uint8Array = new Uint8Array(content);
201
+ const copied = new Uint8Array(uint8Array.length);
202
+ copied.set(uint8Array);
203
+ return { stream: new Blob([copied]).stream(), length: content.byteLength };
204
+ }
205
+ if (ArrayBuffer.isView(content)) {
206
+ const view = new Uint8Array(content.buffer, content.byteOffset, content.byteLength);
207
+ return { stream: new Blob([view]).stream(), length: content.byteLength };
208
+ }
209
+ if (content instanceof Blob) {
210
+ return { stream: content.stream(), length: content.size };
211
+ }
212
+ if (content instanceof Request || content instanceof Response) {
213
+ const arrayBuffer = await content.arrayBuffer();
214
+ return { stream: new Blob([arrayBuffer]).stream(), length: arrayBuffer.byteLength };
215
+ }
216
+ if (typeof content === "object" && content !== null && "arrayBuffer" in content) {
217
+ const fileContent = content;
218
+ const arrayBuffer = await fileContent.arrayBuffer();
219
+ return { stream: new Blob([arrayBuffer]).stream(), length: arrayBuffer.byteLength };
220
+ }
221
+ throw new StorageException("Unsupported content type for upload", "UNSUPPORTED_CONTENT_TYPE");
222
+ }
223
+ }
224
+ BunnyStorage = __legacyDecorateClassTS([
225
+ decorator.storage(),
226
+ __legacyDecorateParamTS(0, inject(AppEnv)),
227
+ __legacyMetadataTS("design:paramtypes", [
228
+ typeof AppEnv === "undefined" ? Object : AppEnv
229
+ ])
230
+ ], BunnyStorage);
231
+ // src/CloudflareStorage.ts
232
+ import { AppEnv as AppEnv2 } from "@ooneex/app-env";
233
+ import { inject as inject2 } from "@ooneex/container";
234
+
235
+ // src/Storage.ts
236
+ import { readdir as readdir2 } from "fs/promises";
237
+ import { basename as basename2, join as join2 } from "path";
238
+
239
+ class Storage {
240
+ client = null;
241
+ getBucket() {
242
+ return this.bucket;
243
+ }
244
+ setBucket(name) {
245
+ this.bucket = name;
246
+ this.client = new Bun.S3Client(this.getOptions());
247
+ return this;
248
+ }
249
+ async list() {
250
+ const client = this.getClient();
251
+ return (await client.list()).contents?.map((content) => content.key) || [];
252
+ }
253
+ async clearBucket() {
254
+ const client = this.getClient();
255
+ const keys = await this.list();
256
+ await Promise.all(keys.map((key) => client.delete(key)));
257
+ return this;
258
+ }
259
+ async exists(key) {
260
+ const client = this.getClient();
261
+ return await client.exists(key);
262
+ }
263
+ async delete(key) {
264
+ const client = this.getClient();
265
+ await client.delete(key);
266
+ }
267
+ async putFile(key, localPath) {
268
+ const file2 = Bun.file(localPath);
269
+ return await this.put(key, file2);
270
+ }
271
+ async putDir(bucket, options) {
272
+ const { path, filter } = options;
273
+ const entries = await readdir2(path, { withFileTypes: true });
274
+ const tasks = [];
275
+ for (const entry of entries) {
276
+ const entryLocalPath = join2(path, entry.name);
277
+ const entryKey = bucket ? `${bucket}/${entry.name}` : entry.name;
278
+ if (filter && !filter.test(entryLocalPath)) {
279
+ continue;
280
+ }
281
+ if (entry.isDirectory()) {
282
+ const subOptions = { path: entryLocalPath };
283
+ if (filter) {
284
+ subOptions.filter = filter;
285
+ }
286
+ tasks.push(this.putDir(entryKey, subOptions));
287
+ } else {
288
+ tasks.push(this.putFile(entryKey, entryLocalPath));
289
+ }
290
+ }
291
+ const results = await Promise.all(tasks);
292
+ return results.reduce((sum, bytes) => sum + bytes, 0);
293
+ }
294
+ async put(key, content) {
295
+ const s3file = this.getS3File(key);
296
+ return await s3file.write(content);
297
+ }
298
+ async getFile(key, options) {
299
+ const arrayBuffer = await this.getAsArrayBuffer(key);
300
+ const filename = options.filename ?? basename2(key);
301
+ const localPath = join2(options.outputDir, filename);
302
+ return await Bun.write(localPath, arrayBuffer);
303
+ }
304
+ async getAsJson(key) {
305
+ const s3file = this.getS3File(key);
306
+ return await s3file.json();
307
+ }
308
+ async getAsArrayBuffer(key) {
309
+ const s3file = this.getS3File(key);
310
+ return await s3file.arrayBuffer();
311
+ }
312
+ getAsStream(key) {
313
+ const s3file = this.getS3File(key);
314
+ return s3file.stream();
315
+ }
316
+ getClient() {
317
+ if (!this.client) {
318
+ this.client = new Bun.S3Client(this.getOptions());
319
+ }
320
+ return this.client;
321
+ }
322
+ getS3File(path) {
323
+ const client = this.getClient();
324
+ return client.file(path);
325
+ }
326
+ }
327
+
328
+ // src/CloudflareStorage.ts
329
+ class CloudflareStorage extends Storage {
330
+ env;
331
+ bucket;
332
+ accessKey;
333
+ secretKey;
334
+ endpoint;
335
+ region;
336
+ constructor(env) {
337
+ super();
338
+ this.env = env;
339
+ const accessKey = this.env.STORAGE_CLOUDFLARE_ACCESS_KEY;
340
+ const secretKey = this.env.STORAGE_CLOUDFLARE_SECRET_KEY;
341
+ const endpoint = this.env.STORAGE_CLOUDFLARE_ENDPOINT;
342
+ if (!accessKey) {
343
+ throw new StorageException("Cloudflare access key is required. Please provide an access key either through the constructor options or set the STORAGE_CLOUDFLARE_ACCESS_KEY environment variable.", "CONFIG_REQUIRED");
344
+ }
345
+ if (!secretKey) {
346
+ throw new StorageException("Cloudflare secret key is required. Please provide a secret key either through the constructor options or set the STORAGE_CLOUDFLARE_SECRET_KEY environment variable.", "CONFIG_REQUIRED");
347
+ }
348
+ if (!endpoint) {
349
+ throw new StorageException("Cloudflare endpoint is required. Please provide an endpoint either through the constructor options or set the STORAGE_CLOUDFLARE_ENDPOINT environment variable.", "CONFIG_REQUIRED");
350
+ }
351
+ this.accessKey = accessKey;
352
+ this.secretKey = secretKey;
353
+ this.endpoint = endpoint;
354
+ this.region = this.env.STORAGE_CLOUDFLARE_REGION || "EEUR";
355
+ }
356
+ getOptions() {
357
+ return {
358
+ accessKeyId: this.accessKey,
359
+ secretAccessKey: this.secretKey,
360
+ endpoint: this.endpoint,
361
+ bucket: this.bucket,
362
+ region: this.region
363
+ };
364
+ }
365
+ }
366
+ CloudflareStorage = __legacyDecorateClassTS([
367
+ decorator.storage(),
368
+ __legacyDecorateParamTS(0, inject2(AppEnv2)),
369
+ __legacyMetadataTS("design:paramtypes", [
370
+ typeof AppEnv2 === "undefined" ? Object : AppEnv2
371
+ ])
372
+ ], CloudflareStorage);
373
+ // src/FilesystemStorage.ts
374
+ import { existsSync, mkdirSync } from "fs";
375
+ import { mkdir, readdir as readdir3, rm, rmdir } from "fs/promises";
376
+ import { basename as basename3, dirname, join as join3 } from "path";
377
+ import { AppEnv as AppEnv3 } from "@ooneex/app-env";
378
+ import { inject as inject3 } from "@ooneex/container";
379
+ class FilesystemStorage extends Storage {
380
+ env;
381
+ bucket;
382
+ storagePath;
383
+ constructor(env) {
384
+ super();
385
+ this.env = env;
386
+ const basePath = this.env.FILESYSTEM_STORAGE_PATH;
387
+ if (!basePath) {
388
+ throw new StorageException("Base path is required. Please provide a base path either through the constructor options or set the FILESYSTEM_STORAGE_PATH environment variable.", "STORAGE_ROOT_DIR_REQUIRED");
389
+ }
390
+ this.storagePath = basePath;
391
+ try {
392
+ if (!existsSync(basePath)) {
393
+ mkdirSync(basePath, { recursive: true });
394
+ }
395
+ } catch (error) {
396
+ throw new StorageException(`Failed to create base storage directory at ${basePath}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_CONFIG_REQUIRED");
397
+ }
398
+ }
399
+ getOptions() {
400
+ return {
401
+ accessKeyId: "filesystem",
402
+ secretAccessKey: "filesystem",
403
+ endpoint: this.storagePath,
404
+ bucket: this.bucket,
405
+ region: "local"
406
+ };
407
+ }
408
+ getBucketPath() {
409
+ if (!this.bucket) {
410
+ throw new StorageException("Bucket name is required. Please call setBucket() first.", "STORAGE_BUCKET_REQUIRED");
411
+ }
412
+ return join3(this.storagePath, this.bucket);
413
+ }
414
+ getFilePath(key) {
415
+ return join3(this.getBucketPath(), key);
416
+ }
417
+ setBucket(name) {
418
+ this.bucket = name;
419
+ const bucketPath = this.getBucketPath();
420
+ try {
421
+ if (!existsSync(bucketPath)) {
422
+ mkdirSync(bucketPath, { recursive: true });
423
+ }
424
+ } catch (error) {
425
+ throw new StorageException(`Failed to create bucket directory at ${bucketPath}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_UPLOAD_FAILED");
426
+ }
427
+ return this;
428
+ }
429
+ async list() {
430
+ const bucketPath = this.getBucketPath();
431
+ if (!existsSync(bucketPath)) {
432
+ return [];
433
+ }
434
+ try {
435
+ const files = await this.listFilesRecursive(bucketPath, bucketPath);
436
+ return files;
437
+ } catch (error) {
438
+ throw new StorageException(`Failed to list files in bucket: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_LIST_FAILED");
439
+ }
440
+ }
441
+ async listFilesRecursive(dir, baseDir) {
442
+ const entries = await readdir3(dir, { withFileTypes: true });
443
+ const results = await Promise.all(entries.map(async (entry) => {
444
+ const fullPath = join3(dir, entry.name);
445
+ if (entry.isDirectory()) {
446
+ return this.listFilesRecursive(fullPath, baseDir);
447
+ }
448
+ return [fullPath.substring(baseDir.length + 1)];
449
+ }));
450
+ return results.flat();
451
+ }
452
+ async clearBucket() {
453
+ const bucketPath = this.getBucketPath();
454
+ if (!existsSync(bucketPath)) {
455
+ return this;
456
+ }
457
+ try {
458
+ await rm(bucketPath, { recursive: true });
459
+ await mkdir(bucketPath, { recursive: true });
460
+ } catch (error) {
461
+ throw new StorageException(`Failed to clear bucket: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_CLEAR_FAILED");
462
+ }
463
+ return this;
464
+ }
465
+ async exists(key) {
466
+ const filePath = this.getFilePath(key);
467
+ const file2 = Bun.file(filePath);
468
+ return await file2.exists();
469
+ }
470
+ async delete(key) {
471
+ const filePath = this.getFilePath(key);
472
+ const file2 = Bun.file(filePath);
473
+ if (!await file2.exists()) {
474
+ return;
475
+ }
476
+ try {
477
+ await file2.delete();
478
+ let parentDir = dirname(filePath);
479
+ const bucketPath = this.getBucketPath();
480
+ while (parentDir !== bucketPath && parentDir !== this.storagePath) {
481
+ try {
482
+ const entries = await readdir3(parentDir);
483
+ if (entries.length === 0) {
484
+ await rmdir(parentDir);
485
+ parentDir = dirname(parentDir);
486
+ } else {
487
+ break;
488
+ }
489
+ } catch {
490
+ break;
491
+ }
492
+ }
493
+ } catch (error) {
494
+ throw new StorageException(`Failed to delete file ${key}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DELETE_FAILED");
495
+ }
496
+ }
497
+ async putFile(key, localPath) {
498
+ const file2 = Bun.file(localPath);
499
+ return await this.put(key, file2);
500
+ }
501
+ async put(key, content) {
502
+ const filePath = this.getFilePath(key);
503
+ const dir = dirname(filePath);
504
+ try {
505
+ if (!existsSync(dir)) {
506
+ await mkdir(dir, { recursive: true });
507
+ }
508
+ } catch (error) {
509
+ throw new StorageException(`Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_UPLOAD_FAILED");
510
+ }
511
+ try {
512
+ let bytesWritten;
513
+ if (typeof content === "string") {
514
+ bytesWritten = await Bun.write(filePath, content);
515
+ } else if (content instanceof ArrayBuffer) {
516
+ bytesWritten = await Bun.write(filePath, content);
517
+ } else if (content instanceof SharedArrayBuffer) {
518
+ const arrayBuffer = new ArrayBuffer(content.byteLength);
519
+ new Uint8Array(arrayBuffer).set(new Uint8Array(content));
520
+ bytesWritten = await Bun.write(filePath, arrayBuffer);
521
+ } else if (content instanceof Request) {
522
+ const arrayBuffer = await content.arrayBuffer();
523
+ bytesWritten = await Bun.write(filePath, arrayBuffer);
524
+ } else if (content instanceof Response) {
525
+ const arrayBuffer = await content.arrayBuffer();
526
+ bytesWritten = await Bun.write(filePath, arrayBuffer);
527
+ } else if (content instanceof Blob) {
528
+ bytesWritten = await Bun.write(filePath, content);
529
+ } else {
530
+ const arrayBuffer = await content.arrayBuffer();
531
+ bytesWritten = await Bun.write(filePath, arrayBuffer);
532
+ }
533
+ return bytesWritten;
534
+ } catch (error) {
535
+ throw new StorageException(`Failed to write file ${key}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_UPLOAD_FAILED");
536
+ }
537
+ }
538
+ async getFile(key, options) {
539
+ const filePath = this.getFilePath(key);
540
+ const file2 = Bun.file(filePath);
541
+ if (!await file2.exists()) {
542
+ throw new StorageException(`File ${key} does not exist`, "FILE_NOT_FOUND");
543
+ }
544
+ const filename = options.filename ?? basename3(key);
545
+ const localPath = join3(options.outputDir, filename);
546
+ const dir = dirname(localPath);
547
+ try {
548
+ if (!existsSync(dir)) {
549
+ await mkdir(dir, { recursive: true });
550
+ }
551
+ } catch (error) {
552
+ throw new StorageException(`Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED");
553
+ }
554
+ try {
555
+ return await Bun.write(localPath, file2);
556
+ } catch (error) {
557
+ throw new StorageException(`Failed to save file ${key} to ${localPath}: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED");
558
+ }
559
+ }
560
+ async getAsJson(key) {
561
+ const filePath = this.getFilePath(key);
562
+ const file2 = Bun.file(filePath);
563
+ if (!await file2.exists()) {
564
+ throw new StorageException(`File ${key} does not exist`, "FILE_NOT_FOUND");
565
+ }
566
+ try {
567
+ return await file2.json();
568
+ } catch (error) {
569
+ throw new StorageException(`Failed to read file ${key} as JSON: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED");
570
+ }
571
+ }
572
+ async getAsArrayBuffer(key) {
573
+ const filePath = this.getFilePath(key);
574
+ const file2 = Bun.file(filePath);
575
+ if (!await file2.exists()) {
576
+ throw new StorageException(`File ${key} does not exist`, "FILE_NOT_FOUND");
577
+ }
578
+ try {
579
+ return await file2.arrayBuffer();
580
+ } catch (error) {
581
+ throw new StorageException(`Failed to read file ${key} as ArrayBuffer: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED");
582
+ }
583
+ }
584
+ getAsStream(key) {
585
+ const filePath = this.getFilePath(key);
586
+ if (!existsSync(filePath)) {
587
+ throw new StorageException(`File ${key} does not exist`, "FILE_NOT_FOUND");
588
+ }
589
+ const file2 = Bun.file(filePath);
590
+ try {
591
+ return file2.stream();
592
+ } catch (error) {
593
+ throw new StorageException(`Failed to read file ${key} as stream: ${error instanceof Error ? error.message : String(error)}`, "STORAGE_DOWNLOAD_FAILED");
594
+ }
595
+ }
596
+ }
597
+ FilesystemStorage = __legacyDecorateClassTS([
598
+ decorator.storage(),
599
+ __legacyDecorateParamTS(0, inject3(AppEnv3)),
600
+ __legacyMetadataTS("design:paramtypes", [
601
+ typeof AppEnv3 === "undefined" ? Object : AppEnv3
602
+ ])
603
+ ], FilesystemStorage);
604
+ export {
605
+ decorator,
606
+ StorageException,
607
+ Storage,
608
+ FilesystemStorage,
609
+ CloudflareStorage,
610
+ BunnyStorage
611
+ };
612
+
613
+ //# debugId=B7060C45594F90B964756E2164756E21
package/dist/index.js.map CHANGED
@@ -2,14 +2,14 @@
2
2
  "version": 3,
3
3
  "sources": ["src/BunnyStorage.ts", "src/decorators.ts", "src/StorageException.ts", "src/CloudflareStorage.ts", "src/Storage.ts", "src/FilesystemStorage.ts"],
4
4
  "sourcesContent": [
5
- "import { readdir } from \"node:fs/promises\";\nimport { basename, join } from \"node:path\";\nimport { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { BunFile, S3File } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { StorageException } from \"./StorageException\";\nimport type { GetFileOptionsType, IStorage, PutDirOptionsType } from \"./types\";\n\ntype BunnyRegionType = \"de\" | \"uk\" | \"ny\" | \"la\" | \"sg\" | \"se\" | \"br\" | \"jh\" | \"syd\";\n\ntype BunnyFileInfoType = {\n Guid: string;\n StorageZoneName: string;\n Path: string;\n ObjectName: string;\n Length: number;\n LastChanged: string;\n ServerId: number;\n ArrayNumber: number;\n IsDirectory: boolean;\n UserId: string;\n ContentType: string;\n DateCreated: string;\n StorageZoneId: number;\n Checksum: string | null;\n ReplicatedZones: string | null;\n};\n\nconst REGION_ENDPOINTS: Record<BunnyRegionType, string> = {\n de: \"storage.bunnycdn.com\",\n uk: \"uk.storage.bunnycdn.com\",\n ny: \"ny.storage.bunnycdn.com\",\n la: \"la.storage.bunnycdn.com\",\n sg: \"sg.storage.bunnycdn.com\",\n se: \"se.storage.bunnycdn.com\",\n br: \"br.storage.bunnycdn.com\",\n jh: \"jh.storage.bunnycdn.com\",\n syd: \"syd.storage.bunnycdn.com\",\n};\n\n@decorator.storage()\nexport class BunnyStorage implements IStorage {\n private bucket = \"\";\n private readonly accessKey: string;\n private readonly storageZone: string;\n private readonly region: BunnyRegionType;\n private readonly baseUrl: string;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n const accessKey = this.env.STORAGE_BUNNY_ACCESS_KEY;\n const storageZone = this.env.STORAGE_BUNNY_STORAGE_ZONE;\n const region = this.env.STORAGE_BUNNY_REGION as BunnyRegionType | undefined;\n\n if (!accessKey) {\n throw new StorageException(\n \"Bunny access key is required. Please provide an access key either through the constructor options or set the STORAGE_BUNNY_ACCESS_KEY environment variable.\",\n );\n }\n if (!storageZone) {\n throw new StorageException(\n \"Bunny storage zone is required. Please provide a storage zone either through the constructor options or set the STORAGE_BUNNY_STORAGE_ZONE environment variable.\",\n );\n }\n\n this.accessKey = accessKey;\n this.storageZone = storageZone;\n this.region = region ?? \"de\";\n this.baseUrl = `https://${REGION_ENDPOINTS[this.region]}`;\n }\n\n public getBucket(): string {\n return this.bucket;\n }\n\n public setBucket(name: string): this {\n this.bucket = name;\n\n return this;\n }\n\n public async list(): Promise<string[]> {\n const path = this.bucket ? `${this.bucket}/` : \"\";\n const url = `${this.baseUrl}/${this.storageZone}/${path}`;\n\n const response = await fetch(url, {\n method: \"GET\",\n headers: {\n AccessKey: this.accessKey,\n accept: \"application/json\",\n },\n });\n\n if (!response.ok) {\n throw new StorageException(`Failed to list files: ${response.status} ${response.statusText}`, {\n status: response.status,\n path,\n });\n }\n\n const files: BunnyFileInfoType[] = await response.json();\n\n return files.filter((file) => !file.IsDirectory).map((file) => file.ObjectName);\n }\n\n public async clearBucket(): Promise<this> {\n const keys = await this.list();\n\n await Promise.all(keys.map((key) => this.delete(key)));\n\n return this;\n }\n\n public async exists(key: string): Promise<boolean> {\n const url = this.buildFileUrl(key);\n\n const response = await fetch(url, {\n method: \"HEAD\",\n headers: {\n AccessKey: this.accessKey,\n },\n });\n\n return response.ok;\n }\n\n public async delete(key: string): Promise<void> {\n const url = this.buildFileUrl(key);\n\n const response = await fetch(url, {\n method: \"DELETE\",\n headers: {\n AccessKey: this.accessKey,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n throw new StorageException(`Failed to delete file: ${response.status} ${response.statusText}`, {\n status: response.status,\n key,\n });\n }\n }\n\n public async putFile(key: string, localPath: string): Promise<number> {\n const file = Bun.file(localPath);\n\n return await this.put(key, file);\n }\n\n public async putDir(bucket: string, options: PutDirOptionsType): Promise<number> {\n const { path, filter } = options;\n const entries = await readdir(path, { withFileTypes: true });\n\n const tasks: Promise<number>[] = [];\n\n for (const entry of entries) {\n const entryLocalPath = join(path, entry.name);\n const entryKey = bucket ? `${bucket}/${entry.name}` : entry.name;\n\n if (filter && !filter.test(entryLocalPath)) {\n continue;\n }\n\n if (entry.isDirectory()) {\n const subOptions: PutDirOptionsType = { path: entryLocalPath };\n if (filter) {\n subOptions.filter = filter;\n }\n tasks.push(this.putDir(entryKey, subOptions));\n } else {\n tasks.push(this.putFile(entryKey, entryLocalPath));\n }\n }\n\n const results = await Promise.all(tasks);\n\n return results.reduce((sum, bytes) => sum + bytes, 0);\n }\n\n public async put(\n key: string,\n content: string | ArrayBufferView | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<number> {\n const url = this.buildFileUrl(key);\n\n let body: BodyInit;\n let contentLength: number;\n\n if (typeof content === \"string\") {\n body = content;\n contentLength = new TextEncoder().encode(content).length;\n } else if (content instanceof ArrayBuffer) {\n body = new Blob([content]);\n contentLength = content.byteLength;\n } else if (content instanceof SharedArrayBuffer) {\n const uint8Array = new Uint8Array(content);\n const copiedArray = new Uint8Array(uint8Array.length);\n copiedArray.set(uint8Array);\n body = new Blob([copiedArray]);\n contentLength = content.byteLength;\n } else if (ArrayBuffer.isView(content)) {\n body = new Blob([new Uint8Array(content.buffer as ArrayBuffer, content.byteOffset, content.byteLength)]);\n contentLength = content.byteLength;\n } else if (content instanceof Blob) {\n body = content;\n contentLength = content.size;\n } else if (content instanceof Request || content instanceof Response) {\n const arrayBuffer = await content.arrayBuffer();\n body = new Blob([arrayBuffer]);\n contentLength = arrayBuffer.byteLength;\n } else if (typeof content === \"object\" && content !== null && \"arrayBuffer\" in content) {\n const fileContent = content as BunFile | S3File;\n const arrayBuffer = await fileContent.arrayBuffer();\n body = new Blob([arrayBuffer]);\n contentLength = arrayBuffer.byteLength;\n } else {\n throw new StorageException(\"Unsupported content type for upload\", { key });\n }\n\n const response = await fetch(url, {\n method: \"PUT\",\n headers: {\n AccessKey: this.accessKey,\n \"Content-Type\": \"application/octet-stream\",\n },\n body,\n });\n\n if (!response.ok) {\n throw new StorageException(`Failed to upload file: ${response.status} ${response.statusText}`, {\n status: response.status,\n key,\n });\n }\n\n return contentLength;\n }\n\n public async getFile(key: string, options: GetFileOptionsType): Promise<number> {\n const arrayBuffer = await this.getAsArrayBuffer(key);\n const filename = options.filename ?? basename(key);\n const localPath = join(options.outputDir, filename);\n\n return await Bun.write(localPath, arrayBuffer);\n }\n\n public async getAsJson<T = unknown>(key: string): Promise<T> {\n const url = this.buildFileUrl(key);\n\n const response = await fetch(url, {\n method: \"GET\",\n headers: {\n AccessKey: this.accessKey,\n },\n });\n\n if (!response.ok) {\n throw new StorageException(`Failed to get file as JSON: ${response.status} ${response.statusText}`, {\n status: response.status,\n key,\n });\n }\n\n return await response.json();\n }\n\n public async getAsArrayBuffer(key: string): Promise<ArrayBuffer> {\n const url = this.buildFileUrl(key);\n\n const response = await fetch(url, {\n method: \"GET\",\n headers: {\n AccessKey: this.accessKey,\n },\n });\n\n if (!response.ok) {\n throw new StorageException(`Failed to get file as ArrayBuffer: ${response.status} ${response.statusText}`, {\n status: response.status,\n key,\n });\n }\n\n return await response.arrayBuffer();\n }\n\n public getAsStream(key: string): ReadableStream {\n const url = this.buildFileUrl(key);\n const accessKey = this.accessKey;\n\n const { readable, writable } = new TransformStream();\n\n fetch(url, {\n method: \"GET\",\n headers: { AccessKey: accessKey },\n }).then((response) => {\n if (!response.ok) {\n writable.abort(\n new StorageException(`Failed to get file as stream: ${response.status} ${response.statusText}`, {\n status: response.status,\n key,\n }),\n );\n return;\n }\n\n if (!response.body) {\n writable.abort(new StorageException(\"Response body is null\", { key }));\n return;\n }\n\n response.body.pipeTo(writable);\n });\n\n return readable;\n }\n\n private buildFileUrl(key: string): string {\n const path = this.bucket ? `${this.bucket}/${key}` : key;\n\n return `${this.baseUrl}/${this.storageZone}/${path}`;\n }\n}\n",
6
- "import { container, EContainerScope, injectable } from \"@ooneex/container\";\nimport type { StorageClassType } from \"./types\";\n\nexport const decorator = {\n storage: (scope: EContainerScope = EContainerScope.Singleton) => {\n return (target: StorageClassType): void => {\n injectable()(target);\n container.add(target, scope);\n };\n },\n};\n",
7
- "import { Exception } from \"@ooneex/exception\";\nimport { HttpStatus } from \"@ooneex/http-status\";\n\nexport class StorageException extends Exception {\n constructor(message: string, data: Record<string, unknown> = {}) {\n super(message, {\n status: HttpStatus.Code.InternalServerError,\n data,\n });\n this.name = \"StorageException\";\n }\n}\n",
8
- "import { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { S3Options } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { Storage } from \"./Storage\";\nimport { StorageException } from \"./StorageException\";\n\n@decorator.storage()\nexport class CloudflareStorage extends Storage {\n protected bucket: string;\n private readonly accessKey: string;\n private readonly secretKey: string;\n private readonly endpoint: string;\n private readonly region: string;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n super();\n\n const accessKey = this.env.STORAGE_CLOUDFLARE_ACCESS_KEY;\n const secretKey = this.env.STORAGE_CLOUDFLARE_SECRET_KEY;\n const endpoint = this.env.STORAGE_CLOUDFLARE_ENDPOINT;\n\n if (!accessKey) {\n throw new StorageException(\n \"Cloudflare access key is required. Please provide an access key either through the constructor options or set the STORAGE_CLOUDFLARE_ACCESS_KEY environment variable.\",\n );\n }\n if (!secretKey) {\n throw new StorageException(\n \"Cloudflare secret key is required. Please provide a secret key either through the constructor options or set the STORAGE_CLOUDFLARE_SECRET_KEY environment variable.\",\n );\n }\n if (!endpoint) {\n throw new StorageException(\n \"Cloudflare endpoint is required. Please provide an endpoint either through the constructor options or set the STORAGE_CLOUDFLARE_ENDPOINT environment variable.\",\n );\n }\n\n this.accessKey = accessKey;\n this.secretKey = secretKey;\n this.endpoint = endpoint;\n this.region = this.env.STORAGE_CLOUDFLARE_REGION || \"EEUR\";\n }\n\n public getOptions(): S3Options {\n return {\n accessKeyId: this.accessKey,\n secretAccessKey: this.secretKey,\n endpoint: this.endpoint,\n bucket: this.bucket,\n region: this.region,\n };\n }\n}\n",
5
+ "import { readdir } from \"node:fs/promises\";\nimport { basename, join } from \"node:path\";\nimport type { ReadableStream as NodeReadableStream } from \"node:stream/web\";\nimport * as BunnyStorageSDK from \"@bunny.net/storage-sdk\";\nimport { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { BunFile, S3File } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { StorageException } from \"./StorageException\";\nimport type { GetFileOptionsType, IStorage, PutDirOptionsType } from \"./types\";\n\ntype BunnyRegionType = \"de\" | \"uk\" | \"ny\" | \"la\" | \"sg\" | \"se\" | \"br\" | \"jh\" | \"syd\";\n\nconst REGION_MAP: Record<BunnyRegionType, BunnyStorageSDK.regions.StorageRegion> = {\n de: BunnyStorageSDK.regions.StorageRegion.Falkenstein,\n uk: BunnyStorageSDK.regions.StorageRegion.London,\n ny: BunnyStorageSDK.regions.StorageRegion.NewYork,\n la: BunnyStorageSDK.regions.StorageRegion.LosAngeles,\n sg: BunnyStorageSDK.regions.StorageRegion.Singapore,\n se: BunnyStorageSDK.regions.StorageRegion.Stockholm,\n br: BunnyStorageSDK.regions.StorageRegion.SaoPaulo,\n jh: BunnyStorageSDK.regions.StorageRegion.Johannesburg,\n syd: BunnyStorageSDK.regions.StorageRegion.Sydney,\n};\n\n@decorator.storage()\nexport class BunnyStorage implements IStorage {\n private bucket = \"\";\n private readonly storageZone: BunnyStorageSDK.zone.StorageZone;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n const accessKey = this.env.STORAGE_BUNNY_ACCESS_KEY;\n const storageZone = this.env.STORAGE_BUNNY_STORAGE_ZONE;\n const region = this.env.STORAGE_BUNNY_REGION as BunnyRegionType | undefined;\n\n if (!accessKey) {\n throw new StorageException(\n \"Bunny access key is required. Please provide an access key either through the constructor options or set the STORAGE_BUNNY_ACCESS_KEY environment variable.\",\n \"API_KEY_REQUIRED\",\n );\n }\n if (!storageZone) {\n throw new StorageException(\n \"Bunny storage zone is required. Please provide a storage zone either through the constructor options or set the STORAGE_BUNNY_STORAGE_ZONE environment variable.\",\n \"STORAGE_ZONE_REQUIRED\",\n );\n }\n\n const sdkRegion = REGION_MAP[region ?? \"de\"];\n this.storageZone = BunnyStorageSDK.zone.connect_with_accesskey(sdkRegion, storageZone, accessKey);\n }\n\n public getBucket(): string {\n return this.bucket;\n }\n\n public setBucket(name: string): this {\n this.bucket = name;\n\n return this;\n }\n\n public async list(): Promise<string[]> {\n const path = this.bucket ? `/${this.bucket}/` : \"/\";\n\n try {\n const files = await BunnyStorageSDK.file.list(this.storageZone, path);\n\n return files.filter((file) => !file.isDirectory).map((file) => file.objectName);\n } catch (error) {\n throw new StorageException(`Failed to list files: ${error instanceof Error ? error.message : String(error)}`, \"STORAGE_LIST_FAILED\", {\n path,\n });\n }\n }\n\n public async clearBucket(): Promise<this> {\n const keys = await this.list();\n\n await Promise.all(keys.map((key) => this.delete(key)));\n\n return this;\n }\n\n public async exists(key: string): Promise<boolean> {\n try {\n await BunnyStorageSDK.file.get(this.storageZone, this.buildFilePath(key));\n\n return true;\n } catch {\n return false;\n }\n }\n\n public async delete(key: string): Promise<void> {\n try {\n await BunnyStorageSDK.file.remove(this.storageZone, this.buildFilePath(key));\n } catch {\n // Ignore errors (file may not exist)\n }\n }\n\n public async putFile(key: string, localPath: string): Promise<number> {\n const file = Bun.file(localPath);\n\n return await this.put(key, file);\n }\n\n public async putDir(bucket: string, options: PutDirOptionsType): Promise<number> {\n const { path, filter } = options;\n const entries = await readdir(path, { withFileTypes: true });\n\n const tasks: Promise<number>[] = [];\n\n for (const entry of entries) {\n const entryLocalPath = join(path, entry.name);\n const entryKey = bucket ? `${bucket}/${entry.name}` : entry.name;\n\n if (filter && !filter.test(entryLocalPath)) {\n continue;\n }\n\n if (entry.isDirectory()) {\n const subOptions: PutDirOptionsType = { path: entryLocalPath };\n if (filter) {\n subOptions.filter = filter;\n }\n tasks.push(this.putDir(entryKey, subOptions));\n } else {\n tasks.push(this.putFile(entryKey, entryLocalPath));\n }\n }\n\n const results = await Promise.all(tasks);\n\n return results.reduce((sum, bytes) => sum + bytes, 0);\n }\n\n public async put(\n key: string,\n content: string | ArrayBufferView | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<number> {\n const { stream, length } = await this.toReadableStream(content);\n const filePath = this.buildFilePath(key);\n\n try {\n await BunnyStorageSDK.file.upload(\n this.storageZone,\n filePath,\n stream as unknown as NodeReadableStream<Uint8Array>,\n );\n } catch (error) {\n throw new StorageException(`Failed to upload file: ${error instanceof Error ? error.message : String(error)}`, \"STORAGE_UPLOAD_FAILED\", {\n key,\n });\n }\n\n return length;\n }\n\n public async getFile(key: string, options: GetFileOptionsType): Promise<number> {\n const arrayBuffer = await this.getAsArrayBuffer(key);\n const filename = options.filename ?? basename(key);\n const localPath = join(options.outputDir, filename);\n\n return await Bun.write(localPath, arrayBuffer);\n }\n\n public async getAsJson<T = unknown>(key: string): Promise<T> {\n try {\n const { stream } = await BunnyStorageSDK.file.download(this.storageZone, this.buildFilePath(key));\n\n return await new Response(stream as unknown as BodyInit).json();\n } catch (error) {\n throw new StorageException(\n `Failed to get file as JSON: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n { key },\n );\n }\n }\n\n public async getAsArrayBuffer(key: string): Promise<ArrayBuffer> {\n try {\n const { stream } = await BunnyStorageSDK.file.download(this.storageZone, this.buildFilePath(key));\n\n return await new Response(stream as unknown as BodyInit).arrayBuffer();\n } catch (error) {\n throw new StorageException(\n `Failed to get file as ArrayBuffer: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n { key },\n );\n }\n }\n\n public getAsStream(key: string): ReadableStream {\n const filePath = this.buildFilePath(key);\n\n const { readable, writable } = new TransformStream();\n\n BunnyStorageSDK.file\n .download(this.storageZone, filePath)\n .then(({ stream }) => {\n (stream as unknown as ReadableStream).pipeTo(writable);\n })\n .catch((error) => {\n writable.abort(\n new StorageException(\n `Failed to get file as stream: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n { key },\n ),\n );\n });\n\n return readable;\n }\n\n private buildFilePath(key: string): string {\n return this.bucket ? `/${this.bucket}/${key}` : `/${key}`;\n }\n\n private async toReadableStream(\n content: string | ArrayBufferView | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<{ stream: ReadableStream<Uint8Array>; length: number }> {\n if (typeof content === \"string\") {\n const encoded = new TextEncoder().encode(content);\n return { stream: new Blob([encoded]).stream(), length: encoded.length };\n }\n\n if (content instanceof ArrayBuffer) {\n return { stream: new Blob([content]).stream(), length: content.byteLength };\n }\n\n if (content instanceof SharedArrayBuffer) {\n const uint8Array = new Uint8Array(content);\n const copied = new Uint8Array(uint8Array.length);\n copied.set(uint8Array);\n return { stream: new Blob([copied]).stream(), length: content.byteLength };\n }\n\n if (ArrayBuffer.isView(content)) {\n const view = new Uint8Array(content.buffer as ArrayBuffer, content.byteOffset, content.byteLength);\n return { stream: new Blob([view]).stream(), length: content.byteLength };\n }\n\n if (content instanceof Blob) {\n return { stream: content.stream(), length: content.size };\n }\n\n if (content instanceof Request || content instanceof Response) {\n const arrayBuffer = await content.arrayBuffer();\n return { stream: new Blob([arrayBuffer]).stream(), length: arrayBuffer.byteLength };\n }\n\n if (typeof content === \"object\" && content !== null && \"arrayBuffer\" in content) {\n const fileContent = content as BunFile | S3File;\n const arrayBuffer = await fileContent.arrayBuffer();\n return { stream: new Blob([arrayBuffer]).stream(), length: arrayBuffer.byteLength };\n }\n\n throw new StorageException(\"Unsupported content type for upload\", \"UNSUPPORTED_CONTENT_TYPE\");\n }\n}\n",
6
+ "import { container, EContainerScope } from \"@ooneex/container\";\nimport type { StorageClassType } from \"./types\";\n\nexport const decorator = {\n storage: (scope: EContainerScope = EContainerScope.Singleton) => {\n return (target: StorageClassType): void => {\n container.add(target, scope);\n };\n },\n};\n",
7
+ "import { Exception } from \"@ooneex/exception\";\nimport { HttpStatus } from \"@ooneex/http-status\";\n\nexport class StorageException extends Exception {\n constructor(message: string, key: string, data: Record<string, unknown> = {}) {\n super(message, {\n key,\n status: HttpStatus.Code.InternalServerError,\n data,\n });\n this.name = \"StorageException\";\n }\n}\n",
8
+ "import { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { S3Options } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { Storage } from \"./Storage\";\nimport { StorageException } from \"./StorageException\";\n\n@decorator.storage()\nexport class CloudflareStorage extends Storage {\n protected bucket: string;\n private readonly accessKey: string;\n private readonly secretKey: string;\n private readonly endpoint: string;\n private readonly region: string;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n super();\n\n const accessKey = this.env.STORAGE_CLOUDFLARE_ACCESS_KEY;\n const secretKey = this.env.STORAGE_CLOUDFLARE_SECRET_KEY;\n const endpoint = this.env.STORAGE_CLOUDFLARE_ENDPOINT;\n\n if (!accessKey) {\n throw new StorageException(\n \"Cloudflare access key is required. Please provide an access key either through the constructor options or set the STORAGE_CLOUDFLARE_ACCESS_KEY environment variable.\",\n \"CONFIG_REQUIRED\",\n );\n }\n if (!secretKey) {\n throw new StorageException(\n \"Cloudflare secret key is required. Please provide a secret key either through the constructor options or set the STORAGE_CLOUDFLARE_SECRET_KEY environment variable.\",\n \"CONFIG_REQUIRED\",\n );\n }\n if (!endpoint) {\n throw new StorageException(\n \"Cloudflare endpoint is required. Please provide an endpoint either through the constructor options or set the STORAGE_CLOUDFLARE_ENDPOINT environment variable.\",\n \"CONFIG_REQUIRED\",\n );\n }\n\n this.accessKey = accessKey;\n this.secretKey = secretKey;\n this.endpoint = endpoint;\n this.region = this.env.STORAGE_CLOUDFLARE_REGION || \"EEUR\";\n }\n\n public getOptions(): S3Options {\n return {\n accessKeyId: this.accessKey,\n secretAccessKey: this.secretKey,\n endpoint: this.endpoint,\n bucket: this.bucket,\n region: this.region,\n };\n }\n}\n",
9
9
  "import { readdir } from \"node:fs/promises\";\nimport { basename, join } from \"node:path\";\nimport type { BunFile, S3File, S3Options } from \"bun\";\nimport type { GetFileOptionsType, IStorage, PutDirOptionsType } from \"./types\";\n\nexport abstract class Storage implements IStorage {\n protected client: Bun.S3Client | null = null;\n public abstract getOptions(): S3Options;\n protected abstract bucket: string;\n\n public getBucket(): string {\n return this.bucket;\n }\n\n public setBucket(name: string): this {\n this.bucket = name;\n this.client = new Bun.S3Client(this.getOptions());\n\n return this;\n }\n\n public async list(): Promise<string[]> {\n const client = this.getClient();\n\n return (await client.list()).contents?.map((content) => content.key) || [];\n }\n\n public async clearBucket(): Promise<this> {\n const client = this.getClient();\n const keys = await this.list();\n\n await Promise.all(keys.map((key) => client.delete(key)));\n\n return this;\n }\n\n public async exists(key: string): Promise<boolean> {\n const client = this.getClient();\n\n return await client.exists(key);\n }\n\n public async delete(key: string): Promise<void> {\n const client = this.getClient();\n\n await client.delete(key);\n }\n\n public async putFile(key: string, localPath: string): Promise<number> {\n const file = Bun.file(localPath);\n\n return await this.put(key, file);\n }\n\n public async putDir(bucket: string, options: PutDirOptionsType): Promise<number> {\n const { path, filter } = options;\n const entries = await readdir(path, { withFileTypes: true });\n\n const tasks: Promise<number>[] = [];\n\n for (const entry of entries) {\n const entryLocalPath = join(path, entry.name);\n const entryKey = bucket ? `${bucket}/${entry.name}` : entry.name;\n\n if (filter && !filter.test(entryLocalPath)) {\n continue;\n }\n\n if (entry.isDirectory()) {\n const subOptions: PutDirOptionsType = { path: entryLocalPath };\n if (filter) {\n subOptions.filter = filter;\n }\n tasks.push(this.putDir(entryKey, subOptions));\n } else {\n tasks.push(this.putFile(entryKey, entryLocalPath));\n }\n }\n\n const results = await Promise.all(tasks);\n\n return results.reduce((sum, bytes) => sum + bytes, 0);\n }\n\n public async put(\n key: string,\n content: string | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<number> {\n const s3file: S3File = this.getS3File(key);\n\n return await s3file.write(content);\n }\n\n public async getFile(key: string, options: GetFileOptionsType): Promise<number> {\n const arrayBuffer = await this.getAsArrayBuffer(key);\n const filename = options.filename ?? basename(key);\n const localPath = join(options.outputDir, filename);\n\n return await Bun.write(localPath, arrayBuffer);\n }\n\n public async getAsJson<T>(key: string): Promise<T> {\n const s3file: S3File = this.getS3File(key);\n\n return await s3file.json();\n }\n\n public async getAsArrayBuffer(key: string): Promise<ArrayBuffer> {\n const s3file: S3File = this.getS3File(key);\n\n return await s3file.arrayBuffer();\n }\n\n public getAsStream(key: string): ReadableStream {\n const s3file: S3File = this.getS3File(key);\n\n return s3file.stream();\n }\n\n protected getClient(): Bun.S3Client {\n if (!this.client) {\n this.client = new Bun.S3Client(this.getOptions());\n }\n\n return this.client;\n }\n\n protected getS3File(path: string): S3File {\n const client = this.getClient();\n\n return client.file(path);\n }\n}\n",
10
- "import { existsSync, mkdirSync } from \"node:fs\";\nimport { mkdir, readdir, rm, rmdir } from \"node:fs/promises\";\nimport { basename, dirname, join } from \"node:path\";\nimport { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { BunFile, S3File, S3Options } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { Storage } from \"./Storage\";\nimport { StorageException } from \"./StorageException\";\nimport type { GetFileOptionsType } from \"./types\";\n\n@decorator.storage()\nexport class FilesystemStorage extends Storage {\n protected bucket: string;\n private readonly storagePath: string;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n super();\n\n const basePath = this.env.FILESYSTEM_STORAGE_PATH;\n\n if (!basePath) {\n throw new StorageException(\n \"Base path is required. Please provide a base path either through the constructor options or set the FILESYSTEM_STORAGE_PATH environment variable.\",\n );\n }\n\n this.storagePath = basePath;\n\n try {\n if (!existsSync(basePath)) {\n mkdirSync(basePath, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create base storage directory at ${basePath}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public getOptions(): S3Options {\n return {\n accessKeyId: \"filesystem\",\n secretAccessKey: \"filesystem\",\n endpoint: this.storagePath,\n bucket: this.bucket,\n region: \"local\",\n };\n }\n\n private getBucketPath(): string {\n if (!this.bucket) {\n throw new StorageException(\"Bucket name is required. Please call setBucket() first.\");\n }\n return join(this.storagePath, this.bucket);\n }\n\n private getFilePath(key: string): string {\n return join(this.getBucketPath(), key);\n }\n\n public override setBucket(name: string): this {\n this.bucket = name;\n\n const bucketPath = this.getBucketPath();\n try {\n if (!existsSync(bucketPath)) {\n mkdirSync(bucketPath, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create bucket directory at ${bucketPath}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n\n return this;\n }\n\n public override async list(): Promise<string[]> {\n const bucketPath = this.getBucketPath();\n\n if (!existsSync(bucketPath)) {\n return [];\n }\n\n try {\n const files = await this.listFilesRecursive(bucketPath, bucketPath);\n return files;\n } catch (error) {\n throw new StorageException(\n `Failed to list files in bucket: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n private async listFilesRecursive(dir: string, baseDir: string): Promise<string[]> {\n const entries = await readdir(dir, { withFileTypes: true });\n\n const results = await Promise.all(\n entries.map(async (entry) => {\n const fullPath = join(dir, entry.name);\n\n if (entry.isDirectory()) {\n return this.listFilesRecursive(fullPath, baseDir);\n }\n\n return [fullPath.substring(baseDir.length + 1)];\n }),\n );\n\n return results.flat();\n }\n\n public override async clearBucket(): Promise<this> {\n const bucketPath = this.getBucketPath();\n\n if (!existsSync(bucketPath)) {\n return this;\n }\n\n try {\n await rm(bucketPath, { recursive: true });\n await mkdir(bucketPath, { recursive: true });\n } catch (error) {\n throw new StorageException(`Failed to clear bucket: ${error instanceof Error ? error.message : String(error)}`);\n }\n\n return this;\n }\n\n public override async exists(key: string): Promise<boolean> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n return await file.exists();\n }\n\n public override async delete(key: string): Promise<void> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n return;\n }\n\n try {\n await file.delete();\n\n let parentDir = dirname(filePath);\n const bucketPath = this.getBucketPath();\n\n while (parentDir !== bucketPath && parentDir !== this.storagePath) {\n try {\n const entries = await readdir(parentDir);\n if (entries.length === 0) {\n await rmdir(parentDir);\n parentDir = dirname(parentDir);\n } else {\n break;\n }\n } catch {\n break;\n }\n }\n } catch (error) {\n throw new StorageException(\n `Failed to delete file ${key}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public override async putFile(key: string, localPath: string): Promise<number> {\n const file = Bun.file(localPath);\n return await this.put(key, file);\n }\n\n public override async put(\n key: string,\n content: string | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<number> {\n const filePath = this.getFilePath(key);\n const dir = dirname(filePath);\n\n try {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n\n try {\n let bytesWritten: number;\n\n if (typeof content === \"string\") {\n bytesWritten = await Bun.write(filePath, content);\n } else if (content instanceof ArrayBuffer) {\n bytesWritten = await Bun.write(filePath, content);\n } else if (content instanceof SharedArrayBuffer) {\n const arrayBuffer = new ArrayBuffer(content.byteLength);\n new Uint8Array(arrayBuffer).set(new Uint8Array(content));\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Request) {\n const arrayBuffer = await content.arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Response) {\n const arrayBuffer = await content.arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Blob) {\n bytesWritten = await Bun.write(filePath, content);\n } else {\n const arrayBuffer = await (content as BunFile | S3File).arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n }\n\n return bytesWritten;\n } catch (error) {\n throw new StorageException(\n `Failed to write file ${key}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public override async getFile(key: string, options: GetFileOptionsType): Promise<number> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`);\n }\n\n const filename = options.filename ?? basename(key);\n const localPath = join(options.outputDir, filename);\n const dir = dirname(localPath);\n\n try {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n\n try {\n return await Bun.write(localPath, file);\n } catch (error) {\n throw new StorageException(\n `Failed to save file ${key} to ${localPath}: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public override async getAsJson<T>(key: string): Promise<T> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`);\n }\n\n try {\n return await file.json();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as JSON: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public override async getAsArrayBuffer(key: string): Promise<ArrayBuffer> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`);\n }\n\n try {\n return await file.arrayBuffer();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as ArrayBuffer: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n\n public override getAsStream(key: string): ReadableStream {\n const filePath = this.getFilePath(key);\n\n if (!existsSync(filePath)) {\n throw new StorageException(`File ${key} does not exist`);\n }\n\n const file = Bun.file(filePath);\n\n try {\n return file.stream();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as stream: ${error instanceof Error ? error.message : String(error)}`,\n );\n }\n }\n}\n"
10
+ "import { existsSync, mkdirSync } from \"node:fs\";\nimport { mkdir, readdir, rm, rmdir } from \"node:fs/promises\";\nimport { basename, dirname, join } from \"node:path\";\nimport { AppEnv } from \"@ooneex/app-env\";\nimport { inject } from \"@ooneex/container\";\nimport type { BunFile, S3File, S3Options } from \"bun\";\nimport { decorator } from \"./decorators\";\nimport { Storage } from \"./Storage\";\nimport { StorageException } from \"./StorageException\";\nimport type { GetFileOptionsType } from \"./types\";\n\n@decorator.storage()\nexport class FilesystemStorage extends Storage {\n protected bucket: string;\n private readonly storagePath: string;\n\n constructor(@inject(AppEnv) private readonly env: AppEnv) {\n super();\n\n const basePath = this.env.FILESYSTEM_STORAGE_PATH;\n\n if (!basePath) {\n throw new StorageException(\n \"Base path is required. Please provide a base path either through the constructor options or set the FILESYSTEM_STORAGE_PATH environment variable.\",\n \"STORAGE_ROOT_DIR_REQUIRED\",\n );\n }\n\n this.storagePath = basePath;\n\n try {\n if (!existsSync(basePath)) {\n mkdirSync(basePath, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create base storage directory at ${basePath}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_CONFIG_REQUIRED\",\n );\n }\n }\n\n public getOptions(): S3Options {\n return {\n accessKeyId: \"filesystem\",\n secretAccessKey: \"filesystem\",\n endpoint: this.storagePath,\n bucket: this.bucket,\n region: \"local\",\n };\n }\n\n private getBucketPath(): string {\n if (!this.bucket) {\n throw new StorageException(\"Bucket name is required. Please call setBucket() first.\", \"STORAGE_BUCKET_REQUIRED\");\n }\n return join(this.storagePath, this.bucket);\n }\n\n private getFilePath(key: string): string {\n return join(this.getBucketPath(), key);\n }\n\n public override setBucket(name: string): this {\n this.bucket = name;\n\n const bucketPath = this.getBucketPath();\n try {\n if (!existsSync(bucketPath)) {\n mkdirSync(bucketPath, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create bucket directory at ${bucketPath}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_UPLOAD_FAILED\",\n );\n }\n\n return this;\n }\n\n public override async list(): Promise<string[]> {\n const bucketPath = this.getBucketPath();\n\n if (!existsSync(bucketPath)) {\n return [];\n }\n\n try {\n const files = await this.listFilesRecursive(bucketPath, bucketPath);\n return files;\n } catch (error) {\n throw new StorageException(\n `Failed to list files in bucket: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_LIST_FAILED\",\n );\n }\n }\n\n private async listFilesRecursive(dir: string, baseDir: string): Promise<string[]> {\n const entries = await readdir(dir, { withFileTypes: true });\n\n const results = await Promise.all(\n entries.map(async (entry) => {\n const fullPath = join(dir, entry.name);\n\n if (entry.isDirectory()) {\n return this.listFilesRecursive(fullPath, baseDir);\n }\n\n return [fullPath.substring(baseDir.length + 1)];\n }),\n );\n\n return results.flat();\n }\n\n public override async clearBucket(): Promise<this> {\n const bucketPath = this.getBucketPath();\n\n if (!existsSync(bucketPath)) {\n return this;\n }\n\n try {\n await rm(bucketPath, { recursive: true });\n await mkdir(bucketPath, { recursive: true });\n } catch (error) {\n throw new StorageException(`Failed to clear bucket: ${error instanceof Error ? error.message : String(error)}`, \"STORAGE_CLEAR_FAILED\");\n }\n\n return this;\n }\n\n public override async exists(key: string): Promise<boolean> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n return await file.exists();\n }\n\n public override async delete(key: string): Promise<void> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n return;\n }\n\n try {\n await file.delete();\n\n let parentDir = dirname(filePath);\n const bucketPath = this.getBucketPath();\n\n while (parentDir !== bucketPath && parentDir !== this.storagePath) {\n try {\n const entries = await readdir(parentDir);\n if (entries.length === 0) {\n await rmdir(parentDir);\n parentDir = dirname(parentDir);\n } else {\n break;\n }\n } catch {\n break;\n }\n }\n } catch (error) {\n throw new StorageException(\n `Failed to delete file ${key}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DELETE_FAILED\",\n );\n }\n }\n\n public override async putFile(key: string, localPath: string): Promise<number> {\n const file = Bun.file(localPath);\n return await this.put(key, file);\n }\n\n public override async put(\n key: string,\n content: string | ArrayBuffer | SharedArrayBuffer | Request | Response | BunFile | S3File | Blob,\n ): Promise<number> {\n const filePath = this.getFilePath(key);\n const dir = dirname(filePath);\n\n try {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_UPLOAD_FAILED\",\n );\n }\n\n try {\n let bytesWritten: number;\n\n if (typeof content === \"string\") {\n bytesWritten = await Bun.write(filePath, content);\n } else if (content instanceof ArrayBuffer) {\n bytesWritten = await Bun.write(filePath, content);\n } else if (content instanceof SharedArrayBuffer) {\n const arrayBuffer = new ArrayBuffer(content.byteLength);\n new Uint8Array(arrayBuffer).set(new Uint8Array(content));\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Request) {\n const arrayBuffer = await content.arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Response) {\n const arrayBuffer = await content.arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n } else if (content instanceof Blob) {\n bytesWritten = await Bun.write(filePath, content);\n } else {\n const arrayBuffer = await (content as BunFile | S3File).arrayBuffer();\n bytesWritten = await Bun.write(filePath, arrayBuffer);\n }\n\n return bytesWritten;\n } catch (error) {\n throw new StorageException(\n `Failed to write file ${key}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_UPLOAD_FAILED\",\n );\n }\n }\n\n public override async getFile(key: string, options: GetFileOptionsType): Promise<number> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`, \"FILE_NOT_FOUND\");\n }\n\n const filename = options.filename ?? basename(key);\n const localPath = join(options.outputDir, filename);\n const dir = dirname(localPath);\n\n try {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n } catch (error) {\n throw new StorageException(\n `Failed to create directory ${dir}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n );\n }\n\n try {\n return await Bun.write(localPath, file);\n } catch (error) {\n throw new StorageException(\n `Failed to save file ${key} to ${localPath}: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n );\n }\n }\n\n public override async getAsJson<T>(key: string): Promise<T> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`, \"FILE_NOT_FOUND\");\n }\n\n try {\n return await file.json();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as JSON: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n );\n }\n }\n\n public override async getAsArrayBuffer(key: string): Promise<ArrayBuffer> {\n const filePath = this.getFilePath(key);\n const file = Bun.file(filePath);\n\n if (!(await file.exists())) {\n throw new StorageException(`File ${key} does not exist`, \"FILE_NOT_FOUND\");\n }\n\n try {\n return await file.arrayBuffer();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as ArrayBuffer: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n );\n }\n }\n\n public override getAsStream(key: string): ReadableStream {\n const filePath = this.getFilePath(key);\n\n if (!existsSync(filePath)) {\n throw new StorageException(`File ${key} does not exist`, \"FILE_NOT_FOUND\");\n }\n\n const file = Bun.file(filePath);\n\n try {\n return file.stream();\n } catch (error) {\n throw new StorageException(\n `Failed to read file ${key} as stream: ${error instanceof Error ? error.message : String(error)}`,\n \"STORAGE_DOWNLOAD_FAILED\",\n );\n }\n }\n}\n"
11
11
  ],
12
- "mappings": "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",
13
- "debugId": "E5B84E64D1323F2864756E2164756E21",
12
+ "mappings": "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",
13
+ "debugId": "B7060C45594F90B964756E2164756E21",
14
14
  "names": []
15
15
  }
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@ooneex/storage",
3
3
  "description": "File storage abstraction supporting local filesystem and cloud providers — upload, download, list, and manage files with a unified bucket-based API",
4
- "version": "1.1.3",
4
+ "version": "1.1.5",
5
5
  "type": "module",
6
6
  "files": [
7
7
  "dist",
@@ -28,10 +28,11 @@
28
28
  "npm:publish": "bun publish --tolerate-republish --force --production --access public"
29
29
  },
30
30
  "dependencies": {
31
- "@ooneex/app-env": "1.2.0",
32
- "@ooneex/container": "1.1.2",
33
- "@ooneex/exception": "1.1.2",
34
- "@ooneex/http-status": "1.1.2"
31
+ "@bunny.net/storage-sdk": "^0.3.1",
32
+ "@ooneex/app-env": "1.2.1",
33
+ "@ooneex/container": "1.2.1",
34
+ "@ooneex/exception": "1.1.3",
35
+ "@ooneex/http-status": "1.1.3"
35
36
  },
36
37
  "keywords": [
37
38
  "blob",