feat: blob size limit with quota (#5524)

fix AFF-506 TOV-342
This commit is contained in:
DarkSky 2024-01-11 10:21:40 +00:00
parent d1c2b2a7b0
commit d6f65ea414
No known key found for this signature in database
GPG Key ID: 97B7D036B1566E9D
8 changed files with 150 additions and 26 deletions

View File

@ -114,6 +114,7 @@
"typescript": "^5.3.2" "typescript": "^5.3.2"
}, },
"ava": { "ava": {
"timeout": "1m",
"extensions": { "extensions": {
"ts": "module" "ts": "module"
}, },

View File

@ -37,6 +37,23 @@ export const Quotas: Quota[] = [
memberLimit: 10, memberLimit: 10,
}, },
}, },
{
feature: QuotaType.RestrictedPlanV1,
type: FeatureKind.Quota,
version: 1,
configs: {
// quota name
name: 'Restricted',
// single blob limit 10MB
blobLimit: OneMB,
// total blob limit 1GB
storageQuota: 10 * OneMB,
// history period of validity 30 days
historyPeriod: 30 * OneDay,
// member limit 10
memberLimit: 10,
},
},
]; ];
export const Quota_FreePlanV1 = { export const Quota_FreePlanV1 = {

View File

@ -44,11 +44,11 @@ export class QuotaManagementService {
const { user: owner } = const { user: owner } =
await this.permissions.getWorkspaceOwner(workspaceId); await this.permissions.getWorkspaceOwner(workspaceId);
if (!owner) throw new NotFoundException('Workspace owner not found'); if (!owner) throw new NotFoundException('Workspace owner not found');
const { storageQuota } = await this.getUserQuota(owner.id); const { storageQuota, blobLimit } = await this.getUserQuota(owner.id);
// get all workspaces size of owner used // get all workspaces size of owner used
const usageSize = await this.getUserUsage(owner.id); const usageSize = await this.getUserUsage(owner.id);
return { quota: storageQuota, size: usageSize }; return { quota: storageQuota, size: usageSize, limit: blobLimit };
} }
async checkBlobQuota(workspaceId: string, size: number) { async checkBlobQuota(workspaceId: string, size: number) {

View File

@ -8,10 +8,16 @@ import { ByteUnit, OneDay, OneKB } from './constant';
export enum QuotaType { export enum QuotaType {
FreePlanV1 = 'free_plan_v1', FreePlanV1 = 'free_plan_v1',
ProPlanV1 = 'pro_plan_v1', ProPlanV1 = 'pro_plan_v1',
// only for test, smaller quota
RestrictedPlanV1 = 'restricted_plan_v1',
} }
const quotaPlan = z.object({ const quotaPlan = z.object({
feature: z.enum([QuotaType.FreePlanV1, QuotaType.ProPlanV1]), feature: z.enum([
QuotaType.FreePlanV1,
QuotaType.ProPlanV1,
QuotaType.RestrictedPlanV1,
]),
configs: z.object({ configs: z.object({
name: z.string(), name: z.string(),
blobLimit: z.number().positive().int(), blobLimit: z.number().positive().int(),

View File

@ -1,4 +1,4 @@
import { ForbiddenException, Logger, UseGuards } from '@nestjs/common'; import { HttpStatus, Logger, UseGuards } from '@nestjs/common';
import { import {
Args, Args,
Float, Float,
@ -9,12 +9,14 @@ import {
ResolveField, ResolveField,
Resolver, Resolver,
} from '@nestjs/graphql'; } from '@nestjs/graphql';
import { GraphQLError } from 'graphql';
import GraphQLUpload from 'graphql-upload/GraphQLUpload.mjs'; import GraphQLUpload from 'graphql-upload/GraphQLUpload.mjs';
import { MakeCache, PreventCache } from '../../../cache'; import { MakeCache, PreventCache } from '../../../cache';
import { CloudThrottlerGuard } from '../../../throttler'; import { CloudThrottlerGuard } from '../../../throttler';
import type { FileUpload } from '../../../types'; import type { FileUpload } from '../../../types';
import { Auth, CurrentUser } from '../../auth'; import { Auth, CurrentUser } from '../../auth';
import { FeatureManagementService, FeatureType } from '../../features';
import { QuotaManagementService } from '../../quota'; import { QuotaManagementService } from '../../quota';
import { WorkspaceBlobStorage } from '../../storage'; import { WorkspaceBlobStorage } from '../../storage';
import { UserType } from '../../users'; import { UserType } from '../../users';
@ -28,10 +30,26 @@ export class WorkspaceBlobResolver {
logger = new Logger(WorkspaceBlobResolver.name); logger = new Logger(WorkspaceBlobResolver.name);
constructor( constructor(
private readonly permissions: PermissionService, private readonly permissions: PermissionService,
private readonly feature: FeatureManagementService,
private readonly quota: QuotaManagementService, private readonly quota: QuotaManagementService,
private readonly storage: WorkspaceBlobStorage private readonly storage: WorkspaceBlobStorage
) {} ) {}
@ResolveField(() => [String], {
description: 'List blobs of workspace',
complexity: 2,
})
async blobs(
@CurrentUser() user: UserType,
@Parent() workspace: WorkspaceType
) {
await this.permissions.checkWorkspace(workspace.id, user.id);
return this.storage
.list(workspace.id)
.then(list => list.map(item => item.key));
}
@ResolveField(() => Int, { @ResolveField(() => Int, {
description: 'Blobs size of workspace', description: 'Blobs size of workspace',
complexity: 2, complexity: 2,
@ -107,16 +125,30 @@ export class WorkspaceBlobResolver {
Permission.Write Permission.Write
); );
const { quota, size } = await this.quota.getWorkspaceUsage(workspaceId); const { quota, size, limit } =
await this.quota.getWorkspaceUsage(workspaceId);
const unlimited = await this.feature.hasWorkspaceFeature(
workspaceId,
FeatureType.UnlimitedWorkspace
);
const checkExceeded = (recvSize: number) => { const checkExceeded = (recvSize: number) => {
if (!quota) { if (!quota) {
throw new ForbiddenException('cannot find user quota'); throw new GraphQLError('cannot find user quota', {
extensions: {
status: HttpStatus[HttpStatus.FORBIDDEN],
code: HttpStatus.FORBIDDEN,
},
});
} }
if (size + recvSize > quota) { const total = size + recvSize;
this.logger.log( // only skip total storage check if workspace has unlimited feature
`storage size limit exceeded: ${size + recvSize} > ${quota}` if (total > quota && !unlimited) {
); this.logger.log(`storage size limit exceeded: ${total} > ${quota}`);
return true;
} else if (recvSize > limit) {
this.logger.log(`blob size limit exceeded: ${recvSize} > ${limit}`);
return true; return true;
} else { } else {
return false; return false;
@ -124,7 +156,12 @@ export class WorkspaceBlobResolver {
}; };
if (checkExceeded(0)) { if (checkExceeded(0)) {
throw new ForbiddenException('storage size limit exceeded'); throw new GraphQLError('storage or blob size limit exceeded', {
extensions: {
status: HttpStatus[HttpStatus.PAYLOAD_TOO_LARGE],
code: HttpStatus.PAYLOAD_TOO_LARGE,
},
});
} }
const buffer = await new Promise<Buffer>((resolve, reject) => { const buffer = await new Promise<Buffer>((resolve, reject) => {
const stream = blob.createReadStream(); const stream = blob.createReadStream();
@ -135,7 +172,14 @@ export class WorkspaceBlobResolver {
// check size after receive each chunk to avoid unnecessary memory usage // check size after receive each chunk to avoid unnecessary memory usage
const bufferSize = chunks.reduce((acc, cur) => acc + cur.length, 0); const bufferSize = chunks.reduce((acc, cur) => acc + cur.length, 0);
if (checkExceeded(bufferSize)) { if (checkExceeded(bufferSize)) {
reject(new ForbiddenException('storage size limit exceeded')); reject(
new GraphQLError('storage or blob size limit exceeded', {
extensions: {
status: HttpStatus[HttpStatus.PAYLOAD_TOO_LARGE],
code: HttpStatus.PAYLOAD_TOO_LARGE,
},
})
);
} }
}); });
stream.on('error', reject); stream.on('error', reject);
@ -143,17 +187,20 @@ export class WorkspaceBlobResolver {
const buffer = Buffer.concat(chunks); const buffer = Buffer.concat(chunks);
if (checkExceeded(buffer.length)) { if (checkExceeded(buffer.length)) {
reject(new ForbiddenException('storage size limit exceeded')); reject(
new GraphQLError('storage limit exceeded', {
extensions: {
status: HttpStatus[HttpStatus.PAYLOAD_TOO_LARGE],
code: HttpStatus.PAYLOAD_TOO_LARGE,
},
})
);
} else { } else {
resolve(buffer); resolve(buffer);
} }
}); });
}); });
if (!(await this.quota.checkBlobQuota(workspaceId, buffer.length))) {
throw new ForbiddenException('blob size limit exceeded');
}
await this.storage.put(workspaceId, blob.filename, buffer); await this.storage.put(workspaceId, blob.filename, buffer);
return blob.filename; return blob.filename;
} }

View File

@ -1,7 +1,6 @@
import { import {
ForbiddenException, ForbiddenException,
HttpStatus, HttpStatus,
InternalServerErrorException,
Logger, Logger,
NotFoundException, NotFoundException,
UseGuards, UseGuards,
@ -398,8 +397,15 @@ export class WorkspaceResolver {
`failed to send ${workspaceId} invite email to ${email}, but successfully revoked permission: ${e}` `failed to send ${workspaceId} invite email to ${email}, but successfully revoked permission: ${e}`
); );
} }
return new GraphQLError(
return new InternalServerErrorException(e); 'failed to send invite email, please try again',
{
extensions: {
status: HttpStatus[HttpStatus.INTERNAL_SERVER_ERROR],
code: HttpStatus.INTERNAL_SERVER_ERROR,
},
}
);
} }
} }
return inviteId; return inviteId;

View File

@ -144,6 +144,9 @@ type WorkspaceType {
publicPages: [WorkspacePage!]! publicPages: [WorkspacePage!]!
histories(guid: String!, before: DateTime, take: Int): [DocHistoryType!]! histories(guid: String!, before: DateTime, take: Int): [DocHistoryType!]!
"""List blobs of workspace"""
blobs: [String!]!
"""Blobs size of workspace""" """Blobs size of workspace"""
blobsSize: Int! blobsSize: Int!
} }

View File

@ -3,6 +3,7 @@ import test from 'ava';
import request from 'supertest'; import request from 'supertest';
import { AppModule } from '../src/app'; import { AppModule } from '../src/app';
import { FeatureManagementService, FeatureType } from '../src/modules/features';
import { QuotaService, QuotaType } from '../src/modules/quota'; import { QuotaService, QuotaType } from '../src/modules/quota';
import { import {
checkBlobSize, checkBlobSize,
@ -15,8 +16,11 @@ import {
signUp, signUp,
} from './utils'; } from './utils';
const OneMB = 1024 * 1024;
let app: INestApplication; let app: INestApplication;
let quota: QuotaService; let quota: QuotaService;
let feature: FeatureManagementService;
test.beforeEach(async () => { test.beforeEach(async () => {
const { app: testApp } = await createTestingApp({ const { app: testApp } = await createTestingApp({
@ -25,6 +29,7 @@ test.beforeEach(async () => {
app = testApp; app = testApp;
quota = app.get(QuotaService); quota = app.get(QuotaService);
feature = app.get(FeatureManagementService);
}); });
test.afterEach.always(async () => { test.afterEach.always(async () => {
@ -163,12 +168,51 @@ test('should be able calc quota after switch plan', async t => {
t.is(size2, 0, 'failed to check pro plan blob size'); t.is(size2, 0, 'failed to check pro plan blob size');
}); });
test('should reject blob exceeded limit', t => { test('should reject blob exceeded limit', async t => {
// TODO const u1 = await signUp(app, 'darksky', 'darksky@affine.pro', '1');
t.true(true);
const workspace1 = await createWorkspace(app, u1.token.token);
await quota.switchUserQuota(u1.id, QuotaType.RestrictedPlanV1);
const buffer1 = Buffer.from(Array.from({ length: OneMB + 1 }, () => 0));
await t.throwsAsync(setBlob(app, u1.token.token, workspace1.id, buffer1));
await quota.switchUserQuota(u1.id, QuotaType.FreePlanV1);
const buffer2 = Buffer.from(Array.from({ length: OneMB + 1 }, () => 0));
await t.notThrowsAsync(setBlob(app, u1.token.token, workspace1.id, buffer2));
const buffer3 = Buffer.from(Array.from({ length: 10 * OneMB + 1 }, () => 0));
await t.throwsAsync(setBlob(app, u1.token.token, workspace1.id, buffer3));
}); });
test('should reject blob exceeded quota', t => { test('should reject blob exceeded quota', async t => {
// TODO const u1 = await signUp(app, 'darksky', 'darksky@affine.pro', '1');
t.true(true);
const workspace = await createWorkspace(app, u1.token.token);
await quota.switchUserQuota(u1.id, QuotaType.RestrictedPlanV1);
const buffer = Buffer.from(Array.from({ length: OneMB }, () => 0));
for (let i = 0; i < 10; i++) {
await t.notThrowsAsync(setBlob(app, u1.token.token, workspace.id, buffer));
}
await t.throwsAsync(setBlob(app, u1.token.token, workspace.id, buffer));
});
test('should accept blob even storage out of quota if workspace has unlimited feature', async t => {
const u1 = await signUp(app, 'darksky', 'darksky@affine.pro', '1');
const workspace = await createWorkspace(app, u1.token.token);
await quota.switchUserQuota(u1.id, QuotaType.RestrictedPlanV1);
feature.addWorkspaceFeatures(workspace.id, FeatureType.UnlimitedWorkspace);
const buffer = Buffer.from(Array.from({ length: OneMB }, () => 0));
for (let i = 0; i < 10; i++) {
await t.notThrowsAsync(setBlob(app, u1.token.token, workspace.id, buffer));
}
await t.notThrowsAsync(setBlob(app, u1.token.token, workspace.id, buffer));
}); });