 31e82fc29a
			
		
	
	31e82fc29a
	
	
	
		
			
			* kill any on utils:api * kill any on timeline test * use optional chain to kill TS2532 on timeline test 変更前: 該当ノートが見つからなければundefinedに対するプロパティアクセスとしてテストがクラッシュ 変更後: 該当ノートが見つからなければoptional chainがundefinedとして評価されるが、strictEqualの右辺がnon-nullableなためアサーションに失敗しテストがクラッシュ * kill `as any` for ApMfmService * kill argument any for api-visibility * kill argument any across a few tests * do not return value that has yielded from `await`-ing `Promise<void>` * force cast * runtime non-null assertion to coerce * rewrite `assert.notEqual(expr, null)` to `assert.ok(expr)` こうすることでassertion type扱いになり、non-nullableになる * change return type of `failedApiCall` to `void` 戻り値がどこにも使われていない * split bindings for exports.ts 型が合わなくて文句を言ってくるので適切に分割 * runtime non-null assertion * runtime non-null assertion * 何故かうまく行かないので、とりあえずXORしてみる * Revert "何故かうまく行かないので、とりあえずXORしてみる" This reverts commit48cf32c930. * castAsErrorで安全ではないキャストを隠蔽 * 型アサーションの追加 * 型アサーションの追加 * 型アサーションの追加 * voidで値を返さない * castAsError * assert.ok => kill nullability * もはや明示的な型の指定は必要ない * castAsError * castAsError * 型アサーションの追加 * nullableを一旦抑止 * 変数を分離して型エラーを排除 * 不要なプロパティを削除する処理を隠蔽してanyを排除 * Repository type * simple type * assert.ok => kill nullability * revert `as any` drop revertsfe95c05b3fpartialy * test: fix invalid assertion partially revertb99b7b5392* test:52d8a54fc7により型が合うようになった部分の`as any`を除去 * format * test: apply https://github.com/misskey-dev/misskey/pull/14054#discussion_r1672369526 (part 1) * test: use non-null assertion to suppress too many error * Update packages/backend/test/utils.ts Co-authored-by: anatawa12 <anatawa12@icloud.com> --------- Co-authored-by: anatawa12 <anatawa12@icloud.com>
		
			
				
	
	
		
			1040 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			1040 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /*
 | ||
|  * SPDX-FileCopyrightText: syuilo and misskey-project
 | ||
|  * SPDX-License-Identifier: AGPL-3.0-only
 | ||
|  */
 | ||
| 
 | ||
| import type { Repository } from "typeorm";
 | ||
| 
 | ||
| process.env.NODE_ENV = 'test';
 | ||
| 
 | ||
| import * as assert from 'assert';
 | ||
| import { MiNote } from '@/models/Note.js';
 | ||
| import { MAX_NOTE_TEXT_LENGTH } from '@/const.js';
 | ||
| import { api, castAsError, initTestDb, post, role, signup, uploadFile, uploadUrl } from '../utils.js';
 | ||
| import type * as misskey from 'misskey-js';
 | ||
| 
 | ||
