add unittest
This commit is contained in:
parent
3613409eed
commit
86b92381e3
|
@ -0,0 +1,369 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { describe, jest, expect, beforeAll, beforeEach, afterEach, afterAll, test } from '@jest/globals';
|
||||||
|
import { Test, TestingModule } from '@nestjs/testing';
|
||||||
|
import { randomString } from '../utils.js';
|
||||||
|
import { NoteMutingService } from '@/core/note/NoteMutingService.js';
|
||||||
|
import {
|
||||||
|
MiNoteMuting,
|
||||||
|
MiNote,
|
||||||
|
MiUser,
|
||||||
|
NoteMutingsRepository,
|
||||||
|
NotesRepository,
|
||||||
|
UsersRepository,
|
||||||
|
} from '@/models/_.js';
|
||||||
|
import { DI } from '@/di-symbols.js';
|
||||||
|
import { GlobalModule } from '@/GlobalModule.js';
|
||||||
|
import { CoreModule } from '@/core/CoreModule.js';
|
||||||
|
import { IdService } from '@/core/IdService.js';
|
||||||
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
|
import { QueryService } from '@/core/QueryService.js';
|
||||||
|
|
||||||
|
process.env.NODE_ENV = 'test';
|
||||||
|
|
||||||
|
describe('NoteMutingService', () => {
|
||||||
|
let app: TestingModule;
|
||||||
|
let service: NoteMutingService;
|
||||||
|
|
||||||
|
// --------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
let notesRepository: NotesRepository;
|
||||||
|
let noteMutingsRepository: NoteMutingsRepository;
|
||||||
|
let usersRepository: UsersRepository;
|
||||||
|
let idService: IdService;
|
||||||
|
let globalEventService: GlobalEventService;
|
||||||
|
let queryService: QueryService;
|
||||||
|
|
||||||
|
// --------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// Helper function to create a user
|
||||||
|
async function createUser(data: Partial<MiUser> = {}): Promise<MiUser> {
|
||||||
|
const user = {
|
||||||
|
id: idService.gen(),
|
||||||
|
username: randomString(),
|
||||||
|
usernameLower: randomString().toLowerCase(),
|
||||||
|
host: null,
|
||||||
|
...data,
|
||||||
|
};
|
||||||
|
|
||||||
|
return await usersRepository.insert(user)
|
||||||
|
.then(x => usersRepository.findOneByOrFail(x.identifiers[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to create a note
|
||||||
|
async function createNote(data: Partial<MiNote> = {}): Promise<MiNote> {
|
||||||
|
return await notesRepository.insert({
|
||||||
|
id: idService.gen(),
|
||||||
|
userId: data.userId ?? (await createUser()).id,
|
||||||
|
text: randomString(),
|
||||||
|
visibility: 'public',
|
||||||
|
...data,
|
||||||
|
})
|
||||||
|
.then(x => notesRepository.findOneByOrFail(x.identifiers[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to create a note muting
|
||||||
|
async function createNoteMuting(data: Partial<MiNoteMuting> = {}): Promise<MiNoteMuting> {
|
||||||
|
const id = idService.gen();
|
||||||
|
const noteMuting = {
|
||||||
|
id,
|
||||||
|
userId: data.userId || (await createUser()).id,
|
||||||
|
noteId: data.noteId || (await createNote()).id,
|
||||||
|
expiresAt: null,
|
||||||
|
...data,
|
||||||
|
};
|
||||||
|
|
||||||
|
return await noteMutingsRepository.insert(noteMuting)
|
||||||
|
.then(x => noteMutingsRepository.findOneByOrFail(x.identifiers[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
// --------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
app = await Test
|
||||||
|
.createTestingModule({
|
||||||
|
imports: [
|
||||||
|
GlobalModule,
|
||||||
|
CoreModule,
|
||||||
|
],
|
||||||
|
})
|
||||||
|
.compile();
|
||||||
|
|
||||||
|
service = app.get(NoteMutingService);
|
||||||
|
idService = app.get(IdService);
|
||||||
|
queryService = app.get(QueryService);
|
||||||
|
globalEventService = app.get(GlobalEventService);
|
||||||
|
notesRepository = app.get(DI.notesRepository);
|
||||||
|
noteMutingsRepository = app.get(DI.noteMutingsRepository);
|
||||||
|
usersRepository = app.get(DI.usersRepository);
|
||||||
|
|
||||||
|
app.enableShutdownHooks();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
// Clean database before each test
|
||||||
|
await noteMutingsRepository.delete({});
|
||||||
|
await notesRepository.delete({});
|
||||||
|
await usersRepository.delete({});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(async () => {
|
||||||
|
// Clean database after each test
|
||||||
|
await noteMutingsRepository.delete({});
|
||||||
|
await notesRepository.delete({});
|
||||||
|
await usersRepository.delete({});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
await app.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
// --------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
describe('create', () => {
|
||||||
|
test('should create a note muting', async () => {
|
||||||
|
// Create a user and a note
|
||||||
|
const user = await createUser();
|
||||||
|
const note = await createNote();
|
||||||
|
|
||||||
|
// Create a note muting
|
||||||
|
await service.create({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
expiresAt: null,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify the note muting was created
|
||||||
|
const noteMuting = await noteMutingsRepository.findOneBy({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(noteMuting).not.toBeNull();
|
||||||
|
expect(noteMuting?.userId).toBe(user.id);
|
||||||
|
expect(noteMuting?.noteId).toBe(note.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should throw NoSuchNoteError if note does not exist', async () => {
|
||||||
|
// Create a user
|
||||||
|
const user = await createUser();
|
||||||
|
const nonexistentNoteId = idService.gen();
|
||||||
|
|
||||||
|
// Attempt to create a note muting with a non-existent note
|
||||||
|
await expect(service.create({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: nonexistentNoteId,
|
||||||
|
expiresAt: null,
|
||||||
|
})).rejects.toThrow(NoteMutingService.NoSuchNoteError);
|
||||||
|
|
||||||
|
// Verify no note muting was created
|
||||||
|
const noteMuting = await noteMutingsRepository.findOneBy({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: nonexistentNoteId,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(noteMuting).toBeNull();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('delete', () => {
|
||||||
|
test('should delete a note muting', async () => {
|
||||||
|
// Create a user, note, and note muting
|
||||||
|
const user = await createUser();
|
||||||
|
const note = await createNote();
|
||||||
|
const noteMuting = await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify the note muting exists
|
||||||
|
const beforeDelete = await noteMutingsRepository.findOneBy({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
});
|
||||||
|
expect(beforeDelete).not.toBeNull();
|
||||||
|
|
||||||
|
// Delete the note muting
|
||||||
|
await service.delete(user.id, note.id);
|
||||||
|
|
||||||
|
// Verify the note muting was deleted
|
||||||
|
const afterDelete = await noteMutingsRepository.findOneBy({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
});
|
||||||
|
expect(afterDelete).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should throw NotMutedError if muting does not exist', async () => {
|
||||||
|
// Create a user and note
|
||||||
|
const user = await createUser();
|
||||||
|
const note = await createNote();
|
||||||
|
|
||||||
|
// Attempt to delete a non-existent note muting
|
||||||
|
await expect(service.delete(user.id, note.id)).rejects.toThrow(NoteMutingService.NotMutedError);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('isMuting', () => {
|
||||||
|
test('should return true if user is muting the note', async () => {
|
||||||
|
// Create a user, note, and note muting
|
||||||
|
const user = await createUser();
|
||||||
|
const note = await createNote();
|
||||||
|
await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check if the user is muting the note
|
||||||
|
const result = await service.isMuting(user.id, note.id);
|
||||||
|
|
||||||
|
expect(result).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should return false if user is not muting the note', async () => {
|
||||||
|
// Create a user and note, but no muting
|
||||||
|
const user = await createUser();
|
||||||
|
const note = await createNote();
|
||||||
|
|
||||||
|
// Check if the user is muting the note
|
||||||
|
const result = await service.isMuting(user.id, note.id);
|
||||||
|
|
||||||
|
expect(result).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getMutingNoteIdsSet', () => {
|
||||||
|
test('should return a set of muted note IDs', async () => {
|
||||||
|
// Create a user and multiple notes
|
||||||
|
const user = await createUser();
|
||||||
|
const note1 = await createNote();
|
||||||
|
const note2 = await createNote();
|
||||||
|
const note3 = await createNote();
|
||||||
|
|
||||||
|
// Create note mutings for two of the notes
|
||||||
|
await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note1.id,
|
||||||
|
});
|
||||||
|
await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note2.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Get the set of muted note IDs
|
||||||
|
const result = await service.getMutingNoteIdsSet(user.id);
|
||||||
|
|
||||||
|
// Verify the result is a Set containing the muted note IDs
|
||||||
|
expect(result).toBeInstanceOf(Set);
|
||||||
|
expect(result.has(note1.id)).toBe(true);
|
||||||
|
expect(result.has(note2.id)).toBe(true);
|
||||||
|
expect(result.has(note3.id)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('listByUserId', () => {
|
||||||
|
test('should return a list of note mutings for a user', async () => {
|
||||||
|
// Create a user and multiple notes
|
||||||
|
const user = await createUser();
|
||||||
|
const note1 = await createNote();
|
||||||
|
const note2 = await createNote();
|
||||||
|
|
||||||
|
// Create note mutings
|
||||||
|
const muting1 = await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note1.id,
|
||||||
|
});
|
||||||
|
const muting2 = await createNoteMuting({
|
||||||
|
userId: user.id,
|
||||||
|
noteId: note2.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Get the list of note mutings
|
||||||
|
const result = await service.listByUserId({ userId: user.id });
|
||||||
|
|
||||||
|
// Verify the result contains the expected mutings
|
||||||
|
expect(result).toHaveLength(2);
|
||||||
|
expect(result.map(m => m.id).sort()).toEqual([muting1.id, muting2.id].sort());
|
||||||
|
expect(result.map(m => m.noteId).sort()).toEqual([note1.id, note2.id].sort());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('cleanupExpiredMutes', () => {
|
||||||
|
test('should delete expired mutes', async () => {
|
||||||
|
// Create users and notes
|
||||||
|
const user1 = await createUser();
|
||||||
|
const user2 = await createUser();
|
||||||
|
const note1 = await createNote();
|
||||||
|
const note2 = await createNote();
|
||||||
|
const note3 = await createNote();
|
||||||
|
|
||||||
|
// Set the expiration date to 1 hour ago
|
||||||
|
const expiredDate = new Date();
|
||||||
|
expiredDate.setHours(expiredDate.getHours() - 1);
|
||||||
|
|
||||||
|
// Set the expiration date to 1 hour in the future
|
||||||
|
const futureDate = new Date();
|
||||||
|
futureDate.setHours(futureDate.getHours() + 1);
|
||||||
|
|
||||||
|
// Create expired note mutings
|
||||||
|
const expiredMuting1 = await createNoteMuting({
|
||||||
|
userId: user1.id,
|
||||||
|
noteId: note1.id,
|
||||||
|
expiresAt: expiredDate,
|
||||||
|
});
|
||||||
|
|
||||||
|
const expiredMuting2 = await createNoteMuting({
|
||||||
|
userId: user1.id,
|
||||||
|
noteId: note2.id,
|
||||||
|
expiresAt: expiredDate,
|
||||||
|
});
|
||||||
|
|
||||||
|
const expiredMuting3 = await createNoteMuting({
|
||||||
|
userId: user2.id,
|
||||||
|
noteId: note3.id,
|
||||||
|
expiresAt: expiredDate,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create non-expired note muting
|
||||||
|
const activeMuting = await createNoteMuting({
|
||||||
|
userId: user2.id,
|
||||||
|
noteId: note1.id,
|
||||||
|
expiresAt: futureDate,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create permanent note muting (no expiration)
|
||||||
|
const permanentMuting = await createNoteMuting({
|
||||||
|
userId: user2.id,
|
||||||
|
noteId: note2.id,
|
||||||
|
expiresAt: null,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify all mutings exist before cleanup
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting1.id })).not.toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting2.id })).not.toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting3.id })).not.toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: activeMuting.id })).not.toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: permanentMuting.id })).not.toBeNull();
|
||||||
|
|
||||||
|
// Run cleanup
|
||||||
|
await service.cleanupExpiredMutes();
|
||||||
|
|
||||||
|
// Verify expired mutings are deleted and others remain
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting1.id })).toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting2.id })).toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: expiredMuting3.id })).toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: activeMuting.id })).not.toBeNull();
|
||||||
|
expect(await noteMutingsRepository.findOneBy({ id: permanentMuting.id })).not.toBeNull();
|
||||||
|
|
||||||
|
// Verify cache is updated by checking isMuting
|
||||||
|
expect(await service.isMuting(user1.id, note1.id)).toBe(false);
|
||||||
|
expect(await service.isMuting(user1.id, note2.id)).toBe(false);
|
||||||
|
expect(await service.isMuting(user2.id, note3.id)).toBe(false);
|
||||||
|
expect(await service.isMuting(user2.id, note1.id)).toBe(true);
|
||||||
|
expect(await service.isMuting(user2.id, note2.id)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in New Issue