mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-25 18:54:52 +00:00 
			
		
		
		
	* feat(misskey-js): `POST admin/roles/create`の型を具象化 * fix * docs: CHANGELOG.md * test(misskey-js): admin/roles/createの型が合うことを表明 * test(misskey-js): single quote * test(misskey-js): 無を読もうとして爆発するのを修正 * test(misskey-js): fix comment
		
			
				
	
	
		
			300 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { enableFetchMocks } from 'jest-fetch-mock';
 | |
| import { APIClient, isAPIError } from '../src/api.js';
 | |
| 
 | |
| enableFetchMocks();
 | |
| 
 | |
| function getFetchCall(call: any[]) {
 | |
| 	const { body, method } = call[1];
 | |
| 	const contentType = call[1].headers['Content-Type'];
 | |
| 	if (
 | |
| 		body == null ||
 | |
| 		(contentType === 'application/json' && typeof body !== 'string') ||
 | |
| 		(contentType === 'multipart/form-data' && !(body instanceof FormData))
 | |
| 	) {
 | |
| 		throw new Error('invalid body');
 | |
| 	}
 | |
| 	return {
 | |
| 		url: call[0],
 | |
| 		method: method,
 | |
| 		contentType: contentType,
 | |
| 		body: body instanceof FormData ? Object.fromEntries(body.entries()) : JSON.parse(body),
 | |
| 	};
 | |
| }
 | |
| 
 | |
| describe('API', () => {
 | |
| 	test('success', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			const body = await req.json();
 | |
| 			if (req.method == 'POST' && req.url == 'https://misskey.test/api/i') {
 | |
| 				if (body.i === 'TOKEN') {
 | |
| 					return JSON.stringify({ id: 'foo' });
 | |
| 				} else {
 | |
| 					return { status: 400 };
 | |
| 				}
 | |
| 			} else {
 | |
| 				return { status: 404 };
 | |
| 			}
 | |
| 		});
 | |
| 
 | |
| 		const cli = new APIClient({
 | |
| 			origin: 'https://misskey.test',
 | |
| 			credential: 'TOKEN',
 | |
| 		});
 | |
| 
 | |
| 		const res = await cli.request('i');
 | |
| 
 | |
| 		expect(res).toEqual({
 | |
| 			id: 'foo'
 | |
| 		});
 | |
| 
 | |
| 		expect(getFetchCall(fetchMock.mock.calls[0])).toEqual({
 | |
| 			url: 'https://misskey.test/api/i',
 | |
| 			method: 'POST',
 | |
| 			contentType: 'application/json',
 | |
| 			body: { i: 'TOKEN' }
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	test('with params', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			const body = await req.json();
 | |
| 			if (req.method == 'POST' && req.url == 'https://misskey.test/api/notes/show') {
 | |
| 				if (body.i === 'TOKEN' && body.noteId === 'aaaaa') {
 | |
| 					return JSON.stringify({ id: 'foo' });
 | |
| 				} else {
 | |
| 					return { status: 400 };
 | |
| 				}
 | |
| 			} else {
 | |
| 				return { 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(getFetchCall(fetchMock.mock.calls[0])).toEqual({
 | |
| 			url: 'https://misskey.test/api/notes/show',
 | |
| 			method: 'POST',
 | |
| 			contentType: 'application/json',
 | |
| 			body: { i: 'TOKEN', noteId: 'aaaaa' }
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	test('multipart/form-data', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			if (req.method == 'POST' && req.url == 'https://misskey.test/api/drive/files/create') {
 | |
| 				if (req.headers.get('Content-Type')?.includes('multipart/form-data')) {
 | |
| 					return JSON.stringify({ id: 'foo' });
 | |
| 				} else {
 | |
| 					return { status: 400 };
 | |
| 				}
 | |
| 			} else {
 | |
| 				return { 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(getFetchCall(fetchMock.mock.calls[0])).toEqual({
 | |
| 			url: 'https://misskey.test/api/drive/files/create',
 | |
| 			method: 'POST',
 | |
| 			contentType: 'multipart/form-data',
 | |
| 			body: {
 | |
| 				i: 'TOKEN',
 | |
| 				file: testFile,
 | |
| 			}
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	test('204 No Content で null が返る', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			if (req.method == 'POST' && req.url == 'https://misskey.test/api/reset-password') {
 | |
| 				return { status: 204 };
 | |
| 			} else {
 | |
| 				return { 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(getFetchCall(fetchMock.mock.calls[0])).toEqual({
 | |
| 			url: 'https://misskey.test/api/reset-password',
 | |
| 			method: 'POST',
 | |
| 			contentType: 'application/json',
 | |
| 			body: { i: 'TOKEN', token: 'aaa', password: 'aaa' }
 | |
| 		});
 | |
| 	});
 | |
| 
 | |
| 	test('インスタンスの credential が指定されていても引数で credential が null ならば null としてリクエストされる', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			const body = await req.json();
 | |
| 			if (req.method == 'POST' && req.url == 'https://misskey.test/api/i') {
 | |
| 				if (typeof body.i === 'string') {
 | |
| 					return JSON.stringify({ id: 'foo' });
 | |
| 				} else {
 | |
| 					return {
 | |
| 						status: 401,
 | |
| 						body: JSON.stringify({
 | |
| 							error: {
 | |
| 								message: 'Credential required.',
 | |
| 								code: 'CREDENTIAL_REQUIRED',
 | |
| 								id: '1384574d-a912-4b81-8601-c7b1c4085df1',
 | |
| 							}
 | |
| 						})
 | |
| 					};
 | |
| 				}
 | |
| 			} else {
 | |
| 				return { 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);
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	test('api error', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			return {
 | |
| 				status: 500,
 | |
| 				body: 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',
 | |
| 					},
 | |
| 				})
 | |
| 			};
 | |
| 		});
 | |
| 
 | |
| 		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');
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	test('network error', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockAbort();
 | |
| 
 | |
| 		try {
 | |
| 			const cli = new APIClient({
 | |
| 				origin: 'https://misskey.test',
 | |
| 				credential: 'TOKEN',
 | |
| 			});
 | |
| 
 | |
| 			await cli.request('i');
 | |
| 		} catch (e) {
 | |
| 			expect(isAPIError(e)).toEqual(false);
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	test('json parse error', async () => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async (req) => {
 | |
| 			return {
 | |
| 				status: 500,
 | |
| 				body: '<html>I AM NOT JSON</html>'
 | |
| 			};
 | |
| 		});
 | |
| 
 | |
| 		try {
 | |
| 			const cli = new APIClient({
 | |
| 				origin: 'https://misskey.test',
 | |
| 				credential: 'TOKEN',
 | |
| 			});
 | |
| 
 | |
| 			await cli.request('i');
 | |
| 		} catch (e) {
 | |
| 			expect(isAPIError(e)).toEqual(false);
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	test('admin/roles/create の型が合う', async() => {
 | |
| 		fetchMock.resetMocks();
 | |
| 		fetchMock.mockResponse(async () => {
 | |
| 			return {
 | |
| 				// 本来返すべき値は`Role`型だが、テストなのでお茶を濁す
 | |
| 				status: 200,
 | |
| 				body: '{}'
 | |
| 			};
 | |
| 		});
 | |
| 
 | |
| 		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',
 | |
| 		});
 | |
| 	})
 | |
| });
 |