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();
|
|
})
|
|
});
|