mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-23 09:44:51 +00:00 
			
		
		
		
	* Revert "fix: 添付ファイルのあるリクエストを受けたときの初動を改善 (#15896)"
This reverts commit 7e8cc4d7c0.
* fix CHANGELOG.md
			
			
This commit is contained in:
		
							parent
							
								
									17a9b08f54
								
							
						
					
					
						commit
						b91a67d74e
					
				
					 7 changed files with 65 additions and 332 deletions
				
			
		|  | @ -229,7 +229,6 @@ | |||
| 		"@types/semver": "7.7.0", | ||||
| 		"@types/simple-oauth2": "5.0.7", | ||||
| 		"@types/sinonjs__fake-timers": "8.1.5", | ||||
| 		"@types/supertest": "6.0.3", | ||||
| 		"@types/tinycolor2": "1.4.6", | ||||
| 		"@types/tmp": "0.2.6", | ||||
| 		"@types/uuid": "^9.0.4", | ||||
|  | @ -247,7 +246,6 @@ | |||
| 		"jest-mock": "29.7.0", | ||||
| 		"nodemon": "3.1.10", | ||||
| 		"pid-port": "1.0.2", | ||||
| 		"simple-oauth2": "5.1.0", | ||||
| 		"supertest": "7.1.0" | ||||
| 		"simple-oauth2": "5.1.0" | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -75,7 +75,7 @@ export class ServerService implements OnApplicationShutdown { | |||
| 	} | ||||
| 
 | ||||
