313 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
import { vi, describe, test, expect } from 'vitest';
 | 
						|
import { APIClient, isAPIError } from '../src/api.js';
 | 
						|
 | 
						|
describe('API', () => {
 | 
						|
	test('success', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async (url, options) => {
 | 
						|
				if (url === 'https://misskey.test/api/i' && options?.method === 'POST') {
 | 
						|
					if (options.body) {
 | 
						|
						const body = JSON.parse(options.body as string);
 | 
						|
						if (body.i === 'TOKEN') {
 | 
						|
							return new Response(JSON.stringify({ id: 'foo' }), { status: 200 });
 | 
						|
						}
 | 
						|
					}
 | 
						|
 | 
						|
					return new Response(null, { status: 400 });
 | 
						|
				}
 | 
						|
 | 
						|
				return new Response(null, { status: 404 });
 | 
						|
			});
 | 
						|
 | 
						|
		const cli = new APIClient({
 | 
						|
			origin: 'https://misskey.test',
 | 
						|
			credential: 'TOKEN',
 | 
						|
		});
 | 
						|
 | 
						|
		const res = await cli.request('i');
 | 
						|
 | 
						|
		expect(res).toEqual({
 | 
						|
			id: 'foo'
 | 
						|
		});
 | 
						|
 | 
						|
		fetch('https://misskey.test/api/i', {
 | 
						|
			method: 'POST',
 | 
						|
		})
 | 
						|
 | 
						|
		expect(fetchMock).toHaveBeenCalledWith('https://misskey.test/api/i', {
 | 
						|
			method: 'POST',
 | 
						|
			headers: {
 | 
						|
				'Content-Type': 'application/json',
 | 
						|
			},
 | 
						|
			credentials: 'omit',
 | 
						|
			cache: 'no-cache',
 | 
						|
			body: JSON.stringify({ i: 'TOKEN' }),
 | 
						|
		});
 | 
						|
 | 
						|
		fetchMock.mockRestore();
 | 
						|
	});
 | 
						|
 | 
						|
	test('with params', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async (url, options) => {
 | 
						|
				if (url === 'https://misskey.test/api/notes/show' && options?.method === 'POST') {
 | 
						|
					if (options.body) {
 | 
						|
						const body = JSON.parse(options.body as string);
 | 
						|
						if (body.i === 'TOKEN' && body.noteId === 'aaaaa') {
 | 
						|
							return new Response(JSON.stringify({ id: 'foo' }), { status: 200 });
 | 
						|
						}
 | 
						|
					}
 | 
						|
					return new Response(null, { status: 400 });
 | 
						|
				}
 | 
						|
				return new Response(null, { status: 404 });
 | 
						|
			});
 | 
						|
 | 
						|
		const cli = new APIClient({
 | 
						|
			origin: 'https://misskey.test',
 | 
						|
			credential: 'TOKEN',
 | 
						|
		});
 | 
						|
 | 
						|
		const res = await cli.request('notes/show', { noteId: 'aaaaa' });
 | 
						|
 | 
						|
		expect(res).toEqual({
 | 
						|
			id: 'foo'
 | 
						|
		});
 | 
						|
 | 
						|
		expect(fetchMock).toHaveBeenCalledWith('https://misskey.test/api/notes/show', {
 | 
						|
			method: 'POST',
 | 
						|
			headers: {
 | 
						|
				'Content-Type': 'application/json',
 | 
						|
			},
 | 
						|
			credentials: 'omit',
 | 
						|
			cache: 'no-cache',
 | 
						|
			body: JSON.stringify({ noteId: 'aaaaa', i: 'TOKEN' }),
 | 
						|
		});
 | 
						|
 | 
						|
		fetchMock.mockRestore();
 | 
						|
	});
 | 
						|
 | 
						|
	test('multipart/form-data', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async (url, options) => {
 | 
						|
				if (url === 'https://misskey.test/api/drive/files/create' && options?.method === 'POST') {
 | 
						|
					if (options.body instanceof FormData) {
 | 
						|
						const file = options.body.get('file');
 | 
						|
						if (file instanceof File && file.name === 'foo.txt') {
 | 
						|
							return new Response(JSON.stringify({ id: 'foo' }), { status: 200 });
 | 
						|
						}
 | 
						|
					}
 | 
						|
					return new Response(null, { status: 400 });
 | 
						|
				}
 | 
						|
				return new Response(null, { status: 404 });
 | 
						|
			});
 | 
						|
 | 
						|
		const cli = new APIClient({
 | 
						|
			origin: 'https://misskey.test',
 | 
						|
			credential: 'TOKEN',
 | 
						|
		});
 | 
						|
 | 
						|
		const testFile = new File([], 'foo.txt');
 | 
						|
 | 
						|
		const res = await cli.request('drive/files/create', {
 | 
						|
			file: testFile,
 | 
						|
			name: null, // nullのパラメータは消える
 | 
						|
		});
 | 
						|
 | 
						|
		expect(res).toEqual({
 | 
						|
			id: 'foo'
 | 
						|
		});
 | 
						|
 | 
						|
		expect(fetchMock).toHaveBeenCalledWith('https://misskey.test/api/drive/files/create', {
 | 
						|
			method: 'POST',
 | 
						|
			body: expect.any(FormData),
 | 
						|
			headers: {},
 | 
						|
			credentials: 'omit',
 | 
						|
			cache: 'no-cache',
 | 
						|
		});
 | 
						|
 | 
						|
		fetchMock.mockRestore();
 | 
						|
	});
 | 
						|
 | 
						|
	test('204 No Content で null が返る', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async (url, options) => {
 | 
						|
				if (url === 'https://misskey.test/api/reset-password' && options?.method === 'POST') {
 | 
						|
					return new Response(null, { status: 204 });
 | 
						|
				}
 | 
						|
				return new Response(null, { status: 404 });
 | 
						|
			});
 | 
						|
 | 
						|
		const cli = new APIClient({
 | 
						|
			origin: 'https://misskey.test',
 | 
						|
			credential: 'TOKEN',
 | 
						|
		});
 | 
						|
 | 
						|
		const res = await cli.request('reset-password', { token: 'aaa', password: 'aaa' });
 | 
						|
 | 
						|
		expect(res).toEqual(null);
 | 
						|
 | 
						|
		expect(fetchMock).toHaveBeenCalledWith('https://misskey.test/api/reset-password', {
 | 
						|
			method: 'POST',
 | 
						|
			headers: {
 | 
						|
				'Content-Type': 'application/json',
 | 
						|
			},
 | 
						|
			credentials: 'omit',
 | 
						|
			cache: 'no-cache',
 | 
						|
			body: JSON.stringify({ token: 'aaa', password: 'aaa', i: 'TOKEN' }),
 | 
						|
		});
 | 
						|
 | 
						|
		fetchMock.mockRestore();
 | 
						|
	});
 | 
						|
 | 
						|
	test('インスタンスの credential が指定されていても引数で credential が null ならば null としてリクエストされる', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async (url, options) => {
 | 
						|
				if (url === 'https://misskey.test/api/i' && options?.method === 'POST') {
 | 
						|
					if (options.body) {
 | 
						|
						const body = JSON.parse(options.body as string);
 | 
						|
						if (typeof body.i === 'string') {
 | 
						|
							return new Response(JSON.stringify({ id: 'foo' }), { status: 200 });
 | 
						|
						} else {
 | 
						|
							return new Response(JSON.stringify({
 | 
						|
								error: {
 | 
						|
									message: 'Credential required.',
 | 
						|
									code: 'CREDENTIAL_REQUIRED',
 | 
						|
									id: '1384574d-a912-4b81-8601-c7b1c4085df1',
 | 
						|
								}
 | 
						|
							}), { status: 401 });
 | 
						|
						}
 | 
						|
					}
 | 
						|
					return new Response(null, { status: 400 });
 | 
						|
				}
 | 
						|
				return new Response(null, { status: 404 });
 | 
						|
			});
 | 
						|
 | 
						|
		try {
 | 
						|
			const cli = new APIClient({
 | 
						|
				origin: 'https://misskey.test',
 | 
						|
				credential: 'TOKEN',
 | 
						|
			});
 | 
						|
 | 
						|
			await cli.request('i', {}, null);
 | 
						|
		} catch (e) {
 | 
						|
			expect(isAPIError(e)).toEqual(true);
 | 
						|
		} finally {
 | 
						|
			fetchMock.mockRestore();
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	test('api error', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async () => {
 | 
						|
				return new Response(JSON.stringify({
 | 
						|
					error: {
 | 
						|
						message: 'Internal error occurred. Please contact us if the error persists.',
 | 
						|
						code: 'INTERNAL_ERROR',
 | 
						|
						id: '5d37dbcb-891e-41ca-a3d6-e690c97775ac',
 | 
						|
						kind: 'server',
 | 
						|
					},
 | 
						|
				}), { status: 500 });
 | 
						|
			});
 | 
						|
 | 
						|
		try {
 | 
						|
			const cli = new APIClient({
 | 
						|
				origin: 'https://misskey.test',
 | 
						|
				credential: 'TOKEN',
 | 
						|
			});
 | 
						|
 | 
						|
			await cli.request('i');
 | 
						|
		} catch (e: any) {
 | 
						|
			expect(isAPIError(e)).toEqual(true);
 | 
						|
			expect(e.id).toEqual('5d37dbcb-891e-41ca-a3d6-e690c97775ac');
 | 
						|
		} finally {
 | 
						|
			fetchMock.mockRestore();
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	test('network error', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async () => {
 | 
						|
				throw new Error('Network error');
 | 
						|
			});
 | 
						|
 | 
						|
		try {
 | 
						|
			const cli = new APIClient({
 | 
						|
				origin: 'https://misskey.test',
 | 
						|
				credential: 'TOKEN',
 | 
						|
			});
 | 
						|
 | 
						|
			await cli.request('i');
 | 
						|
		} catch (e) {
 | 
						|
			expect(isAPIError(e)).toEqual(false);
 | 
						|
		} finally {
 | 
						|
			fetchMock.mockRestore();
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	test('json parse error', async () => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async () => {
 | 
						|
				return new Response('<html>I AM NOT JSON</html>', { status: 500 });
 | 
						|
			});
 | 
						|
 | 
						|
		try {
 | 
						|
			const cli = new APIClient({
 | 
						|
				origin: 'https://misskey.test',
 | 
						|
				credential: 'TOKEN',
 | 
						|
			});
 | 
						|
 | 
						|
			await cli.request('i');
 | 
						|
		} catch (e) {
 | 
						|
			expect(isAPIError(e)).toEqual(false);
 | 
						|
		} finally {
 | 
						|
			fetchMock.mockRestore();
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	test('admin/roles/create の型が合う', async() => {
 | 
						|
		const fetchMock = vi
 | 
						|
			.spyOn(globalThis, 'fetch')
 | 
						|
			.mockImplementation(async () => {
 | 
						|
				// 本来返すべき値は`Role`型だが、テストなのでお茶を濁す
 | 
						|
				return new Response('{}', { status: 200 });
 | 
						|
			});
 | 
						|
 | 
						|
		const cli = new APIClient({
 | 
						|
			origin: 'https://misskey.test',
 | 
						|
			credential: 'TOKEN',
 | 
						|
		});
 | 
						|
		await cli.request('admin/roles/create', {
 | 
						|
			name: 'aaa',
 | 
						|
			asBadge: false,
 | 
						|
			canEditMembersByModerator: false,
 | 
						|
			color: '#123456',
 | 
						|
			condFormula: {},
 | 
						|
			description: '',
 | 
						|
			displayOrder: 0,
 | 
						|
			iconUrl: '',
 | 
						|
			isAdministrator: false,
 | 
						|
			isExplorable: false,
 | 
						|
			isModerator: false,
 | 
						|
			isPublic: false,
 | 
						|
			policies: {
 | 
						|
				ltlAvailable: {
 | 
						|
					value: true,
 | 
						|
					priority: 0,
 | 
						|
					useDefault: false,
 | 
						|
				},
 | 
						|
			},
 | 
						|
			target: 'manual',
 | 
						|
		});
 | 
						|
 | 
						|
		fetchMock.mockRestore();
 | 
						|
	})
 | 
						|
});
 |