mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-23 09:44:51 +00:00 
			
		
		
		
	merge: Merge the two File fixes from 2024.5.0 (!997)
View MR for information: https://activitypub.software/TransFem-org/Sharkey/-/merge_requests/997 Approved-by: Hazelnoot <acomputerdog@gmail.com> Approved-by: dakkar <dakkar@thenautilus.net>
This commit is contained in:
		
						commit
						45ecb08a42
					
				
					 7 changed files with 156 additions and 148 deletions
				
			
		|  | @ -7,7 +7,7 @@ import cluster from 'node:cluster'; | ||||||
| import * as fs from 'node:fs'; | import * as fs from 'node:fs'; | ||||||
| import { fileURLToPath } from 'node:url'; | import { fileURLToPath } from 'node:url'; | ||||||
| import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | ||||||
| import Fastify, { FastifyInstance } from 'fastify'; | import Fastify, { type FastifyInstance } from 'fastify'; | ||||||
| import fastifyStatic from '@fastify/static'; | import fastifyStatic from '@fastify/static'; | ||||||
| import fastifyRawBody from 'fastify-raw-body'; | import fastifyRawBody from 'fastify-raw-body'; | ||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
|  | @ -75,7 +75,7 @@ export class ServerService implements OnApplicationShutdown { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async launch() { | 	public async launch(): Promise<void> { | ||||||
| 		const fastify = Fastify({ | 		const fastify = Fastify({ | ||||||
| 			trustProxy: true, | 			trustProxy: true, | ||||||
| 			logger: false, | 			logger: false, | ||||||
|  | @ -135,8 +135,8 @@ export class ServerService implements OnApplicationShutdown { | ||||||
| 				reply.header('content-type', 'text/plain; charset=utf-8'); | 				reply.header('content-type', 'text/plain; charset=utf-8'); | ||||||
| 				reply.header('link', `<${encodeURI(location)}>; rel="canonical"`); | 				reply.header('link', `<${encodeURI(location)}>; rel="canonical"`); | ||||||
| 				done(null, [ | 				done(null, [ | ||||||
| 					'Refusing to relay remote ActivityPub object lookup.', | 					"Refusing to relay remote ActivityPub object lookup.", | ||||||
| 					'', | 					"", | ||||||
| 					`Please remove 'application/activity+json' and 'application/ld+json' from the Accept header or fetch using the authoritative URL at ${location}.`, | 					`Please remove 'application/activity+json' and 'application/ld+json' from the Accept header or fetch using the authoritative URL at ${location}.`, | ||||||
| 				].join('\n')); | 				].join('\n')); | ||||||
| 			}); | 			}); | ||||||
|  | @ -304,7 +304,6 @@ export class ServerService implements OnApplicationShutdown { | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		await fastify.ready(); | 		await fastify.ready(); | ||||||
| 		return fastify; |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	@bindThis | 	@bindThis | ||||||
|  | @ -313,6 +312,13 @@ export class ServerService implements OnApplicationShutdown { | ||||||
| 		await this.#fastify.close(); | 		await this.#fastify.close(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/** | ||||||
|  | 	 * Get the Fastify instance for testing. | ||||||
|  | 	 */ | ||||||
|  | 	public get fastify(): FastifyInstance { | ||||||
|  | 		return this.#fastify; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	async onApplicationShutdown(signal: string): Promise<void> { | 	async onApplicationShutdown(signal: string): Promise<void> { | ||||||
| 		await this.dispose(); | 		await this.dispose(); | ||||||
|  |  | ||||||
|  | @ -6,11 +6,8 @@ | ||||||
| import { randomUUID } from 'node:crypto'; | import { randomUUID } from 'node:crypto'; | ||||||
| import * as fs from 'node:fs'; | import * as fs from 'node:fs'; | ||||||
| import * as stream from 'node:stream/promises'; | import * as stream from 'node:stream/promises'; | ||||||
| import { Transform } from 'node:stream'; |  | ||||||
| import { type MultipartFile } from '@fastify/multipart'; |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as Sentry from '@sentry/node'; | import * as Sentry from '@sentry/node'; | ||||||
| import { AttachmentFile } from '@/server/api/endpoint-base.js'; |  | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { getIpHash } from '@/misc/get-ip-hash.js'; | import { getIpHash } from '@/misc/get-ip-hash.js'; | ||||||
| import type { MiLocalUser, MiUser } from '@/models/User.js'; | import type { MiLocalUser, MiUser } from '@/models/User.js'; | ||||||
|  | @ -19,7 +16,7 @@ import type Logger from '@/logger.js'; | ||||||
| import type { MiMeta, UserIpsRepository } from '@/models/_.js'; | import type { MiMeta, UserIpsRepository } from '@/models/_.js'; | ||||||
| import { createTemp } from '@/misc/create-temp.js'; | import { createTemp } from '@/misc/create-temp.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { type RolePolicies, RoleService } from '@/core/RoleService.js'; | import { RoleService } from '@/core/RoleService.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import { sendRateLimitHeaders } from '@/misc/rate-limit-utils.js'; | import { sendRateLimitHeaders } from '@/misc/rate-limit-utils.js'; | ||||||
| import { SkRateLimiterService } from '@/server/SkRateLimiterService.js'; | import { SkRateLimiterService } from '@/server/SkRateLimiterService.js'; | ||||||
|  | @ -194,6 +191,18 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		const [path, cleanup] = await createTemp(); | ||||||
|  | 		await stream.pipeline(multipartData.file, fs.createWriteStream(path)); | ||||||
|  | 
 | ||||||
|  | 		// ファイルサイズが制限を超えていた場合
 | ||||||
|  | 		// なお truncated はストリームを読み切ってからでないと機能しないため、stream.pipeline より後にある必要がある
 | ||||||
|  | 		if (multipartData.file.truncated) { | ||||||
|  | 			cleanup(); | ||||||
|  | 			reply.code(413); | ||||||
|  | 			reply.send(); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		const fields = {} as Record<string, unknown>; | 		const fields = {} as Record<string, unknown>; | ||||||
| 		for (const [k, v] of Object.entries(multipartData.fields)) { | 		for (const [k, v] of Object.entries(multipartData.fields)) { | ||||||
| 			fields[k] = typeof v === 'object' && 'value' in v ? v.value : undefined; | 			fields[k] = typeof v === 'object' && 'value' in v ? v.value : undefined; | ||||||
|  | @ -204,13 +213,18 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 			? request.headers.authorization.slice(7) | 			? request.headers.authorization.slice(7) | ||||||
| 			: fields['i']; | 			: fields['i']; | ||||||
| 		if (token != null && typeof token !== 'string') { | 		if (token != null && typeof token !== 'string') { | ||||||
|  | 			cleanup(); | ||||||
| 			reply.code(400); | 			reply.code(400); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 		this.authenticateService.authenticate(token).then(([user, app]) => { | 		this.authenticateService.authenticate(token).then(([user, app]) => { | ||||||
| 			this.call(endpoint, user, app, fields, multipartData, request, reply).then((res) => { | 			this.call(endpoint, user, app, fields, { | ||||||
|  | 				name: multipartData.filename, | ||||||
|  | 				path: path, | ||||||
|  | 			}, request, reply).then((res) => { | ||||||
| 				this.send(reply, res); | 				this.send(reply, res); | ||||||
| 			}).catch((err: ApiError) => { | 			}).catch((err: ApiError) => { | ||||||
|  | 				cleanup(); | ||||||
| 				this.#sendApiError(reply, err); | 				this.#sendApiError(reply, err); | ||||||
| 			}); | 			}); | ||||||
| 
 | 
 | ||||||
|  | @ -218,6 +232,7 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 				this.logIp(request, user); | 				this.logIp(request, user); | ||||||
| 			} | 			} | ||||||
| 		}).catch(err => { | 		}).catch(err => { | ||||||
|  | 			cleanup(); | ||||||
| 			this.#sendAuthenticationError(reply, err); | 			this.#sendAuthenticationError(reply, err); | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
|  | @ -278,7 +293,10 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 		user: MiLocalUser | null | undefined, | 		user: MiLocalUser | null | undefined, | ||||||
| 		token: MiAccessToken | null | undefined, | 		token: MiAccessToken | null | undefined, | ||||||
| 		data: any, | 		data: any, | ||||||
| 		multipartFile: MultipartFile | null, | 		file: { | ||||||
|  | 			name: string; | ||||||
|  | 			path: string; | ||||||
|  | 		} | null, | ||||||
| 		request: FastifyRequest<{ Body: Record<string, unknown> | undefined, Querystring: Record<string, unknown> }>, | 		request: FastifyRequest<{ Body: Record<string, unknown> | undefined, Querystring: Record<string, unknown> }>, | ||||||
| 		reply: FastifyReply, | 		reply: FastifyReply, | ||||||
| 	) { | 	) { | ||||||
|  | @ -354,37 +372,6 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		// Cast non JSON input
 |  | ||||||
| 		if ((ep.meta.requireFile || request.method === 'GET') && ep.params.properties) { |  | ||||||
| 			for (const k of Object.keys(ep.params.properties)) { |  | ||||||
| 				const param = ep.params.properties![k]; |  | ||||||
| 				if (['boolean', 'number', 'integer'].includes(param.type ?? '') && typeof data[k] === 'string') { |  | ||||||
| 					try { |  | ||||||
| 						data[k] = JSON.parse(data[k]); |  | ||||||
| 					} catch (e) { |  | ||||||
| 						throw new ApiError({ |  | ||||||
| 							message: 'Invalid param.', |  | ||||||
| 							code: 'INVALID_PARAM', |  | ||||||
| 							id: '0b5f1631-7c1a-41a6-b399-cce335f34d85', |  | ||||||
| 						}, { |  | ||||||
| 							param: k, |  | ||||||
| 							reason: `cannot cast to ${param.type}`, |  | ||||||
| 						}); |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if (token && ((ep.meta.kind && !token.permission.some(p => p === ep.meta.kind)) |  | ||||||
| 			|| (!ep.meta.kind && (ep.meta.requireCredential || ep.meta.requireModerator || ep.meta.requireAdmin)))) { |  | ||||||
| 			throw new ApiError({ |  | ||||||
| 				message: 'Your app does not have the necessary permissions to use this endpoint.', |  | ||||||
| 				code: 'PERMISSION_DENIED', |  | ||||||
| 				kind: 'permission', |  | ||||||
| 				id: '1370e5b7-d4eb-4566-bb1d-7748ee6a1838', |  | ||||||
| 			}); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if ((ep.meta.requireModerator || ep.meta.requireAdmin) && (this.meta.rootUserId !== user!.id)) { | 		if ((ep.meta.requireModerator || ep.meta.requireAdmin) && (this.meta.rootUserId !== user!.id)) { | ||||||
| 			const myRoles = await this.roleService.getUserRoles(user!.id); | 			const myRoles = await this.roleService.getUserRoles(user!.id); | ||||||
| 			if (ep.meta.requireModerator && !myRoles.some(r => r.isModerator || r.isAdministrator)) { | 			if (ep.meta.requireModerator && !myRoles.some(r => r.isModerator || r.isAdministrator)) { | ||||||
|  | @ -418,91 +405,49 @@ export class ApiCallService implements OnApplicationShutdown { | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		let attachmentFile: AttachmentFile | null = null; | 		if (token && ((ep.meta.kind && !token.permission.some(p => p === ep.meta.kind)) | ||||||
| 		let cleanup = () => {}; | 			|| (!ep.meta.kind && (ep.meta.requireCredential || ep.meta.requireModerator || ep.meta.requireAdmin)))) { | ||||||
| 		if (ep.meta.requireFile && request.method === 'POST' && multipartFile) { | 			throw new ApiError({ | ||||||
| 			const policies = await this.roleService.getUserPolicies(user!.id); | 				message: 'Your app does not have the necessary permissions to use this endpoint.', | ||||||
| 			const result = await this.handleAttachmentFile( | 				code: 'PERMISSION_DENIED', | ||||||
| 				Math.min((policies.maxFileSizeMb * 1024 * 1024), this.config.maxFileSize), | 				kind: 'permission', | ||||||
| 				multipartFile, | 				id: '1370e5b7-d4eb-4566-bb1d-7748ee6a1838', | ||||||
| 			); | 			}); | ||||||
| 			attachmentFile = result.attachmentFile; | 		} | ||||||
| 			cleanup = result.cleanup; | 
 | ||||||
|  | 		// Cast non JSON input
 | ||||||
|  | 		if ((ep.meta.requireFile || request.method === 'GET') && ep.params.properties) { | ||||||
|  | 			for (const k of Object.keys(ep.params.properties)) { | ||||||
|  | 				const param = ep.params.properties![k]; | ||||||
|  | 				if (['boolean', 'number', 'integer'].includes(param.type ?? '') && typeof data[k] === 'string') { | ||||||
|  | 					try { | ||||||
|  | 						data[k] = JSON.parse(data[k]); | ||||||
|  | 					} catch (e) { | ||||||
|  | 						throw new ApiError({ | ||||||
|  | 							message: 'Invalid param.', | ||||||
|  | 							code: 'INVALID_PARAM', | ||||||
|  | 							id: '0b5f1631-7c1a-41a6-b399-cce335f34d85', | ||||||
|  | 						}, { | ||||||
|  | 							param: k, | ||||||
|  | 							reason: `cannot cast to ${param.type}`, | ||||||
|  | 						}); | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		// API invoking
 | 		// API invoking
 | ||||||
| 		if (this.config.sentryForBackend) { | 		if (this.config.sentryForBackend) { | ||||||
| 			return await Sentry.startSpan({ | 			return await Sentry.startSpan({ | ||||||
| 				name: 'API: ' + ep.name, | 				name: 'API: ' + ep.name, | ||||||
| 			}, () => { | 			}, () => ep.exec(data, user, token, file, request.ip, request.headers) | ||||||
| 				return ep.exec(data, user, token, attachmentFile, request.ip, request.headers) | 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id))); | ||||||
| 					.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)) |  | ||||||
| 					.finally(() => cleanup()); |  | ||||||
| 			}); |  | ||||||
| 		} else { | 		} else { | ||||||
| 			return await ep.exec(data, user, token, attachmentFile, request.ip, request.headers) | 			return await ep.exec(data, user, token, file, request.ip, request.headers) | ||||||
| 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)) | 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)); | ||||||
| 				.finally(() => cleanup()); |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	@bindThis |  | ||||||
| 	private async handleAttachmentFile( |  | ||||||
| 		fileSizeLimit: number, |  | ||||||
| 		multipartFile: MultipartFile, |  | ||||||
| 	) { |  | ||||||
| 		function createTooLongError() { |  | ||||||
| 			return new ApiError({ |  | ||||||
| 				httpStatusCode: 413, |  | ||||||
| 				kind: 'client', |  | ||||||
| 				message: 'File size is too large.', |  | ||||||
| 				code: 'FILE_SIZE_TOO_LARGE', |  | ||||||
| 				id: 'ff827ce8-9b4b-4808-8511-422222a3362f', |  | ||||||
| 			}); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		function createLimitStream(limit: number) { |  | ||||||
| 			let total = 0; |  | ||||||
| 
 |  | ||||||
| 			return new Transform({ |  | ||||||
| 				transform(chunk, _, callback) { |  | ||||||
| 					total += chunk.length; |  | ||||||
| 					if (total > limit) { |  | ||||||
| 						callback(createTooLongError()); |  | ||||||
| 					} else { |  | ||||||
| 						callback(null, chunk); |  | ||||||
| 					} |  | ||||||
| 				}, |  | ||||||
| 			}); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		const [path, cleanup] = await createTemp(); |  | ||||||
| 		try { |  | ||||||
| 			await stream.pipeline( |  | ||||||
| 				multipartFile.file, |  | ||||||
| 				createLimitStream(fileSizeLimit), |  | ||||||
| 				fs.createWriteStream(path), |  | ||||||
| 			); |  | ||||||
| 
 |  | ||||||
| 			// ファイルサイズが制限を超えていた場合
 |  | ||||||
| 			// なお truncated はストリームを読み切ってからでないと機能しないため、stream.pipeline より後にある必要がある
 |  | ||||||
| 			if (multipartFile.file.truncated) { |  | ||||||
| 				throw createTooLongError(); |  | ||||||
| 			} |  | ||||||
| 		} catch (err) { |  | ||||||
| 			cleanup(); |  | ||||||
| 			throw err; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		return { |  | ||||||
| 			attachmentFile: { |  | ||||||
| 				name: multipartFile.filename, |  | ||||||
| 				path, |  | ||||||
| 			}, |  | ||||||
| 			cleanup, |  | ||||||
| 		}; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public dispose(): void { | 	public dispose(): void { | ||||||
| 		clearInterval(this.userIpHistoriesClearIntervalId); | 		clearInterval(this.userIpHistoriesClearIntervalId); | ||||||
|  |  | ||||||
|  | @ -21,23 +21,23 @@ ajv.addFormat('misskey:id', /^[a-zA-Z0-9]+$/); | ||||||
| 
 | 
 | ||||||
| export type Response = Record<string, any> | void; | export type Response = Record<string, any> | void; | ||||||
| 
 | 
 | ||||||
| export type AttachmentFile = { | type File = { | ||||||
| 	name: string | null; | 	name: string | null; | ||||||
| 	path: string; | 	path: string; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // TODO: paramsの型をT['params']のスキーマ定義から推論する
 | // TODO: paramsの型をT['params']のスキーマ定義から推論する
 | ||||||
| type Executor<T extends IEndpointMeta, Ps extends Schema> = | type Executor<T extends IEndpointMeta, Ps extends Schema> = | ||||||
| 	(params: SchemaType<Ps>, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: AttachmentFile, cleanup?: () => any, ip?: string | null, headers?: Record<string, string> | null) => | 	(params: SchemaType<Ps>, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: File, cleanup?: () => any, ip?: string | null, headers?: Record<string, string> | null) => | ||||||
| 	Promise<T['res'] extends undefined ? Response : SchemaType<NonNullable<T['res']>>>; | 		Promise<T['res'] extends undefined ? Response : SchemaType<NonNullable<T['res']>>>; | ||||||
| 
 | 
 | ||||||
| export abstract class Endpoint<T extends IEndpointMeta, Ps extends Schema> { | export abstract class Endpoint<T extends IEndpointMeta, Ps extends Schema> { | ||||||
| 	public exec: (params: any, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: AttachmentFile, ip?: string | null, headers?: Record<string, string> | null) => Promise<any>; | 	public exec: (params: any, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: File, ip?: string | null, headers?: Record<string, string> | null) => Promise<any>; | ||||||
| 
 | 
 | ||||||
| 	constructor(meta: T, paramDef: Ps, cb: Executor<T, Ps>) { | 	constructor(meta: T, paramDef: Ps, cb: Executor<T, Ps>) { | ||||||
| 		const validate = ajv.compile(paramDef); | 		const validate = ajv.compile(paramDef); | ||||||
| 
 | 
 | ||||||
| 		this.exec = (params: any, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: AttachmentFile, ip?: string | null, headers?: Record<string, string> | null) => { | 		this.exec = (params: any, user: T['requireCredential'] extends true ? MiLocalUser : MiLocalUser | null, token: MiAccessToken | null, file?: File, ip?: string | null, headers?: Record<string, string> | null) => { | ||||||
| 			let cleanup: undefined | (() => void) = undefined; | 			let cleanup: undefined | (() => void) = undefined; | ||||||
| 
 | 
 | ||||||
| 			if (meta.requireFile) { | 			if (meta.requireFile) { | ||||||
|  |  | ||||||
|  | @ -67,6 +67,7 @@ export const meta = { | ||||||
| 			message: 'Cannot upload the file because it exceeds the maximum file size.', | 			message: 'Cannot upload the file because it exceeds the maximum file size.', | ||||||
| 			code: 'MAX_FILE_SIZE_EXCEEDED', | 			code: 'MAX_FILE_SIZE_EXCEEDED', | ||||||
| 			id: 'b9d8c348-33f0-4673-b9a9-5d4da058977a', | 			id: 'b9d8c348-33f0-4673-b9a9-5d4da058977a', | ||||||
|  | 			httpStatusCode: 413, | ||||||
| 		}, | 		}, | ||||||
| 	}, | 	}, | ||||||
| } as const; | } as const; | ||||||
|  |  | ||||||
|  | @ -159,8 +159,8 @@ describe('API', () => { | ||||||
| 			user: { token: application3 }, | 			user: { token: application3 }, | ||||||
| 		}, { | 		}, { | ||||||
| 			status: 403, | 			status: 403, | ||||||
| 			code: 'PERMISSION_DENIED', | 			code: 'ROLE_PERMISSION_DENIED', | ||||||
| 			id: '1370e5b7-d4eb-4566-bb1d-7748ee6a1838', | 			id: 'c3d38592-54c0-429d-be96-5636b0431a61', | ||||||
| 		}); | 		}); | ||||||
| 
 | 
 | ||||||
| 		await failedApiCall({ | 		await failedApiCall({ | ||||||
|  |  | ||||||
|  | @ -3,29 +3,31 @@ | ||||||
|  * SPDX-License-Identifier: AGPL-3.0-only |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { S3Client } from '@aws-sdk/client-s3'; |  | ||||||
| import { Test, TestingModule } from '@nestjs/testing'; | import { Test, TestingModule } from '@nestjs/testing'; | ||||||
| import { mockClient } from 'aws-sdk-client-mock'; |  | ||||||
| import { FastifyInstance } from 'fastify'; | import { FastifyInstance } from 'fastify'; | ||||||
| import request from 'supertest'; | import request from 'supertest'; | ||||||
|  | import { randomString } from '../../../../../utils.js'; | ||||||
| import { CoreModule } from '@/core/CoreModule.js'; | import { CoreModule } from '@/core/CoreModule.js'; | ||||||
| import { RoleService } from '@/core/RoleService.js'; | import { RoleService } from '@/core/RoleService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { GlobalModule } from '@/GlobalModule.js'; | import { GlobalModule } from '@/GlobalModule.js'; | ||||||
| import { MiRole, UserProfilesRepository, UsersRepository } from '@/models/_.js'; | import { DriveFoldersRepository, MiDriveFolder, MiRole, UserProfilesRepository, UsersRepository } from '@/models/_.js'; | ||||||
| import { MiUser } from '@/models/User.js'; | import { MiUser } from '@/models/User.js'; | ||||||
| import { ServerModule } from '@/server/ServerModule.js'; | import { ServerModule } from '@/server/ServerModule.js'; | ||||||
| import { ServerService } from '@/server/ServerService.js'; | import { ServerService } from '@/server/ServerService.js'; | ||||||
|  | import { IdService } from '@/core/IdService.js'; | ||||||
| 
 | 
 | ||||||
| describe('/drive/files/create', () => { | describe('/drive/files/create', () => { | ||||||
| 	let module: TestingModule; | 	let module: TestingModule; | ||||||
| 	let server: FastifyInstance; | 	let server: FastifyInstance; | ||||||
| 	const s3Mock = mockClient(S3Client); |  | ||||||
| 	let roleService: RoleService; | 	let roleService: RoleService; | ||||||
|  | 	let idService: IdService; | ||||||
| 
 | 
 | ||||||
| 	let root: MiUser; | 	let root: MiUser; | ||||||
| 	let role_tinyAttachment: MiRole; | 	let role_tinyAttachment: MiRole; | ||||||
| 
 | 
 | ||||||
|  | 	let folder: MiDriveFolder; | ||||||
|  | 
 | ||||||
| 	beforeAll(async () => { | 	beforeAll(async () => { | ||||||
| 		module = await Test.createTestingModule({ | 		module = await Test.createTestingModule({ | ||||||
| 			imports: [GlobalModule, CoreModule, ServerModule], | 			imports: [GlobalModule, CoreModule, ServerModule], | ||||||
|  | @ -33,21 +35,34 @@ describe('/drive/files/create', () => { | ||||||
| 		module.enableShutdownHooks(); | 		module.enableShutdownHooks(); | ||||||
| 
 | 
 | ||||||
| 		const serverService = module.get<ServerService>(ServerService); | 		const serverService = module.get<ServerService>(ServerService); | ||||||
| 		server = await serverService.launch(); | 		await serverService.launch(); | ||||||
|  | 		server = serverService.fastify; | ||||||
|  | 
 | ||||||
|  | 		idService = module.get(IdService); | ||||||
| 
 | 
 | ||||||
| 		const usersRepository = module.get<UsersRepository>(DI.usersRepository); | 		const usersRepository = module.get<UsersRepository>(DI.usersRepository); | ||||||
|  | 		await usersRepository.delete({}); | ||||||
| 		root = await usersRepository.insert({ | 		root = await usersRepository.insert({ | ||||||
| 			id: 'root', | 			id: idService.gen(), | ||||||
| 			username: 'root', | 			username: 'root', | ||||||
| 			usernameLower: 'root', | 			usernameLower: 'root', | ||||||
| 			token: '1234567890123456', | 			token: '1234567890123456', | ||||||
| 		}).then(x => usersRepository.findOneByOrFail(x.identifiers[0])); | 		}).then(x => usersRepository.findOneByOrFail(x.identifiers[0])); | ||||||
| 
 | 
 | ||||||
| 		const userProfilesRepository = module.get<UserProfilesRepository>(DI.userProfilesRepository); | 		const userProfilesRepository = module.get<UserProfilesRepository>(DI.userProfilesRepository); | ||||||
|  | 		await userProfilesRepository.delete({}); | ||||||
| 		await userProfilesRepository.insert({ | 		await userProfilesRepository.insert({ | ||||||
| 			userId: root.id, | 			userId: root.id, | ||||||
| 		}); | 		}); | ||||||
| 
 | 
 | ||||||
|  | 		const driveFoldersRepository = module.get<DriveFoldersRepository>(DI.driveFoldersRepository); | ||||||
|  | 		folder = await driveFoldersRepository.insertOne({ | ||||||
|  | 			id: idService.gen(), | ||||||
|  | 			name: 'root-folder', | ||||||
|  | 			parentId: null, | ||||||
|  | 			userId: root.id, | ||||||
|  | 		}); | ||||||
|  | 
 | ||||||
| 		roleService = module.get<RoleService>(RoleService); | 		roleService = module.get<RoleService>(RoleService); | ||||||
| 		role_tinyAttachment = await roleService.create({ | 		role_tinyAttachment = await roleService.create({ | ||||||
| 			name: 'test-role001', | 			name: 'test-role001', | ||||||
|  | @ -65,8 +80,8 @@ describe('/drive/files/create', () => { | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	beforeEach(async () => { | 	beforeEach(async () => { | ||||||
| 		s3Mock.reset(); | 		await roleService.unassign(root.id, role_tinyAttachment.id).catch(() => { | ||||||
| 		await roleService.unassign(root.id, role_tinyAttachment.id).catch(() => {}); | 		}); | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	afterAll(async () => { | 	afterAll(async () => { | ||||||
|  | @ -74,35 +89,76 @@ describe('/drive/files/create', () => { | ||||||
| 		await module.close(); | 		await module.close(); | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	test('200 ok', async () => { | 	async function postFile(props: { | ||||||
| 		const result = await request(server.server) | 		name: string, | ||||||
|  | 		comment: string, | ||||||
|  | 		isSensitive: boolean, | ||||||
|  | 		force: boolean, | ||||||
|  | 		fileContent: Buffer | string, | ||||||
|  | 	}) { | ||||||
|  | 		const { name, comment, isSensitive, force, fileContent } = props; | ||||||
|  | 
 | ||||||
|  | 		return await request(server.server) | ||||||
| 			.post('/api/drive/files/create') | 			.post('/api/drive/files/create') | ||||||
| 			.set('Content-Type', 'multipart/form-data') | 			.set('Content-Type', 'multipart/form-data') | ||||||
| 			.set('Authorization', `Bearer ${root.token}`) | 			.attach('file', fileContent) | ||||||
| 			.attach('file', Buffer.from('a'.repeat(1024 * 1024))); | 			.field('name', name) | ||||||
|  | 			.field('comment', comment) | ||||||
|  | 			.field('isSensitive', isSensitive) | ||||||
|  | 			.field('force', force) | ||||||
|  | 			.field('folderId', folder.id) | ||||||
|  | 			.field('i', root.token ?? ''); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	test('200 ok', async () => { | ||||||
|  | 		const name = randomString(); | ||||||
|  | 		const comment = randomString(); | ||||||
|  | 		const result = await postFile({ | ||||||
|  | 			name: name, | ||||||
|  | 			comment: comment, | ||||||
|  | 			isSensitive: true, | ||||||
|  | 			force: true, | ||||||
|  | 			fileContent: Buffer.from('a'.repeat(1000 * 1000)), | ||||||
|  | 		}); | ||||||
| 		expect(result.statusCode).toBe(200); | 		expect(result.statusCode).toBe(200); | ||||||
|  | 		expect(result.body.name).toBe(name + '.unknown'); | ||||||
|  | 		expect(result.body.comment).toBe(comment); | ||||||
|  | 		expect(result.body.isSensitive).toBe(true); | ||||||
|  | 		expect(result.body.folderId).toBe(folder.id); | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	test('200 ok(with role)', async () => { | 	test('200 ok(with role)', async () => { | ||||||
| 		await roleService.assign(root.id, role_tinyAttachment.id); | 		await roleService.assign(root.id, role_tinyAttachment.id); | ||||||
| 
 | 
 | ||||||
| 		const result = await request(server.server) | 		const name = randomString(); | ||||||
| 			.post('/api/drive/files/create') | 		const comment = randomString(); | ||||||
| 			.set('Content-Type', 'multipart/form-data') | 		const result = await postFile({ | ||||||
| 			.set('Authorization', `Bearer ${root.token}`) | 			name: name, | ||||||
| 			.attach('file', Buffer.from('a'.repeat(10))); | 			comment: comment, | ||||||
|  | 			isSensitive: true, | ||||||
|  | 			force: true, | ||||||
|  | 			fileContent: Buffer.from('a'.repeat(10)), | ||||||
|  | 		}); | ||||||
| 		expect(result.statusCode).toBe(200); | 		expect(result.statusCode).toBe(200); | ||||||
|  | 		expect(result.body.name).toBe(name + '.unknown'); | ||||||
|  | 		expect(result.body.comment).toBe(comment); | ||||||
|  | 		expect(result.body.isSensitive).toBe(true); | ||||||
|  | 		expect(result.body.folderId).toBe(folder.id); | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	test('413 too large', async () => { | 	test('413 too large', async () => { | ||||||
| 		await roleService.assign(root.id, role_tinyAttachment.id); | 		await roleService.assign(root.id, role_tinyAttachment.id); | ||||||
| 
 | 
 | ||||||
| 		const result = await request(server.server) | 		const name = randomString(); | ||||||
| 			.post('/api/drive/files/create') | 		const comment = randomString(); | ||||||
| 			.set('Content-Type', 'multipart/form-data') | 		const result = await postFile({ | ||||||
| 			.set('Authorization', `Bearer ${root.token}`) | 			name: name, | ||||||
| 			.attach('file', Buffer.from('a'.repeat(11))); | 			comment: comment, | ||||||
|  | 			isSensitive: true, | ||||||
|  | 			force: true, | ||||||
|  | 			fileContent: Buffer.from('a'.repeat(11)), | ||||||
|  | 		}); | ||||||
| 		expect(result.statusCode).toBe(413); | 		expect(result.statusCode).toBe(413); | ||||||
| 		expect(result.body.error.code).toBe('FILE_SIZE_TOO_LARGE'); | 		expect(result.body.error.code).toBe('MAX_FILE_SIZE_EXCEEDED'); | ||||||
| 	}); | 	}); | ||||||
| }); | }); | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							|  | @ -21215,7 +21215,7 @@ snapshots: | ||||||
|       formidable: 3.5.4 |       formidable: 3.5.4 | ||||||
|       methods: 1.1.2 |       methods: 1.1.2 | ||||||
|       mime: 2.6.0 |       mime: 2.6.0 | ||||||
|       qs: 6.13.0 |       qs: 6.14.0 | ||||||
|     transitivePeerDependencies: |     transitivePeerDependencies: | ||||||
|       - supports-color |       - supports-color | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue