diff --git a/README.md b/README.md index 96d9e6d35..518fcdc44 100644 --- a/README.md +++ b/README.md @@ -147,6 +147,7 @@ With `rdme`, you have access to a variety of tools to manage your API definition - [Validation](#validating-an-api-definition) ✅ - [Reduction](#reducing-an-api-definition) 📉 - [Inspection](#inspecting-an-api-definition) 🔍 +- [Conversion](#converting-an-api-definition) ⏩ `rdme` supports [OpenAPI 3.1](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md), [OpenAPI 3.0](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md), and [Swagger 2.x](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/2.0.md). @@ -262,12 +263,12 @@ The command will ask you a couple questions about how you wish to reduce the fil - The input API definition is called `petstore.json` - The file is reduced to only the `/pet/{id}` path and the `GET` and `PUT` methods -- The output file is called `petstore-reduced.json` +- The output file is called `petstore.reduced.json` Here's what the resulting command looks like: ``` -rdme openapi:reduce petstore.json --path /pet/{id} --method get --method put --out petstore-reduced.json +rdme openapi:reduce petstore.json --path /pet/{id} --method get --method put --out petstore.reduced.json ``` As with the `openapi` command, you can also [omit the file path](#omitting-the-file-path). @@ -288,6 +289,22 @@ rdme openapi:inspect [url-or-local-path-to-file] --feature circularRefs --featur As with the `openapi` command, you can also [omit the file path](#omitting-the-file-path). +#### Converting an API definition + + + +You can also convert any Swagger or Postman Collection to an OpenAPI 3.0 definition. + +```sh +rdme openapi:convert [url-or-local-path-to-file] +``` + +Similar to the `openapi` command, you can also [omit the file path](#omitting-the-file-path). + +> **Note:** +> +> All of our OpenAPI commands already do this conversion automatically, but in case you need to utilize this exclusive functionality outside of the context of those, you can. + ### Docs (a.k.a. Guides) 📖 The Markdown files will require YAML front matter with certain ReadMe documentation attributes. Check out [our docs](https://docs.readme.com/docs/rdme#markdown-file-setup) for more info on setting up your front matter. diff --git a/__tests__/__snapshots__/index.test.ts.snap b/__tests__/__snapshots__/index.test.ts.snap index 524461b6b..6a75493cd 100644 --- a/__tests__/__snapshots__/index.test.ts.snap +++ b/__tests__/__snapshots__/index.test.ts.snap @@ -30,6 +30,7 @@ Options Related commands + $ rdme openapi:convert Convert a Swagger or Postman Collection to OpenAPI. $ rdme openapi:inspect Analyze an OpenAPI/Swagger definition for various OpenAPI and ReadMe feature usage. $ rdme openapi:reduce Reduce an OpenAPI definition into a smaller subset. @@ -67,6 +68,7 @@ Options Related commands + $ rdme openapi:convert Convert a Swagger or Postman Collection to OpenAPI. $ rdme openapi:inspect Analyze an OpenAPI/Swagger definition for various OpenAPI and ReadMe feature usage. $ rdme openapi:reduce Reduce an OpenAPI definition into a smaller subset. @@ -104,6 +106,7 @@ Options Related commands + $ rdme openapi:convert Convert a Swagger or Postman Collection to OpenAPI. $ rdme openapi:inspect Analyze an OpenAPI/Swagger definition for various OpenAPI and ReadMe feature usage. $ rdme openapi:reduce Reduce an OpenAPI definition into a smaller subset. diff --git a/__tests__/cmds/openapi/convert.test.ts b/__tests__/cmds/openapi/convert.test.ts new file mode 100644 index 000000000..93f6256ed --- /dev/null +++ b/__tests__/cmds/openapi/convert.test.ts @@ -0,0 +1,81 @@ +import fs from 'fs'; + +import prompts from 'prompts'; + +import OpenAPIConvertCommand from '../../../src/cmds/openapi/convert'; + +const convert = new OpenAPIConvertCommand(); + +const successfulConversion = () => 'Your converted API definition has been saved to output.json!'; + +const testWorkingDir = process.cwd(); + +describe('rdme openapi:convert', () => { + afterEach(() => { + process.chdir(testWorkingDir); + + jest.clearAllMocks(); + }); + + describe('converting', () => { + it.each([ + ['Swagger 2.0', 'json', '2.0'], + ['Swagger 2.0', 'yaml', '2.0'], + ])('should support reducing a %s definition (format: %s)', async (_, format, specVersion) => { + const spec = require.resolve(`@readme/oas-examples/${specVersion}/${format}/petstore-simple.${format}`); + + let reducedSpec; + fs.writeFileSync = jest.fn((fileName, data) => { + reducedSpec = JSON.parse(data as string); + }); + + prompts.inject(['output.json']); + + await expect( + convert.run({ + spec, + }) + ).resolves.toBe(successfulConversion()); + + expect(fs.writeFileSync).toHaveBeenCalledWith('output.json', expect.any(String)); + expect(reducedSpec.tags).toHaveLength(1); + expect(Object.keys(reducedSpec.paths)).toStrictEqual(['/pet/{petId}']); + expect(Object.keys(reducedSpec.paths['/pet/{petId}'])).toStrictEqual(['get', 'post', 'delete']); + }); + + it('should convert with no prompts via opts', async () => { + const spec = 'petstore-simple.json'; + + let reducedSpec; + fs.writeFileSync = jest.fn((fileName, data) => { + reducedSpec = JSON.parse(data as string); + }); + + await expect( + convert.run({ + spec, + workingDirectory: require.resolve(`@readme/oas-examples/2.0/json/${spec}`).replace(spec, ''), + out: 'output.json', + }) + ).resolves.toBe(successfulConversion()); + + expect(fs.writeFileSync).toHaveBeenCalledWith('output.json', expect.any(String)); + expect(Object.keys(reducedSpec.paths)).toStrictEqual(['/pet/{petId}']); + expect(Object.keys(reducedSpec.paths['/pet/{petId}'])).toStrictEqual(['get', 'post', 'delete']); + }); + }); + + describe('error handling', () => { + it.each([['json'], ['yaml']])('should fail if given an OpenAPI 3.0 definition (format: %s)', async format => { + const spec = require.resolve(`@readme/oas-examples/3.0/${format}/petstore.${format}`); + + await expect( + convert.run({ + spec, + }) + ).rejects.toStrictEqual( + new Error("Sorry, this API definition is already an OpenAPI definition and doesn't need to be converted.") + ); + }); + }); +}); diff --git a/__tests__/lib/__snapshots__/commands.test.ts.snap b/__tests__/lib/__snapshots__/commands.test.ts.snap index 68cb4d75b..cd6a6744b 100644 --- a/__tests__/lib/__snapshots__/commands.test.ts.snap +++ b/__tests__/lib/__snapshots__/commands.test.ts.snap @@ -29,6 +29,11 @@ exports[`utils #listByCategory should list commands by category 1`] = ` "hidden": false, "name": "openapi", }, + { + "description": "Convert a Swagger or Postman Collection to OpenAPI.", + "hidden": false, + "name": "openapi:convert", + }, { "description": "Analyze an OpenAPI/Swagger definition for various OpenAPI and ReadMe feature usage.", "hidden": false, diff --git a/src/cmds/index.ts b/src/cmds/index.ts index f4d93b583..074ae5121 100644 --- a/src/cmds/index.ts +++ b/src/cmds/index.ts @@ -12,6 +12,7 @@ import LogoutCommand from './logout'; import OASCommand from './oas'; import OpenCommand from './open'; import OpenAPICommand from './openapi'; +import OpenAPIConvertCommand from './openapi/convert'; import OpenAPIInspectCommand from './openapi/inspect'; import OpenAPIReduceCommand from './openapi/reduce'; import OpenAPIValidateCommand from './openapi/validate'; @@ -47,6 +48,7 @@ const commands = { open: OpenCommand, openapi: OpenAPICommand, + 'openapi:convert': OpenAPIConvertCommand, 'openapi:inspect': OpenAPIInspectCommand, 'openapi:reduce': OpenAPIReduceCommand, 'openapi:validate': OpenAPIValidateCommand, diff --git a/src/cmds/openapi/convert.ts b/src/cmds/openapi/convert.ts new file mode 100644 index 000000000..a063a50c4 --- /dev/null +++ b/src/cmds/openapi/convert.ts @@ -0,0 +1,90 @@ +import type { CommandOptions } from '../../lib/baseCommand'; + +import fs from 'fs'; +import path from 'path'; + +import chalk from 'chalk'; +import prompts from 'prompts'; + +import Command, { CommandCategories } from '../../lib/baseCommand'; +import { checkFilePath } from '../../lib/checkFile'; +import prepareOas from '../../lib/prepareOas'; +import promptTerminal from '../../lib/promptWrapper'; + +export interface Options { + spec?: string; + out?: string; + workingDirectory?: string; +} + +export default class OpenAPIConvertCommand extends Command { + constructor() { + super(); + + this.command = 'openapi:convert'; + this.usage = 'openapi:convert [file|url] [options]'; + this.description = 'Convert a Swagger or Postman Collection to OpenAPI.'; + this.cmdCategory = CommandCategories.APIS; + + this.hiddenArgs = ['spec']; + this.args = [ + { + name: 'spec', + type: String, + defaultOption: true, + }, + { + name: 'out', + type: String, + description: 'Output file path to write converted file to', + }, + { + name: 'workingDirectory', + type: String, + description: 'Working directory (for usage with relative external references)', + }, + ]; + } + + async run(opts: CommandOptions) { + await super.run(opts); + + const { spec, workingDirectory } = opts; + + if (workingDirectory) { + process.chdir(workingDirectory); + } + + const { preparedSpec, specPath, specType } = await prepareOas(spec, 'openapi:convert', { convertToLatest: true }); + const parsedPreparedSpec = JSON.parse(preparedSpec); + + if (specType === 'OpenAPI') { + throw new Error("Sorry, this API definition is already an OpenAPI definition and doesn't need to be converted."); + } + + prompts.override({ + outputPath: opts.out, + }); + + const promptResults = await promptTerminal([ + { + type: 'text', + name: 'outputPath', + message: 'Enter the path to save your converted API definition to:', + initial: () => { + const extension = path.extname(specPath); + return `${path.basename(specPath).split(extension)[0]}.openapi${extension}`; + }, + validate: value => checkFilePath(value), + }, + ]); + + Command.debug(`saving converted spec to ${promptResults.outputPath}`); + + fs.writeFileSync(promptResults.outputPath, JSON.stringify(parsedPreparedSpec, null, 2)); + + Command.debug('converted spec saved'); + + return Promise.resolve(chalk.green(`Your converted API definition has been saved to ${promptResults.outputPath}!`)); + } +} diff --git a/src/cmds/openapi/index.ts b/src/cmds/openapi/index.ts index bfb920174..62b1f71fc 100644 --- a/src/cmds/openapi/index.ts +++ b/src/cmds/openapi/index.ts @@ -135,7 +135,7 @@ export default class OpenAPICommand extends Command { // Reason we're hardcoding in command here is because `swagger` command // relies on this and we don't want to use `swagger` in this function - const { bundledSpec, specPath, specType, specVersion } = await prepareOas(spec, 'openapi'); + const { preparedSpec, specPath, specType, specVersion } = await prepareOas(spec, 'openapi'); if (useSpecVersion) { Command.info( @@ -218,7 +218,7 @@ export default class OpenAPICommand extends Command { }); }; - const registryUUID = await streamSpecToRegistry(bundledSpec); + const registryUUID = await streamSpecToRegistry(preparedSpec); const options: RequestInit = { headers: cleanHeaders( diff --git a/src/cmds/openapi/inspect.ts b/src/cmds/openapi/inspect.ts index 13557c581..9ed88c409 100644 --- a/src/cmds/openapi/inspect.ts +++ b/src/cmds/openapi/inspect.ts @@ -242,9 +242,9 @@ export default class OpenAPIInspectCommand extends Command { process.chdir(workingDirectory); } - const { bundledSpec, definitionVersion } = await prepareOas(spec, 'openapi:inspect', { convertToLatest: true }); + const { preparedSpec, definitionVersion } = await prepareOas(spec, 'openapi:inspect', { convertToLatest: true }); this.definitionVersion = definitionVersion.version; - const parsedBundledSpec = JSON.parse(bundledSpec); + const parsedPreparedSpec = JSON.parse(preparedSpec); const spinner = ora({ ...oraOptions() }); if (features?.length) { @@ -257,7 +257,7 @@ export default class OpenAPIInspectCommand extends Command { spinner.start('Analyzing your API definition for OpenAPI and ReadMe feature usage...'); } - const analysis = await analyzeOas(parsedBundledSpec).catch(err => { + const analysis = await analyzeOas(parsedPreparedSpec).catch(err => { Command.debug(`analyzer err: ${err.message}`); spinner.fail(); throw err; diff --git a/src/cmds/openapi/reduce.ts b/src/cmds/openapi/reduce.ts index de213e972..a8bc5c360 100644 --- a/src/cmds/openapi/reduce.ts +++ b/src/cmds/openapi/reduce.ts @@ -80,8 +80,8 @@ export default class OpenAPIReduceCommand extends Command { process.chdir(workingDirectory); } - const { bundledSpec, specPath, specType } = await prepareOas(spec, 'openapi:reduce'); - const parsedBundledSpec = JSON.parse(bundledSpec); + const { preparedSpec, specPath, specType } = await prepareOas(spec, 'openapi:reduce'); + const parsedPreparedSpec = JSON.parse(preparedSpec); if (specType !== 'OpenAPI') { throw new Error('Sorry, this reducer feature in rdme only supports OpenAPI 3.0+ definitions.'); @@ -117,7 +117,7 @@ export default class OpenAPIReduceCommand extends Command { choices: () => { const tags: string[] = JSONPath({ path: '$..paths[*].tags', - json: parsedBundledSpec, + json: parsedPreparedSpec, resultType: 'value', }).flat(); @@ -133,7 +133,7 @@ export default class OpenAPIReduceCommand extends Command { message: 'Choose which paths to reduce by:', min: 1, choices: () => { - return Object.keys(parsedBundledSpec.paths).map(p => ({ + return Object.keys(parsedPreparedSpec.paths).map(p => ({ title: p, value: p, })); @@ -147,7 +147,7 @@ export default class OpenAPIReduceCommand extends Command { choices: (prev, values) => { const paths: string[] = values.paths; let methods = paths - .map((p: string) => Object.keys(parsedBundledSpec.paths[p] || {})) + .map((p: string) => Object.keys(parsedPreparedSpec.paths[p] || {})) .flat() .filter((method: string) => method.toLowerCase() !== 'parameters'); @@ -173,7 +173,7 @@ export default class OpenAPIReduceCommand extends Command { message: 'Enter the path to save your reduced API definition to:', initial: () => { const extension = path.extname(specPath); - return `${path.basename(specPath).split(extension)[0]}-reduced${extension}`; + return `${path.basename(specPath).split(extension)[0]}.reduced${extension}`; }, validate: value => checkFilePath(value), }, @@ -196,7 +196,7 @@ export default class OpenAPIReduceCommand extends Command { let reducedSpec; try { - reducedSpec = oasReducer(parsedBundledSpec, { + reducedSpec = oasReducer(parsedPreparedSpec, { tags: promptResults.tags || [], paths: (promptResults.paths || []).reduce((acc: Record, p: string) => { acc[p] = promptResults.methods; diff --git a/src/lib/logger.ts b/src/lib/logger.ts index b2e2cdc69..13d496f50 100644 --- a/src/lib/logger.ts +++ b/src/lib/logger.ts @@ -13,9 +13,17 @@ const debugPackage = debugModule(config.get('cli')); /** * Wrapper for debug statements. */ -function debug(input: string) { +// eslint-disable-next-line @typescript-eslint/no-explicit-any +function debug(input: any) { /* istanbul ignore next */ - if (isGHA() && !isTest()) core.debug(`rdme: ${input}`); + if (isGHA() && !isTest()) { + if (typeof input === 'object') { + core.debug(`rdme: ${JSON.stringify(input)}`); + } else { + core.debug(`rdme: ${input}`); + } + } + return debugPackage(input); } diff --git a/src/lib/prepareOas.ts b/src/lib/prepareOas.ts index 4603f9ffd..6b4e324bd 100644 --- a/src/lib/prepareOas.ts +++ b/src/lib/prepareOas.ts @@ -1,3 +1,5 @@ +import type { OASDocument } from 'oas/dist/rmoas.types'; + import chalk from 'chalk'; import OASNormalize, { getAPIDefinitionType } from 'oas-normalize'; import ora from 'ora'; @@ -35,7 +37,7 @@ const capitalizeSpecType = (type: string) => */ export default async function prepareOas( path: string, - command: 'openapi' | 'openapi:inspect' | 'openapi:reduce' | 'openapi:validate', + command: 'openapi' | 'openapi:convert' | 'openapi:inspect' | 'openapi:reduce' | 'openapi:validate', opts: { /** * Optionally convert the supplied or discovered API definition to the latest OpenAPI release. @@ -65,13 +67,13 @@ export default async function prepareOas( const fileFindingSpinner = ora({ text: 'Looking for API definitions...', ...oraOptions() }).start(); - let action: 'inspect' | 'reduce' | 'upload' | 'validate'; + let action: 'convert' | 'inspect' | 'reduce' | 'upload' | 'validate'; switch (command) { case 'openapi': action = 'upload'; break; default: - action = command.split(':')[1] as 'inspect' | 'reduce' | 'validate'; + action = command.split(':')[1] as 'convert' | 'inspect' | 'reduce' | 'validate'; } const jsonAndYamlFiles = readdirRecursive('.', true).filter( @@ -164,7 +166,7 @@ export default async function prepareOas( }); // If we were supplied a Postman collection this will **always** convert it to OpenAPI 3.0. - const api = await oas.validate({ convertToLatest: opts.convertToLatest }).catch((err: Error) => { + const api: OASDocument = await oas.validate({ convertToLatest: opts.convertToLatest }).catch((err: Error) => { spinner.fail(); debug(`raw validation error object: ${JSON.stringify(err)}`); throw err; @@ -180,18 +182,22 @@ export default async function prepareOas( const specVersion: string = api.info.version; debug(`version in spec: ${specVersion}`); - let bundledSpec = ''; + let preparedSpec = ''; if (['openapi', 'openapi:inspect', 'openapi:reduce'].includes(command)) { - bundledSpec = await oas.bundle().then(res => { + preparedSpec = await oas.bundle().then(res => { return JSON.stringify(res); }); debug('spec bundled'); + } else if (command === 'openapi:convert') { + // As `openapi:convert` is purely for converting a spec to OpenAPI we don't need to do any + // bundling work as those'll be handled in other commands. + preparedSpec = JSON.stringify(api); } return { - bundledSpec, + preparedSpec, specPath, specType, /**