diff --git a/src/common/generatorResolver.js b/src/common/generatorResolver.js index 0e70eb8..27b3663 100644 --- a/src/common/generatorResolver.js +++ b/src/common/generatorResolver.js @@ -1,115 +1,75 @@ const fs = require("fs"); const path = require("path"); const os = require("os"); -const shell = require("shelljs"); -const URL = require("url").URL; - const log = require("./log"); +const { isUrl } = require("./util"); +const { gitClone, installPackage, installDependencies } = + require("./shell").shellWithOptions(log.shellOptions); -let npmPackage; -let temporaryFolder; - -function isUrl(maybeUrl) { - try { - new URL(maybeUrl); - return true; - } catch (err) { - return false; - } -} - -function getGenerator(generatorPathOrUrl) { - let generatorPath = path.resolve(generatorPathOrUrl); +function validateGenerator(generatorPath) { if (!fs.existsSync(generatorPath)) { - if (isUrl(generatorPathOrUrl)) { - generatorPath = cloneGenerator(generatorPathOrUrl); - } else { - generatorPath = installGeneratorFromNPM(generatorPathOrUrl); - } + throw new Error( + `Generator path ${generatorPath} does not exist, check that the path points to a valid generator` + ); } - return generatorPath; -} - -function cloneGenerator(generatorPathOrUrl) { - // if the generator is specified as a git URL, clone it into a temporary directory - if (!generatorPathOrUrl.endsWith(".git")) { + if (!fs.existsSync(`${generatorPath}/template`)) { throw new Error( - `Generator URL ${generatorPathOrUrl} does not end with ".git", check that the URL points to a valid generator` + `Generator path ${generatorPath} does not contain a template folder, check that the path points to a valid generator` ); } - temporaryFolder = fs.mkdtempSync(path.join(os.tmpdir(), "generator")); - log.verbose( - `Cloning generator from ${generatorPathOrUrl} to ${temporaryFolder}` - ); - shell.exec( - `git clone ${generatorPathOrUrl} ${temporaryFolder}`, - log.shellOptions - ); - const currentPath = process.cwd(); - shell.cd(temporaryFolder, log.shellOptions); - installGeneratorDependencies(); - shell.cd(currentPath, log.shellOptions); - return temporaryFolder; } -function installGeneratorFromNPM(generatorPathOrUrl) { - log.verbose(`Checking if npm package ${generatorPathOrUrl} is installed`); - const currentPath = process.cwd(); - shell.cd(__dirname, log.shellOptions); - if ( - !shell - .exec(`npm list --depth=0`, log.shellOptions) - .stdout.match( - new RegExp(`^.*${generatorPathOrUrl}@\\d+\\.\\d+\\.\\d+$`, "m") - ) - ) { - npmPackage = generatorPathOrUrl; +function getGenerator(generatorPathOrUrl) { + // if this is a folder, and contains a template sub-folder, assume that the generator + // has been supplied as a path + if (fs.existsSync(generatorPathOrUrl)) { + validateGenerator(generatorPathOrUrl); + return { + path: generatorPathOrUrl, + dispose: () => {}, + }; + } + + // if the generator is specified as a git URL, clone it into a temporary directory + if (isUrl(generatorPathOrUrl)) { + const temporaryFolder = fs.mkdtempSync(path.join(os.tmpdir(), "generator")); log.verbose( - `npm package ${generatorPathOrUrl} doesn't exist, installing package` + `Cloning generator from ${generatorPathOrUrl} to ${temporaryFolder}` ); - if ( - shell.exec(`npm install ${generatorPathOrUrl}`, log.shellOptions).code !== - 0 - ) { - throw new Error( - `No local generator or npm package found using '${generatorPathOrUrl}', check that it points to a local generator or npm package` - ); - } + gitClone(generatorPathOrUrl, temporaryFolder); + + log.verbose("Installing generator dependencies"); + installDependencies(temporaryFolder); + return { + path: temporaryFolder, + dispose: () => { + log.verbose(`Removing temporary folder ${temporaryFolder}`); + fs.rmSync(temporaryFolder, { recursive: true }); + }, + }; } - const generatorPath = path.resolve( - __dirname, - path.join("..", "node_modules", generatorPathOrUrl) - ); - shell.cd(generatorPath, log.shellOptions); - installGeneratorDependencies(); - shell.cd(currentPath, log.shellOptions); - return generatorPath; -} -function installGeneratorDependencies() { - log.verbose("Installing generator dependencies"); - shell.exec(`npm pkg delete scripts.prepare`, log.shellOptions); // Do not run husky preparation script as it will cause unnecessary errors - shell.exec(`npm install`, log.shellOptions); + // assume that this must be an npm package, installing into a temporary directory + const temporaryFolder = fs.mkdtempSync(path.join(os.tmpdir(), "generator")); + log.verbose(`Installing generator from npm into ${temporaryFolder}`); + installPackage(generatorPathOrUrl, temporaryFolder); + + // NOTE, there is no need to install dependencies, these will automatically be installed + return { + path: path.join(temporaryFolder, "node_modules", generatorPathOrUrl), + dispose: () => { + log.verbose(`Removing temporary folder ${temporaryFolder}`); + fs.rmSync(temporaryFolder, { recursive: true }); + }, + }; } -function cleanup() { - if (temporaryFolder) { - log.verbose(`Removing temporary folder ${temporaryFolder}`); - fs.rmSync(temporaryFolder, { recursive: true }); - temporaryFolder = null; - } - if (npmPackage) { - const currentPath = process.cwd(); - shell.cd(__dirname, log.shellOptions); - log.verbose(`Removing npm package ${npmPackage}`); - shell.exec(`npm uninstall ${npmPackage}`, log.shellOptions); - shell.cd(currentPath, log.shellOptions); - npmPackage = null; - } +function resolveAndValidate(generatorPathOrUrl) { + const generator = getGenerator(generatorPathOrUrl); + // ensure paths are absolute, and validate contents + generator.path = path.resolve(generator.path); + validateGenerator(generator.path); + return generator; } -module.exports = { - isUrl, - getGenerator, - cleanup, -}; +module.exports = { getGenerator: resolveAndValidate }; diff --git a/src/common/shell.js b/src/common/shell.js new file mode 100644 index 0000000..93192d9 --- /dev/null +++ b/src/common/shell.js @@ -0,0 +1,65 @@ +// This module exports a number of functions that are used to execute shell commands. Each function +// ensures that the working directory is restored to its original value after the command has executed. + +const shell = require("shelljs"); + +function gitClone(url, path, shellOptions) { + shell.exec(`git clone ${url} ${path}`, shellOptions); +} + +function listInstalledPackages(path, shellOptions) { + const currentPath = process.cwd(); + shell.cd(path, shellOptions); + const code = shell.exec(`npm list --depth=0`, shellOptions); + shell.cd(currentPath, shellOptions); + return code; +} + +function installPackage(packageName, path, shellOptions) { + const currentPath = process.cwd(); + shell.cd(path, shellOptions); + const code = shell.exec(`npm install ${packageName}`, shellOptions); + shell.cd(currentPath, shellOptions); + return code; +} + +function installDependencies(path, shellOptions) { + const currentPath = process.cwd(); + shell.cd(path, shellOptions); + // Do not run husky preparation script as it will cause unnecessary errors + shell.exec(`npm pkg delete scripts.prepare`, shellOptions); + shell.exec(`npm install`, shellOptions); + shell.cd(currentPath, shellOptions); +} + +function uninstallPackage(npmPackage, path, shellOptions) { + const currentPath = process.cwd(); + shell.cd(path, shellOptions); + shell.exec(`npm uninstall ${npmPackage}`, shellOptions); + shell.cd(currentPath, shellOptions); +} + +// use partial application to adapt a function to a new signature, with a 'fixed' value +// for the first argument +const partial = + (fn, lastArg) => + (...args) => + fn(...args, lastArg); + +const exportedFns = { + gitClone, + listInstalledPackages, + installPackage, + installDependencies, + uninstallPackage, +}; + +module.exports = { + ...exportedFns, + // export a version of each function that has the shellOptions argument pre-filled + shellWithOptions: (options) => + Object.keys(exportedFns).reduce((acc, key) => { + acc[key] = partial(exportedFns[key], options); + return acc; + }, {}), +}; diff --git a/src/common/util.js b/src/common/util.js new file mode 100644 index 0000000..fc11388 --- /dev/null +++ b/src/common/util.js @@ -0,0 +1,14 @@ +const URL = require("url").URL; + +function isUrl(maybeUrl) { + try { + new URL(maybeUrl); + return true; + } catch (err) { + return false; + } +} + +module.exports = { + isUrl, +}; diff --git a/src/forge/index.js b/src/forge/index.js index 62e7855..6603cc7 100644 --- a/src/forge/index.js +++ b/src/forge/index.js @@ -41,7 +41,7 @@ const forgeCommand = function (program) { log.setLogLevel(options.logLevel); const generatorPath = generatorResolver.getGenerator(generatorPathOrUrl); - const configFile = path.join(generatorPath, "config.json"); + const configFile = path.join(generatorPath.path, "config.json"); // re-configure the command to generate the API command.allowUnknownOption(false).action(async (_, __, options) => { diff --git a/src/generate.js b/src/generate.js index 02be1b3..cb90921 100644 --- a/src/generate.js +++ b/src/generate.js @@ -12,6 +12,7 @@ const { parse } = require("yaml"); const generatorResolver = require("./common/generatorResolver"); const helpers = require("./helpers"); +const { isUrl } = require("./common/util"); const log = require("./common/log"); const transformers = require("./transformers"); const SwaggerParser = require("@apidevtools/swagger-parser"); @@ -25,7 +26,7 @@ Object.keys(helpers).forEach((helperName) => { async function loadSchema(schemaPathOrUrl) { const isYml = schemaPathOrUrl.endsWith(".yml") || schemaPathOrUrl.endsWith(".yaml"); - const schema = generatorResolver.isUrl(schemaPathOrUrl) + const schema = isUrl(schemaPathOrUrl) ? await fetch(schemaPathOrUrl).then((d) => { if (d.status === 200) { return d.text(); @@ -51,20 +52,6 @@ function cloneSchema(schema) { return JSON.parse(JSON.stringify(schema)); } -function validateGenerator(generatorPath) { - if (!fs.existsSync(generatorPath)) { - throw new Error( - `Generator path ${generatorPath} does not exist, check that the path points to a valid generator` - ); - } - - if (!fs.existsSync(`${generatorPath}/template`)) { - throw new Error( - `Generator path ${generatorPath} does not contain a template folder, check that the path points to a valid generator` - ); - } -} - function getFileName(fileName, tagName = "") { let newFileName = fileName.slice(0, fileName.indexOf(".")); if (tagName !== "") newFileName += helpers.capitalizeFirst(tagName); @@ -168,15 +155,14 @@ function getFilesInFolders(basePath, partialPath = "") { async function generate(schemaPathOrUrl, generatorPathOrUrl, options) { log.logTitle(); let exception = null; + let generator = null; let numberOfDiscoveredModels = 0; let numberOfDiscoveredEndpoints = 0; try { log.standard(`Loading generator from '${generatorPathOrUrl}'`); - let generatorPath = generatorResolver.getGenerator(generatorPathOrUrl); - - log.standard("Validating generator"); - validateGenerator(generatorPath); + generator = generatorResolver.getGenerator(generatorPathOrUrl); + const generatorPath = generator.path; // load the OpenAPI schema log.standard(`Loading schema from '${schemaPathOrUrl}'`); @@ -287,7 +273,7 @@ async function generate(schemaPathOrUrl, generatorPathOrUrl, options) { } catch (e) { exception = e; } finally { - generatorResolver.cleanup(); + generator.dispose(); } if (exception === null) { diff --git a/src/generatorOptions/generatorOptions.js b/src/generatorOptions/generatorOptions.js index 4422082..ba70f41 100644 --- a/src/generatorOptions/generatorOptions.js +++ b/src/generatorOptions/generatorOptions.js @@ -3,14 +3,15 @@ const path = require("path"); const { Option, Command } = require("commander"); -const generatorResolver = require("../common/generatorResolver"); +const { getGenerator } = require("../common/generatorResolver"); const generatorOptionsPrefix = "generator."; // generates the help text for the additional options that a generator supports async function generatorOptionsHelp(generatorPathOrUrl) { let optionsHelp = ""; - const generatorPath = generatorResolver.getGenerator(generatorPathOrUrl); + const generator = getGenerator(generatorPathOrUrl); + const generatorPath = generator.path; const configFile = path.join(generatorPath, "config.json"); if (!fs.existsSync(configFile)) { @@ -34,7 +35,7 @@ async function generatorOptionsHelp(generatorPathOrUrl) { optionsHelp += lines.join("\n"); } - generatorResolver.cleanup(); + generator.dispose(); return optionsHelp; } diff --git a/test/common/generatorResolver.test.js b/test/common/generatorResolver.test.js new file mode 100644 index 0000000..d36c637 --- /dev/null +++ b/test/common/generatorResolver.test.js @@ -0,0 +1,67 @@ +const path = require("path"); +const fs = require("fs"); +const { getGenerator } = require("../../src/common/generatorResolver"); +const log = require("../../src/common/log"); +log.setLogLevel("quiet"); + +describe("generatorResolver", () => { + describe("generator specified as a filepath", () => { + it("should return the filepath given", async () => { + const generatorPath = path.join(__dirname, "validGenerator"); + const generator = await getGenerator(generatorPath); + expect(generator.path).toEqual(generatorPath); + }); + + it("should throw an error if the filepath doesn't point to a valid generator", async () => { + // we'll just point to some random directory + const generatorPath = path.join(__dirname); + expect(() => { + getGenerator(generatorPath); + }).toThrow(); + }); + }); + + describe("generator specified as a git repo", () => { + it("should clone the repo into a temporary directory", async () => { + const generator = await getGenerator( + "https://github.com/ScottLogic/openapi-forge-csharp.git" + ); + // validate via the presence of a package.json file + const packageJsonPath = path.join(generator.path, "package.json"); + expect(fs.existsSync(packageJsonPath)).toEqual(true); + const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, "utf8")); + expect(packageJson.name).toEqual("openapi-forge-csharp"); + + // clean up + generator.dispose(); + }); + + it("should cleanup on disposal", async () => { + const generator = await getGenerator( + "https://github.com/ScottLogic/openapi-forge-csharp.git" + ); + generator.dispose(); + expect(fs.existsSync(generator.path)).toEqual(false); + }); + }); + + describe("generator specified as an npm package", () => { + it("should install the package in a temporary folder", async () => { + const generator = await getGenerator("openapi-forge-typescript"); + // validate via the presence of a package.json file + const packageJsonPath = path.join(generator.path, "package.json"); + expect(fs.existsSync(packageJsonPath)).toEqual(true); + const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, "utf8")); + expect(packageJson.name).toEqual("openapi-forge-typescript"); + + // clean up + generator.dispose(); + }); + + it("should cleanup on disposal", async () => { + const generator = await getGenerator("openapi-forge-typescript"); + generator.dispose(); + expect(fs.existsSync(generator.path)).toEqual(false); + }); + }); +}); diff --git a/test/common/validGenerator/template/.gitignore b/test/common/validGenerator/template/.gitignore new file mode 100644 index 0000000..86d0cb2 --- /dev/null +++ b/test/common/validGenerator/template/.gitignore @@ -0,0 +1,4 @@ +# Ignore everything in this directory +* +# Except this file +!.gitignore \ No newline at end of file diff --git a/test/generate.test.js b/test/generate.test.js index 9765d47..5ee48bd 100644 --- a/test/generate.test.js +++ b/test/generate.test.js @@ -2,6 +2,7 @@ const fs = require("fs"); const path = require("path"); const Handlebars = require("handlebars"); +const util = require("../src/common/util"); const generate = require("../src/generate"); const generatorResolver = require("../src/common/generatorResolver"); const minimatch = require("minimatch"); @@ -10,6 +11,7 @@ jest.mock("fs"); jest.mock("path"); jest.mock("handlebars"); jest.mock("minimatch"); +jest.mock("../src/common/util"); jest.mock("../src/common/generatorResolver"); describe("generate", () => { @@ -25,8 +27,11 @@ describe("generate", () => { path.resolve.mockImplementation((path) => path); fs.existsSync.mockReturnValue(true); fs.readFileSync.mockReturnValue(fakeSchema); - generatorResolver.isUrl.mockReturnValue(false); - generatorResolver.getGenerator.mockImplementation((path) => path); + util.isUrl.mockReturnValue(false); + generatorResolver.getGenerator.mockImplementation((path) => ({ + path, + dispose: () => {}, + })); Handlebars.compile.mockReturnValue(() => outCode); const generatorPackage = { diff --git a/test/generatorOptions.test.js b/test/generatorOptions.test.js index bc6c7bd..1f89578 100644 --- a/test/generatorOptions.test.js +++ b/test/generatorOptions.test.js @@ -11,7 +11,10 @@ jest.mock("../src/common/generatorResolver"); describe("generate", () => { beforeAll(() => { - generatorResolver.getGenerator.mockImplementation((path) => path); + generatorResolver.getGenerator.mockImplementation((path) => ({ + path, + dispose: () => {}, + })); }); beforeEach(() => {