Skip to content

Commit c72d4ce

Browse files
authored
Apply OpenAPI overlay actions (#150)
* Add OpenAPI overlay option * Added JsonPath-plus library * Updated types * Added overlay metrics * Updated readme
1 parent ee6c4a3 commit c72d4ce

20 files changed

+1035
-9
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
## unreleased
22

3+
## [1.25.0] - 2025-01-01
4+
5+
- CLI - Added option to apply OpenAPI overlay actions
6+
37
## [1.24.2] - 2024-10-07
48

59
- fix: scalar $ref: >-

bin/__snapshots__/cli.test.js.snap

+32
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,7 @@ Options:
8686
-k, --casingFile <casingFile> the file to specify casing rules
8787
-f, --filterFile <filterFile> the file to specify filter rules
8888
-g, --generateFile <generateFile> the file to specify generate rules
89+
-l, --overlayFile <overlayFile> the file to specify OpenAPI overlay changes
8990
-c, --configFile <configFile> the file with the OpenAPI-format CLI options
9091
--no-sort don't sort the OpenAPI file
9192
--keepComments don't remove the comments from the OpenAPI YAML file (default: false)
@@ -284,6 +285,37 @@ OpenAPI-Format CLI settings:
284285
"
285286
`;
286287
288+
exports[`openapi-format CLI command should use the overlayFile 1`] = `
289+
"================================================================================
290+
OpenAPI-Format CLI settings:
291+
- Overlay file: test/overlay-combi/overlay.yaml
292+
- Input file: test/overlay-combi/input.yaml
293+
================================================================================
294+
✅ OpenAPI formatted successfully
295+
================================================================================
296+
"
297+
`;
298+
299+
exports[`openapi-format CLI command should use the overlayFile with verbose 1`] = `
300+
"================================================================================
301+
OpenAPI-Format CLI settings:
302+
- Overlay file: test/overlay-combi/overlay.yaml
303+
- Input file: test/overlay-combi/input.yaml
304+
================================================================================
305+
OpenAPI Overlay actions summary:
306+
- Total actions: 4
307+
- Applied actions: 3
308+
- Unused actions: 1
309+
================================================================================
310+
Unused overlay actions:
311+
- Target: $.server[*]
312+
Type: remove
313+
================================================================================
314+
✅ OpenAPI formatted successfully
315+
================================================================================
316+
"
317+
`;
318+
287319
exports[`openapi-format CLI command should use the rename 1`] = `
288320
"================================================================================
289321
OpenAPI-Format CLI settings:

bin/cli.js

+53
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ program
2121
.option('-k, --casingFile <casingFile>', 'the file to specify casing rules')
2222
.option('-f, --filterFile <filterFile>', 'the file to specify filter rules')
2323
.option('-g, --generateFile <generateFile>', 'the file to specify generate rules')
24+
.option('-l, --overlayFile <overlayFile>', 'the file to specify OpenAPI overlay changes')
2425
.option('-c, --configFile <configFile>', 'the file with the OpenAPI-format CLI options')
2526
.option('--no-sort', `don't sort the OpenAPI file`)
2627
.option('--keepComments', `don't remove the comments from the OpenAPI YAML file`, false)
@@ -198,6 +199,22 @@ async function run(oaFile, options) {
198199
}
199200
}
200201

202+
// Set OpenAPI overlay actions
203+
if (options && options.overlayFile) {
204+
infoOut(`- Overlay file:\t\t${options.overlayFile}`); // LOG - Casing file
205+
try {
206+
let overlayOptions = {overlaySet: {}};
207+
overlayOptions.overlaySet = await openapiFormat.parseFile(options.overlayFile);
208+
options = Object.assign({}, options, overlayOptions);
209+
} catch (err) {
210+
console.error('\x1b[31m', `Overlay file error - no such file or directory "${options.overlayOptions}"`);
211+
if (options.verbose >= 1) {
212+
console.error(err);
213+
}
214+
process.exit(1);
215+
}
216+
}
217+
201218
let resObj = {};
202219
let output = {};
203220
let input = {};
@@ -234,6 +251,19 @@ async function run(oaFile, options) {
234251
resObj = resFilter.data;
235252
}
236253

254+
// Apply OpenAPI overlay actions
255+
if (options.overlaySet) {
256+
const resOverlay = await openapiFormat.openapiOverlay(resObj, options);
257+
if (resOverlay?.resultData &&
258+
(resOverlay.resultData.unusedActions || resOverlay.resultData.appliedActions || resOverlay.resultData.totalActions)) {
259+
cliLog.unusedActions = resOverlay.resultData.unusedActions || [];
260+
cliLog.totalUsedActions = resOverlay.resultData.totalUsedActions || 0;
261+
cliLog.totalUnusedActions = resOverlay.resultData.totalUnusedActions || 0;
262+
cliLog.totalActions = resOverlay.resultData.totalActions || 0;
263+
}
264+
resObj = resOverlay.data;
265+
}
266+
237267
// Format & Order OpenAPI document
238268
if (options.sort === true) {
239269
const resFormat = await openapiFormat.openapiSort(resObj, options);
@@ -327,6 +357,29 @@ async function run(oaFile, options) {
327357
}
328358
}
329359

360+
// Show unused components
361+
if (options.overlaySet && (cliLog?.totalActions || cliLog?.appliedActions || cliLog?.unusedActions)) {
362+
// Log summary of actions
363+
logOut(`${consoleLine}`, options.verbose); // LOG - horizontal rule
364+
logOut(`OpenAPI Overlay actions summary:`, options.verbose);
365+
logOut(`- Total actions: \t${cliLog.totalActions}`, options.verbose);
366+
logOut(`- Applied actions: \t${cliLog.totalUsedActions}`, options.verbose);
367+
logOut(`- Unused actions: \t${cliLog.totalUnusedActions}`, options.verbose);
368+
369+
const cliOut = [];
370+
cliLog.unusedActions.forEach(action => {
371+
const description = action.description || 'No description provided';
372+
cliOut.push(`- Target: ${action.target}\n Type: ${action.update ? 'update' : action.remove ? 'remove' : 'unknown'}`);
373+
});
374+
375+
if (cliLog.unusedActions.length > 0) {
376+
// Log unused actions
377+
logOut(`${consoleLine}`, options.verbose); // LOG - horizontal rule
378+
logOut(`Unused overlay actions:`, options.verbose);
379+
logOut(cliOut.join('\n'), options.verbose);
380+
}
381+
}
382+
330383
// Final result
331384
infoOut(`\x1b[32m${consoleLine}\x1b[0m`); // LOG - horizontal rule
332385
infoOut(`\x1b[32m✅ OpenAPI ${outputLogFiltered}formatted successfully\x1b[0m`, 99); // LOG - success message

bin/cli.test.js

+30
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,36 @@ describe('openapi-format CLI command', () => {
191191
expect(sanitize(result.stderr)).toStrictEqual(sanitize(output));
192192
});
193193

194+
it('should use the overlayFile', async () => {
195+
const path = `test/overlay-combi`;
196+
const inputFile = `${path}/input.yaml`;
197+
const outputFile = `${path}/output.yaml`;
198+
const output = await getLocalFile(outputFile);
199+
const setting = `${path}/overlay.yaml`;
200+
201+
let result = await testUtils.cli([inputFile, `--overlayFile ${setting}`, `--no-sort`], '.');
202+
// console.log('result', result)
203+
expect(result.code).toBe(0);
204+
expect(result.stdout).toContain('formatted successfully');
205+
expect(result.stdout).toMatchSnapshot();
206+
expect(sanitize(result.stderr)).toStrictEqual(sanitize(output));
207+
});
208+
209+
it('should use the overlayFile with verbose', async () => {
210+
const path = `test/overlay-combi`;
211+
const inputFile = `${path}/input.yaml`;
212+
const outputFile = `${path}/output.yaml`;
213+
const output = await getLocalFile(outputFile);
214+
const setting = `${path}/overlay.yaml`;
215+
216+
let result = await testUtils.cli([inputFile, `--overlayFile ${setting}`, `--no-sort`, `-v`], '.');
217+
// console.log('result', result)
218+
expect(result.code).toBe(0);
219+
expect(result.stdout).toContain('formatted successfully');
220+
expect(result.stdout).toMatchSnapshot();
221+
expect(sanitize(result.stderr)).toStrictEqual(sanitize(output));
222+
});
223+
194224
it('should bundle reference', async () => {
195225
const path = `test/yaml-ref-quotes`;
196226
const inputFile = `${path}/input.yaml`;

openapi-format.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ const {parseFile, writeFile, stringify, detectFormat, parseString, analyzeOpenAp
3232
const {parseTpl, getOperation} = require('./utils/parseTpl');
3333
const {writePaths, writeComponents, writeSplitOpenAPISpec} = require('./utils/split');
3434
const {dirname, extname} = require('path');
35+
const { openapiOverlay, resolveJsonPath, resolveJsonPathValue} = require('./utils/overlay');
3536

3637
/**
3738
* OpenAPI sort function
@@ -1145,6 +1146,7 @@ module.exports = {
11451146
openapiGenerate: openapiGenerate,
11461147
openapiSort: openapiSort,
11471148
openapiChangeCase: openapiChangeCase,
1149+
openapiOverlay: openapiOverlay,
11481150
openapiSplit: openapiSplit,
11491151
openapiConvertVersion: openapiConvertVersion,
11501152
openapiRename: openapiRename,
@@ -1155,5 +1157,7 @@ module.exports = {
11551157
writeFile: writeFile,
11561158
detectFormat: detectFormat,
11571159
analyzeOpenApi: analyzeOpenApi,
1158-
changeCase: changeCase
1160+
changeCase: changeCase,
1161+
resolveJsonPath: resolveJsonPath,
1162+
resolveJsonPathValue: resolveJsonPathValue,
11591163
};

package-lock.json

+86-6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@
3737
"api-ref-bundler": "^0.4.3",
3838
"case-anything": "2.1.10",
3939
"commander": "^7.2.0",
40+
"jsonpath-plus": "^10.2.0",
4041
"neotraverse": "^0.6.18"
4142
},
4243
"devDependencies": {

0 commit comments

Comments
 (0)