| 	@bindThis | ||||
| 	public async launch() { | ||||
| 	public async launch(): Promise<void> { | ||||
| 		const fastify = Fastify({ | ||||
| 			trustProxy: true, | ||||
| 			logger: false, | ||||
|  | @ -135,8 +135,8 @@ export class ServerService implements OnApplicationShutdown { | |||
| 				reply.header('content-type', 'text/plain; charset=utf-8'); | ||||
| 				reply.header('link', `<${encodeURI(location)}>; rel="canonical"`); | ||||
| 				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}.`, | ||||
| 				].join('\n')); | ||||
| 			}); | ||||
|  | @ -304,7 +304,6 @@ export class ServerService implements OnApplicationShutdown { | |||
| 		} | ||||
| 
 | ||||
| 		await fastify.ready(); | ||||
| 		return fastify; | ||||
| 	} | ||||
| 
 | ||||
| 	@bindThis | ||||
|  |  | |||
|  | @ -6,11 +6,8 @@ | |||
| import { randomUUID } from 'node:crypto'; | ||||
| import * as fs from 'node:fs'; | ||||
| 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 * as Sentry from '@sentry/node'; | ||||
| import { AttachmentFile } from '@/server/api/endpoint-base.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { getIpHash } from '@/misc/get-ip-hash.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 { createTemp } from '@/misc/create-temp.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 { sendRateLimitHeaders } from '@/misc/rate-limit-utils.js'; | ||||
| import { SkRateLimiterService } from '@/server/SkRateLimiterService.js'; | ||||
|  | @ -194,6 +191,18 @@ export class ApiCallService implements OnApplicationShutdown { | |||
| 			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>; | ||||
| 		for (const [k, v] of Object.entries(multipartData.fields)) { | ||||
| 			fields[k] = typeof v === 'object' && 'value' in v ? v.value : undefined; | ||||
|  | @ -208,7 +217,10 @@ export class ApiCallService implements OnApplicationShutdown { | |||
| 			return; | ||||
| 		} | ||||
| 		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); | ||||
| 			}).catch((err: ApiError) => { | ||||
| 				this.#sendApiError(reply, err); | ||||
|  | @ -278,7 +290,10 @@ export class ApiCallService implements OnApplicationShutdown { | |||
| 		user: MiLocalUser | null | undefined, | ||||
| 		token: MiAccessToken | null | undefined, | ||||
| 		data: any, | ||||
| 		multipartFile: MultipartFile | null, | ||||
| 		file: { | ||||
| 			name: string; | ||||
| 			path: string; | ||||
| 		} | null, | ||||
| 		request: FastifyRequest<{ Body: Record<string, unknown> | undefined, Querystring: Record<string, unknown> }>, | ||||
| 		reply: FastifyReply, | ||||
| 	) { | ||||
|  | @ -354,37 +369,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)) { | ||||
| 			const myRoles = await this.roleService.getUserRoles(user!.id); | ||||
| 			if (ep.meta.requireModerator && !myRoles.some(r => r.isModerator || r.isAdministrator)) { | ||||
|  | @ -418,91 +402,49 @@ export class ApiCallService implements OnApplicationShutdown { | |||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		let attachmentFile: AttachmentFile | null = null; | ||||
| 		let cleanup = () => {}; | ||||
| 		if (ep.meta.requireFile && request.method === 'POST' && multipartFile) { | ||||
| 			const policies = await this.roleService.getUserPolicies(user!.id); | ||||
| 			const result = await this.handleAttachmentFile( | ||||
| 				Math.min((policies.maxFileSizeMb * 1024 * 1024), this.config.maxFileSize), | ||||
| 				multipartFile, | ||||
| 			); | ||||
| 			attachmentFile = result.attachmentFile; | ||||
| 			cleanup = result.cleanup; | ||||
| 		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', | ||||
| 			}); | ||||
| 		} | ||||
| 
 | ||||
| 		// 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
 | ||||
| 		if (this.config.sentryForBackend) { | ||||
| 			return await Sentry.startSpan({ | ||||
| 				name: 'API: ' + ep.name, | ||||
| 			}, () => { | ||||
| 				return ep.exec(data, user, token, attachmentFile, request.ip, request.headers) | ||||
| 					.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)) | ||||
| 					.finally(() => cleanup()); | ||||
| 			}); | ||||
| 			}, () => ep.exec(data, user, token, file, request.ip, request.headers) | ||||
| 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id))); | ||||
| 		} else { | ||||
| 			return await ep.exec(data, user, token, attachmentFile, request.ip, request.headers) | ||||
| 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)) | ||||
| 				.finally(() => cleanup()); | ||||
| 			return await ep.exec(data, user, token, file, request.ip, request.headers) | ||||
| 				.catch((err: Error) => this.#onExecError(ep, data, err, user?.id)); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	@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 | ||||
| 	public dispose(): void { | ||||
| 		clearInterval(this.userIpHistoriesClearIntervalId); | ||||
|  |  | |||
|  | @ -21,23 +21,23 @@ ajv.addFormat('misskey:id', /^[a-zA-Z0-9]+$/); | |||
| 
 | ||||
| export type Response = Record<string, any> | void; | ||||
| 
 | ||||
| export type AttachmentFile = { | ||||
| type File = { | ||||
| 	name: string | null; | ||||
| 	path: string; | ||||
| }; | ||||
| 
 | ||||
| // TODO: paramsの型をT['params']のスキーマ定義から推論する
 | ||||
| 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) => | ||||
| 	Promise<T['res'] extends undefined ? Response : SchemaType<NonNullable<T['res']>>>; | ||||
| 	(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']>>>; | ||||
| 
 | ||||
| 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>) { | ||||
| 		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; | ||||
| 
 | ||||
| 			if (meta.requireFile) { | ||||
|  |  | |||
|  | @ -159,8 +159,8 @@ describe('API', () => { | |||
| 			user: { token: application3 }, | ||||
| 		}, { | ||||
| 			status: 403, | ||||
| 			code: 'PERMISSION_DENIED', | ||||
| 			id: '1370e5b7-d4eb-4566-bb1d-7748ee6a1838', | ||||
| 			code: 'ROLE_PERMISSION_DENIED', | ||||
| 			id: 'c3d38592-54c0-429d-be96-5636b0431a61', | ||||
| 		}); | ||||
| 
 | ||||
| 		await failedApiCall({ | ||||
|  |  | |||
|  | @ -1,108 +0,0 @@ | |||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
| 
 | ||||
| import { S3Client } from '@aws-sdk/client-s3'; | ||||
| import { Test, TestingModule } from '@nestjs/testing'; | ||||
| import { mockClient } from 'aws-sdk-client-mock'; | ||||
| import { FastifyInstance } from 'fastify'; | ||||
| import request from 'supertest'; | ||||
| import { CoreModule } from '@/core/CoreModule.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { GlobalModule } from '@/GlobalModule.js'; | ||||
| import { MiRole, UserProfilesRepository, UsersRepository } from '@/models/_.js'; | ||||
| import { MiUser } from '@/models/User.js'; | ||||
| import { ServerModule } from '@/server/ServerModule.js'; | ||||
| import { ServerService } from '@/server/ServerService.js'; | ||||
| 
 | ||||
| describe('/drive/files/create', () => { | ||||
| 	let module: TestingModule; | ||||
| 	let server: FastifyInstance; | ||||
| 	const s3Mock = mockClient(S3Client); | ||||
| 	let roleService: RoleService; | ||||
| 
 | ||||
| 	let root: MiUser; | ||||
| 	let role_tinyAttachment: MiRole; | ||||
| 
 | ||||
| 	beforeAll(async () => { | ||||
| 		module = await Test.createTestingModule({ | ||||
| 			imports: [GlobalModule, CoreModule, ServerModule], | ||||
| 		}).compile(); | ||||
| 		module.enableShutdownHooks(); | ||||
| 
 | ||||
| 		const serverService = module.get<ServerService>(ServerService); | ||||
| 		server = await serverService.launch(); | ||||
| 
 | ||||
| 		const usersRepository = module.get<UsersRepository>(DI.usersRepository); | ||||
| 		root = await usersRepository.insert({ | ||||
| 			id: 'root', | ||||
| 			username: 'root', | ||||
| 			usernameLower: 'root', | ||||
| 			token: '1234567890123456', | ||||
| 		}).then(x => usersRepository.findOneByOrFail(x.identifiers[0])); | ||||
| 
 | ||||
| 		const userProfilesRepository = module.get<UserProfilesRepository>(DI.userProfilesRepository); | ||||
| 		await userProfilesRepository.insert({ | ||||
| 			userId: root.id, | ||||
| 		}); | ||||
| 
 | ||||
| 		roleService = module.get<RoleService>(RoleService); | ||||
| 		role_tinyAttachment = await roleService.create({ | ||||
| 			name: 'test-role001', | ||||
| 			description: 'Test role001 description', | ||||
| 			target: 'manual', | ||||
| 			policies: { | ||||
| 				maxFileSizeMb: { | ||||
| 					useDefault: false, | ||||
| 					priority: 1, | ||||
| 					// 10byte
 | ||||
| 					value: 10 / 1024 / 1024, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}); | ||||
| 	}); | ||||
| 
 | ||||
| 	beforeEach(async () => { | ||||
| 		s3Mock.reset(); | ||||
| 		await roleService.unassign(root.id, role_tinyAttachment.id).catch(() => {}); | ||||
| 	}); | ||||
| 
 | ||||
| 	afterAll(async () => { | ||||
| 		await server.close(); | ||||
| 		await module.close(); | ||||
| 	}); | ||||
| 
 | ||||
| 	test('200 ok', async () => { | ||||
| 		const result = await request(server.server) | ||||
| 			.post('/api/drive/files/create') | ||||
| 			.set('Content-Type', 'multipart/form-data') | ||||
| 			.set('Authorization', `Bearer ${root.token}`) | ||||
| 			.attach('file', Buffer.from('a'.repeat(1024 * 1024))); | ||||
| 		expect(result.statusCode).toBe(200); | ||||
| 	}); | ||||
| 
 | ||||
| 	test('200 ok(with role)', async () => { | ||||
| 		await roleService.assign(root.id, role_tinyAttachment.id); | ||||
| 
 | ||||
| 		const result = await request(server.server) | ||||
| 			.post('/api/drive/files/create') | ||||
| 			.set('Content-Type', 'multipart/form-data') | ||||
| 			.set('Authorization', `Bearer ${root.token}`) | ||||
| 			.attach('file', Buffer.from('a'.repeat(10))); | ||||
| 		expect(result.statusCode).toBe(200); | ||||
| 	}); | ||||
| 
 | ||||
| 	test('413 too large', async () => { | ||||
| 		await roleService.assign(root.id, role_tinyAttachment.id); | ||||
| 
 | ||||
| 		const result = await request(server.server) | ||||
| 			.post('/api/drive/files/create') | ||||
| 			.set('Content-Type', 'multipart/form-data') | ||||
| 			.set('Authorization', `Bearer ${root.token}`) | ||||
| 			.attach('file', Buffer.from('a'.repeat(11))); | ||||
| 		expect(result.statusCode).toBe(413); | ||||
| 		expect(result.body.error.code).toBe('FILE_SIZE_TOO_LARGE'); | ||||
| 	}); | ||||
| }); | ||||
							
								
								
									
										98
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										98
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							|  | @ -652,9 +652,6 @@ importers: | |||
|       '@types/sinonjs__fake-timers': | ||||
|         specifier: 8.1.5 | ||||
|         version: 8.1.5 | ||||
|       '@types/supertest': | ||||
|         specifier: 6.0.3 | ||||
|         version: 6.0.3 | ||||
|       '@types/tinycolor2': | ||||
|         specifier: 1.4.6 | ||||
|         version: 1.4.6 | ||||
|  | @ -709,9 +706,6 @@ importers: | |||
|       simple-oauth2: | ||||
|         specifier: 5.1.0 | ||||
|         version: 5.1.0 | ||||
|       supertest: | ||||
|         specifier: 7.1.0 | ||||
|         version: 7.1.0 | ||||
| 
 | ||||
|   packages/frontend: | ||||
|     dependencies: | ||||
|  | @ -3034,9 +3028,6 @@ packages: | |||
|     peerDependencies: | ||||
|       '@opentelemetry/api': ^1.1.0 | ||||
| 
 | ||||
|   '@paralleldrive/cuid2@2.2.2': | ||||
|     resolution: {integrity: sha512-ZOBkgDwEdoYVlSeRbYYXs0S9MejQofiVYoTbKzy/6GQa39/q5tQU2IX46+shYnUkpEl3wc+J6wRlar7r2EK2xA==} | ||||
| 
 | ||||
|   '@parcel/watcher-android-arm64@2.5.1': | ||||
|     resolution: {integrity: sha512-KF8+j9nNbUN8vzOFDpRMsaKBHZ/mcjEjMToVMJOhTozkDonQFFrRcfdLWn6yWKCmJKmdVxSgHiYvTCef4/qcBA==} | ||||
|     engines: {node: '>= 10.0.0'} | ||||
|  | @ -4150,9 +4141,6 @@ packages: | |||
|   '@types/cookie@0.6.0': | ||||
|     resolution: {integrity: sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==} | ||||
| 
 | ||||
|   '@types/cookiejar@2.1.5': | ||||
|     resolution: {integrity: sha512-he+DHOWReW0nghN24E1WUqM0efK4kI9oTqDm6XmK8ZPe2djZ90BSNdGnIyCLzCPw7/pogPlGbzI2wHGGmi4O/Q==} | ||||
| 
 | ||||
|   '@types/debug@4.1.12': | ||||
|     resolution: {integrity: sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==} | ||||
| 
 | ||||
|  | @ -4237,9 +4225,6 @@ packages: | |||
|   '@types/mdx@2.0.3': | ||||
|     resolution: {integrity: sha512-IgHxcT3RC8LzFLhKwP3gbMPeaK7BM9eBH46OdapPA7yvuIUJ8H6zHZV53J8hGZcTSnt95jANt+rTBNUUc22ACQ==} | ||||
| 
 | ||||
|   '@types/methods@1.1.4': | ||||
|     resolution: {integrity: sha512-ymXWVrDiCxTBE3+RIrrP533E70eA+9qu7zdWoHuOmGujkYtzf4HQF96b8nwHLqhuf4ykX61IGRIB38CC6/sImQ==} | ||||
| 
 | ||||
|   '@types/micromatch@4.0.9': | ||||
|     resolution: {integrity: sha512-7V+8ncr22h4UoYRLnLXSpTxjQrNUXtWHGeMPRJt1nULXI57G9bIcpyrHlmrQ7QK24EyyuXvYcSSWAM8GA9nqCg==} | ||||
| 
 | ||||
|  | @ -4372,12 +4357,6 @@ packages: | |||
|   '@types/statuses@2.0.4': | ||||
|     resolution: {integrity: sha512-eqNDvZsCNY49OAXB0Firg/Sc2BgoWsntsLUdybGFOhAfCD6QJ2n9HXUIHGqt5qjrxmMv4wS8WLAw43ZkKcJ8Pw==} | ||||
| 
 | ||||
|   '@types/superagent@8.1.9': | ||||
|     resolution: {integrity: sha512-pTVjI73witn+9ILmoJdajHGW2jkSaOzhiFYF1Rd3EQ94kymLqB9PjD9ISg7WaALC7+dCHT0FGe9T2LktLq/3GQ==} | ||||
| 
 | ||||
|   '@types/supertest@6.0.3': | ||||
|     resolution: {integrity: sha512-8WzXq62EXFhJ7QsH3Ocb/iKQ/Ty9ZVWnVzoTKc9tyyFRRF3a74Tk2+TLFgaFFw364Ere+npzHKEJ6ga2LzIL7w==} | ||||
| 
 | ||||
|   '@types/tedious@4.0.14': | ||||
|     resolution: {integrity: sha512-KHPsfX/FoVbUGbyYvk1q9MMQHLPeRZhRJZdO45Q4YjvFkv4hMNghCWTvy7rdKessBsmtz4euWCWAB6/tVpI1Iw==} | ||||
| 
 | ||||
|  | @ -5425,9 +5404,6 @@ packages: | |||
|   compare-versions@6.1.1: | ||||
|     resolution: {integrity: sha512-4hm4VPpIecmlg59CHXnRDnqGplJFrbLG4aFEl5vl6cK1u76ws3LLvX7ikFnTDl5vo39sjWD6AaDPYodJp/NNHg==} | ||||
| 
 | ||||
|   component-emitter@1.3.1: | ||||
|     resolution: {integrity: sha512-T0+barUSQRTUQASh8bx02dl+DhF54GtIDY13Y3m9oWTklKbb3Wv974meRpeZ3lp1JpLVECWWNHC4vaG2XHXouQ==} | ||||
| 
 | ||||
|   compress-commons@6.0.2: | ||||
|     resolution: {integrity: sha512-6FqVXeETqWPoGcfzrXb37E50NP0LXT8kAMu5ooZayhWWdgEY4lBEEcbQNXtkuKQsGduxiIcI4gOTsxTmuq/bSg==} | ||||
|     engines: {node: '>= 14'} | ||||
|  | @ -5486,9 +5462,6 @@ packages: | |||
|     resolution: {integrity: sha512-9Kr/j4O16ISv8zBBhJoi4bXOYNTkFLOqSL3UDB0njXxCXNezjeyVrJyGOWtgfs/q2km1gwBcfH8q1yEGoMYunA==} | ||||
|     engines: {node: '>=18'} | ||||
| 
 | ||||
|   cookiejar@2.1.4: | ||||
|     resolution: {integrity: sha512-LDx6oHrK+PhzLKJU9j5S7/Y3jM/mUHvD/DeI1WQmJn652iPC5Y4TBzC9l+5OMOXlyTTA+SmVUPm0HQUwpD5Jqw==} | ||||
| 
 | ||||
|   core-util-is@1.0.2: | ||||
|     resolution: {integrity: sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==} | ||||
| 
 | ||||
|  | @ -5800,9 +5773,6 @@ packages: | |||
|   devlop@1.1.0: | ||||
|     resolution: {integrity: sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA==} | ||||
| 
 | ||||
|   dezalgo@1.0.4: | ||||
|     resolution: {integrity: sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==} | ||||
| 
 | ||||
|   diff-match-patch@1.0.5: | ||||
|     resolution: {integrity: sha512-IayShXAgj/QMXgB0IWmKx+rOPuGMhqm5w6jvFxmVenXKIzRqTAAsbBPT3kWQeGANj3jGgvcvv4yK6SxqYmikgw==} | ||||
| 
 | ||||
|  | @ -6446,10 +6416,6 @@ packages: | |||
|     resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} | ||||
|     engines: {node: '>=12.20.0'} | ||||
| 
 | ||||
|   formidable@3.5.4: | ||||
|     resolution: {integrity: sha512-YikH+7CUTOtP44ZTnUhR7Ic2UASBPOqmaRkRKxRbywPTe5VxF7RRCck4af9wutiZ/QKM5nME9Bie2fFaPz5Gug==} | ||||
|     engines: {node: '>=14.0.0'} | ||||
| 
 | ||||
|   forwarded-parse@2.1.2: | ||||
|     resolution: {integrity: sha512-alTFZZQDKMporBH77856pXgzhEzaUVmLCDk+egLgIgHst3Tpndzz8MnKe+GzRJRfvVdn69HhpW7cmXzvtLvJAw==} | ||||
| 
 | ||||
|  | @ -9843,14 +9809,6 @@ packages: | |||
|     peerDependencies: | ||||
|       postcss: ^8.4.31 | ||||
| 
 | ||||
|   superagent@9.0.2: | ||||
|     resolution: {integrity: sha512-xuW7dzkUpcJq7QnhOsnNUgtYp3xRwpt2F7abdRYIpCsAt0hhUqia0EdxyXZQQpNmGtsCzYHryaKSV3q3GJnq7w==} | ||||
|     engines: {node: '>=14.18.0'} | ||||
| 
 | ||||
|   supertest@7.1.0: | ||||
|     resolution: {integrity: sha512-5QeSO8hSrKghtcWEoPiO036fxH0Ii2wVQfFZSP0oqQhmjk8bOLhDFXr4JrvaFmPuEWUoq4znY3uSi8UzLKxGqw==} | ||||
|     engines: {node: '>=14.18.0'} | ||||
| 
 | ||||
|   supports-color@5.5.0: | ||||
|     resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} | ||||
|     engines: {node: '>=4'} | ||||
|  | @ -12980,10 +12938,6 @@ snapshots: | |||
|       '@opentelemetry/api': 1.9.0 | ||||
|       '@opentelemetry/core': 1.30.1(@opentelemetry/api@1.9.0) | ||||
| 
 | ||||
|   '@paralleldrive/cuid2@2.2.2': | ||||
|     dependencies: | ||||
|       '@noble/hashes': 1.7.1 | ||||
| 
 | ||||
|   '@parcel/watcher-android-arm64@2.5.1': | ||||
|     optional: true | ||||
| 
 | ||||
|  | @ -14342,8 +14296,6 @@ snapshots: | |||
| 
 | ||||
|   '@types/cookie@0.6.0': {} | ||||
| 
 | ||||
|   '@types/cookiejar@2.1.5': {} | ||||
| 
 | ||||
|   '@types/debug@4.1.12': | ||||
|     dependencies: | ||||
|       '@types/ms': 0.7.34 | ||||
|  | @ -14434,8 +14386,6 @@ snapshots: | |||
| 
 | ||||
|   '@types/mdx@2.0.3': {} | ||||
| 
 | ||||
|   '@types/methods@1.1.4': {} | ||||
| 
 | ||||
|   '@types/micromatch@4.0.9': | ||||
|     dependencies: | ||||
|       '@types/braces': 3.0.1 | ||||
|  | @ -14568,18 +14518,6 @@ snapshots: | |||
| 
 | ||||
|   '@types/statuses@2.0.4': {} | ||||
| 
 | ||||
|   '@types/superagent@8.1.9': | ||||
|     dependencies: | ||||
|       '@types/cookiejar': 2.1.5 | ||||
|       '@types/methods': 1.1.4 | ||||
|       '@types/node': 22.15.2 | ||||
|       form-data: 4.0.2 | ||||
| 
 | ||||
|   '@types/supertest@6.0.3': | ||||
|     dependencies: | ||||
|       '@types/methods': 1.1.4 | ||||
|       '@types/superagent': 8.1.9 | ||||
| 
 | ||||
|   '@types/tedious@4.0.14': | ||||
|     dependencies: | ||||
|       '@types/node': 22.15.2 | ||||
|  | @ -15905,8 +15843,6 @@ snapshots: | |||
| 
 | ||||
|   compare-versions@6.1.1: {} | ||||
| 
 | ||||
|   component-emitter@1.3.1: {} | ||||
| 
 | ||||
|   compress-commons@6.0.2: | ||||
|     dependencies: | ||||
|       crc-32: 1.2.2 | ||||
|  | @ -15960,8 +15896,6 @@ snapshots: | |||
| 
 | ||||
|   cookie@1.0.2: {} | ||||
| 
 | ||||
|   cookiejar@2.1.4: {} | ||||
| 
 | ||||
|   core-util-is@1.0.2: {} | ||||
| 
 | ||||
|   core-util-is@1.0.3: {} | ||||
|  | @ -16347,11 +16281,6 @@ snapshots: | |||
|     dependencies: | ||||
|       dequal: 2.0.3 | ||||
| 
 | ||||
|   dezalgo@1.0.4: | ||||
|     dependencies: | ||||
|       asap: 2.0.6 | ||||
|       wrappy: 1.0.2 | ||||
| 
 | ||||
|   diff-match-patch@1.0.5: {} | ||||
| 
 | ||||
|   diff-sequences@29.6.3: {} | ||||
|  | @ -17246,12 +17175,6 @@ snapshots: | |||
|     dependencies: | ||||
|       fetch-blob: 3.2.0 | ||||
| 
 | ||||
|   formidable@3.5.4: | ||||
|     dependencies: | ||||
|       '@paralleldrive/cuid2': 2.2.2 | ||||
|       dezalgo: 1.0.4 | ||||
|       once: 1.4.0 | ||||
| 
 | ||||
|   forwarded-parse@2.1.2: {} | ||||
| 
 | ||||
|   forwarded@0.2.0: {} | ||||
|  | @ -21205,27 +21128,6 @@ snapshots: | |||
|       postcss: 8.5.3 | ||||
|       postcss-selector-parser: 6.1.2 | ||||
| 
 | ||||
|   superagent@9.0.2: | ||||
|     dependencies: | ||||
|       component-emitter: 1.3.1 | ||||
|       cookiejar: 2.1.4 | ||||
|       debug: 4.4.0(supports-color@8.1.1) | ||||
|       fast-safe-stringify: 2.1.1 | ||||
|       form-data: 4.0.2 | ||||
|       formidable: 3.5.4 | ||||
|       methods: 1.1.2 | ||||
|       mime: 2.6.0 | ||||
|       qs: 6.13.0 | ||||
|     transitivePeerDependencies: | ||||
|       - supports-color | ||||
| 
 | ||||
|   supertest@7.1.0: | ||||
|     dependencies: | ||||
|       methods: 1.1.2 | ||||
|       superagent: 9.0.2 | ||||
|     transitivePeerDependencies: | ||||
|       - supports-color | ||||
| 
 | ||||
|   supports-color@5.5.0: | ||||
|     dependencies: | ||||
|       has-flag: 3.0.0 | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue