aws-delivlib 14.1.112 → 14.2.0
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/lib/build-spec.d.ts +1 -0
- package/lib/build-spec.js +1 -1
- package/lib/shellable.d.ts +24 -1
- package/lib/shellable.js +17 -7
- package/package.json +1 -1
package/lib/build-spec.d.ts
CHANGED
package/lib/build-spec.js
CHANGED
|
@@ -167,4 +167,4 @@ function renameKey(xs, orig, rename) {
|
|
|
167
167
|
}
|
|
168
168
|
return ret;
|
|
169
169
|
}
|
|
170
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"build-spec.js","sourceRoot":"","sources":["build-spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { mapValues, noUndefined } from './util';\n\n\nconst MAGIC_ARTIFACT_NAME = 'PRIMARY';\n\n/**\n * Class to model a buildspec version 0.2\n *\n * Artifact handling is a little special: CodeBuild will interpret the\n * 'artifacts' section differently depending on whether there are secondary\n * artifacts or not.\n *\n * If there is only one artifact, the single artifact must go into the top-level\n * 'artifacts' section. If there are multiple artifacts, all of them must go\n * into the 'secondary-artifacts' section. Upon rendering to JSON, the caller\n * must supply the name of the primary artifact (it's determined by\n * the CodePipeline Action that invokes the CodeBuild Project that uses this\n * buildspec).\n *\n * INVARIANT: in-memory, the BuildSpec will treat all artifacts the same (as\n * a bag of secondary artifacts). At the edges (construction or rendering),\n * if there's only a single artifact it will be rendered to the primary\n * artifact.\n */\nexport class BuildSpec {\n  public static literal(struct: BuildSpecStruct) {\n    return new BuildSpec(struct);\n  }\n\n  public static simple(props: SimpleBuildSpecProps) {\n    // We merge the primary artifact into the secondary artifacts under a special key\n    // They will be compacted back together during rendering.\n    const artifactDirectories = Object.assign({},\n      props.additionalArtifactDirectories || {},\n      props.artifactDirectory ? { [MAGIC_ARTIFACT_NAME]: props.artifactDirectory } : {},\n    );\n\n    let artifacts: PrimaryArtifactStruct | undefined;\n    if (Object.keys(artifactDirectories || {}).length > 0) {\n      artifacts = {\n        'secondary-artifacts': mapValues(artifactDirectories!, d => ({\n          'base-directory': d,\n          'files': ['**/*'],\n        })),\n      };\n    }\n\n    return new BuildSpec({\n      version: '0.2',\n      phases: noUndefined({\n        install: props.install !== undefined ? { commands: props.install } : undefined,\n        pre_build: props.preBuild !== undefined ? { commands: props.preBuild } : undefined,\n        build: props.build !== undefined ? { commands: props.build } : undefined,\n      }),\n      artifacts,\n      reports: props.reports,\n    });\n  }\n\n  public static empty() {\n    return new BuildSpec({ version: '0.2' });\n  }\n\n  private constructor(private readonly spec: BuildSpecStruct) {\n  }\n\n  public get additionalArtifactNames(): string[] {\n    return Object.keys(this.spec.artifacts && this.spec.artifacts['secondary-artifacts'] || {}).filter(n => n !== MAGIC_ARTIFACT_NAME);\n  }\n\n  public merge(other: BuildSpec): BuildSpec {\n    return new BuildSpec({\n      'version': '0.2',\n      'run-as': mergeObj(this.spec['run-as'], other.spec['run-as'], equalObjects),\n      'env': mergeObj(this.spec.env, other.spec.env, (a, b) => ({\n        'parameter-store': mergeDict(a['parameter-store'], b['parameter-store'], equalObjects),\n        'variables': mergeDict(a.variables, b.variables, equalObjects),\n      })),\n      'phases': mergeDict(this.spec.phases, other.spec.phases, (a, b) => ({\n        'run-as': mergeObj(this.spec['run-as'], other.spec['run-as'], equalObjects),\n        'commands': mergeList(a.commands, b.commands)!,\n        'finally': mergeList(a.finally, b.finally),\n      })),\n      'artifacts': mergeObj(this.spec.artifacts, other.spec.artifacts, mergeArtifacts),\n      'cache': mergeObj(this.spec.cache, other.spec.cache, (a, b) => ({\n        paths: mergeList(a.paths, b.paths)!,\n      })),\n      'reports': mergeDict(this.spec.reports, other.spec.reports, (a, b) => {\n        throw new Error(`Reports must have unique names, got ${a} and ${b}`);\n      }),\n    });\n\n    function mergeArtifacts(a: PrimaryArtifactStruct, b: PrimaryArtifactStruct): PrimaryArtifactStruct {\n      if (a.files || b.files) {\n        throw new Error('None of the BuildSpecs may have a primary artifact.');\n      }\n\n      const artifacts = Object.assign({}, a['secondary-artifacts'] || {});\n      for (const [k, v] of Object.entries(b['secondary-artifacts'] || {})) {\n        if (k in artifacts) {\n          throw new Error(`There is already an artifact with name ${k}`);\n        }\n        artifacts[k] = v;\n      }\n      return Object.assign({}, a, { 'secondary-artifacts': artifacts });\n    }\n\n\n    function equalObjects(a: string, b: string) {\n      if (a !== b) {\n        throw new Error(`Can't merge two different values for the same key: ${JSON.stringify(a)}, ${JSON.stringify(b)}`);\n      }\n      return b;\n    }\n\n    function mergeObj<T>(a: T | undefined, b: T | undefined, fn: (a: T, b: T) => T): T | undefined {\n      if (a === undefined) { return b; }\n      if (b === undefined) { return a; }\n      return fn(a, b);\n    }\n\n    function mergeDict<T>(as: {[k: string]: T} | undefined, bs: {[k: string]: T} | undefined, fn: (a: T, b: T) => T) {\n      return mergeObj(as, bs, (a, b) => {\n        const ret = Object.assign({}, a);\n        for (const [k, v] of Object.entries(b)) {\n          if (ret[k]) {\n            ret[k] = fn(ret[k], v);\n          } else {\n            ret[k] = v;\n          }\n        }\n        return ret;\n      });\n    }\n\n    function mergeList<T>(as: T[] | undefined, bs: T[] | undefined): T[] | undefined {\n      return mergeObj(as, bs, (a, b) => a.concat(b));\n    }\n  }\n\n  public render(options: BuildSpecRenderOptions = {}): BuildSpecStruct {\n    return Object.assign({}, this.spec, { artifacts: this.renderArtifacts(options) });\n  }\n\n  private renderArtifacts(options: BuildSpecRenderOptions): PrimaryArtifactStruct | undefined {\n    if (!this.spec.artifacts || !this.spec.artifacts['secondary-artifacts']) { return this.spec.artifacts; }\n\n    // Simplify a single \"secondary-artifacts\" to a single primary artifact (regardless of the name)\n    const singleArt = dictSingletonValue(this.spec.artifacts['secondary-artifacts']);\n    if (singleArt) { return singleArt; }\n\n    // Otherwise rename a 'PRIMARY' key if it exists\n    if (MAGIC_ARTIFACT_NAME in this.spec.artifacts['secondary-artifacts']) {\n      if (!options.primaryArtifactName) {\n        throw new Error(`Replacement name for ${MAGIC_ARTIFACT_NAME} artifact not supplied`);\n      }\n\n      return { 'secondary-artifacts': renameKey(this.spec.artifacts['secondary-artifacts'], MAGIC_ARTIFACT_NAME, options.primaryArtifactName) };\n    }\n\n    return this.spec.artifacts;\n  }\n}\n\nexport interface SimpleBuildSpecProps {\n  install?: string[];\n  preBuild?: string[];\n  build?: string[];\n  reports?: {[key: string]: ReportStruct};\n  artifactDirectory?: string;\n\n  /**\n   * Where the directories for each artifact are\n   *\n   * Use special name PRIMARY to refer to the primary artifact. Will be\n   * replaced with the actual artifact name when the build spec is synthesized.\n   */\n  additionalArtifactDirectories?: {[id: string]: string};\n}\n\nexport interface BuildSpecStruct {\n  version: '0.2';\n  'run-as'?: string;\n  env?: EnvStruct;\n  phases?: {[key: string]: PhaseStruct};\n  artifacts?: PrimaryArtifactStruct;\n  cache?: CacheStruct;\n  reports?: {[key: string]: ReportStruct};\n}\n\nexport interface EnvStruct {\n  variables?: {[key: string]: string};\n  'parameter-store'?: {[key: string]: string};\n}\n\nexport interface PhaseStruct {\n  'run-as'?: string;\n  commands: string[];\n  finally?: string[];\n}\n\nexport interface ReportStruct {\n  files?: string[];\n  'base-directory'?: string;\n  'discard-paths'?: 'yes' | 'no';\n  'file-format'?: 'CucumberJson' | 'JunitXml' | 'NunitXml' | 'TestNGXml' | 'VisualStudioTrx';\n}\n\nexport interface ArtifactStruct {\n  files?: string[];\n  name?: string;\n  'base-directory'?: string;\n  'discard-paths'?: 'yes' | 'no';\n}\n\nexport interface PrimaryArtifactStruct extends ArtifactStruct {\n  'secondary-artifacts'?: {[key: string]: ArtifactStruct};\n}\n\nexport interface CacheStruct {\n  paths: string[];\n}\n\nexport interface BuildSpecRenderOptions {\n  /**\n   * Replace PRIMARY artifact name with this\n   *\n   * Cannot use the special term PRIMARY if this is not supplied.\n   *\n   * @default  Cannot use PRIMARY\n   */\n  primaryArtifactName?: string;\n}\n\n/**\n * If the dict is a singleton dict, return the value of the first key, otherwise return undefined\n */\nfunction dictSingletonValue<T>(xs: {[key: string]: T}): T | undefined {\n  const keys = Object.keys(xs);\n  if (keys.length === 1) {\n    return xs[keys[0]];\n  }\n  return undefined;\n}\n\nfunction renameKey<T>(xs: {[key: string]: T}, orig: string, rename: string): {[key: string]: T} {\n  const ret = Object.assign({}, xs);\n  if (orig in ret) {\n    ret[rename] = ret[orig];\n    delete ret[orig];\n  }\n  return ret;\n}\n"]}
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"build-spec.js","sourceRoot":"","sources":["build-spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { mapValues, noUndefined } from './util';\n\n\nconst MAGIC_ARTIFACT_NAME = 'PRIMARY';\n\n/**\n * Class to model a buildspec version 0.2\n *\n * Artifact handling is a little special: CodeBuild will interpret the\n * 'artifacts' section differently depending on whether there are secondary\n * artifacts or not.\n *\n * If there is only one artifact, the single artifact must go into the top-level\n * 'artifacts' section. If there are multiple artifacts, all of them must go\n * into the 'secondary-artifacts' section. Upon rendering to JSON, the caller\n * must supply the name of the primary artifact (it's determined by\n * the CodePipeline Action that invokes the CodeBuild Project that uses this\n * buildspec).\n *\n * INVARIANT: in-memory, the BuildSpec will treat all artifacts the same (as\n * a bag of secondary artifacts). At the edges (construction or rendering),\n * if there's only a single artifact it will be rendered to the primary\n * artifact.\n */\nexport class BuildSpec {\n  public static literal(struct: BuildSpecStruct) {\n    return new BuildSpec(struct);\n  }\n\n  public static simple(props: SimpleBuildSpecProps) {\n    // We merge the primary artifact into the secondary artifacts under a special key\n    // They will be compacted back together during rendering.\n    const artifactDirectories = Object.assign({},\n      props.additionalArtifactDirectories || {},\n      props.artifactDirectory ? { [MAGIC_ARTIFACT_NAME]: props.artifactDirectory } : {},\n    );\n\n    let artifacts: PrimaryArtifactStruct | undefined;\n    if (Object.keys(artifactDirectories || {}).length > 0) {\n      artifacts = {\n        'secondary-artifacts': mapValues(artifactDirectories!, d => ({\n          'base-directory': d,\n          'files': ['**/*'],\n        })),\n      };\n    }\n\n    return new BuildSpec({\n      version: '0.2',\n      phases: noUndefined({\n        install: props.install !== undefined ? { commands: props.install } : undefined,\n        pre_build: props.preBuild !== undefined ? { commands: props.preBuild } : undefined,\n        build: props.build !== undefined ? { commands: props.build } : undefined,\n      }),\n      artifacts,\n      reports: props.reports,\n    });\n  }\n\n  public static empty() {\n    return new BuildSpec({ version: '0.2' });\n  }\n\n  private constructor(private readonly spec: BuildSpecStruct) {\n  }\n\n  public get additionalArtifactNames(): string[] {\n    return Object.keys(this.spec.artifacts && this.spec.artifacts['secondary-artifacts'] || {}).filter(n => n !== MAGIC_ARTIFACT_NAME);\n  }\n\n  public merge(other: BuildSpec): BuildSpec {\n    return new BuildSpec({\n      'version': '0.2',\n      'run-as': mergeObj(this.spec['run-as'], other.spec['run-as'], equalObjects),\n      'env': mergeObj(this.spec.env, other.spec.env, (a, b) => ({\n        'parameter-store': mergeDict(a['parameter-store'], b['parameter-store'], equalObjects),\n        'variables': mergeDict(a.variables, b.variables, equalObjects),\n      })),\n      'phases': mergeDict(this.spec.phases, other.spec.phases, (a, b) => ({\n        'run-as': mergeObj(this.spec['run-as'], other.spec['run-as'], equalObjects),\n        'commands': mergeList(a.commands, b.commands)!,\n        'finally': mergeList(a.finally, b.finally),\n      })),\n      'artifacts': mergeObj(this.spec.artifacts, other.spec.artifacts, mergeArtifacts),\n      'cache': mergeObj(this.spec.cache, other.spec.cache, (a, b) => ({\n        paths: mergeList(a.paths, b.paths)!,\n      })),\n      'reports': mergeDict(this.spec.reports, other.spec.reports, (a, b) => {\n        throw new Error(`Reports must have unique names, got ${a} and ${b}`);\n      }),\n    });\n\n    function mergeArtifacts(a: PrimaryArtifactStruct, b: PrimaryArtifactStruct): PrimaryArtifactStruct {\n      if (a.files || b.files) {\n        throw new Error('None of the BuildSpecs may have a primary artifact.');\n      }\n\n      const artifacts = Object.assign({}, a['secondary-artifacts'] || {});\n      for (const [k, v] of Object.entries(b['secondary-artifacts'] || {})) {\n        if (k in artifacts) {\n          throw new Error(`There is already an artifact with name ${k}`);\n        }\n        artifacts[k] = v;\n      }\n      return Object.assign({}, a, { 'secondary-artifacts': artifacts });\n    }\n\n\n    function equalObjects(a: string, b: string) {\n      if (a !== b) {\n        throw new Error(`Can't merge two different values for the same key: ${JSON.stringify(a)}, ${JSON.stringify(b)}`);\n      }\n      return b;\n    }\n\n    function mergeObj<T>(a: T | undefined, b: T | undefined, fn: (a: T, b: T) => T): T | undefined {\n      if (a === undefined) { return b; }\n      if (b === undefined) { return a; }\n      return fn(a, b);\n    }\n\n    function mergeDict<T>(as: {[k: string]: T} | undefined, bs: {[k: string]: T} | undefined, fn: (a: T, b: T) => T) {\n      return mergeObj(as, bs, (a, b) => {\n        const ret = Object.assign({}, a);\n        for (const [k, v] of Object.entries(b)) {\n          if (ret[k]) {\n            ret[k] = fn(ret[k], v);\n          } else {\n            ret[k] = v;\n          }\n        }\n        return ret;\n      });\n    }\n\n    function mergeList<T>(as: T[] | undefined, bs: T[] | undefined): T[] | undefined {\n      return mergeObj(as, bs, (a, b) => a.concat(b));\n    }\n  }\n\n  public render(options: BuildSpecRenderOptions = {}): BuildSpecStruct {\n    return Object.assign({}, this.spec, { artifacts: this.renderArtifacts(options) });\n  }\n\n  private renderArtifacts(options: BuildSpecRenderOptions): PrimaryArtifactStruct | undefined {\n    if (!this.spec.artifacts || !this.spec.artifacts['secondary-artifacts']) { return this.spec.artifacts; }\n\n    // Simplify a single \"secondary-artifacts\" to a single primary artifact (regardless of the name)\n    const singleArt = dictSingletonValue(this.spec.artifacts['secondary-artifacts']);\n    if (singleArt) { return singleArt; }\n\n    // Otherwise rename a 'PRIMARY' key if it exists\n    if (MAGIC_ARTIFACT_NAME in this.spec.artifacts['secondary-artifacts']) {\n      if (!options.primaryArtifactName) {\n        throw new Error(`Replacement name for ${MAGIC_ARTIFACT_NAME} artifact not supplied`);\n      }\n\n      return { 'secondary-artifacts': renameKey(this.spec.artifacts['secondary-artifacts'], MAGIC_ARTIFACT_NAME, options.primaryArtifactName) };\n    }\n\n    return this.spec.artifacts;\n  }\n}\n\nexport interface SimpleBuildSpecProps {\n  install?: string[];\n  preBuild?: string[];\n  build?: string[];\n  reports?: {[key: string]: ReportStruct};\n  artifactDirectory?: string;\n\n  /**\n   * Where the directories for each artifact are\n   *\n   * Use special name PRIMARY to refer to the primary artifact. Will be\n   * replaced with the actual artifact name when the build spec is synthesized.\n   */\n  additionalArtifactDirectories?: {[id: string]: string};\n}\n\nexport interface BuildSpecStruct {\n  version: '0.2';\n  'run-as'?: string;\n  env?: EnvStruct;\n  phases?: {[key: string]: PhaseStruct};\n  artifacts?: PrimaryArtifactStruct;\n  cache?: CacheStruct;\n  reports?: {[key: string]: ReportStruct};\n}\n\nexport interface EnvStruct {\n  variables?: {[key: string]: string};\n  'parameter-store'?: {[key: string]: string};\n  'exported-variables'?: string[];\n}\n\nexport interface PhaseStruct {\n  'run-as'?: string;\n  commands: string[];\n  finally?: string[];\n}\n\nexport interface ReportStruct {\n  files?: string[];\n  'base-directory'?: string;\n  'discard-paths'?: 'yes' | 'no';\n  'file-format'?: 'CucumberJson' | 'JunitXml' | 'NunitXml' | 'TestNGXml' | 'VisualStudioTrx';\n}\n\nexport interface ArtifactStruct {\n  files?: string[];\n  name?: string;\n  'base-directory'?: string;\n  'discard-paths'?: 'yes' | 'no';\n}\n\nexport interface PrimaryArtifactStruct extends ArtifactStruct {\n  'secondary-artifacts'?: {[key: string]: ArtifactStruct};\n}\n\nexport interface CacheStruct {\n  paths: string[];\n}\n\nexport interface BuildSpecRenderOptions {\n  /**\n   * Replace PRIMARY artifact name with this\n   *\n   * Cannot use the special term PRIMARY if this is not supplied.\n   *\n   * @default  Cannot use PRIMARY\n   */\n  primaryArtifactName?: string;\n}\n\n/**\n * If the dict is a singleton dict, return the value of the first key, otherwise return undefined\n */\nfunction dictSingletonValue<T>(xs: {[key: string]: T}): T | undefined {\n  const keys = Object.keys(xs);\n  if (keys.length === 1) {\n    return xs[keys[0]];\n  }\n  return undefined;\n}\n\nfunction renameKey<T>(xs: {[key: string]: T}, orig: string, rename: string): {[key: string]: T} {\n  const ret = Object.assign({}, xs);\n  if (orig in ret) {\n    ret[rename] = ret[orig];\n    delete ret[orig];\n  }\n  return ret;\n}\n"]}
|
package/lib/shellable.d.ts
CHANGED
|
@@ -2,6 +2,10 @@ import { Duration, aws_cloudwatch as cloudwatch, aws_codebuild as cbuild, aws_co
|
|
|
2
2
|
import { Construct } from 'constructs';
|
|
3
3
|
import { BuildSpec } from './build-spec';
|
|
4
4
|
export interface ShellableOptions {
|
|
5
|
+
/**
|
|
6
|
+
* Description for the CodeBuild Project
|
|
7
|
+
*/
|
|
8
|
+
readonly description?: string;
|
|
5
9
|
/**
|
|
6
10
|
* Source for the CodeBuild project
|
|
7
11
|
*
|
|
@@ -106,6 +110,24 @@ export interface ShellableOptions {
|
|
|
106
110
|
*/
|
|
107
111
|
alarmEvaluationPeriods?: number;
|
|
108
112
|
secondaryArtifactNames?: string[];
|
|
113
|
+
/**
|
|
114
|
+
* Clarify whether this Shellable produces any artifacts
|
|
115
|
+
*
|
|
116
|
+
* @default true
|
|
117
|
+
*/
|
|
118
|
+
readonly producesArtifacts?: boolean;
|
|
119
|
+
/**
|
|
120
|
+
* Namespace to use when adding as an action to the pipeline
|
|
121
|
+
*
|
|
122
|
+
* @default No namespace
|
|
123
|
+
*/
|
|
124
|
+
readonly actionNamespace?: string;
|
|
125
|
+
/**
|
|
126
|
+
* Additional environment variables to set from the pipeline action
|
|
127
|
+
*
|
|
128
|
+
* @default No environment variables
|
|
129
|
+
*/
|
|
130
|
+
readonly pipelineEnvironmentVars?: Record<string, string>;
|
|
109
131
|
}
|
|
110
132
|
/**
|
|
111
133
|
* Properties used to create a Shellable
|
|
@@ -201,6 +223,7 @@ export interface AssumeRole {
|
|
|
201
223
|
* Supports both Windows and Linux computes.
|
|
202
224
|
*/
|
|
203
225
|
export declare class Shellable extends Construct {
|
|
226
|
+
private readonly props;
|
|
204
227
|
readonly project: cbuild.Project;
|
|
205
228
|
readonly role: iam.IRole;
|
|
206
229
|
/**
|
|
@@ -209,7 +232,7 @@ export declare class Shellable extends Construct {
|
|
|
209
232
|
readonly alarm: cloudwatch.Alarm;
|
|
210
233
|
private readonly platform;
|
|
211
234
|
private readonly buildSpec;
|
|
212
|
-
private readonly outputArtifactName
|
|
235
|
+
private readonly outputArtifactName?;
|
|
213
236
|
constructor(parent: Construct, id: string, props: ShellableProps);
|
|
214
237
|
addToPipeline(stage: cpipeline.IStage, name: string, inputArtifact: cpipeline.Artifact, runOrder?: number): cpipeline_actions.CodeBuildAction;
|
|
215
238
|
/**
|
package/lib/shellable.js
CHANGED
|
@@ -47,8 +47,9 @@ const S3_KEY_ENV = 'SCRIPT_S3_KEY';
|
|
|
47
47
|
*/
|
|
48
48
|
class Shellable extends constructs_1.Construct {
|
|
49
49
|
constructor(parent, id, props) {
|
|
50
|
-
var _a, _b;
|
|
50
|
+
var _a, _b, _c;
|
|
51
51
|
super(parent, id);
|
|
52
|
+
this.props = props;
|
|
52
53
|
this.platform = props.platform || ShellPlatform.LinuxUbuntu;
|
|
53
54
|
const entrypoint = path.join(props.scriptDirectory, props.entrypoint);
|
|
54
55
|
if (!fs.existsSync(entrypoint)) {
|
|
@@ -57,8 +58,8 @@ class Shellable extends constructs_1.Construct {
|
|
|
57
58
|
const asset = new aws_cdk_lib_1.aws_s3_assets.Asset(this, 'ScriptDirectory', {
|
|
58
59
|
path: props.scriptDirectory,
|
|
59
60
|
});
|
|
60
|
-
this.outputArtifactName = `Artifact_${this.node.addr}
|
|
61
|
-
if (this.outputArtifactName.length > 100) {
|
|
61
|
+
this.outputArtifactName = ((_a = props.producesArtifacts) !== null && _a !== void 0 ? _a : true) ? `Artifact_${this.node.addr}` : undefined;
|
|
62
|
+
if (this.outputArtifactName && this.outputArtifactName.length > 100) {
|
|
62
63
|
throw new Error(`Whoops, too long: ${this.outputArtifactName}`);
|
|
63
64
|
}
|
|
64
65
|
this.buildSpec = build_spec_1.BuildSpec.simple({
|
|
@@ -69,6 +70,7 @@ class Shellable extends constructs_1.Construct {
|
|
|
69
70
|
const environmentSecretsAsSecretNames = this.convertEnvironmentSecretArnsToSecretNames(props.environmentSecrets);
|
|
70
71
|
this.project = new aws_cdk_lib_1.aws_codebuild.Project(this, 'Resource', {
|
|
71
72
|
projectName: props.buildProjectName,
|
|
73
|
+
description: props.description,
|
|
72
74
|
source: props.source,
|
|
73
75
|
environment: {
|
|
74
76
|
buildImage: this.platform.buildImage,
|
|
@@ -89,12 +91,12 @@ class Shellable extends constructs_1.Construct {
|
|
|
89
91
|
this.role = this.project.role; // not undefined, as it's a new Project
|
|
90
92
|
asset.grantRead(this.role);
|
|
91
93
|
// Grant read access to secrets
|
|
92
|
-
Object.entries((
|
|
94
|
+
Object.entries((_b = props.environmentSecrets) !== null && _b !== void 0 ? _b : {}).forEach(([name, secretArn]) => {
|
|
93
95
|
const secret = aws_cdk_lib_1.aws_secretsmanager.Secret.fromSecretCompleteArn(this, `${name}Secret`, secretArn);
|
|
94
96
|
secret.grantRead(this.role);
|
|
95
97
|
});
|
|
96
98
|
// Grant read access to parameters
|
|
97
|
-
Object.entries((
|
|
99
|
+
Object.entries((_c = props.environmentParameters) !== null && _c !== void 0 ? _c : {}).forEach(([name, parameterName]) => {
|
|
98
100
|
const parameter = aws_cdk_lib_1.aws_ssm.StringParameter.fromStringParameterName(this, `${name}Parameter`, parameterName);
|
|
99
101
|
parameter.grantRead(this.role);
|
|
100
102
|
});
|
|
@@ -113,12 +115,20 @@ class Shellable extends constructs_1.Construct {
|
|
|
113
115
|
});
|
|
114
116
|
}
|
|
115
117
|
addToPipeline(stage, name, inputArtifact, runOrder) {
|
|
118
|
+
var _a;
|
|
116
119
|
const codeBuildAction = new aws_cdk_lib_1.aws_codepipeline_actions.CodeBuildAction({
|
|
117
120
|
actionName: name,
|
|
118
121
|
project: this.project,
|
|
119
122
|
runOrder,
|
|
120
123
|
input: inputArtifact,
|
|
121
|
-
|
|
124
|
+
variablesNamespace: this.props.actionNamespace,
|
|
125
|
+
environmentVariables: this.props.pipelineEnvironmentVars
|
|
126
|
+
? Object.fromEntries(Object.entries(this.props.pipelineEnvironmentVars)
|
|
127
|
+
.map(([k, v]) => [k, { type: aws_cdk_lib_1.aws_codebuild.BuildEnvironmentVariableType.PLAINTEXT, value: v }]))
|
|
128
|
+
: undefined,
|
|
129
|
+
outputs: this.outputArtifactName
|
|
130
|
+
? [this.outputArtifactName, ...(_a = this.buildSpec.additionalArtifactNames) !== null && _a !== void 0 ? _a : []].map(n => new aws_cdk_lib_1.aws_codepipeline.Artifact(n))
|
|
131
|
+
: undefined,
|
|
122
132
|
});
|
|
123
133
|
stage.addAction(codeBuildAction);
|
|
124
134
|
return codeBuildAction;
|
|
@@ -272,4 +282,4 @@ class WindowsPlatform extends ShellPlatform {
|
|
|
272
282
|
}
|
|
273
283
|
}
|
|
274
284
|
exports.WindowsPlatform = WindowsPlatform;
|
|
275
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shellable.js","sourceRoot":"","sources":["shellable.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport {\n  Duration,\n  aws_cloudwatch as cloudwatch, aws_codebuild as cbuild,\n  aws_codepipeline as cpipeline, aws_codepipeline_actions as cpipeline_actions,\n  aws_iam as iam, aws_s3_assets as assets, aws_secretsmanager, aws_ssm,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { BuildSpec } from './build-spec';\nimport { renderEnvironmentVariables } from './util';\n\nconst S3_BUCKET_ENV = 'SCRIPT_S3_BUCKET';\nconst S3_KEY_ENV = 'SCRIPT_S3_KEY';\n\nexport interface ShellableOptions {\n\n  /**\n   * Source for the CodeBuild project\n   *\n   * @default no source\n   */\n  source?: cbuild.ISource;\n\n  /**\n   * What platform to us to run the scripts on\n   *\n   * @default ShellPlatform.LinuxUbuntu\n   */\n  platform?: ShellPlatform;\n\n  /**\n   * Additional environment variables to set.\n   *\n   * @default No additional environment variables\n   */\n  environment?: { [key: string]: string | undefined };\n\n  /**\n   * Environment variables with secrets manager values. The values must be complete Secret Manager ARNs.\n   *\n   * @default no additional environment variables\n   */\n  environmentSecrets?: { [key: string]: string };\n\n  /**\n   * Environment variables with SSM parameter values.\n   *\n   * @default no additional environment variables\n   */\n  environmentParameters?: { [key: string]: string };\n\n  /**\n   * The compute type to use for the build container.\n   *\n   * Note that not all combinations are available. For example,\n   * Windows images cannot be run on ComputeType.Small.\n   *\n   * @default ComputeType.Medium\n   */\n  computeType?: cbuild.ComputeType;\n\n  /**\n   * Indicates how the project builds Docker images. Specify true to enable\n   * running the Docker daemon inside a Docker container. This value must be\n   * set to true only if this build project will be used to build Docker\n   * images, and the specified build environment image is not one provided by\n   * AWS CodeBuild with Docker support. Otherwise, all associated builds that\n   * attempt to interact with the Docker daemon will fail.\n   *\n   * @default false\n   */\n  privileged?: boolean;\n\n  /**\n   * The name for the build project.\n   *\n   * @default a name is generated by CloudFormation.\n   */\n  buildProjectName?: string;\n\n  /**\n   * Indicates if Regional AWS STS endpoints should be used instead\n   * of the global endpoint. Specify true to use Regional AWS STS endpoints.\n   *\n   * @default false\n   */\n  useRegionalStsEndpoints?: boolean;\n\n  /**\n   * Can be used to run this build using a specific IAM role. This can be used,\n   * for example, to execute in the context of another account (e.g. to run\n   * tests in isolation).\n   */\n  assumeRole?: AssumeRole;\n\n  /**\n   * Additional buildspec (for artifacts etc.)\n   *\n   * @default No additional buildspec\n   */\n  buildSpec?: BuildSpec;\n\n  /**\n   * The timeout of the build.\n   *\n   * @default the CodeBuild default (1 hour)\n   */\n  timeout?: Duration;\n\n  /**\n   * Alarm period.\n   *\n   * @default 300 seconds (5 minutes)\n   */\n  alarmPeriod?: Duration;\n\n  /**\n   * Alarm threshold.\n   * @default 1\n   */\n  alarmThreshold?: number;\n\n  /**\n   * Alarm evaluation periods.\n   * @default 1\n   */\n  alarmEvaluationPeriods?: number;\n\n  secondaryArtifactNames?: string[];\n}\n\n/**\n * Properties used to create a Shellable\n */\nexport interface ShellableProps extends ShellableOptions {\n  /**\n   * Directory with the scripts.\n   *\n   * The whole directory will be uploaded.\n   */\n  scriptDirectory: string;\n\n  /**\n   * Filename of the initial script to start, relative to scriptDirectory.\n   */\n  entrypoint: string;\n}\n\nexport interface AssumeRole {\n  /**\n   * The Amazon Resource Name (ARN) of the role to assume.\n   */\n  roleArn: string;\n\n  /**\n   * An identifier for the assumed role session.\n   *\n   * Use  the  role  session name to uniquely identify a session when the same\n   * role is assumed by different principals or for different reasons. In\n   * cross-account scenarios, the role session name is visible to, and can be\n   * logged by the account that owns the role.  The role session name is also\n   * used in the ARN of the assumed role principal. This means that subsequent\n   * cross-account API requests using the tem- porary security credentials will\n   * expose the role session name to the external account in their CloudTrail\n   * logs.\n   *\n   * The regex used to validate this parameter is a string of characters\n   * consisting  of upper- and lower-case alphanumeric characters with no\n   * spaces. You can also include underscores or any of the following\n   * characters: =,.@-\n   */\n  sessionName: string;\n\n  /**\n   * A  unique  identifier  that  is  used by third parties when assuming roles\n   * in their customers' accounts. For each  role  that  the  third party can\n   * assume, they should instruct their customers to ensure the role's trust\n   * policy checks for the external ID that the third  party generated.  Each\n   * time the third party assumes the role, they should pass the customer's\n   * external ID. The external ID is useful in  order to  help  third  parties\n   * bind a role to the customer who created it. For more information about the\n   * external ID, see How to Use an Exter- nal  ID  When Granting Access to Your\n   * AWS Resources to a Third Party in the IAM User Guide .\n   *\n   * This parameter must be a string of characters consisting  of upper- and\n   * lower-case alphanumeric characters with no spaces. You can also include\n   * underscores or  any  of  the  following characters: =,.@:/-\n   */\n  externalId?: string;\n\n  /**\n   * When a profie name is configured, an assumed role configuration will be created\n   * in the shared aws configuration file (~/.aws/config). This is in contrary of simply invoking\n   * an `sts assume-role` command that creates a session with a fixed expiry date.\n   *\n   * Using a profile will delegate credential refreshing to the SDK/CLI.\n   * This is needed to support long running sessions that needs sessions that are longer than\n   * the session duration that can be configured with a `sts assume-role`.\n   *\n   * The application code will access to this profile in the `AWS_PROFILE` env variable.\n   *\n   * Only relevant if `refresh` is specified.\n   *\n   * @see https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-files.html\n   *\n   * @default 'long-running-profile'\n   */\n  profileName?: string;\n\n  /**\n   * Specify this if you have a long running execution that needs long running sessions.\n   * This will create a profile and use it to delegate credential refreshing to the SDK/CLI\n   *\n   * @default false\n   */\n  refresh?: boolean;\n\n}\n\n/**\n * A CodeBuild project that runs arbitrary scripts.\n *\n * The scripts to be run are specified by supplying a directory.\n * All files in the directory are uploaded, then the script designated\n * as the entry point is started.\n *\n * The script is executed in the directory where the build project's\n * input is stored. The directory where the script files are stored\n * is in the $SCRIPT_DIR environment variable.\n *\n * Supports both Windows and Linux computes.\n */\nexport class Shellable extends Construct {\n  public readonly project: cbuild.Project;\n  public readonly role: iam.IRole;\n\n  /**\n   * CloudWatch alarm that will be triggered if this action fails.\n   */\n  public readonly alarm: cloudwatch.Alarm;\n\n  private readonly platform: ShellPlatform;\n  private readonly buildSpec: BuildSpec;\n\n  private readonly outputArtifactName: string;\n\n  constructor(parent: Construct, id: string, props: ShellableProps) {\n    super(parent, id);\n\n    this.platform = props.platform || ShellPlatform.LinuxUbuntu;\n\n    const entrypoint = path.join(props.scriptDirectory, props.entrypoint);\n    if (!fs.existsSync(entrypoint)) {\n      throw new Error(`Cannot find test entrypoint: ${entrypoint}`);\n    }\n\n    const asset = new assets.Asset(this, 'ScriptDirectory', {\n      path: props.scriptDirectory,\n    });\n\n    this.outputArtifactName = `Artifact_${this.node.addr}`;\n    if (this.outputArtifactName.length > 100) {\n      throw new Error(`Whoops, too long: ${this.outputArtifactName}`);\n    }\n\n    this.buildSpec = BuildSpec.simple({\n      install: this.platform.installCommands(),\n      preBuild: this.platform.prebuildCommands(props.assumeRole, props.useRegionalStsEndpoints),\n      build: this.platform.buildCommands(props.entrypoint),\n    }).merge(props.buildSpec || BuildSpec.empty());\n\n    const environmentSecretsAsSecretNames = this.convertEnvironmentSecretArnsToSecretNames(props.environmentSecrets);\n\n    this.project = new cbuild.Project(this, 'Resource', {\n      projectName: props.buildProjectName,\n      source: props.source,\n      environment: {\n        buildImage: this.platform.buildImage,\n        computeType: props.computeType || cbuild.ComputeType.MEDIUM,\n        privileged: props.privileged,\n      },\n      environmentVariables: {\n        [S3_BUCKET_ENV]: { value: asset.s3BucketName },\n        [S3_KEY_ENV]: { value: asset.s3ObjectKey },\n        ...renderEnvironmentVariables(props.environment),\n        ...renderEnvironmentVariables(environmentSecretsAsSecretNames, cbuild.BuildEnvironmentVariableType.SECRETS_MANAGER),\n        ...renderEnvironmentVariables(props.environmentParameters, cbuild.BuildEnvironmentVariableType.PARAMETER_STORE),\n      },\n      timeout: props.timeout,\n      buildSpec: cbuild.BuildSpec.fromObject(this.buildSpec.render({ primaryArtifactName: this.outputArtifactName })),\n      ssmSessionPermissions: true,\n    });\n\n    this.role = this.project.role!; // not undefined, as it's a new Project\n    asset.grantRead(this.role);\n\n    // Grant read access to secrets\n    Object.entries(props.environmentSecrets ?? {}).forEach(([name, secretArn]) => {\n      const secret = aws_secretsmanager.Secret.fromSecretCompleteArn(this, `${name}Secret`, secretArn);\n      secret.grantRead(this.role);\n    });\n\n    // Grant read access to parameters\n    Object.entries(props.environmentParameters ?? {}).forEach(([name, parameterName]) => {\n      const parameter = aws_ssm.StringParameter.fromStringParameterName(this, `${name}Parameter`, parameterName);\n      parameter.grantRead(this.role);\n    });\n\n    if (props.assumeRole) {\n      this.role.addToPrincipalPolicy(new iam.PolicyStatement({\n        actions: ['sts:AssumeRole'],\n        resources: [props.assumeRole.roleArn],\n      }));\n    }\n\n    this.alarm = new cloudwatch.Alarm(this, 'Alarm', {\n      metric: this.project.metricFailedBuilds({ period: props.alarmPeriod || Duration.seconds(300) }),\n      threshold: props.alarmThreshold || 1,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: props.alarmEvaluationPeriods || 1,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    });\n  }\n\n  public addToPipeline(stage: cpipeline.IStage, name: string, inputArtifact: cpipeline.Artifact, runOrder?: number):\n  cpipeline_actions.CodeBuildAction {\n    const codeBuildAction = new cpipeline_actions.CodeBuildAction({\n      actionName: name,\n      project: this.project,\n      runOrder,\n      input: inputArtifact,\n      outputs: [new cpipeline.Artifact(this.outputArtifactName)].concat(\n        this.buildSpec.additionalArtifactNames.map(artifactName => new cpipeline.Artifact(artifactName))),\n    });\n    stage.addAction(codeBuildAction);\n    return codeBuildAction;\n  }\n\n  /**\n   * The contract of `environmentSecrets` is that the values are complete Secret ARNs;\n   * however, the CodeBuild construct expects secret names as the inputs for environment variables.\n   * This method converts the environment secrets from ARNs to names.\n   */\n  private convertEnvironmentSecretArnsToSecretNames(environmentSecrets?: { [key: string]: string }) {\n    if (!environmentSecrets) {\n      return undefined;\n    }\n\n    const out: { [key: string]: string } = { };\n    Object.entries(environmentSecrets ?? {}).forEach(([name, secretArn]) => {\n      const secret = aws_secretsmanager.Secret.fromSecretCompleteArn(this, `${name}SecretFromArn`, secretArn);\n      out[name] = secret.secretName;\n    });\n    return out;\n  }\n}\n\n/**\n * Platform archetype\n */\nexport enum PlatformType {\n  Linux = 'Linux',\n  Windows = 'Windows'\n}\n\n/**\n * The platform type to run the scripts on\n */\nexport abstract class ShellPlatform {\n  /**\n   * Return a default Ubuntu Linux platform\n   */\n  public static get LinuxUbuntu(): ShellPlatform {\n    // Cannot be static member because of initialization order\n    return new LinuxPlatform(cbuild.LinuxBuildImage.STANDARD_4_0);\n  }\n\n  /**\n   * Return a default Windows platform\n   */\n  public static get Windows(): ShellPlatform {\n    // Cannot be static member because of initialization order\n    return new WindowsPlatform(cbuild.WindowsBuildImage.WIN_SERVER_CORE_2019_BASE);\n  }\n\n  constructor(public readonly buildImage: cbuild.IBuildImage) {\n  }\n\n  /**\n   * Retrn commands to prepare the host for the shellable.\n   */\n  public abstract installCommands(): string[] | undefined;\n\n  /**\n   * Return commands to download the script bundle\n   */\n  public abstract prebuildCommands(assumeRole?: AssumeRole, useRegionalStsEndpoints?: boolean): string[];\n\n  /**\n   * Return commands to start the entrypoint script\n   */\n  public abstract buildCommands(entrypoint: string): string[];\n\n  /**\n   * Type of platform\n   */\n  public abstract get platformType(): PlatformType;\n}\n\n/**\n * A Linux Platform\n */\nexport class LinuxPlatform extends ShellPlatform {\n  public readonly platformType = PlatformType.Linux;\n\n  public installCommands(): string[] | undefined {\n    return undefined;\n  }\n\n  public prebuildCommands(assumeRole?: AssumeRole, useRegionalStsEndpoints?: boolean): string[] {\n    const lines = new Array<string>();\n    // Better echo the location here; if this fails, the error message only contains\n    // the unexpanded variables by default. It might fail if you're running an old\n    // definition of the CodeBuild project--the permissions will have been changed\n    // to only allow downloading the very latest version.\n    lines.push(`echo \"Downloading scripts from s3://\\${${S3_BUCKET_ENV}}/\\${${S3_KEY_ENV}}\"`);\n    lines.push(`aws s3 cp s3://\\${${S3_BUCKET_ENV}}/\\${${S3_KEY_ENV}} /tmp`);\n    lines.push('mkdir -p /tmp/scriptdir');\n    lines.push(`unzip /tmp/$(basename \\$${S3_KEY_ENV}) -d /tmp/scriptdir`);\n\n    if (assumeRole) {\n\n      if (assumeRole.refresh) {\n\n        const awsHome = '~/.aws';\n\n        const profileName = assumeRole.profileName ?? 'long-running-profile';\n\n        lines.push(`mkdir -p ${awsHome}`);\n        lines.push(`touch ${awsHome}/credentials`);\n        lines.push(`config=${awsHome}/config`);\n        lines.push(`echo [profile ${profileName}]>> $\\{config\\}`);\n        lines.push('echo credential_source = EcsContainer >> $\\{config\\}');\n        lines.push(`echo role_session_name = ${assumeRole.sessionName} >> $\\{config\\}`);\n        lines.push(`echo role_arn = ${assumeRole.roleArn} >> $config`);\n\n        if (assumeRole.externalId) {\n          lines.push(`echo external_id = ${assumeRole.externalId} >> $config`);\n        }\n\n        // let the application code know which role is being used.\n        lines.push(`export AWS_PROFILE=${profileName}`);\n\n        // force the AWS SDK for JavaScript to actually load the config file (do automatically so users don't forget)\n        lines.push('export AWS_SDK_LOAD_CONFIG=1');\n\n      } else {\n\n        const externalId = assumeRole.externalId ? `--external-id \"${assumeRole.externalId}\"` : '';\n        const StsEndpoints = useRegionalStsEndpoints ? 'regional' : 'legacy';\n\n        lines.push('creds=$(mktemp -d)/creds.json');\n        lines.push(`AWS_STS_REGIONAL_ENDPOINTS=${StsEndpoints} aws sts assume-role --role-arn \"${assumeRole.roleArn}\" --role-session-name \"${assumeRole.sessionName}\" ${externalId} > $creds`);\n        lines.push('export AWS_ACCESS_KEY_ID=\"$(cat ${creds} | grep \"AccessKeyId\" | cut -d\\'\"\\' -f 4)\"');\n        lines.push('export AWS_SECRET_ACCESS_KEY=\"$(cat ${creds} | grep \"SecretAccessKey\" | cut -d\\'\"\\' -f 4)\"');\n        lines.push('export AWS_SESSION_TOKEN=\"$(cat ${creds} | grep \"SessionToken\" | cut -d\\'\"\\' -f 4)\"');\n      }\n    }\n\n    return lines;\n  }\n\n  public buildCommands(entrypoint: string): string[] {\n    return [\n      'export SCRIPT_DIR=/tmp/scriptdir',\n      `echo \"Running ${entrypoint}\"`,\n      `/bin/bash /tmp/scriptdir/${entrypoint}`,\n    ];\n  }\n}\n\n/**\n * A Windows Platform\n */\nexport class WindowsPlatform extends ShellPlatform {\n  public readonly platformType = PlatformType.Windows;\n\n  public installCommands(): string[] | undefined {\n    return [\n      // Update the image's nodejs to the latest LTS release.\n      'Import-Module \"C:\\\\ProgramData\\\\chocolatey\\\\helpers\\\\chocolateyProfile.psm1\"',\n      'C:\\\\ProgramData\\\\chocolatey\\\\bin\\\\choco.exe upgrade nodejs-lts -y',\n    ];\n  }\n\n  public prebuildCommands(assumeRole?: AssumeRole, _useRegionalStsEndpoints?: boolean): string[] {\n    if (assumeRole) {\n      throw new Error('assumeRole is not supported on Windows: https://github.com/cdklabs/aws-delivlib/issues/57');\n    }\n\n    return [\n      // Would love to do downloading here and executing in the next step,\n      // but I don't know how to propagate the value of $TEMPDIR.\n      //\n      // Punting for someone who knows PowerShell well enough.\n    ];\n  }\n\n  public buildCommands(entrypoint: string): string[] {\n    return [\n      'Set-Variable -Name TEMPDIR -Value (New-TemporaryFile).DirectoryName',\n      `aws s3 cp s3://$env:${S3_BUCKET_ENV}/$env:${S3_KEY_ENV} $TEMPDIR\\\\scripts.zip`,\n      'New-Item -ItemType Directory -Path $TEMPDIR\\\\scriptdir',\n      'Expand-Archive -Path $TEMPDIR/scripts.zip -DestinationPath $TEMPDIR\\\\scriptdir',\n      '$env:SCRIPT_DIR = \"$TEMPDIR\\\\scriptdir\"',\n      `& $TEMPDIR\\\\scriptdir\\\\${entrypoint}`,\n    ];\n  }\n}\n"]}
|
|
285
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shellable.js","sourceRoot":"","sources":["shellable.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport {\n  Duration,\n  aws_cloudwatch as cloudwatch, aws_codebuild as cbuild,\n  aws_codepipeline as cpipeline, aws_codepipeline_actions as cpipeline_actions,\n  aws_iam as iam, aws_s3_assets as assets, aws_secretsmanager, aws_ssm,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { BuildSpec } from './build-spec';\nimport { renderEnvironmentVariables } from './util';\n\nconst S3_BUCKET_ENV = 'SCRIPT_S3_BUCKET';\nconst S3_KEY_ENV = 'SCRIPT_S3_KEY';\n\nexport interface ShellableOptions {\n  /**\n   * Description for the CodeBuild Project\n   */\n  readonly description?: string;\n\n  /**\n   * Source for the CodeBuild project\n   *\n   * @default no source\n   */\n  source?: cbuild.ISource;\n\n  /**\n   * What platform to us to run the scripts on\n   *\n   * @default ShellPlatform.LinuxUbuntu\n   */\n  platform?: ShellPlatform;\n\n  /**\n   * Additional environment variables to set.\n   *\n   * @default No additional environment variables\n   */\n  environment?: { [key: string]: string | undefined };\n\n  /**\n   * Environment variables with secrets manager values. The values must be complete Secret Manager ARNs.\n   *\n   * @default no additional environment variables\n   */\n  environmentSecrets?: { [key: string]: string };\n\n  /**\n   * Environment variables with SSM parameter values.\n   *\n   * @default no additional environment variables\n   */\n  environmentParameters?: { [key: string]: string };\n\n  /**\n   * The compute type to use for the build container.\n   *\n   * Note that not all combinations are available. For example,\n   * Windows images cannot be run on ComputeType.Small.\n   *\n   * @default ComputeType.Medium\n   */\n  computeType?: cbuild.ComputeType;\n\n  /**\n   * Indicates how the project builds Docker images. Specify true to enable\n   * running the Docker daemon inside a Docker container. This value must be\n   * set to true only if this build project will be used to build Docker\n   * images, and the specified build environment image is not one provided by\n   * AWS CodeBuild with Docker support. Otherwise, all associated builds that\n   * attempt to interact with the Docker daemon will fail.\n   *\n   * @default false\n   */\n  privileged?: boolean;\n\n  /**\n   * The name for the build project.\n   *\n   * @default a name is generated by CloudFormation.\n   */\n  buildProjectName?: string;\n\n  /**\n   * Indicates if Regional AWS STS endpoints should be used instead\n   * of the global endpoint. Specify true to use Regional AWS STS endpoints.\n   *\n   * @default false\n   */\n  useRegionalStsEndpoints?: boolean;\n\n  /**\n   * Can be used to run this build using a specific IAM role. This can be used,\n   * for example, to execute in the context of another account (e.g. to run\n   * tests in isolation).\n   */\n  assumeRole?: AssumeRole;\n\n  /**\n   * Additional buildspec (for artifacts etc.)\n   *\n   * @default No additional buildspec\n   */\n  buildSpec?: BuildSpec;\n\n  /**\n   * The timeout of the build.\n   *\n   * @default the CodeBuild default (1 hour)\n   */\n  timeout?: Duration;\n\n  /**\n   * Alarm period.\n   *\n   * @default 300 seconds (5 minutes)\n   */\n  alarmPeriod?: Duration;\n\n  /**\n   * Alarm threshold.\n   * @default 1\n   */\n  alarmThreshold?: number;\n\n  /**\n   * Alarm evaluation periods.\n   * @default 1\n   */\n  alarmEvaluationPeriods?: number;\n\n  secondaryArtifactNames?: string[];\n\n  /**\n   * Clarify whether this Shellable produces any artifacts\n   *\n   * @default true\n   */\n  readonly producesArtifacts?: boolean;\n\n  /**\n   * Namespace to use when adding as an action to the pipeline\n   *\n   * @default No namespace\n   */\n  readonly actionNamespace?: string;\n\n  /**\n   * Additional environment variables to set from the pipeline action\n   *\n   * @default No environment variables\n   */\n  readonly pipelineEnvironmentVars?: Record<string, string>;\n}\n\n/**\n * Properties used to create a Shellable\n */\nexport interface ShellableProps extends ShellableOptions {\n  /**\n   * Directory with the scripts.\n   *\n   * The whole directory will be uploaded.\n   */\n  scriptDirectory: string;\n\n  /**\n   * Filename of the initial script to start, relative to scriptDirectory.\n   */\n  entrypoint: string;\n}\n\nexport interface AssumeRole {\n  /**\n   * The Amazon Resource Name (ARN) of the role to assume.\n   */\n  roleArn: string;\n\n  /**\n   * An identifier for the assumed role session.\n   *\n   * Use  the  role  session name to uniquely identify a session when the same\n   * role is assumed by different principals or for different reasons. In\n   * cross-account scenarios, the role session name is visible to, and can be\n   * logged by the account that owns the role.  The role session name is also\n   * used in the ARN of the assumed role principal. This means that subsequent\n   * cross-account API requests using the tem- porary security credentials will\n   * expose the role session name to the external account in their CloudTrail\n   * logs.\n   *\n   * The regex used to validate this parameter is a string of characters\n   * consisting  of upper- and lower-case alphanumeric characters with no\n   * spaces. You can also include underscores or any of the following\n   * characters: =,.@-\n   */\n  sessionName: string;\n\n  /**\n   * A  unique  identifier  that  is  used by third parties when assuming roles\n   * in their customers' accounts. For each  role  that  the  third party can\n   * assume, they should instruct their customers to ensure the role's trust\n   * policy checks for the external ID that the third  party generated.  Each\n   * time the third party assumes the role, they should pass the customer's\n   * external ID. The external ID is useful in  order to  help  third  parties\n   * bind a role to the customer who created it. For more information about the\n   * external ID, see How to Use an Exter- nal  ID  When Granting Access to Your\n   * AWS Resources to a Third Party in the IAM User Guide .\n   *\n   * This parameter must be a string of characters consisting  of upper- and\n   * lower-case alphanumeric characters with no spaces. You can also include\n   * underscores or  any  of  the  following characters: =,.@:/-\n   */\n  externalId?: string;\n\n  /**\n   * When a profie name is configured, an assumed role configuration will be created\n   * in the shared aws configuration file (~/.aws/config). This is in contrary of simply invoking\n   * an `sts assume-role` command that creates a session with a fixed expiry date.\n   *\n   * Using a profile will delegate credential refreshing to the SDK/CLI.\n   * This is needed to support long running sessions that needs sessions that are longer than\n   * the session duration that can be configured with a `sts assume-role`.\n   *\n   * The application code will access to this profile in the `AWS_PROFILE` env variable.\n   *\n   * Only relevant if `refresh` is specified.\n   *\n   * @see https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-files.html\n   *\n   * @default 'long-running-profile'\n   */\n  profileName?: string;\n\n  /**\n   * Specify this if you have a long running execution that needs long running sessions.\n   * This will create a profile and use it to delegate credential refreshing to the SDK/CLI\n   *\n   * @default false\n   */\n  refresh?: boolean;\n\n}\n\n/**\n * A CodeBuild project that runs arbitrary scripts.\n *\n * The scripts to be run are specified by supplying a directory.\n * All files in the directory are uploaded, then the script designated\n * as the entry point is started.\n *\n * The script is executed in the directory where the build project's\n * input is stored. The directory where the script files are stored\n * is in the $SCRIPT_DIR environment variable.\n *\n * Supports both Windows and Linux computes.\n */\nexport class Shellable extends Construct {\n  public readonly project: cbuild.Project;\n  public readonly role: iam.IRole;\n\n  /**\n   * CloudWatch alarm that will be triggered if this action fails.\n   */\n  public readonly alarm: cloudwatch.Alarm;\n\n  private readonly platform: ShellPlatform;\n  private readonly buildSpec: BuildSpec;\n\n  private readonly outputArtifactName?: string;\n\n  constructor(parent: Construct, id: string, private readonly props: ShellableProps) {\n    super(parent, id);\n\n    this.platform = props.platform || ShellPlatform.LinuxUbuntu;\n\n    const entrypoint = path.join(props.scriptDirectory, props.entrypoint);\n    if (!fs.existsSync(entrypoint)) {\n      throw new Error(`Cannot find test entrypoint: ${entrypoint}`);\n    }\n\n    const asset = new assets.Asset(this, 'ScriptDirectory', {\n      path: props.scriptDirectory,\n    });\n\n    this.outputArtifactName = (props.producesArtifacts ?? true) ? `Artifact_${this.node.addr}` : undefined;\n    if (this.outputArtifactName && this.outputArtifactName.length > 100) {\n      throw new Error(`Whoops, too long: ${this.outputArtifactName}`);\n    }\n\n    this.buildSpec = BuildSpec.simple({\n      install: this.platform.installCommands(),\n      preBuild: this.platform.prebuildCommands(props.assumeRole, props.useRegionalStsEndpoints),\n      build: this.platform.buildCommands(props.entrypoint),\n    }).merge(props.buildSpec || BuildSpec.empty());\n\n    const environmentSecretsAsSecretNames = this.convertEnvironmentSecretArnsToSecretNames(props.environmentSecrets);\n\n    this.project = new cbuild.Project(this, 'Resource', {\n      projectName: props.buildProjectName,\n      description: props.description,\n      source: props.source,\n      environment: {\n        buildImage: this.platform.buildImage,\n        computeType: props.computeType || cbuild.ComputeType.MEDIUM,\n        privileged: props.privileged,\n      },\n      environmentVariables: {\n        [S3_BUCKET_ENV]: { value: asset.s3BucketName },\n        [S3_KEY_ENV]: { value: asset.s3ObjectKey },\n        ...renderEnvironmentVariables(props.environment),\n        ...renderEnvironmentVariables(environmentSecretsAsSecretNames, cbuild.BuildEnvironmentVariableType.SECRETS_MANAGER),\n        ...renderEnvironmentVariables(props.environmentParameters, cbuild.BuildEnvironmentVariableType.PARAMETER_STORE),\n      },\n      timeout: props.timeout,\n      buildSpec: cbuild.BuildSpec.fromObject(this.buildSpec.render({ primaryArtifactName: this.outputArtifactName })),\n      ssmSessionPermissions: true,\n    });\n\n    this.role = this.project.role!; // not undefined, as it's a new Project\n    asset.grantRead(this.role);\n\n    // Grant read access to secrets\n    Object.entries(props.environmentSecrets ?? {}).forEach(([name, secretArn]) => {\n      const secret = aws_secretsmanager.Secret.fromSecretCompleteArn(this, `${name}Secret`, secretArn);\n      secret.grantRead(this.role);\n    });\n\n    // Grant read access to parameters\n    Object.entries(props.environmentParameters ?? {}).forEach(([name, parameterName]) => {\n      const parameter = aws_ssm.StringParameter.fromStringParameterName(this, `${name}Parameter`, parameterName);\n      parameter.grantRead(this.role);\n    });\n\n    if (props.assumeRole) {\n      this.role.addToPrincipalPolicy(new iam.PolicyStatement({\n        actions: ['sts:AssumeRole'],\n        resources: [props.assumeRole.roleArn],\n      }));\n    }\n\n    this.alarm = new cloudwatch.Alarm(this, 'Alarm', {\n      metric: this.project.metricFailedBuilds({ period: props.alarmPeriod || Duration.seconds(300) }),\n      threshold: props.alarmThreshold || 1,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      evaluationPeriods: props.alarmEvaluationPeriods || 1,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    });\n  }\n\n  public addToPipeline(stage: cpipeline.IStage, name: string, inputArtifact: cpipeline.Artifact, runOrder?: number):\n  cpipeline_actions.CodeBuildAction {\n    const codeBuildAction = new cpipeline_actions.CodeBuildAction({\n      actionName: name,\n      project: this.project,\n      runOrder,\n      input: inputArtifact,\n      variablesNamespace: this.props.actionNamespace,\n      environmentVariables: this.props.pipelineEnvironmentVars\n        ? Object.fromEntries(Object.entries(this.props.pipelineEnvironmentVars)\n          .map(([k, v]) => ([k, { type: cbuild.BuildEnvironmentVariableType.PLAINTEXT, value: v }] as const)))\n        : undefined,\n      outputs: this.outputArtifactName\n        ? [this.outputArtifactName, ...this.buildSpec.additionalArtifactNames ?? []].map(n => new cpipeline.Artifact(n))\n        : undefined,\n    });\n    stage.addAction(codeBuildAction);\n    return codeBuildAction;\n  }\n\n  /**\n   * The contract of `environmentSecrets` is that the values are complete Secret ARNs;\n   * however, the CodeBuild construct expects secret names as the inputs for environment variables.\n   * This method converts the environment secrets from ARNs to names.\n   */\n  private convertEnvironmentSecretArnsToSecretNames(environmentSecrets?: { [key: string]: string }) {\n    if (!environmentSecrets) {\n      return undefined;\n    }\n\n    const out: { [key: string]: string } = { };\n    Object.entries(environmentSecrets ?? {}).forEach(([name, secretArn]) => {\n      const secret = aws_secretsmanager.Secret.fromSecretCompleteArn(this, `${name}SecretFromArn`, secretArn);\n      out[name] = secret.secretName;\n    });\n    return out;\n  }\n}\n\n/**\n * Platform archetype\n */\nexport enum PlatformType {\n  Linux = 'Linux',\n  Windows = 'Windows'\n}\n\n/**\n * The platform type to run the scripts on\n */\nexport abstract class ShellPlatform {\n  /**\n   * Return a default Ubuntu Linux platform\n   */\n  public static get LinuxUbuntu(): ShellPlatform {\n    // Cannot be static member because of initialization order\n    return new LinuxPlatform(cbuild.LinuxBuildImage.STANDARD_4_0);\n  }\n\n  /**\n   * Return a default Windows platform\n   */\n  public static get Windows(): ShellPlatform {\n    // Cannot be static member because of initialization order\n    return new WindowsPlatform(cbuild.WindowsBuildImage.WIN_SERVER_CORE_2019_BASE);\n  }\n\n  constructor(public readonly buildImage: cbuild.IBuildImage) {\n  }\n\n  /**\n   * Retrn commands to prepare the host for the shellable.\n   */\n  public abstract installCommands(): string[] | undefined;\n\n  /**\n   * Return commands to download the script bundle\n   */\n  public abstract prebuildCommands(assumeRole?: AssumeRole, useRegionalStsEndpoints?: boolean): string[];\n\n  /**\n   * Return commands to start the entrypoint script\n   */\n  public abstract buildCommands(entrypoint: string): string[];\n\n  /**\n   * Type of platform\n   */\n  public abstract get platformType(): PlatformType;\n}\n\n/**\n * A Linux Platform\n */\nexport class LinuxPlatform extends ShellPlatform {\n  public readonly platformType = PlatformType.Linux;\n\n  public installCommands(): string[] | undefined {\n    return undefined;\n  }\n\n  public prebuildCommands(assumeRole?: AssumeRole, useRegionalStsEndpoints?: boolean): string[] {\n    const lines = new Array<string>();\n    // Better echo the location here; if this fails, the error message only contains\n    // the unexpanded variables by default. It might fail if you're running an old\n    // definition of the CodeBuild project--the permissions will have been changed\n    // to only allow downloading the very latest version.\n    lines.push(`echo \"Downloading scripts from s3://\\${${S3_BUCKET_ENV}}/\\${${S3_KEY_ENV}}\"`);\n    lines.push(`aws s3 cp s3://\\${${S3_BUCKET_ENV}}/\\${${S3_KEY_ENV}} /tmp`);\n    lines.push('mkdir -p /tmp/scriptdir');\n    lines.push(`unzip /tmp/$(basename \\$${S3_KEY_ENV}) -d /tmp/scriptdir`);\n\n    if (assumeRole) {\n\n      if (assumeRole.refresh) {\n\n        const awsHome = '~/.aws';\n\n        const profileName = assumeRole.profileName ?? 'long-running-profile';\n\n        lines.push(`mkdir -p ${awsHome}`);\n        lines.push(`touch ${awsHome}/credentials`);\n        lines.push(`config=${awsHome}/config`);\n        lines.push(`echo [profile ${profileName}]>> $\\{config\\}`);\n        lines.push('echo credential_source = EcsContainer >> $\\{config\\}');\n        lines.push(`echo role_session_name = ${assumeRole.sessionName} >> $\\{config\\}`);\n        lines.push(`echo role_arn = ${assumeRole.roleArn} >> $config`);\n\n        if (assumeRole.externalId) {\n          lines.push(`echo external_id = ${assumeRole.externalId} >> $config`);\n        }\n\n        // let the application code know which role is being used.\n        lines.push(`export AWS_PROFILE=${profileName}`);\n\n        // force the AWS SDK for JavaScript to actually load the config file (do automatically so users don't forget)\n        lines.push('export AWS_SDK_LOAD_CONFIG=1');\n\n      } else {\n\n        const externalId = assumeRole.externalId ? `--external-id \"${assumeRole.externalId}\"` : '';\n        const StsEndpoints = useRegionalStsEndpoints ? 'regional' : 'legacy';\n\n        lines.push('creds=$(mktemp -d)/creds.json');\n        lines.push(`AWS_STS_REGIONAL_ENDPOINTS=${StsEndpoints} aws sts assume-role --role-arn \"${assumeRole.roleArn}\" --role-session-name \"${assumeRole.sessionName}\" ${externalId} > $creds`);\n        lines.push('export AWS_ACCESS_KEY_ID=\"$(cat ${creds} | grep \"AccessKeyId\" | cut -d\\'\"\\' -f 4)\"');\n        lines.push('export AWS_SECRET_ACCESS_KEY=\"$(cat ${creds} | grep \"SecretAccessKey\" | cut -d\\'\"\\' -f 4)\"');\n        lines.push('export AWS_SESSION_TOKEN=\"$(cat ${creds} | grep \"SessionToken\" | cut -d\\'\"\\' -f 4)\"');\n      }\n    }\n\n    return lines;\n  }\n\n  public buildCommands(entrypoint: string): string[] {\n    return [\n      'export SCRIPT_DIR=/tmp/scriptdir',\n      `echo \"Running ${entrypoint}\"`,\n      `/bin/bash /tmp/scriptdir/${entrypoint}`,\n    ];\n  }\n}\n\n/**\n * A Windows Platform\n */\nexport class WindowsPlatform extends ShellPlatform {\n  public readonly platformType = PlatformType.Windows;\n\n  public installCommands(): string[] | undefined {\n    return [\n      // Update the image's nodejs to the latest LTS release.\n      'Import-Module \"C:\\\\ProgramData\\\\chocolatey\\\\helpers\\\\chocolateyProfile.psm1\"',\n      'C:\\\\ProgramData\\\\chocolatey\\\\bin\\\\choco.exe upgrade nodejs-lts -y',\n    ];\n  }\n\n  public prebuildCommands(assumeRole?: AssumeRole, _useRegionalStsEndpoints?: boolean): string[] {\n    if (assumeRole) {\n      throw new Error('assumeRole is not supported on Windows: https://github.com/cdklabs/aws-delivlib/issues/57');\n    }\n\n    return [\n      // Would love to do downloading here and executing in the next step,\n      // but I don't know how to propagate the value of $TEMPDIR.\n      //\n      // Punting for someone who knows PowerShell well enough.\n    ];\n  }\n\n  public buildCommands(entrypoint: string): string[] {\n    return [\n      'Set-Variable -Name TEMPDIR -Value (New-TemporaryFile).DirectoryName',\n      `aws s3 cp s3://$env:${S3_BUCKET_ENV}/$env:${S3_KEY_ENV} $TEMPDIR\\\\scripts.zip`,\n      'New-Item -ItemType Directory -Path $TEMPDIR\\\\scriptdir',\n      'Expand-Archive -Path $TEMPDIR/scripts.zip -DestinationPath $TEMPDIR\\\\scriptdir',\n      '$env:SCRIPT_DIR = \"$TEMPDIR\\\\scriptdir\"',\n      `& $TEMPDIR\\\\scriptdir\\\\${entrypoint}`,\n    ];\n  }\n}\n"]}
|