diff options
author | Hayden Young <hi@hbjy.dev> | 2021-10-14 23:04:03 +0100 |
---|---|---|
committer | Hayden Young <hi@hbjy.dev> | 2021-10-14 23:04:03 +0100 |
commit | e496a15f2d5e8440c026398bde0fb02f4a67bad1 (patch) | |
tree | a701b7580c1cbbbfc7bfce8a08fb4611aecf4c45 /cdn/src | |
parent | Merge pull request #438 from Mr2u/dev (diff) | |
download | server-e496a15f2d5e8440c026398bde0fb02f4a67bad1.tar.xz |
feat: implement an S3-based storage API
Diffstat (limited to 'cdn/src')
-rw-r--r-- | cdn/src/util/S3Storage.ts | 53 | ||||
-rw-r--r-- | cdn/src/util/Storage.ts | 32 |
2 files changed, 83 insertions, 2 deletions
diff --git a/cdn/src/util/S3Storage.ts b/cdn/src/util/S3Storage.ts new file mode 100644 index 00000000..33a6ba39 --- /dev/null +++ b/cdn/src/util/S3Storage.ts @@ -0,0 +1,53 @@ +import { S3 } from "@aws-sdk/client-s3"; +import { Readable, Stream } from "stream"; +import { Storage } from "./Storage"; + +const readableToBuffer = (readable: Readable): Promise<Buffer> => + new Promise((resolve, reject) => { + const chunks: Buffer[] = []; + readable.on('data', chunk => chunks.push(chunk)); + readable.on('error', reject); + readable.on('end', () => resolve(Buffer.concat(chunks))); + }); + +export class S3Storage implements Storage { + public constructor( + private client: S3, + private basePath: string, + private bucket: string + ) {} + + async set(path: string, data: Buffer): Promise<void> { + await this.client.putObject({ + Bucket: this.bucket, + Key: `${this.basePath}${path}`, + Body: data + }); + } + + async get(path: string): Promise<Buffer | null> { + try { + const s3Object = await this.client.getObject({ + Bucket: this.bucket, + Key: `${this.basePath}${path}` + }); + + if (!s3Object.Body) return null; + + const body = s3Object.Body; + + return await readableToBuffer(<Readable> body); + } catch(err) { + console.error(`[CDN] Unable to get S3 object at path ${path}.`); + console.error(err); + return null; + } + } + + async delete(path: string): Promise<void> { + await this.client.deleteObject({ + Bucket: this.bucket, + Key: `${this.basePath}${path}` + }); + } +} diff --git a/cdn/src/util/Storage.ts b/cdn/src/util/Storage.ts index 91f841a6..acef9df3 100644 --- a/cdn/src/util/Storage.ts +++ b/cdn/src/util/Storage.ts @@ -2,6 +2,8 @@ import { FileStorage } from "./FileStorage"; import path from "path"; import fse from "fs-extra"; import { bgCyan, black } from "nanocolors"; +import { S3 } from '@aws-sdk/client-s3'; +import { S3Storage } from "./S3Storage"; process.cwd(); export interface Storage { @@ -10,10 +12,10 @@ export interface Storage { delete(path: string): Promise<void>; } -var storage: Storage; +let storage: Storage; if (process.env.STORAGE_PROVIDER === "file" || !process.env.STORAGE_PROVIDER) { - var location = process.env.STORAGE_LOCATION; + let location = process.env.STORAGE_LOCATION; if (location) { location = path.resolve(location); } else { @@ -24,6 +26,32 @@ if (process.env.STORAGE_PROVIDER === "file" || !process.env.STORAGE_PROVIDER) { process.env.STORAGE_LOCATION = location; storage = new FileStorage(); +} else if (process.env.STORAGE_PROVIDER === "s3") { + const + region = process.env.STORAGE_REGION, + bucket = process.env.STORAGE_BUCKET; + + if (!region) { + console.error(`[CDN] You must provide a region when using the S3 storage provider.`); + process.exit(1); + } + + if (!bucket) { + console.error(`[CDN] You must provide a bucket when using the S3 storage provider.`); + process.exit(1); + } + + // in the S3 provider, this should be the root path in the bucket + let location = process.env.STORAGE_LOCATION; + + if (!location) { + console.warn(`[CDN] STORAGE_LOCATION unconfigured for S3 provider, defaulting to '/'...`); + location = "/"; + } + + const client = new S3({ region }); + + storage = new S3Storage(client, location, bucket); } export { storage }; |