| describe('Note', () => {
 | ||
| 	let Notes: Repository<MiNote>;
 | ||
| 
 | ||
| 	let root: misskey.entities.SignupResponse;
 | ||
| 	let alice: misskey.entities.SignupResponse;
 | ||
| 	let bob: misskey.entities.SignupResponse;
 | ||
| 	let tom: misskey.entities.SignupResponse;
 | ||
| 
 | ||
| 	beforeAll(async () => {
 | ||
| 		const connection = await initTestDb(true);
 | ||
| 		Notes = connection.getRepository(MiNote);
 | ||
| 		root = await signup({ username: 'root' });
 | ||
| 		alice = await signup({ username: 'alice' });
 | ||
| 		bob = await signup({ username: 'bob' });
 | ||
| 		tom = await signup({ username: 'tom', host: 'example.com' });
 | ||
| 	}, 1000 * 60 * 2);
 | ||
| 
 | ||
| 	test('投稿できる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: 'test',
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, post.text);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('ファイルを添付できる', async () => {
 | ||
| 		const file = await uploadUrl(alice, 'https://raw.githubusercontent.com/misskey-dev/misskey/develop/packages/backend/test/resources/192.jpg');
 | ||
| 
 | ||
| 		const res = await api('notes/create', {
 | ||
| 			fileIds: [file.id],
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.deepStrictEqual(res.body.createdNote.fileIds, [file.id]);
 | ||
| 	}, 1000 * 10);
 | ||
| 
 | ||
| 	test('他人のファイルで怒られる', async () => {
 | ||
| 		const file = await uploadUrl(bob, 'https://raw.githubusercontent.com/misskey-dev/misskey/develop/packages/backend/test/resources/192.jpg');
 | ||
| 
 | ||
| 		const res = await api('notes/create', {
 | ||
| 			text: 'test',
 | ||
| 			fileIds: [file.id],
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 		assert.strictEqual(castAsError(res.body).error.code, 'NO_SUCH_FILE');
 | ||
| 		assert.strictEqual(castAsError(res.body).error.id, 'b6992544-63e7-67f0-fa7f-32444b1b5306');
 | ||
| 	}, 1000 * 10);
 | ||
| 
 | ||
| 	test('存在しないファイルで怒られる', async () => {
 | ||
| 		const res = await api('notes/create', {
 | ||
| 			text: 'test',
 | ||
| 			fileIds: ['000000000000000000000000'],
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 		assert.strictEqual(castAsError(res.body).error.code, 'NO_SUCH_FILE');
 | ||
| 		assert.strictEqual(castAsError(res.body).error.id, 'b6992544-63e7-67f0-fa7f-32444b1b5306');
 | ||
| 	});
 | ||
| 
 | ||
| 	test('不正なファイルIDで怒られる', async () => {
 | ||
| 		const res = await api('notes/create', {
 | ||
| 			fileIds: ['kyoppie'],
 | ||
| 		}, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 		assert.strictEqual(castAsError(res.body).error.code, 'NO_SUCH_FILE');
 | ||
| 		assert.strictEqual(castAsError(res.body).error.id, 'b6992544-63e7-67f0-fa7f-32444b1b5306');
 | ||
| 	});
 | ||
| 
 | ||
| 	test('返信できる', async () => {
 | ||
| 		const bobPost = await post(bob, {
 | ||
| 			text: 'foo',
 | ||
| 		});
 | ||
| 
 | ||
| 		const alicePost = {
 | ||
| 			text: 'bar',
 | ||
| 			replyId: bobPost.id,
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', alicePost, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, alicePost.text);
 | ||
| 		assert.strictEqual(res.body.createdNote.replyId, alicePost.replyId);
 | ||
| 		assert.ok(res.body.createdNote.reply);
 | ||
| 		assert.strictEqual(res.body.createdNote.reply.text, bobPost.text);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('renoteできる', async () => {
 | ||
| 		const bobPost = await post(bob, {
 | ||
| 			text: 'test',
 | ||
| 		});
 | ||
| 
 | ||
| 		const alicePost = {
 | ||
| 			renoteId: bobPost.id,
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', alicePost, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.renoteId, alicePost.renoteId);
 | ||
| 		assert.ok(res.body.createdNote.renote);
 | ||
| 		assert.strictEqual(res.body.createdNote.renote.text, bobPost.text);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('引用renoteできる', async () => {
 | ||
| 		const bobPost = await post(bob, {
 | ||
| 			text: 'test',
 | ||
| 		});
 | ||
| 
 | ||
| 		const alicePost = {
 | ||
| 			text: 'test',
 | ||
| 			renoteId: bobPost.id,
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', alicePost, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, alicePost.text);
 | ||
| 		assert.strictEqual(res.body.createdNote.renoteId, alicePost.renoteId);
 | ||
| 		assert.ok(res.body.createdNote.renote);
 | ||
| 		assert.strictEqual(res.body.createdNote.renote.text, bobPost.text);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('引用renoteで空白文字のみで構成されたtextにするとレスポンスがtext: nullになる', async () => {
 | ||
| 		const bobPost = await post(bob, {
 | ||
| 			text: 'test',
 | ||
| 		});
 | ||
| 		const res = await api('notes/create', {
 | ||
| 			text: ' ',
 | ||
| 			renoteId: bobPost.id,
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, null);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('visibility: followersでrenoteできる', async () => {
 | ||
| 		const createRes = await api('notes/create', {
 | ||
| 			text: 'test',
 | ||
| 			visibility: 'followers',
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(createRes.status, 200);
 | ||
| 
 | ||
| 		const renoteId = createRes.body.createdNote.id;
 | ||
| 		const renoteRes = await api('notes/create', {
 | ||
| 			visibility: 'followers',
 | ||
| 			renoteId,
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(renoteRes.status, 200);
 | ||
| 		assert.strictEqual(renoteRes.body.createdNote.renoteId, renoteId);
 | ||
| 		assert.strictEqual(renoteRes.body.createdNote.visibility, 'followers');
 | ||
| 
 | ||
| 		const deleteRes = await api('notes/delete', {
 | ||
| 			noteId: renoteRes.body.createdNote.id,
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(deleteRes.status, 204);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('visibility: followersなノートに対してフォロワーはリプライできる', async () => {
 | ||
| 		await api('following/create', {
 | ||
| 			userId: alice.id,
 | ||
| 		}, bob);
 | ||
| 
 | ||
| 		const aliceNote = await api('notes/create', {
 | ||
| 			text: 'direct note to bob',
 | ||
| 			visibility: 'followers',
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(aliceNote.status, 200);
 | ||
| 
 | ||
| 		const replyId = aliceNote.body.createdNote.id;
 | ||
| 		const bobReply = await api('notes/create', {
 | ||
| 			text: 'reply to alice note',
 | ||
| 			replyId,
 | ||
| 		}, bob);
 | ||
| 
 | ||
| 		assert.strictEqual(bobReply.status, 200);
 | ||
| 		assert.strictEqual(bobReply.body.createdNote.replyId, replyId);
 | ||
| 
 | ||
| 		await api('following/delete', {
 | ||
| 			userId: alice.id,
 | ||
| 		}, bob);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('visibility: followersなノートに対してフォロワーでないユーザーがリプライしようとすると怒られる', async () => {
 | ||
| 		const aliceNote = await api('notes/create', {
 | ||
| 			text: 'direct note to bob',
 | ||
| 			visibility: 'followers',
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(aliceNote.status, 200);
 | ||
| 
 | ||
| 		const bobReply = await api('notes/create', {
 | ||
| 			text: 'reply to alice note',
 | ||
| 			replyId: aliceNote.body.createdNote.id,
 | ||
| 		}, bob);
 | ||
| 
 | ||
| 		assert.strictEqual(bobReply.status, 400);
 | ||
| 		assert.strictEqual(castAsError(bobReply.body).error.code, 'CANNOT_REPLY_TO_AN_INVISIBLE_NOTE');
 | ||
| 	});
 | ||
| 
 | ||
| 	test('visibility: specifiedなノートに対してvisibility: specifiedで返信できる', async () => {
 | ||
| 		const aliceNote = await api('notes/create', {
 | ||
| 			text: 'direct note to bob',
 | ||
| 			visibility: 'specified',
 | ||
| 			visibleUserIds: [bob.id],
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(aliceNote.status, 200);
 | ||
| 
 | ||
| 		const bobReply = await api('notes/create', {
 | ||
| 			text: 'reply to alice note',
 | ||
| 			replyId: aliceNote.body.createdNote.id,
 | ||
| 			visibility: 'specified',
 | ||
| 			visibleUserIds: [alice.id],
 | ||
| 		}, bob);
 | ||
| 
 | ||
| 		assert.strictEqual(bobReply.status, 200);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('visibility: specifiedなノートに対してvisibility: follwersで返信しようとすると怒られる', async () => {
 | ||
| 		const aliceNote = await api('notes/create', {
 | ||
| 			text: 'direct note to bob',
 | ||
| 			visibility: 'specified',
 | ||
| 			visibleUserIds: [bob.id],
 | ||
| 		}, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(aliceNote.status, 200);
 | ||
| 
 | ||
| 		const bobReply = await api('notes/create', {
 | ||
| 			text: 'reply to alice note with visibility: followers',
 | ||
| 			replyId: aliceNote.body.createdNote.id,
 | ||
| 			visibility: 'followers',
 | ||
| 		}, bob);
 | ||
| 
 | ||
| 		assert.strictEqual(bobReply.status, 400);
 | ||
| 		assert.strictEqual(castAsError(bobReply.body).error.code, 'CANNOT_REPLY_TO_SPECIFIED_VISIBILITY_NOTE_WITH_EXTENDED_VISIBILITY');
 | ||
| 	});
 | ||
| 
 | ||
| 	test('文字数ぎりぎりで怒られない', async () => {
 | ||
| 		const post = {
 | ||
| 			text: '!'.repeat(MAX_NOTE_TEXT_LENGTH), // 3000文字
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('文字数オーバーで怒られる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: '!'.repeat(MAX_NOTE_TEXT_LENGTH + 1), // 3001文字
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('存在しないリプライ先で怒られる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: 'test',
 | ||
| 			replyId: '000000000000000000000000',
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('存在しないrenote対象で怒られる', async () => {
 | ||
| 		const post = {
 | ||
| 			renoteId: '000000000000000000000000',
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('不正なリプライ先IDで怒られる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: 'test',
 | ||
| 			replyId: 'foo',
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('不正なrenote対象IDで怒られる', async () => {
 | ||
| 		const post = {
 | ||
| 			renoteId: 'foo',
 | ||
| 		};
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 		assert.strictEqual(res.status, 400);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('存在しないユーザーにメンションできる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: '@ghost yo',
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, post.text);
 | ||
| 	});
 | ||
| 
 | ||
| 	test('同じユーザーに複数メンションしても内部的にまとめられる', async () => {
 | ||
| 		const post = {
 | ||
| 			text: '@bob @bob @bob yo',
 | ||
| 		};
 | ||
| 
 | ||
| 		const res = await api('notes/create', post, alice);
 | ||
| 
 | ||
| 		assert.strictEqual(res.status, 200);
 | ||
| 		assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 		assert.strictEqual(res.body.createdNote.text, post.text);
 | ||
| 
 | ||
| 		const noteDoc = await Notes.findOneBy({ id: res.body.createdNote.id });
 | ||
| 		assert.ok(noteDoc);
 | ||
| 		assert.deepStrictEqual(noteDoc.mentions, [bob.id]);
 | ||
| 	});
 | ||
| 
 | ||
| 	describe('添付ファイル情報', () => {
 | ||
| 		test('ファイルを添付した場合、投稿成功時にファイル情報入りのレスポンスが帰ってくる', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 			const res = await api('notes/create', {
 | ||
| 				fileIds: [file.body!.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 			assert.ok(res.body.createdNote.files);
 | ||
| 			assert.strictEqual(res.body.createdNote.files.length, 1);
 | ||
| 			assert.strictEqual(res.body.createdNote.files[0].id, file.body!.id);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ファイルを添付した場合、タイムラインでファイル情報入りのレスポンスが帰ってくる', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 			const createdNote = await api('notes/create', {
 | ||
| 				fileIds: [file.body!.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(createdNote.status, 200);
 | ||
| 
 | ||
| 			const res = await api('notes', {
 | ||
| 				withFiles: true,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(Array.isArray(res.body), true);
 | ||
| 			const myNote = res.body.find(note => note.id === createdNote.body.createdNote.id);
 | ||
| 			assert.ok(myNote);
 | ||
| 			assert.ok(myNote.files);
 | ||
| 			assert.strictEqual(myNote.files.length, 1);
 | ||
| 			assert.strictEqual(myNote.files[0].id, file.body!.id);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ファイルが添付されたノートをリノートした場合、タイムラインでファイル情報入りのレスポンスが帰ってくる', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 			const createdNote = await api('notes/create', {
 | ||
| 				fileIds: [file.body!.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(createdNote.status, 200);
 | ||
| 
 | ||
| 			const renoted = await api('notes/create', {
 | ||
| 				renoteId: createdNote.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(renoted.status, 200);
 | ||
| 
 | ||
| 			const res = await api('notes', {
 | ||
| 				renote: true,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(Array.isArray(res.body), true);
 | ||
| 			const myNote = res.body.find((note: { id: string }) => note.id === renoted.body.createdNote.id);
 | ||
| 			assert.ok(myNote);
 | ||
| 			assert.ok(myNote.renote);
 | ||
| 			assert.ok(myNote.renote.files);
 | ||
| 			assert.strictEqual(myNote.renote.files.length, 1);
 | ||
| 			assert.strictEqual(myNote.renote.files[0].id, file.body!.id);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ファイルが添付されたノートに返信した場合、タイムラインでファイル情報入りのレスポンスが帰ってくる', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 			const createdNote = await api('notes/create', {
 | ||
| 				fileIds: [file.body!.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(createdNote.status, 200);
 | ||
| 
 | ||
| 			const reply = await api('notes/create', {
 | ||
| 				replyId: createdNote.body.createdNote.id,
 | ||
| 				text: 'this is reply',
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(reply.status, 200);
 | ||
| 
 | ||
| 			const res = await api('notes', {
 | ||
| 				reply: true,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(Array.isArray(res.body), true);
 | ||
| 			const myNote = res.body.find((note: { id: string }) => note.id === reply.body.createdNote.id);
 | ||
| 			assert.ok(myNote);
 | ||
| 			assert.ok(myNote.reply);
 | ||
| 			assert.ok(myNote.reply.files);
 | ||
| 			assert.strictEqual(myNote.reply.files.length, 1);
 | ||
| 			assert.strictEqual(myNote.reply.files[0].id, file.body!.id);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ファイルが添付されたノートへの返信をリノートした場合、タイムラインでファイル情報入りのレスポンスが帰ってくる', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 			const createdNote = await api('notes/create', {
 | ||
| 				fileIds: [file.body!.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(createdNote.status, 200);
 | ||
| 
 | ||
| 			const reply = await api('notes/create', {
 | ||
| 				replyId: createdNote.body.createdNote.id,
 | ||
| 				text: 'this is reply',
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(reply.status, 200);
 | ||
| 
 | ||
| 			const renoted = await api('notes/create', {
 | ||
| 				renoteId: reply.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(renoted.status, 200);
 | ||
| 
 | ||
| 			const res = await api('notes', {
 | ||
| 				renote: true,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(Array.isArray(res.body), true);
 | ||
| 			const myNote = res.body.find((note: { id: string }) => note.id === renoted.body.createdNote.id);
 | ||
| 			assert.ok(myNote);
 | ||
| 			assert.ok(myNote.renote);
 | ||
| 			assert.ok(myNote.renote.reply);
 | ||
| 			assert.ok(myNote.renote.reply.files);
 | ||
| 			assert.strictEqual(myNote.renote.reply.files.length, 1);
 | ||
| 			assert.strictEqual(myNote.renote.reply.files[0].id, file.body!.id);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('NSFWが強制されている場合変更できない', async () => {
 | ||
| 			const file = await uploadFile(alice);
 | ||
| 
 | ||
| 			const res = await api('admin/roles/create', {
 | ||
| 				name: 'test',
 | ||
| 				description: '',
 | ||
| 				color: null,
 | ||
| 				iconUrl: null,
 | ||
| 				displayOrder: 0,
 | ||
| 				target: 'manual',
 | ||
| 				condFormula: {},
 | ||
| 				isAdministrator: false,
 | ||
| 				isModerator: false,
 | ||
| 				isPublic: false,
 | ||
| 				isExplorable: false,
 | ||
| 				asBadge: false,
 | ||
| 				canEditMembersByModerator: false,
 | ||
| 				policies: {
 | ||
| 					alwaysMarkNsfw: {
 | ||
| 						useDefault: false,
 | ||
| 						priority: 0,
 | ||
| 						value: true,
 | ||
| 					},
 | ||
| 				},
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 
 | ||
| 			const assign = await api('admin/roles/assign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(assign.status, 204);
 | ||
| 			assert.strictEqual(file.body!.isSensitive, false);
 | ||
| 
 | ||
| 			const nsfwfile = await uploadFile(alice);
 | ||
| 
 | ||
| 			assert.strictEqual(nsfwfile.status, 200);
 | ||
| 			assert.strictEqual(nsfwfile.body!.isSensitive, true);
 | ||
| 
 | ||
| 			const liftnsfw = await api('drive/files/update', {
 | ||
| 				fileId: nsfwfile.body!.id,
 | ||
| 				isSensitive: false,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(liftnsfw.status, 400);
 | ||
| 			assert.strictEqual(castAsError(liftnsfw.body).error.code, 'RESTRICTED_BY_ROLE');
 | ||
| 
 | ||
| 			const oldaddnsfw = await api('drive/files/update', {
 | ||
| 				fileId: file.body!.id,
 | ||
| 				isSensitive: true,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(oldaddnsfw.status, 200);
 | ||
| 
 | ||
| 			await api('admin/roles/unassign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			await api('admin/roles/delete', {
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 		});
 | ||
| 	});
 | ||
| 
 | ||
| 	describe('notes/create', () => {
 | ||
| 		test('投票を添付できる', async () => {
 | ||
| 			const res = await api('notes/create', {
 | ||
| 				text: 'test',
 | ||
| 				poll: {
 | ||
| 					choices: ['foo', 'bar'],
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 			assert.strictEqual(typeof res.body === 'object' && !Array.isArray(res.body), true);
 | ||
| 			assert.strictEqual(res.body.createdNote.poll != null, true);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('投票の選択肢が無くて怒られる', async () => {
 | ||
| 			const res = await api('notes/create', {
 | ||
| 				// @ts-expect-error poll must not be empty
 | ||
| 				poll: {},
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('投票の選択肢が無くて怒られる (空の配列)', async () => {
 | ||
| 			const res = await api('notes/create', {
 | ||
| 				poll: {
 | ||
| 					choices: [],
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('投票の選択肢が1つで怒られる', async () => {
 | ||
| 			const res = await api('notes/create', {
 | ||
| 				poll: {
 | ||
| 					choices: ['Strawberry Pasta'],
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('投票できる', async () => {
 | ||
| 			const { body } = await api('notes/create', {
 | ||
| 				text: 'test',
 | ||
| 				poll: {
 | ||
| 					choices: ['sakura', 'izumi', 'ako'],
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			const res = await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 1,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 204);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('複数投票できない', async () => {
 | ||
| 			const { body } = await api('notes/create', {
 | ||
| 				text: 'test',
 | ||
| 				poll: {
 | ||
| 					choices: ['sakura', 'izumi', 'ako'],
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 0,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			const res = await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 2,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('許可されている場合は複数投票できる', async () => {
 | ||
| 			const { body } = await api('notes/create', {
 | ||
| 				text: 'test',
 | ||
| 				poll: {
 | ||
| 					choices: ['sakura', 'izumi', 'ako'],
 | ||
| 					multiple: true,
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 0,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 1,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			const res = await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 2,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 204);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('締め切られている場合は投票できない', async () => {
 | ||
| 			const { body } = await api('notes/create', {
 | ||
| 				text: 'test',
 | ||
| 				poll: {
 | ||
| 					choices: ['sakura', 'izumi', 'ako'],
 | ||
| 					expiredAfter: 1,
 | ||
| 				},
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const res = await api('notes/polls/vote', {
 | ||
| 				noteId: body.createdNote.id,
 | ||
| 				choice: 1,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('センシティブな投稿はhomeになる (単語指定)', async () => {
 | ||
| 			const sensitive = await api('admin/update-meta', {
 | ||
| 				sensitiveWords: [
 | ||
| 					'test',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(sensitive.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note1 = await api('notes/create', {
 | ||
| 				text: 'hogetesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note1.status, 200);
 | ||
| 			assert.strictEqual(note1.body.createdNote.visibility, 'home');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('センシティブな投稿はhomeになる (正規表現)', async () => {
 | ||
| 			const sensitive = await api('admin/update-meta', {
 | ||
| 				sensitiveWords: [
 | ||
| 					'/Test/i',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(sensitive.status, 204);
 | ||
| 
 | ||
| 			const note2 = await api('notes/create', {
 | ||
| 				text: 'hogetesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note2.status, 200);
 | ||
| 			assert.strictEqual(note2.body.createdNote.visibility, 'home');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('センシティブな投稿はhomeになる (スペースアンド)', async () => {
 | ||
| 			const sensitive = await api('admin/update-meta', {
 | ||
| 				sensitiveWords: [
 | ||
| 					'Test hoge',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(sensitive.status, 204);
 | ||
| 
 | ||
| 			const note2 = await api('notes/create', {
 | ||
| 				text: 'hogeTesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note2.status, 200);
 | ||
| 			assert.strictEqual(note2.body.createdNote.visibility, 'home');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('禁止ワードを含む投稿はエラーになる (単語指定)', async () => {
 | ||
| 			const prohibited = await api('admin/update-meta', {
 | ||
| 				prohibitedWords: [
 | ||
| 					'test',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(prohibited.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note1 = await api('notes/create', {
 | ||
| 				text: 'hogetesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note1.status, 400);
 | ||
| 			assert.strictEqual(castAsError(note1.body).error.code, 'CONTAINS_PROHIBITED_WORDS');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('禁止ワードを含む投稿はエラーになる (正規表現)', async () => {
 | ||
| 			const prohibited = await api('admin/update-meta', {
 | ||
| 				prohibitedWords: [
 | ||
| 					'/Test/i',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(prohibited.status, 204);
 | ||
| 
 | ||
| 			const note2 = await api('notes/create', {
 | ||
| 				text: 'hogetesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note2.status, 400);
 | ||
| 			assert.strictEqual(castAsError(note2.body).error.code, 'CONTAINS_PROHIBITED_WORDS');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('禁止ワードを含む投稿はエラーになる (スペースアンド)', async () => {
 | ||
| 			const prohibited = await api('admin/update-meta', {
 | ||
| 				prohibitedWords: [
 | ||
| 					'Test hoge',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(prohibited.status, 204);
 | ||
| 
 | ||
| 			const note2 = await api('notes/create', {
 | ||
| 				text: 'hogeTesthuge',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note2.status, 400);
 | ||
| 			assert.strictEqual(castAsError(note2.body).error.code, 'CONTAINS_PROHIBITED_WORDS');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('禁止ワードを含んでるリモートノートもエラーになる', async () => {
 | ||
| 			const prohibited = await api('admin/update-meta', {
 | ||
| 				prohibitedWords: [
 | ||
| 					'test',
 | ||
| 				],
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(prohibited.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note1 = await api('notes/create', {
 | ||
| 				text: 'hogetesthuge',
 | ||
| 			}, tom);
 | ||
| 
 | ||
| 			assert.strictEqual(note1.status, 400);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('メンションの数が上限を超えるとエラーになる', async () => {
 | ||
| 			const res = await api('admin/roles/create', {
 | ||
| 				name: 'test',
 | ||
| 				description: '',
 | ||
| 				color: null,
 | ||
| 				iconUrl: null,
 | ||
| 				displayOrder: 0,
 | ||
| 				target: 'manual',
 | ||
| 				condFormula: {},
 | ||
| 				isAdministrator: false,
 | ||
| 				isModerator: false,
 | ||
| 				isPublic: false,
 | ||
| 				isExplorable: false,
 | ||
| 				asBadge: false,
 | ||
| 				canEditMembersByModerator: false,
 | ||
| 				policies: {
 | ||
| 					mentionLimit: {
 | ||
| 						useDefault: false,
 | ||
| 						priority: 1,
 | ||
| 						value: 0,
 | ||
| 					},
 | ||
| 				},
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const assign = await api('admin/roles/assign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(assign.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note = await api('notes/create', {
 | ||
| 				text: '@bob potentially annoying text',
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note.status, 400);
 | ||
| 			assert.strictEqual(castAsError(note.body).error.code, 'CONTAINS_TOO_MANY_MENTIONS');
 | ||
| 
 | ||
| 			await api('admin/roles/unassign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			await api('admin/roles/delete', {
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ダイレクト投稿もエラーになる', async () => {
 | ||
| 			const res = await api('admin/roles/create', {
 | ||
| 				name: 'test',
 | ||
| 				description: '',
 | ||
| 				color: null,
 | ||
| 				iconUrl: null,
 | ||
| 				displayOrder: 0,
 | ||
| 				target: 'manual',
 | ||
| 				condFormula: {},
 | ||
| 				isAdministrator: false,
 | ||
| 				isModerator: false,
 | ||
| 				isPublic: false,
 | ||
| 				isExplorable: false,
 | ||
| 				asBadge: false,
 | ||
| 				canEditMembersByModerator: false,
 | ||
| 				policies: {
 | ||
| 					mentionLimit: {
 | ||
| 						useDefault: false,
 | ||
| 						priority: 1,
 | ||
| 						value: 0,
 | ||
| 					},
 | ||
| 				},
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const assign = await api('admin/roles/assign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(assign.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note = await api('notes/create', {
 | ||
| 				text: 'potentially annoying text',
 | ||
| 				visibility: 'specified',
 | ||
| 				visibleUserIds: [bob.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note.status, 400);
 | ||
| 			assert.strictEqual(castAsError(note.body).error.code, 'CONTAINS_TOO_MANY_MENTIONS');
 | ||
| 
 | ||
| 			await api('admin/roles/unassign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			await api('admin/roles/delete', {
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('ダイレクトの宛先とメンションが同じ場合は重複してカウントしない', async () => {
 | ||
| 			const res = await api('admin/roles/create', {
 | ||
| 				name: 'test',
 | ||
| 				description: '',
 | ||
| 				color: null,
 | ||
| 				iconUrl: null,
 | ||
| 				displayOrder: 0,
 | ||
| 				target: 'manual',
 | ||
| 				condFormula: {},
 | ||
| 				isAdministrator: false,
 | ||
| 				isModerator: false,
 | ||
| 				isPublic: false,
 | ||
| 				isExplorable: false,
 | ||
| 				asBadge: false,
 | ||
| 				canEditMembersByModerator: false,
 | ||
| 				policies: {
 | ||
| 					mentionLimit: {
 | ||
| 						useDefault: false,
 | ||
| 						priority: 1,
 | ||
| 						value: 1,
 | ||
| 					},
 | ||
| 				},
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 200);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const assign = await api('admin/roles/assign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			assert.strictEqual(assign.status, 204);
 | ||
| 
 | ||
| 			await new Promise(x => setTimeout(x, 2));
 | ||
| 
 | ||
| 			const note = await api('notes/create', {
 | ||
| 				text: '@bob potentially annoying text',
 | ||
| 				visibility: 'specified',
 | ||
| 				visibleUserIds: [bob.id],
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(note.status, 200);
 | ||
| 
 | ||
| 			await api('admin/roles/unassign', {
 | ||
| 				userId: alice.id,
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 
 | ||
| 			await api('admin/roles/delete', {
 | ||
| 				roleId: res.body.id,
 | ||
| 			}, root);
 | ||
| 		});
 | ||
| 	});
 | ||
| 
 | ||
| 	describe('notes/delete', () => {
 | ||
| 		test('delete a reply', async () => {
 | ||
| 			const mainNoteRes = await api('notes/create', {
 | ||
| 				text: 'main post',
 | ||
| 			}, alice);
 | ||
| 			const replyOneRes = await api('notes/create', {
 | ||
| 				text: 'reply one',
 | ||
| 				replyId: mainNoteRes.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 			const replyTwoRes = await api('notes/create', {
 | ||
| 				text: 'reply two',
 | ||
| 				replyId: mainNoteRes.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			const deleteOneRes = await api('notes/delete', {
 | ||
| 				noteId: replyOneRes.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(deleteOneRes.status, 204);
 | ||
| 			let mainNote = await Notes.findOneBy({ id: mainNoteRes.body.createdNote.id });
 | ||
| 			assert.ok(mainNote);
 | ||
| 			assert.strictEqual(mainNote.repliesCount, 1);
 | ||
| 
 | ||
| 			const deleteTwoRes = await api('notes/delete', {
 | ||
| 				noteId: replyTwoRes.body.createdNote.id,
 | ||
| 			}, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(deleteTwoRes.status, 204);
 | ||
| 			mainNote = await Notes.findOneBy({ id: mainNoteRes.body.createdNote.id });
 | ||
| 			assert.ok(mainNote);
 | ||
| 			assert.strictEqual(mainNote.repliesCount, 0);
 | ||
| 		});
 | ||
| 	});
 | ||
| 
 | ||
| 	describe('notes/translate', () => {
 | ||
| 		describe('翻訳機能の利用が許可されていない場合', () => {
 | ||
| 			let cannotTranslateRole: misskey.entities.Role;
 | ||
| 
 | ||
| 			beforeAll(async () => {
 | ||
| 				cannotTranslateRole = await role(root, {}, { canUseTranslator: false });
 | ||
| 				await api('admin/roles/assign', { roleId: cannotTranslateRole.id, userId: alice.id }, root);
 | ||
| 			});
 | ||
| 
 | ||
| 			test('翻訳機能の利用が許可されていない場合翻訳できない', async () => {
 | ||
| 				const aliceNote = await post(alice, { text: 'Hello' });
 | ||
| 				const res = await api('notes/translate', {
 | ||
| 					noteId: aliceNote.id,
 | ||
| 					targetLang: 'ja',
 | ||
| 				}, alice);
 | ||
| 
 | ||
| 				assert.strictEqual(res.status, 400);
 | ||
| 				assert.strictEqual(castAsError(res.body).error.code, 'UNAVAILABLE');
 | ||
| 			});
 | ||
| 
 | ||
| 			afterAll(async () => {
 | ||
| 				await api('admin/roles/unassign', { roleId: cannotTranslateRole.id, userId: alice.id }, root);
 | ||
| 			});
 | ||
| 		});
 | ||
| 
 | ||
| 		test('存在しないノートは翻訳できない', async () => {
 | ||
| 			const res = await api('notes/translate', { noteId: 'foo', targetLang: 'ja' }, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 			assert.strictEqual(castAsError(res.body).error.code, 'NO_SUCH_NOTE');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('不可視なノートは翻訳できない', async () => {
 | ||
| 			const aliceNote = await post(alice, { visibility: 'followers', text: 'Hello' });
 | ||
| 			const bobTranslateAttempt = await api('notes/translate', { noteId: aliceNote.id, targetLang: 'ja' }, bob);
 | ||
| 
 | ||
| 			assert.strictEqual(bobTranslateAttempt.status, 400);
 | ||
| 			assert.strictEqual(castAsError(bobTranslateAttempt.body).error.code, 'CANNOT_TRANSLATE_INVISIBLE_NOTE');
 | ||
| 		});
 | ||
| 
 | ||
| 		test('text: null なノートを翻訳すると空のレスポンスが返ってくる', async () => {
 | ||
| 			const aliceNote = await post(alice, { text: null, poll: { choices: ['kinoko', 'takenoko'] } });
 | ||
| 			const res = await api('notes/translate', { noteId: aliceNote.id, targetLang: 'ja' }, alice);
 | ||
| 
 | ||
| 			assert.strictEqual(res.status, 204);
 | ||
| 		});
 | ||
| 
 | ||
| 		test('サーバーに DeepL 認証キーが登録されていない場合翻訳できない', async () => {
 | ||
| 			const aliceNote = await post(alice, { text: 'Hello' });
 | ||
| 			const res = await api('notes/translate', { noteId: aliceNote.id, targetLang: 'ja' }, alice);
 | ||
| 
 | ||
| 			// NOTE: デフォルトでは登録されていないので落ちる
 | ||
| 			assert.strictEqual(res.status, 400);
 | ||
| 			assert.strictEqual(castAsError(res.body).error.code, 'UNAVAILABLE');
 | ||
| 		});
 | ||
| 	});
 | ||
| });
 |