@hey-api/json-schema-ref-parser 1.3.0 → 1.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@hey-api/json-schema-ref-parser",
3
- "version": "1.3.0",
3
+ "version": "1.3.1",
4
4
  "description": "Parse, Resolve, and Dereference JSON Schema $ref pointers",
5
5
  "keywords": [
6
6
  "$ref",
@@ -0,0 +1,11 @@
1
+ {
2
+ "schemas": {
3
+ "Foo": {
4
+ "$ref": "#/schemas/Bar"
5
+ },
6
+ "Bar": {
7
+ "description": "ok",
8
+ "$ref": "#/schemas/Foo"
9
+ }
10
+ }
11
+ }
@@ -0,0 +1,78 @@
1
+ {
2
+ "openapi": "3.0.0",
3
+ "info": {
4
+ "title": "Test API",
5
+ "version": "1.0.0"
6
+ },
7
+ "paths": {
8
+ "/resolution": {
9
+ "get": {
10
+ "summary": "Get resolution step",
11
+ "responses": {
12
+ "200": {
13
+ "description": "Success",
14
+ "content": {
15
+ "application/json": {
16
+ "schema": {
17
+ "$ref": "#/components/schemas/ResolutionStep"
18
+ }
19
+ }
20
+ }
21
+ }
22
+ }
23
+ }
24
+ },
25
+ "/action": {
26
+ "get": {
27
+ "summary": "Get action info",
28
+ "responses": {
29
+ "200": {
30
+ "description": "Success",
31
+ "content": {
32
+ "application/json": {
33
+ "schema": {
34
+ "$ref": "#/components/schemas/ActionInfo"
35
+ }
36
+ }
37
+ }
38
+ }
39
+ }
40
+ }
41
+ }
42
+ },
43
+ "components": {
44
+ "schemas": {
45
+ "ActionInfo": {
46
+ "type": "object",
47
+ "properties": {
48
+ "ActionId": {
49
+ "type": "string"
50
+ }
51
+ }
52
+ },
53
+ "ResolutionStep": {
54
+ "type": "object",
55
+ "properties": {
56
+ "ResolutionType": {
57
+ "oneOf": [
58
+ {
59
+ "$ref": "#/components/schemas/ResolutionType"
60
+ }
61
+ ]
62
+ },
63
+ "ActionName": {
64
+ "type": "string"
65
+ }
66
+ }
67
+ },
68
+ "ResolutionType": {
69
+ "type": "string",
70
+ "enum": [
71
+ "ContactVendor",
72
+ "ResetToDefaults",
73
+ "RetryOperation"
74
+ ]
75
+ }
76
+ }
77
+ }
78
+ }
@@ -0,0 +1,48 @@
1
+ {
2
+ "paths": {
3
+ "/test1/{pathId}": {
4
+ "get": {
5
+ "summary": "First endpoint using the same pathId schema",
6
+ "parameters": [
7
+ {
8
+ "$ref": "#/components/parameters/pathId"
9
+ }
10
+ ],
11
+ "responses": {
12
+ "200": {
13
+ "description": "Test 1 response"
14
+ }
15
+ }
16
+ }
17
+ },
18
+ "/test2/{pathId}": {
19
+ "get": {
20
+ "summary": "Second endpoint using the same pathId schema",
21
+ "parameters": [
22
+ {
23
+ "$ref": "#/components/parameters/pathId"
24
+ }
25
+ ],
26
+ "responses": {
27
+ "200": {
28
+ "description": "Test 2 response"
29
+ }
30
+ }
31
+ }
32
+ }
33
+ },
34
+ "components": {
35
+ "parameters": {
36
+ "pathId": {
37
+ "name": "pathId",
38
+ "in": "path",
39
+ "required": true,
40
+ "schema": {
41
+ "type": "string",
42
+ "format": "uuid",
43
+ "description": "Unique identifier for the path"
44
+ }
45
+ }
46
+ }
47
+ }
48
+ }
@@ -0,0 +1,87 @@
1
+ {
2
+ "openapi": "3.0.0",
3
+ "info": {
4
+ "title": "Redfish-like API",
5
+ "version": "1.0.0",
6
+ "description": "Test API simulating Redfish structure with versioned schemas"
7
+ },
8
+ "paths": {
9
+ "/redfish/v1/Systems": {
10
+ "get": {
11
+ "summary": "Get Systems",
12
+ "responses": {
13
+ "200": {
14
+ "description": "Success",
15
+ "content": {
16
+ "application/json": {
17
+ "schema": {
18
+ "$ref": "#/components/schemas/ResolutionStep_v1_0_1_ResolutionStep"
19
+ }
20
+ }
21
+ }
22
+ },
23
+ "default": {
24
+ "description": "Error"
25
+ }
26
+ }
27
+ }
28
+ },
29
+ "/redfish/v1/Actions": {
30
+ "post": {
31
+ "summary": "Submit Action",
32
+ "responses": {
33
+ "200": {
34
+ "description": "Success",
35
+ "content": {
36
+ "application/json": {
37
+ "schema": {
38
+ "$ref": "#/components/schemas/ResolutionStep_v1_0_1_ActionParameters"
39
+ }
40
+ }
41
+ }
42
+ }
43
+ }
44
+ }
45
+ }
46
+ },
47
+ "components": {
48
+ "schemas": {
49
+ "ResolutionStep_v1_0_1_ActionParameters": {
50
+ "type": "object",
51
+ "properties": {
52
+ "ActionId": {
53
+ "type": "string"
54
+ },
55
+ "ActionType": {
56
+ "$ref": "#/components/schemas/ResolutionStep_v1_0_1_ResolutionType"
57
+ }
58
+ }
59
+ },
60
+ "ResolutionStep_v1_0_1_ResolutionStep": {
61
+ "type": "object",
62
+ "properties": {
63
+ "ResolutionType": {
64
+ "oneOf": [
65
+ {
66
+ "$ref": "#/components/schemas/ResolutionStep_v1_0_1_ResolutionType"
67
+ }
68
+ ]
69
+ },
70
+ "ActionName": {
71
+ "type": "string",
72
+ "description": "Name of the action"
73
+ }
74
+ }
75
+ },
76
+ "ResolutionStep_v1_0_1_ResolutionType": {
77
+ "type": "string",
78
+ "enum": [
79
+ "ContactVendor",
80
+ "ResetToDefaults",
81
+ "RetryOperation"
82
+ ],
83
+ "description": "Types of resolution actions"
84
+ }
85
+ }
86
+ }
87
+ }
@@ -1,8 +1,38 @@
1
+ import fs from 'node:fs';
1
2
  import path from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
2
4
 
3
5
  import { $RefParser } from '..';
4
6
  import { getSpecsPath } from './utils';
5
7
 
8
+ const __filename = fileURLToPath(import.meta.url);
9
+ const __dirname = path.dirname(__filename);
10
+
11
+ const getSnapshotsPath = () => path.join(__dirname, '__snapshots__');
12
+ const getTempSnapshotsPath = () => path.join(__dirname, '.gen', 'snapshots');
13
+
14
+ /**
15
+ * Helper function to compare a bundled schema with a snapshot file.
16
+ * Handles writing the schema to a temp file and comparing with the snapshot.
17
+ *
18
+ * @param schema - The bundled schema to compare
19
+ * @param snapshotName - The name of the snapshot file (e.g., 'circular-ref-with-description.json')
20
+ */
21
+ const expectBundledSchemaToMatchSnapshot = async (schema: unknown, snapshotName: string) => {
22
+ const outputPath = path.join(getTempSnapshotsPath(), snapshotName);
23
+ const snapshotPath = path.join(getSnapshotsPath(), snapshotName);
24
+
25
+ // Ensure directory exists
26
+ fs.mkdirSync(path.dirname(outputPath), { recursive: true });
27
+
28
+ // Write the bundled result
29
+ const content = JSON.stringify(schema, null, 2);
30
+ fs.writeFileSync(outputPath, content);
31
+
32
+ // Compare with snapshot
33
+ await expect(content).toMatchFileSnapshot(snapshotPath);
34
+ };
35
+
6
36
  describe('bundle', () => {
7
37
  it('handles circular reference with description', async () => {
8
38
  const refParser = new $RefParser();
@@ -12,17 +42,8 @@ describe('bundle', () => {
12
42
  'circular-ref-with-description.json',
13
43
  );
14
44
  const schema = await refParser.bundle({ pathOrUrlOrSchema });
15
- expect(schema).toEqual({
16
- schemas: {
17
- Bar: {
18
- $ref: '#/schemas/Foo',
19
- description: 'ok',
20
- },
21
- Foo: {
22
- $ref: '#/schemas/Bar',
23
- },
24
- },
25
- });
45
+
46
+ await expectBundledSchemaToMatchSnapshot(schema, 'circular-ref-with-description.json');
26
47
  });
27
48
 
28
49
  it('bundles multiple references to the same file correctly', async () => {
@@ -32,28 +53,341 @@ describe('bundle', () => {
32
53
  'json-schema-ref-parser',
33
54
  'multiple-refs.json',
34
55
  );
35
- const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
36
-
37
- // Both parameters should now be $ref to the same internal definition
38
- const firstParam = schema.paths['/test1/{pathId}'].get.parameters[0];
39
- const secondParam = schema.paths['/test2/{pathId}'].get.parameters[0];
40
-
41
- // The $ref should match the output structure in file_context_0
42
- expect(firstParam.$ref).toBe('#/components/parameters/path-parameter_pathId');
43
- expect(secondParam.$ref).toBe('#/components/parameters/path-parameter_pathId');
44
-
45
- // The referenced parameter should exist and match the expected structure
46
- expect(schema.components).toBeDefined();
47
- expect(schema.components.parameters).toBeDefined();
48
- expect(schema.components.parameters['path-parameter_pathId']).toEqual({
49
- in: 'path',
50
- name: 'pathId',
51
- required: true,
52
- schema: {
53
- description: 'Unique identifier for the path',
54
- format: 'uuid',
55
- type: 'string',
56
- },
56
+ const schema = await refParser.bundle({ pathOrUrlOrSchema });
57
+
58
+ await expectBundledSchemaToMatchSnapshot(schema, 'multiple-refs.json');
59
+ });
60
+
61
+ it('hoists sibling schemas from external files', async () => {
62
+ const refParser = new $RefParser();
63
+ const pathOrUrlOrSchema = path.join(
64
+ getSpecsPath(),
65
+ 'json-schema-ref-parser',
66
+ 'main-with-external-siblings.json',
67
+ );
68
+ const schema = await refParser.bundle({ pathOrUrlOrSchema });
69
+
70
+ await expectBundledSchemaToMatchSnapshot(schema, 'main-with-external-siblings.json');
71
+ });
72
+
73
+ it('hoists sibling schemas from YAML files with versioned names (Redfish-like)', async () => {
74
+ const refParser = new $RefParser();
75
+ const pathOrUrlOrSchema = path.join(
76
+ getSpecsPath(),
77
+ 'json-schema-ref-parser',
78
+ 'redfish-like.yaml',
79
+ );
80
+ const schema = await refParser.bundle({ pathOrUrlOrSchema });
81
+
82
+ await expectBundledSchemaToMatchSnapshot(schema, 'redfish-like.json');
83
+ });
84
+
85
+ describe('sibling schema resolution', () => {
86
+ const specsDir = path.join(getSpecsPath(), 'json-schema-ref-parser');
87
+
88
+ const findSchemaByValue = (
89
+ schemas: Record<string, any>,
90
+ predicate: (value: any) => boolean,
91
+ ): [string, any] | undefined => {
92
+ for (const [name, value] of Object.entries(schemas)) {
93
+ if (predicate(value)) {
94
+ return [name, value];
95
+ }
96
+ }
97
+ return undefined;
98
+ };
99
+
100
+ it('hoists sibling schemas through a bare $ref wrapper chain', async () => {
101
+ const refParser = new $RefParser();
102
+ const pathOrUrlOrSchema = path.join(specsDir, 'sibling-schema-root.json');
103
+ const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
104
+
105
+ expect(schema.components).toBeDefined();
106
+ expect(schema.components.schemas).toBeDefined();
107
+
108
+ const schemas = schema.components.schemas;
109
+
110
+ const mainSchema = findSchemaByValue(
111
+ schemas,
112
+ (v) => v.type === 'object' && v.properties?.name,
113
+ );
114
+ expect(mainSchema).toBeDefined();
115
+ const [mainName, mainValue] = mainSchema!;
116
+ expect(mainValue.type).toBe('object');
117
+ expect(mainValue.properties.name).toEqual({ type: 'string' });
118
+
119
+ const enumSchema = findSchemaByValue(
120
+ schemas,
121
+ (v) => Array.isArray(v.enum) && v.enum.includes('active'),
122
+ );
123
+ expect(enumSchema).toBeDefined();
124
+ const [enumName, enumValue] = enumSchema!;
125
+ expect(enumValue.type).toBe('string');
126
+ expect(enumValue.enum).toEqual(['active', 'inactive', 'pending']);
127
+
128
+ // The main schema's status property should reference the hoisted enum
129
+ expect(mainValue.properties.status.$ref).toBe(`#/components/schemas/${enumName}`);
130
+
131
+ // The root path's schema ref should point to the hoisted main schema
132
+ const rootRef = schema.paths['/test'].get.responses['200'].content['application/json'].schema;
133
+ expect(rootRef.$ref).toBe(`#/components/schemas/${mainName}`);
134
+ });
135
+
136
+ it('hoists sibling schemas through an extended $ref wrapper chain', async () => {
137
+ const refParser = new $RefParser();
138
+ const pathOrUrlOrSchema = path.join(specsDir, 'sibling-schema-extended-root.json');
139
+ const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => {});
140
+
141
+ try {
142
+ const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
143
+
144
+ expect(schema.components).toBeDefined();
145
+ expect(schema.components.schemas).toBeDefined();
146
+
147
+ const schemas = schema.components.schemas;
148
+
149
+ // The main schema should be hoisted (with the extra description merged in)
150
+ const mainSchema = findSchemaByValue(
151
+ schemas,
152
+ (v) =>
153
+ v.description === 'Wrapper that extends the versioned schema' ||
154
+ (v.type === 'object' && v.properties?.name),
155
+ );
156
+ expect(mainSchema).toBeDefined();
157
+
158
+ // The sibling enum must also be hoisted (this was the bug — it was lost before the fix)
159
+ const enumSchema = findSchemaByValue(
160
+ schemas,
161
+ (v) => Array.isArray(v.enum) && v.enum.includes('active'),
162
+ );
163
+ expect(enumSchema).toBeDefined();
164
+ const [, enumValue] = enumSchema!;
165
+ expect(enumValue.type).toBe('string');
166
+ expect(enumValue.enum).toEqual(['active', 'inactive', 'pending']);
167
+
168
+ // No "Skipping unresolvable $ref" warnings should have been emitted
169
+ const unresolvableWarnings = warnSpy.mock.calls.filter(
170
+ (args) => typeof args[0] === 'string' && args[0].includes('Skipping unresolvable $ref'),
171
+ );
172
+ expect(unresolvableWarnings).toHaveLength(0);
173
+ } finally {
174
+ warnSpy.mockRestore();
175
+ }
176
+ });
177
+
178
+ it('hoists sibling schemas from a direct reference (no wrapper)', async () => {
179
+ const refParser = new $RefParser();
180
+ const pathOrUrlOrSchema = path.join(specsDir, 'sibling-schema-direct-root.json');
181
+ const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
182
+
183
+ expect(schema.components).toBeDefined();
184
+ expect(schema.components.schemas).toBeDefined();
185
+
186
+ const schemas = schema.components.schemas;
187
+
188
+ const mainSchema = findSchemaByValue(
189
+ schemas,
190
+ (v) => v.type === 'object' && v.properties?.name,
191
+ );
192
+ expect(mainSchema).toBeDefined();
193
+
194
+ const enumSchema = findSchemaByValue(
195
+ schemas,
196
+ (v) => Array.isArray(v.enum) && v.enum.includes('active'),
197
+ );
198
+ expect(enumSchema).toBeDefined();
199
+ const [enumName, enumValue] = enumSchema!;
200
+ expect(enumValue.enum).toEqual(['active', 'inactive', 'pending']);
201
+
202
+ const [, mainValue] = mainSchema!;
203
+ expect(mainValue.properties.status.$ref).toBe(`#/components/schemas/${enumName}`);
204
+ });
205
+
206
+ it('hoists multiple sibling schemas through an extended wrapper', async () => {
207
+ const refParser = new $RefParser();
208
+ const pathOrUrlOrSchema = path.join(specsDir, 'sibling-schema-multi-root.json');
209
+ const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => {});
210
+
211
+ try {
212
+ const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
213
+
214
+ expect(schema.components).toBeDefined();
215
+ expect(schema.components.schemas).toBeDefined();
216
+
217
+ const schemas = schema.components.schemas;
218
+
219
+ const mainSchema = findSchemaByValue(
220
+ schemas,
221
+ (v) => v.type === 'object' && v.properties?.health,
222
+ );
223
+ expect(mainSchema).toBeDefined();
224
+
225
+ const statusEnum = findSchemaByValue(
226
+ schemas,
227
+ (v) => Array.isArray(v.enum) && v.enum.includes('enabled'),
228
+ );
229
+ expect(statusEnum).toBeDefined();
230
+ expect(statusEnum![1].enum).toEqual(['enabled', 'disabled', 'standby']);
231
+
232
+ const healthEnum = findSchemaByValue(
233
+ schemas,
234
+ (v) => Array.isArray(v.enum) && v.enum.includes('ok'),
235
+ );
236
+ expect(healthEnum).toBeDefined();
237
+ expect(healthEnum![1].enum).toEqual(['ok', 'warning', 'critical']);
238
+
239
+ const [, mainValue] = mainSchema!;
240
+ expect(mainValue.properties.status.$ref).toBe(`#/components/schemas/${statusEnum![0]}`);
241
+ expect(mainValue.properties.health.$ref).toBe(`#/components/schemas/${healthEnum![0]}`);
242
+
243
+ const unresolvableWarnings = warnSpy.mock.calls.filter(
244
+ (args) => typeof args[0] === 'string' && args[0].includes('Skipping unresolvable $ref'),
245
+ );
246
+ expect(unresolvableWarnings).toHaveLength(0);
247
+ } finally {
248
+ warnSpy.mockRestore();
249
+ }
250
+ });
251
+
252
+ it('handles multiple external files with same-named sibling schemas', async () => {
253
+ const refParser = new $RefParser();
254
+ const pathOrUrlOrSchema = path.join(specsDir, 'sibling-schema-collision-root.json');
255
+ const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => {});
256
+
257
+ try {
258
+ const schema = (await refParser.bundle({ pathOrUrlOrSchema })) as any;
259
+
260
+ expect(schema.components).toBeDefined();
261
+ expect(schema.components.schemas).toBeDefined();
262
+
263
+ const schemas = schema.components.schemas;
264
+ const schemaNames = Object.keys(schemas);
265
+
266
+ const mainSchemaKey = schemaNames.find((name) => name.includes('MainSchema'));
267
+ const otherSchemaKey = schemaNames.find((name) => name.includes('OtherSchema'));
268
+
269
+ expect(mainSchemaKey).toBeDefined();
270
+ expect(otherSchemaKey).toBeDefined();
271
+
272
+ const statusSchemas = schemaNames.filter((name) => name.includes('Status'));
273
+ expect(statusSchemas.length).toBeGreaterThanOrEqual(2);
274
+
275
+ const statusValues = statusSchemas.map((name) => schemas[name]);
276
+ const stringStatus = statusValues.find((v: any) => v.type === 'string');
277
+ const integerStatus = statusValues.find((v: any) => v.type === 'integer');
278
+
279
+ expect(stringStatus).toBeDefined();
280
+ expect(integerStatus).toBeDefined();
281
+ expect(stringStatus!.enum).toEqual(['active', 'inactive']);
282
+ expect(integerStatus!.enum).toEqual([0, 1, 2]);
283
+
284
+ const mainSchemaValue = schemas[mainSchemaKey!];
285
+ const mainStatusRef = mainSchemaValue.properties.status.$ref;
286
+ expect(mainStatusRef).toMatch(/^#\/components\/schemas\/.*Status/);
287
+
288
+ const referencedStatus = schemas[mainStatusRef.replace('#/components/schemas/', '')];
289
+ expect(referencedStatus).toBeDefined();
290
+ expect(referencedStatus.type).toBe('string');
291
+ expect(referencedStatus.enum).toEqual(['active', 'inactive']);
292
+
293
+ const otherSchemaValue = schemas[otherSchemaKey!];
294
+ const otherStatusRef = otherSchemaValue.properties.code.$ref;
295
+ expect(otherStatusRef).toMatch(/^#\/components\/schemas\/.*Status/);
296
+
297
+ const referencedOtherStatus = schemas[otherStatusRef.replace('#/components/schemas/', '')];
298
+ expect(referencedOtherStatus).toBeDefined();
299
+ expect(referencedOtherStatus.type).toBe('integer');
300
+ expect(referencedOtherStatus.enum).toEqual([0, 1, 2]);
301
+
302
+ const unresolvableWarnings = warnSpy.mock.calls.filter(
303
+ (args) => typeof args[0] === 'string' && args[0].includes('Skipping unresolvable $ref'),
304
+ );
305
+ expect(unresolvableWarnings).toHaveLength(0);
306
+ } finally {
307
+ warnSpy.mockRestore();
308
+ }
309
+ });
310
+ });
311
+
312
+ describe('mergeMany', () => {
313
+ it('merges paths with non-conflicting methods under the same path', async () => {
314
+ const refParser = new $RefParser();
315
+ const spec1 = {
316
+ info: { title: 'Spec 1', version: '1.0.0' },
317
+ paths: {
318
+ '/pet/{petId}': {
319
+ post: {
320
+ operationId: 'updatePetWithForm',
321
+ responses: { '405': { description: 'Invalid input' } },
322
+ },
323
+ },
324
+ },
325
+ swagger: '2.0',
326
+ };
327
+ const spec2 = {
328
+ info: { title: 'Spec 2', version: '1.0.0' },
329
+ paths: {
330
+ '/pet/{petId}': {
331
+ delete: {
332
+ operationId: 'deletePet',
333
+ responses: {
334
+ '400': { description: 'Invalid ID supplied' },
335
+ '404': { description: 'Pet not found' },
336
+ },
337
+ },
338
+ },
339
+ },
340
+ swagger: '2.0',
341
+ };
342
+
343
+ const merged = (await refParser.bundleMany({ pathOrUrlOrSchemas: [spec1, spec2] })) as any;
344
+
345
+ // Both methods should be under the same path (no prefix added)
346
+ expect(merged.paths['/pet/{petId}']).toBeDefined();
347
+ expect(merged.paths['/pet/{petId}'].post).toBeDefined();
348
+ expect(merged.paths['/pet/{petId}'].delete).toBeDefined();
349
+
350
+ // No prefixed path should be created
351
+ const pathKeys = Object.keys(merged.paths);
352
+ expect(pathKeys).toHaveLength(1);
353
+ });
354
+
355
+ it('adds prefix to path when HTTP methods conflict', async () => {
356
+ const refParser = new $RefParser();
357
+ const spec1 = {
358
+ info: { title: 'Spec 1', version: '1.0.0' },
359
+ paths: {
360
+ '/pet/{petId}': {
361
+ get: {
362
+ operationId: 'getPetById',
363
+ responses: { '200': { description: 'OK' } },
364
+ },
365
+ },
366
+ },
367
+ swagger: '2.0',
368
+ };
369
+ const spec2 = {
370
+ info: { title: 'Spec 2', version: '1.0.0' },
371
+ paths: {
372
+ '/pet/{petId}': {
373
+ get: {
374
+ operationId: 'getPet',
375
+ responses: { '200': { description: 'Success' } },
376
+ },
377
+ },
378
+ },
379
+ swagger: '2.0',
380
+ };
381
+
382
+ const merged = (await refParser.bundleMany({ pathOrUrlOrSchemas: [spec1, spec2] })) as any;
383
+
384
+ // The conflicting path should be prefixed
385
+ const pathKeys = Object.keys(merged.paths);
386
+ expect(pathKeys).toHaveLength(2);
387
+ expect(merged.paths['/pet/{petId}']).toBeDefined();
388
+ const prefixedKey = pathKeys.find((k) => k !== '/pet/{petId}');
389
+ expect(prefixedKey).toBeDefined();
390
+ expect(merged.paths[prefixedKey!].get).toBeDefined();
57
391
  });
58
392
  });
59
393
  });