UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

85 lines 12.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.invokeBuiltinHooks = invokeBuiltinHooks; const path = require("path"); const toolkit_lib_1 = require("@aws-cdk/toolkit-lib"); const os_1 = require("./os"); const util_1 = require("../../util"); /** * Invoke hooks for the given init template * * Sometimes templates need more complex logic than just replacing tokens. A 'hook' can be * used to do additional processing other than copying files. * * Hooks used to be defined externally to the CLI, by running arbitrarily * substituted shell scripts in the target directory. * * In practice, they're all TypeScript files and all the same, and the dynamism * that the original solution allowed wasn't used at all. Worse, since the CLI * is now bundled the hooks can't even reuse code from the CLI libraries at all * anymore, so all shared code would have to be copy/pasted. * * Bundle hooks as built-ins into the CLI, so they get bundled and can take advantage * of all shared code. */ async function invokeBuiltinHooks(ioHelper, target, context) { switch (target.language) { case 'csharp': if (['app', 'sample-app'].includes(target.templateName)) { return dotnetAddProject(ioHelper, target.targetDirectory, context); } break; case 'fsharp': if (['app', 'sample-app'].includes(target.templateName)) { return dotnetAddProject(ioHelper, target.targetDirectory, context, 'fsproj'); } break; case 'python': // We can't call this file 'requirements.template.txt' because Dependabot needs to be able to find it. // Therefore, keep the in-repo name but still substitute placeholders. await context.substitutePlaceholdersIn('requirements.txt'); break; case 'java': // We can't call this file 'pom.template.xml'... for the same reason as Python above. await context.substitutePlaceholdersIn('pom.xml'); break; case 'javascript': case 'typescript': // See above, but for 'package.json'. await context.substitutePlaceholdersIn('package.json', 'README.md'); } } async function dotnetAddProject(ioHelper, targetDirectory, context, ext = 'csproj') { const pname = context.placeholder('name.PascalCased'); const slnPath = path.join(targetDirectory, 'src', `${pname}.sln`); const csprojPath = path.join(targetDirectory, 'src', pname, `${pname}.${ext}`); // We retry this command a couple of times. It usually never fails, except on CI where // we sometimes see: // // System.IO.IOException: The system cannot open the device or file specified. : 'NuGet-Migrations' // // This error can be caused by lack of permissions on a temporary directory, // but in our case it's intermittent so my guess is it is caused by multiple // invocations of the .NET CLI running in parallel, and trampling on each // other creating a Mutex. There is no fix, and it is annoyingly breaking our // CI regularly. Retry a couple of times to increase reliability. // // - https://github.com/dotnet/sdk/issues/43750 // - https://github.com/dotnet/runtime/issues/80619 // - https://github.com/dotnet/runtime/issues/91987 const MAX_ATTEMPTS = 3; for (let attempt = 1;; attempt++) { try { await (0, os_1.shell)(ioHelper, ['dotnet', 'sln', slnPath, 'add', csprojPath]); return; } catch (e) { if (attempt === MAX_ATTEMPTS) { throw new toolkit_lib_1.ToolkitError(`Could not add project ${pname}.${ext} to solution ${pname}.sln. ${(0, util_1.formatErrorMessage)(e)}`); } // Sleep for a bit then try again await new Promise(resolve => setTimeout(resolve, 1000)); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"init-hooks.js","sourceRoot":"","sources":["init-hooks.ts"],"names":[],"mappings":";;AAkDA,gDA8BC;AAhFD,6BAA6B;AAC7B,sDAAoD;AACpD,6BAA6B;AAE7B,qCAAgD;AA6BhD;;;;;;;;;;;;;;;;GAgBG;AACI,KAAK,UAAU,kBAAkB,CAAC,QAAkB,EAAE,MAAkB,EAAE,OAAoB;IACnG,QAAQ,MAAM,CAAC,QAAQ,EAAE,CAAC;QACxB,KAAK,QAAQ;YACX,IAAI,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE,CAAC;gBACxD,OAAO,gBAAgB,CAAC,QAAQ,EAAE,MAAM,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;YACrE,CAAC;YACD,MAAM;QAER,KAAK,QAAQ;YACX,IAAI,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE,CAAC;gBACxD,OAAO,gBAAgB,CAAC,QAAQ,EAAE,MAAM,CAAC,eAAe,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;YAC/E,CAAC;YACD,MAAM;QAER,KAAK,QAAQ;YACX,sGAAsG;YACtG,sEAAsE;YACtE,MAAM,OAAO,CAAC,wBAAwB,CAAC,kBAAkB,CAAC,CAAC;YAC3D,MAAM;QAER,KAAK,MAAM;YACT,qFAAqF;YACrF,MAAM,OAAO,CAAC,wBAAwB,CAAC,SAAS,CAAC,CAAC;YAClD,MAAM;QAER,KAAK,YAAY,CAAC;QAClB,KAAK,YAAY;YACf,qCAAqC;YACrC,MAAM,OAAO,CAAC,wBAAwB,CAAC,cAAc,EAAE,WAAW,CAAC,CAAC;IACxE,CAAC;AACH,CAAC;AAED,KAAK,UAAU,gBAAgB,CAAC,QAAkB,EAAE,eAAuB,EAAE,OAAoB,EAAE,GAAG,GAAG,QAAQ;IAC/G,MAAM,KAAK,GAAG,OAAO,CAAC,WAAW,CAAC,kBAAkB,CAAC,CAAC;IACtD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,KAAK,EAAE,GAAG,KAAK,MAAM,CAAC,CAAC;IAClE,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,KAAK,IAAI,GAAG,EAAE,CAAC,CAAC;IAE/E,sFAAsF;IACtF,oBAAoB;IACpB,EAAE;IACF,qGAAqG;IACrG,EAAE;IACF,4EAA4E;IAC5E,4EAA4E;IAC5E,yEAAyE;IACzE,6EAA6E;IAC7E,iEAAiE;IACjE,EAAE;IACF,+CAA+C;IAC/C,mDAAmD;IACnD,mDAAmD;IACnD,MAAM,YAAY,GAAG,CAAC,CAAC;IACvB,KAAK,IAAI,OAAO,GAAG,CAAC,GAAI,OAAO,EAAE,EAAE,CAAC;QAClC,IAAI,CAAC;YACH,MAAM,IAAA,UAAK,EAAC,QAAQ,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC;YACrE,OAAO;QACT,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,IAAI,OAAO,KAAK,YAAY,EAAE,CAAC;gBAC7B,MAAM,IAAI,0BAAY,CAAC,yBAAyB,KAAK,IAAI,GAAG,gBAAgB,KAAK,SAAS,IAAA,yBAAkB,EAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACrH,CAAC;YAED,iCAAiC;YACjC,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC;AACH,CAAC","sourcesContent":["import * as path from 'path';\nimport { ToolkitError } from '@aws-cdk/toolkit-lib';\nimport { shell } from './os';\nimport type { IoHelper } from '../../api-private';\nimport { formatErrorMessage } from '../../util';\n\nexport type SubstitutePlaceholders = (...fileNames: string[]) => Promise<void>;\n\n/**\n * Helpers passed to hook functions\n */\nexport interface HookContext {\n  /**\n   * Callback function to replace placeholders on arbitrary files\n   *\n   * This makes token substitution available to non-`.template` files.\n   */\n  readonly substitutePlaceholdersIn: SubstitutePlaceholders;\n\n  /**\n   * Return a single placeholder\n   */\n  placeholder(name: string): string;\n}\n\nexport type InvokeHook = (targetDirectory: string, context: HookContext) => Promise<void>;\n\nexport interface HookTarget {\n  readonly targetDirectory: string;\n  readonly templateName: string;\n  readonly language: string;\n}\n\n/**\n * Invoke hooks for the given init template\n *\n * Sometimes templates need more complex logic than just replacing tokens. A 'hook' can be\n * used to do additional processing other than copying files.\n *\n * Hooks used to be defined externally to the CLI, by running arbitrarily\n * substituted shell scripts in the target directory.\n *\n * In practice, they're all TypeScript files and all the same, and the dynamism\n * that the original solution allowed wasn't used at all. Worse, since the CLI\n * is now bundled the hooks can't even reuse code from the CLI libraries at all\n * anymore, so all shared code would have to be copy/pasted.\n *\n * Bundle hooks as built-ins into the CLI, so they get bundled and can take advantage\n * of all shared code.\n */\nexport async function invokeBuiltinHooks(ioHelper: IoHelper, target: HookTarget, context: HookContext) {\n  switch (target.language) {\n    case 'csharp':\n      if (['app', 'sample-app'].includes(target.templateName)) {\n        return dotnetAddProject(ioHelper, target.targetDirectory, context);\n      }\n      break;\n\n    case 'fsharp':\n      if (['app', 'sample-app'].includes(target.templateName)) {\n        return dotnetAddProject(ioHelper, target.targetDirectory, context, 'fsproj');\n      }\n      break;\n\n    case 'python':\n      // We can't call this file 'requirements.template.txt' because Dependabot needs to be able to find it.\n      // Therefore, keep the in-repo name but still substitute placeholders.\n      await context.substitutePlaceholdersIn('requirements.txt');\n      break;\n\n    case 'java':\n      // We can't call this file 'pom.template.xml'... for the same reason as Python above.\n      await context.substitutePlaceholdersIn('pom.xml');\n      break;\n\n    case 'javascript':\n    case 'typescript':\n      // See above, but for 'package.json'.\n      await context.substitutePlaceholdersIn('package.json', 'README.md');\n  }\n}\n\nasync function dotnetAddProject(ioHelper: IoHelper, targetDirectory: string, context: HookContext, ext = 'csproj') {\n  const pname = context.placeholder('name.PascalCased');\n  const slnPath = path.join(targetDirectory, 'src', `${pname}.sln`);\n  const csprojPath = path.join(targetDirectory, 'src', pname, `${pname}.${ext}`);\n\n  // We retry this command a couple of times. It usually never fails, except on CI where\n  // we sometimes see:\n  //\n  //   System.IO.IOException: The system cannot open the device or file specified. : 'NuGet-Migrations'\n  //\n  // This error can be caused by lack of permissions on a temporary directory,\n  // but in our case it's intermittent so my guess is it is caused by multiple\n  // invocations of the .NET CLI running in parallel, and trampling on each\n  // other creating a Mutex. There is no fix, and it is annoyingly breaking our\n  // CI regularly. Retry a couple of times to increase reliability.\n  //\n  // - https://github.com/dotnet/sdk/issues/43750\n  // - https://github.com/dotnet/runtime/issues/80619\n  // - https://github.com/dotnet/runtime/issues/91987\n  const MAX_ATTEMPTS = 3;\n  for (let attempt = 1; ; attempt++) {\n    try {\n      await shell(ioHelper, ['dotnet', 'sln', slnPath, 'add', csprojPath]);\n      return;\n    } catch (e: any) {\n      if (attempt === MAX_ATTEMPTS) {\n        throw new ToolkitError(`Could not add project ${pname}.${ext} to solution ${pname}.sln. ${formatErrorMessage(e)}`);\n      }\n\n      // Sleep for a bit then try again\n      await new Promise(resolve => setTimeout(resolve, 1000));\n    }\n  }\n}\n"]}