-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathindex.test.ts
175 lines (144 loc) · 5.65 KB
/
index.test.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
import type { HarRequest, Request } from '@readme/httpsnippet';
import type { ClientPlugin } from '@readme/httpsnippet/targets';
import type { OASDocument } from 'oas/types';
import { readdirSync } from 'node:fs';
import fs from 'node:fs/promises';
import path from 'node:path';
import { HTTPSnippet, addClientPlugin } from '@readme/httpsnippet';
import readme from '@readme/oas-examples/3.0/json/readme.json' with { type: 'json' };
import toBeAValidOpenAPIDefinition from 'jest-expect-openapi';
import { describe, beforeEach, expect, it } from 'vitest';
import plugin from '../src/index.js';
expect.extend({ toBeAValidOpenAPIDefinition });
const DATASETS_DIR = path.join(__dirname, '__datasets__');
const SNIPPETS = readdirSync(DATASETS_DIR);
export interface SnippetMock {
definition: OASDocument;
har: HarRequest;
}
function getSnippetDataset(snippet): Promise<SnippetMock> {
return import(path.join(DATASETS_DIR, snippet, 'index')).then(r => r.default);
}
describe('httpsnippet-client-api', () => {
beforeEach(() => {
try {
addClientPlugin(plugin as ClientPlugin);
} catch (err) {
if (err.message !== 'The supplied custom target client already exists, please use a different key') {
throw err;
}
}
});
it('should have info', () => {
expect(plugin).toHaveProperty('target', 'node');
expect(plugin.client).toHaveProperty('info');
expect(plugin.client.info).toStrictEqual({
key: 'api',
title: 'API',
link: 'https://npm.im/api',
description: 'Automatic SDK generation from an OpenAPI definition.',
extname: '.js',
installation: 'npx api install "{packageName}"',
});
});
it('should error if no `api` config was supplied', async () => {
const { har } = await getSnippetDataset('petstore');
const snippet = new HTTPSnippet(har);
expect(() => snippet.convert('node', 'api')).toThrow(/must have an `api` config supplied/);
expect(() => snippet.convert('node', 'api', {})).toThrow(/must have an `api` config supplied/);
});
it('should error if no `api.definition` was supplied', async () => {
const { har } = await getSnippetDataset('petstore');
const snippet = new HTTPSnippet(har);
expect(() =>
snippet.convert('node', 'api', {
api: {
registryURI: '@developers/v2.0#17273l2glm9fq4l5',
},
}),
).toThrow(/must have an `api.definition` option supplied/);
});
it('should error if no `api.registryURI` was supplied', async () => {
const { har } = await getSnippetDataset('petstore');
const snippet = new HTTPSnippet(har);
expect(() =>
snippet.convert('node', 'api', {
api: {
definition: readme,
},
}),
).toThrow(/must have an `api.registryURI` option supplied/);
});
// This test should fail because the url in the HAR is missing `/v1` in the path.
it('should error if no matching operation was found in the supplied API definition', () => {
const har = {
httpVersion: 'HTTP/1.1',
method: 'GET',
queryString: [
{ name: 'perPage', value: '10' },
{ name: 'page', value: '1' },
],
url: 'https://dash.readme.com/api/api-specification',
};
const snippet = new HTTPSnippet(har as Request);
expect(() =>
snippet.convert('node', 'api', {
api: {
definition: readme,
registryURI: '@developers/v2.0#17273l2glm9fq4l5',
},
}),
).toThrow(/unable to locate a matching operation/i);
});
describe('snippets', () => {
describe.each(SNIPPETS)('%s', snippet => {
it('should generate the expected snippet', async () => {
const mock = await getSnippetDataset(snippet);
// `OpenAPIParser.validate()` updates the spec that's passed and we just want to validate
// it here so we need to clone the object.
const spec = JSON.parse(JSON.stringify(mock.definition));
await expect(spec).toBeAValidOpenAPIDefinition();
const expected = await fs.readFile(path.join(DATASETS_DIR, snippet, 'output.js'), 'utf-8');
const code = new HTTPSnippet(mock.har).convert('node', 'api', {
api: {
definition: mock.definition,
registryURI: `@${snippet}/v2.0#17273l2glm9fq4l5`,
},
})[0];
expect(`${code}\n`).toStrictEqual(expected);
});
});
describe('custom variable names', () => {
it('should support custom SDK variable names', async () => {
const mock = await getSnippetDataset('petstore');
const code = new HTTPSnippet(mock.har).convert('node', 'api', {
api: {
definition: mock.definition,
identifier: 'developers',
registryURI: '@developers/v2.0#17273l2glm9fq4l5',
},
})[0];
expect(code).toBe(`import developers from '@api/developers';
developers.auth('123');
developers.findPetsByStatus({status: 'available', accept: 'application/xml'})
.then(({ data }) => console.log(data))
.catch(err => console.error(err));`);
});
it('should make an unsafe variable name safe', async () => {
const mock = await getSnippetDataset('petstore');
const code = new HTTPSnippet(mock.har).convert('node', 'api', {
api: {
definition: mock.definition,
identifier: 'metro-transit',
registryURI: '@metro-transit/v2.0#17273l2glm9fq4l5',
},
})[0];
expect(code).toBe(`import metroTransit from '@api/metro-transit';
metroTransit.auth('123');
metroTransit.findPetsByStatus({status: 'available', accept: 'application/xml'})
.then(({ data }) => console.log(data))
.catch(err => console.error(err));`);
});
});
});
});