mirror of
https://github.com/spacebarchat/server.git
synced 2024-11-14 06:32:36 +01:00
feat: implement an S3-based storage API
This commit is contained in:
parent
2bf3ee4290
commit
e496a15f2d
9891
cdn/package-lock.json
generated
9891
cdn/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -42,6 +42,8 @@
|
|||||||
"ts-patch": "^1.4.4"
|
"ts-patch": "^1.4.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@aws-sdk/client-s3": "^3.36.1",
|
||||||
|
"@aws-sdk/node-http-handler": "^3.36.0",
|
||||||
"@fosscord/util": "file:../util",
|
"@fosscord/util": "file:../util",
|
||||||
"body-parser": "^1.19.0",
|
"body-parser": "^1.19.0",
|
||||||
"btoa": "^1.2.1",
|
"btoa": "^1.2.1",
|
||||||
|
53
cdn/src/util/S3Storage.ts
Normal file
53
cdn/src/util/S3Storage.ts
Normal file
@ -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}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -2,6 +2,8 @@ import { FileStorage } from "./FileStorage";
|
|||||||
import path from "path";
|
import path from "path";
|
||||||
import fse from "fs-extra";
|
import fse from "fs-extra";
|
||||||
import { bgCyan, black } from "nanocolors";
|
import { bgCyan, black } from "nanocolors";
|
||||||
|
import { S3 } from '@aws-sdk/client-s3';
|
||||||
|
import { S3Storage } from "./S3Storage";
|
||||||
process.cwd();
|
process.cwd();
|
||||||
|
|
||||||
export interface Storage {
|
export interface Storage {
|
||||||
@ -10,10 +12,10 @@ export interface Storage {
|
|||||||
delete(path: string): Promise<void>;
|
delete(path: string): Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
var storage: Storage;
|
let storage: Storage;
|
||||||
|
|
||||||
if (process.env.STORAGE_PROVIDER === "file" || !process.env.STORAGE_PROVIDER) {
|
if (process.env.STORAGE_PROVIDER === "file" || !process.env.STORAGE_PROVIDER) {
|
||||||
var location = process.env.STORAGE_LOCATION;
|
let location = process.env.STORAGE_LOCATION;
|
||||||
if (location) {
|
if (location) {
|
||||||
location = path.resolve(location);
|
location = path.resolve(location);
|
||||||
} else {
|
} else {
|
||||||
@ -24,6 +26,32 @@ if (process.env.STORAGE_PROVIDER === "file" || !process.env.STORAGE_PROVIDER) {
|
|||||||
process.env.STORAGE_LOCATION = location;
|
process.env.STORAGE_LOCATION = location;
|
||||||
|
|
||||||
storage = new FileStorage();
|
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 };
|
export { storage };
|
||||||
|
Loading…
Reference in New Issue
Block a user