mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-25 18:54:52 +00:00 
			
		
		
		
	* fix(backend): send Delete activity of a note to users who renoted or replied to it * Update CHANGELOG.md
		
			
				
	
	
		
			383 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			383 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import assert, { rejects, strictEqual } from 'node:assert';
 | |
| import * as Misskey from 'misskey-js';
 | |
| import { addCustomEmoji, createAccount, createModerator, deepStrictEqualWithExcludedFields, type LoginUser, resolveRemoteNote, resolveRemoteUser, sleep, uploadFile } from './utils.js';
 | |
| 
 | |
| describe('Note', () => {
 | |
| 	let alice: LoginUser, bob: LoginUser;
 | |
| 	let bobInA: Misskey.entities.UserDetailedNotMe, aliceInB: Misskey.entities.UserDetailedNotMe;
 | |
| 
 | |
| 	beforeAll(async () => {
 | |
| 		[alice, bob] = await Promise.all([
 | |
| 			createAccount('a.test'),
 | |
| 			createAccount('b.test'),
 | |
| 		]);
 | |
| 
 | |
| 		[bobInA, aliceInB] = await Promise.all([
 | |
| 			resolveRemoteUser('b.test', bob.id, alice),
 | |
| 			resolveRemoteUser('a.test', alice.id, bob),
 | |
| 		]);
 | |
| 	});
 | |
| 
 | |
| 	describe('Note content', () => {
 | |
| 		test('Consistency of Public Note', async () => {
 | |
| 			const image = await uploadFile('a.test', alice);
 | |
| 			const note = (await alice.client.request('notes/create', {
 | |
| 				text: 'I am Alice!',
 | |
| 				fileIds: [image.id],
 | |
| 				poll: {
 | |
| 					choices: ['neko', 'inu'],
 | |
| 					multiple: false,
 | |
| 					expiredAfter: 60 * 60 * 1000,
 | |
| 				},
 | |
| 			})).createdNote;
 | |
| 
 | |
| 			const resolvedNote = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 			deepStrictEqualWithExcludedFields(note, resolvedNote, [
 | |
| 				'id',
 | |
| 				'emojis',
 | |
| 				/** Consistency of files is checked at {@link file://./drive.test.ts}, so let's skip. */
 | |
| 				'fileIds',
 | |
| 				'files',
 | |
| 				/** @see https://github.com/misskey-dev/misskey/issues/12409 */
 | |
| 				'reactionAcceptance',
 | |
| 				'userId',
 | |
| 				'user',
 | |
| 				'uri',
 | |
| 			]);
 | |
| 			strictEqual(aliceInB.id, resolvedNote.userId);
 | |
| 		});
 | |
| 
 | |
| 		test('Consistency of reply', async () => {
 | |
| 			const _replyedNote = (await alice.client.request('notes/create', {
 | |
| 				text: 'a',
 | |
| 			})).createdNote;
 | |
| 			const note = (await alice.client.request('notes/create', {
 | |
| 				text: 'b',
 | |
| 				replyId: _replyedNote.id,
 | |
| 			})).createdNote;
 | |
| 			// NOTE: the repliedCount is incremented, so fetch again
 | |
| 			const replyedNote = await alice.client.request('notes/show', { noteId: _replyedNote.id });
 | |
| 			strictEqual(replyedNote.repliesCount, 1);
 | |
| 
 | |
| 			const resolvedNote = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 			deepStrictEqualWithExcludedFields(note, resolvedNote, [
 | |
| 				'id',
 | |
| 				'emojis',
 | |
| 				'reactionAcceptance',
 | |
| 				'replyId',
 | |
| 				'reply',
 | |
| 				'userId',
 | |
| 				'user',
 | |
| 				'uri',
 | |
| 			]);
 | |
| 			assert(resolvedNote.replyId != null);
 | |
| 			assert(resolvedNote.reply != null);
 | |
| 			deepStrictEqualWithExcludedFields(replyedNote, resolvedNote.reply, [
 | |
| 				'id',
 | |
| 				// TODO: why clippedCount loses consistency?
 | |
| 				'clippedCount',
 | |
| 				'emojis',
 | |
| 				'userId',
 | |
| 				'user',
 | |
| 				'uri',
 | |
| 				// flaky because this is parallelly incremented, so let's check it below
 | |
| 				'repliesCount',
 | |
| 			]);
 | |
| 			strictEqual(aliceInB.id, resolvedNote.userId);
 | |
| 
 | |
| 			await sleep();
 | |
| 
 | |
| 			const resolvedReplyedNote = await bob.client.request('notes/show', { noteId: resolvedNote.replyId });
 | |
| 			strictEqual(resolvedReplyedNote.repliesCount, 1);
 | |
| 		});
 | |
| 
 | |
| 		test('Consistency of Renote', async () => {
 | |
| 			// NOTE: the renoteCount is not incremented, so no need to fetch again
 | |
| 			const renotedNote = (await alice.client.request('notes/create', {
 | |
| 				text: 'a',
 | |
| 			})).createdNote;
 | |
| 			const note = (await alice.client.request('notes/create', {
 | |
| 				text: 'b',
 | |
| 				renoteId: renotedNote.id,
 | |
| 			})).createdNote;
 | |
| 
 | |
| 			const resolvedNote = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 			deepStrictEqualWithExcludedFields(note, resolvedNote, [
 | |
| 				'id',
 | |
| 				'emojis',
 | |
| 				'reactionAcceptance',
 | |
| 				'renoteId',
 | |
| 				'renote',
 | |
| 				'userId',
 | |
| 				'user',
 | |
| 				'uri',
 | |
| 			]);
 | |
| 			assert(resolvedNote.renoteId != null);
 | |
| 			assert(resolvedNote.renote != null);
 | |
| 			deepStrictEqualWithExcludedFields(renotedNote, resolvedNote.renote, [
 | |
| 				'id',
 | |
| 				'emojis',
 | |
| 				'userId',
 | |
| 				'user',
 | |
| 				'uri',
 | |
| 			]);
 | |
| 			strictEqual(aliceInB.id, resolvedNote.userId);
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	describe('Other props', () => {
 | |
| 		test('localOnly', async () => {
 | |
| 			const note = (await alice.client.request('notes/create', { text: 'a', localOnly: true })).createdNote;
 | |
| 			rejects(
 | |
| 				async () => await bob.client.request('ap/show', { uri: `https://a.test/notes/${note.id}` }),
 | |
| 				(err: any) => {
 | |
| 					strictEqual(err.code, 'REQUEST_FAILED');
 | |
| 					return true;
 | |
| 				},
 | |
| 			);
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	describe('Deletion', () => {
 | |
| 		describe('Check Delete is delivered', () => {
 | |
| 			describe('To followers', () => {
 | |
| 				let carol: LoginUser;
 | |
| 
 | |
| 				beforeAll(async () => {
 | |
| 					carol = await createAccount('a.test');
 | |
| 
 | |
| 					await carol.client.request('following/create', { userId: bobInA.id });
 | |
| 					await sleep();
 | |
| 				});
 | |
| 
 | |
| 				test('Check', async () => {
 | |
| 					const note = (await bob.client.request('notes/create', { text: 'I\'m Bob.' })).createdNote;
 | |
| 					const noteInA = await resolveRemoteNote('b.test', note.id, carol);
 | |
| 					await bob.client.request('notes/delete', { noteId: note.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await rejects(
 | |
| 						async () => await carol.client.request('notes/show', { noteId: noteInA.id }),
 | |
| 						(err: any) => {
 | |
| 							strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 							return true;
 | |
| 						},
 | |
| 					);
 | |
| 				});
 | |
| 
 | |
| 				afterAll(async () => {
 | |
| 					await carol.client.request('following/delete', { userId: bobInA.id });
 | |
| 					await sleep();
 | |
| 				});
 | |
| 			});
 | |
| 
 | |
| 			describe('To renoted and not followed user', () => {
 | |
| 				test('Check', async () => {
 | |
| 					const note = (await bob.client.request('notes/create', { text: 'I\'m Bob.' })).createdNote;
 | |
| 					const noteInA = await resolveRemoteNote('b.test', note.id, alice);
 | |
| 					await alice.client.request('notes/create', { renoteId: noteInA.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await bob.client.request('notes/delete', { noteId: note.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await rejects(
 | |
| 						async () => await alice.client.request('notes/show', { noteId: noteInA.id }),
 | |
| 						(err: any) => {
 | |
| 							strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 							return true;
 | |
| 						},
 | |
| 					);
 | |
| 				});
 | |
| 			});
 | |
| 
 | |
| 			describe('To replied and not followed user', () => {
 | |
| 				test('Check', async () => {
 | |
| 					const note = (await bob.client.request('notes/create', { text: 'I\'m Bob.' })).createdNote;
 | |
| 					const noteInA = await resolveRemoteNote('b.test', note.id, alice);
 | |
| 					await alice.client.request('notes/create', { text: 'Hello Bob!', replyId: noteInA.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await bob.client.request('notes/delete', { noteId: note.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await rejects(
 | |
| 						async () => await alice.client.request('notes/show', { noteId: noteInA.id }),
 | |
| 						(err: any) => {
 | |
| 							strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 							return true;
 | |
| 						},
 | |
| 					);
 | |
| 				});
 | |
| 			});
 | |
| 
 | |
| 			/**
 | |
| 			 * FIXME: not delivered
 | |
| 			 * @see https://github.com/misskey-dev/misskey/issues/15548
 | |
| 			 */
 | |
| 			describe('To only resolved and not followed user', () => {
 | |
| 				test.failing('Check', async () => {
 | |
| 					const note = (await bob.client.request('notes/create', { text: 'I\'m Bob.' })).createdNote;
 | |
| 					const noteInA = await resolveRemoteNote('b.test', note.id, alice);
 | |
| 					await sleep();
 | |
| 
 | |
| 					await bob.client.request('notes/delete', { noteId: note.id });
 | |
| 					await sleep();
 | |
| 
 | |
| 					await rejects(
 | |
| 						async () => await alice.client.request('notes/show', { noteId: noteInA.id }),
 | |
| 						(err: any) => {
 | |
| 							strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 							return true;
 | |
| 						},
 | |
| 					);
 | |
| 				});
 | |
| 			});
 | |
| 		});
 | |
| 
 | |
| 		describe('Deletion of remote user\'s note for moderation', () => {
 | |
| 			let note: Misskey.entities.Note;
 | |
| 
 | |
| 			test('Alice post is deleted in B', async () => {
 | |
| 				note = (await alice.client.request('notes/create', { text: 'Hello' })).createdNote;
 | |
| 				const noteInB = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				const bMod = await createModerator('b.test');
 | |
| 				await bMod.client.request('notes/delete', { noteId: noteInB.id });
 | |
| 				await rejects(
 | |
| 					async () => await bob.client.request('notes/show', { noteId: noteInB.id }),
 | |
| 					(err: any) => {
 | |
| 						strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 						return true;
 | |
| 					},
 | |
| 				);
 | |
| 			});
 | |
| 
 | |
| 			/**
 | |
| 			 * FIXME: implement soft deletion as well as user?
 | |
| 			 *        @see https://github.com/misskey-dev/misskey/issues/11437
 | |
| 			 */
 | |
| 			test.failing('Not found even if resolve again', async () => {
 | |
| 				const noteInB = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				await rejects(
 | |
| 					async () => await bob.client.request('notes/show', { noteId: noteInB.id }),
 | |
| 					(err: any) => {
 | |
| 						strictEqual(err.code, 'NO_SUCH_NOTE');
 | |
| 						return true;
 | |
| 					},
 | |
| 				);
 | |
| 			});
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	describe('Reaction', () => {
 | |
| 		describe('Consistency', () => {
 | |
| 			test('Unicode reaction', async () => {
 | |
| 				const note = (await alice.client.request('notes/create', { text: 'a' })).createdNote;
 | |
| 				const resolvedNote = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				const reaction = '😅';
 | |
| 				await bob.client.request('notes/reactions/create', { noteId: resolvedNote.id, reaction });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const reactions = await alice.client.request('notes/reactions', { noteId: note.id });
 | |
| 				strictEqual(reactions.length, 1);
 | |
| 				strictEqual(reactions[0].type, reaction);
 | |
| 				strictEqual(reactions[0].user.id, bobInA.id);
 | |
| 			});
 | |
| 
 | |
| 			test('Custom emoji reaction', async () => {
 | |
| 				const note = (await alice.client.request('notes/create', { text: 'a' })).createdNote;
 | |
| 				const resolvedNote = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				const emoji = await addCustomEmoji('b.test');
 | |
| 				await bob.client.request('notes/reactions/create', { noteId: resolvedNote.id, reaction: `:${emoji.name}:` });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const reactions = await alice.client.request('notes/reactions', { noteId: note.id });
 | |
| 				strictEqual(reactions.length, 1);
 | |
| 				strictEqual(reactions[0].type, `:${emoji.name}@b.test:`);
 | |
| 				strictEqual(reactions[0].user.id, bobInA.id);
 | |
| 			});
 | |
| 		});
 | |
| 
 | |
| 		describe('Acceptance', () => {
 | |
| 			test('Even if likeOnly, remote users can react with custom emoji, but it is converted to like', async () => {
 | |
| 				const note = (await alice.client.request('notes/create', { text: 'a', reactionAcceptance: 'likeOnly' })).createdNote;
 | |
| 				const noteInB = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				const emoji = await addCustomEmoji('b.test');
 | |
| 				await bob.client.request('notes/reactions/create', { noteId: noteInB.id, reaction: `:${emoji.name}:` });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const reactions = await alice.client.request('notes/reactions', { noteId: note.id });
 | |
| 				strictEqual(reactions.length, 1);
 | |
| 				strictEqual(reactions[0].type, '❤');
 | |
| 			});
 | |
| 
 | |
| 			/**
 | |
| 			 * TODO: this may be unexpected behavior?
 | |
| 			 *       @see https://github.com/misskey-dev/misskey/issues/12409
 | |
| 			 */
 | |
| 			test('Even if nonSensitiveOnly, remote users can react with sensitive emoji, and it is not converted', async () => {
 | |
| 				const note = (await alice.client.request('notes/create', { text: 'a', reactionAcceptance: 'nonSensitiveOnly' })).createdNote;
 | |
| 				const noteInB = await resolveRemoteNote('a.test', note.id, bob);
 | |
| 				const emoji = await addCustomEmoji('b.test', { isSensitive: true });
 | |
| 				await bob.client.request('notes/reactions/create', { noteId: noteInB.id, reaction: `:${emoji.name}:` });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const reactions = await alice.client.request('notes/reactions', { noteId: note.id });
 | |
| 				strictEqual(reactions.length, 1);
 | |
| 				strictEqual(reactions[0].type, `:${emoji.name}@b.test:`);
 | |
| 			});
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	describe('Poll', () => {
 | |
| 		describe('Any remote user\'s vote is delivered to the author', () => {
 | |
| 			let carol: LoginUser;
 | |
| 
 | |
| 			beforeAll(async () => {
 | |
| 				carol = await createAccount('a.test');
 | |
| 			});
 | |
| 
 | |
| 			test('Bob creates poll and receives a vote from Carol', async () => {
 | |
| 				const note = (await bob.client.request('notes/create', { poll: { choices: ['inu', 'neko'] } })).createdNote;
 | |
| 				const noteInA = await resolveRemoteNote('b.test', note.id, carol);
 | |
| 				await carol.client.request('notes/polls/vote', { noteId: noteInA.id, choice: 0 });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const noteAfterVote = await bob.client.request('notes/show', { noteId: note.id });
 | |
| 				assert(noteAfterVote.poll != null);
 | |
| 				strictEqual(noteAfterVote.poll.choices[0].votes, 1);
 | |
| 				strictEqual(noteAfterVote.poll.choices[1].votes, 0);
 | |
| 			});
 | |
| 		});
 | |
| 
 | |
| 		describe('Local user\'s vote is delivered to the author\'s remote followers', () => {
 | |
| 			let bobRemoteFollower: LoginUser, localVoter: LoginUser;
 | |
| 
 | |
| 			beforeAll(async () => {
 | |
| 				[
 | |
| 					bobRemoteFollower,
 | |
| 					localVoter,
 | |
| 				] = await Promise.all([
 | |
| 					createAccount('a.test'),
 | |
| 					createAccount('b.test'),
 | |
| 				]);
 | |
| 
 | |
| 				await bobRemoteFollower.client.request('following/create', { userId: bobInA.id });
 | |
| 				await sleep();
 | |
| 			});
 | |
| 
 | |
| 			test('A vote in Bob\'s server is delivered to Bob\'s remote followers', async () => {
 | |
| 				const note = (await bob.client.request('notes/create', { poll: { choices: ['inu', 'neko'] } })).createdNote;
 | |
| 				// NOTE: resolve before voting
 | |
| 				const noteInA = await resolveRemoteNote('b.test', note.id, bobRemoteFollower);
 | |
| 				await localVoter.client.request('notes/polls/vote', { noteId: note.id, choice: 0 });
 | |
| 				await sleep();
 | |
| 
 | |
| 				const noteAfterVote = await bobRemoteFollower.client.request('notes/show', { noteId: noteInA.id });
 | |
| 				assert(noteAfterVote.poll != null);
 | |
| 				strictEqual(noteAfterVote.poll.choices[0].votes, 1);
 | |
| 				strictEqual(noteAfterVote.poll.choices[1].votes, 0);
 | |
| 			});
 | |
| 		});
 | |
| 	});
 | |
| });
 |