UNPKG

21.7 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.mergeBuildSpecs = exports.BuildSpec = void 0;
5const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6const core_1 = require("@aws-cdk/core");
7const yaml_cfn = require("./private/yaml-cfn");
8/**
9 * BuildSpec for CodeBuild projects
10 */
11class BuildSpec {
12 constructor() {
13 }
14 static fromObject(value) {
15 return new ObjectBuildSpec(value);
16 }
17 /**
18 * Create a buildspec from an object that will be rendered as YAML in the resulting CloudFormation template.
19 *
20 * @param value the object containing the buildspec that will be rendered as YAML
21 */
22 static fromObjectToYaml(value) {
23 return new YamlBuildSpec(value);
24 }
25 /**
26 * Use a file from the source as buildspec
27 *
28 * Use this if you want to use a file different from 'buildspec.yml'`
29 */
30 static fromSourceFilename(filename) {
31 return new FilenameBuildSpec(filename);
32 }
33}
34exports.BuildSpec = BuildSpec;
35_a = JSII_RTTI_SYMBOL_1;
36BuildSpec[_a] = { fqn: "@aws-cdk/aws-codebuild.BuildSpec", version: "1.181.1" };
37/**
38 * BuildSpec that just returns the input unchanged
39 */
40class FilenameBuildSpec extends BuildSpec {
41 constructor(filename) {
42 super();
43 this.filename = filename;
44 this.isImmediate = false;
45 }
46 toBuildSpec() {
47 return this.filename;
48 }
49 toString() {
50 return `<buildspec file: ${this.filename}>`;
51 }
52}
53/**
54 * BuildSpec that understands about structure
55 */
56class ObjectBuildSpec extends BuildSpec {
57 constructor(spec) {
58 super();
59 this.spec = spec;
60 this.isImmediate = true;
61 }
62 toBuildSpec() {
63 // We have to pretty-print the buildspec, otherwise
64 // CodeBuild will not recognize it as an inline buildspec.
65 return core_1.Lazy.uncachedString({
66 produce: (ctx) => core_1.Stack.of(ctx.scope).toJsonString(this.spec, 2),
67 });
68 }
69}
70/**
71 * BuildSpec that exports into YAML format
72 */
73class YamlBuildSpec extends BuildSpec {
74 constructor(spec) {
75 super();
76 this.spec = spec;
77 this.isImmediate = true;
78 }
79 toBuildSpec() {
80 return yaml_cfn.serialize(this.spec);
81 }
82}
83/**
84 * Merge two buildspecs into a new BuildSpec by doing a deep merge
85 *
86 * We decided to disallow merging of artifact specs, which is
87 * actually impossible since we can't merge two buildspecs with a
88 * single primary output into a buildspec with multiple outputs.
89 * In case of multiple outputs they must have identifiers but we won't have that information.
90 *
91 * In case of test reports we replace the whole object with the RHS (instead of recursively merging)
92*/
93function mergeBuildSpecs(lhs, rhs) {
94 if (!(lhs instanceof ObjectBuildSpec) || !(rhs instanceof ObjectBuildSpec)) {
95 throw new Error('Can only merge buildspecs created using BuildSpec.fromObject()');
96 }
97 if (lhs.spec.version === '0.1') {
98 throw new Error('Cannot extend buildspec at version "0.1". Set the version to "0.2" or higher instead.');
99 }
100 if (lhs.spec.artifacts && rhs.spec.artifacts) {
101 // We decided to disallow merging of artifact specs, which is
102 // actually impossible since we can't merge two buildspecs with a
103 // single primary output into a buildspec with multiple outputs.
104 // In case of multiple outputs they must have identifiers but we won't have that information.
105 throw new Error('Only one build spec is allowed to specify artifacts.');
106 }
107 const lhsSpec = JSON.parse(JSON.stringify(lhs.spec));
108 const rhsSpec = JSON.parse(JSON.stringify(rhs.spec));
109 normalizeSpec(lhsSpec);
110 normalizeSpec(rhsSpec);
111 const merged = mergeDeep(lhsSpec, rhsSpec);
112 // In case of test reports we replace the whole object with the RHS (instead of recursively merging)
113 if (lhsSpec.reports && rhsSpec.reports) {
114 merged.reports = { ...lhsSpec.reports, ...rhsSpec.reports };
115 }
116 return new ObjectBuildSpec(merged);
117}
118exports.mergeBuildSpecs = mergeBuildSpecs;
119/*
120 * Normalizes the build spec
121 * The CodeBuild runtime allows fields that are defined as string[] to be strings
122 * and interprets them as singleton lists.
123 * When merging we need to normalize this to have the correct concat semantics
124 */
125function normalizeSpec(spec) {
126 if (spec.env && typeof spec.env['exported-variables'] === 'string') {
127 spec.env['exported-variables'] = [spec.env['exported-variables']];
128 }
129 for (const key in spec.phases) {
130 if (Object.prototype.hasOwnProperty.call(spec.phases, key)) {
131 normalizeSpecPhase(spec.phases[key]);
132 }
133 }
134 if (spec.reports) {
135 for (const key in spec.reports) {
136 if (Object.prototype.hasOwnProperty.call(spec.reports, key)) {
137 const report = spec.reports[key];
138 if (typeof report.files === 'string') {
139 report.files = [report.files];
140 }
141 }
142 }
143 }
144 if (spec.artifacts) {
145 if (typeof spec.artifacts.files === 'string') {
146 spec.artifacts.files = [spec.artifacts.files];
147 }
148 for (const key in spec.artifacts['secondary-artifacts']) {
149 if (Object.prototype.hasOwnProperty.call(spec.artifacts['secondary-artifacts'], key)) {
150 const secArtifact = spec.artifacts['secondary-artifacts'][key];
151 if (typeof secArtifact.files === 'string') {
152 secArtifact.files = [secArtifact.files];
153 }
154 }
155 }
156 }
157 if (spec.cache && typeof spec.cache.paths === 'string') {
158 spec.cache.paths = [spec.cache.paths];
159 }
160}
161function normalizeSpecPhase(phase) {
162 if (phase.commands && typeof phase.commands === 'string') {
163 phase.commands = [phase.commands];
164 }
165 if (phase.finally && typeof phase.finally === 'string') {
166 phase.finally = [phase.finally];
167 }
168}
169function mergeDeep(lhs, rhs) {
170 if (Array.isArray(lhs) && Array.isArray(rhs)) {
171 return [...lhs, ...rhs];
172 }
173 if (Array.isArray(lhs) || Array.isArray(rhs)) {
174 return rhs;
175 }
176 const isObject = (obj) => obj && typeof obj === 'object';
177 if (isObject(lhs) && isObject(rhs)) {
178 const ret = { ...lhs };
179 for (const k of Object.keys(rhs)) {
180 ret[k] = k in lhs ? mergeDeep(lhs[k], rhs[k]) : rhs[k];
181 }
182 return ret;
183 }
184 return rhs;
185}
186;
187//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"build-spec.js","sourceRoot":"","sources":["build-spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { IResolveContext, Lazy, Stack } from '@aws-cdk/core';\nimport * as yaml_cfn from './private/yaml-cfn';\n\n/**\n * BuildSpec for CodeBuild projects\n */\nexport abstract class BuildSpec {\n  public static fromObject(value: { [key: string]: any }): BuildSpec {\n    return new ObjectBuildSpec(value);\n  }\n\n  /**\n   * Create a buildspec from an object that will be rendered as YAML in the resulting CloudFormation template.\n   *\n   * @param value the object containing the buildspec that will be rendered as YAML\n   */\n  public static fromObjectToYaml(value: { [key: string]: any }): BuildSpec {\n    return new YamlBuildSpec(value);\n  }\n\n  /**\n   * Use a file from the source as buildspec\n   *\n   * Use this if you want to use a file different from 'buildspec.yml'`\n   */\n  public static fromSourceFilename(filename: string): BuildSpec {\n    return new FilenameBuildSpec(filename);\n  }\n\n  /**\n   * Whether the buildspec is directly available or deferred until build-time\n   */\n  public abstract readonly isImmediate: boolean;\n\n  protected constructor() {\n  }\n\n  /**\n   * Render the represented BuildSpec\n   */\n  public abstract toBuildSpec(): string;\n}\n\n/**\n * BuildSpec that just returns the input unchanged\n */\nclass FilenameBuildSpec extends BuildSpec {\n  public readonly isImmediate: boolean = false;\n\n  constructor(private readonly filename: string) {\n    super();\n  }\n\n  public toBuildSpec(): string {\n    return this.filename;\n  }\n\n  public toString() {\n    return `<buildspec file: ${this.filename}>`;\n  }\n}\n\n/**\n * BuildSpec that understands about structure\n */\nclass ObjectBuildSpec extends BuildSpec {\n  public readonly isImmediate: boolean = true;\n\n  constructor(public readonly spec: { [key: string]: any }) {\n    super();\n  }\n\n  public toBuildSpec(): string {\n    // We have to pretty-print the buildspec, otherwise\n    // CodeBuild will not recognize it as an inline buildspec.\n    return Lazy.uncachedString({\n      produce: (ctx: IResolveContext) =>\n        Stack.of(ctx.scope).toJsonString(this.spec, 2),\n    });\n  }\n}\n\n/**\n * BuildSpec that exports into YAML format\n */\nclass YamlBuildSpec extends BuildSpec {\n  public readonly isImmediate: boolean = true;\n\n  constructor(public readonly spec: { [key: string]: any }) {\n    super();\n  }\n\n  public toBuildSpec(): string {\n    return yaml_cfn.serialize(this.spec);\n  }\n}\n\n/**\n * Merge two buildspecs into a new BuildSpec by doing a deep merge\n *\n * We decided to disallow merging of artifact specs, which is\n * actually impossible since we can't merge two buildspecs with a\n * single primary output into a buildspec with multiple outputs.\n * In case of multiple outputs they must have identifiers but we won't have that information.\n *\n * In case of test reports we replace the whole object with the RHS (instead of recursively merging)\n*/\nexport function mergeBuildSpecs(lhs: BuildSpec, rhs: BuildSpec): BuildSpec {\n  if (!(lhs instanceof ObjectBuildSpec) || !(rhs instanceof ObjectBuildSpec)) {\n    throw new Error('Can only merge buildspecs created using BuildSpec.fromObject()');\n  }\n\n  if (lhs.spec.version === '0.1') {\n    throw new Error('Cannot extend buildspec at version \"0.1\". Set the version to \"0.2\" or higher instead.');\n  }\n  if (lhs.spec.artifacts && rhs.spec.artifacts) {\n    // We decided to disallow merging of artifact specs, which is\n    // actually impossible since we can't merge two buildspecs with a\n    // single primary output into a buildspec with multiple outputs.\n    // In case of multiple outputs they must have identifiers but we won't have that information.\n    throw new Error('Only one build spec is allowed to specify artifacts.');\n  }\n\n  const lhsSpec = JSON.parse(JSON.stringify(lhs.spec));\n  const rhsSpec = JSON.parse(JSON.stringify(rhs.spec));\n\n  normalizeSpec(lhsSpec);\n  normalizeSpec(rhsSpec);\n\n  const merged = mergeDeep(lhsSpec, rhsSpec);\n\n  // In case of test reports we replace the whole object with the RHS (instead of recursively merging)\n  if (lhsSpec.reports && rhsSpec.reports) {\n    merged.reports = { ...lhsSpec.reports, ...rhsSpec.reports };\n  }\n\n  return new ObjectBuildSpec(merged);\n}\n\n/*\n * Normalizes the build spec\n * The CodeBuild runtime allows fields that are defined as string[] to be strings\n * and interprets them as singleton lists.\n * When merging we need to normalize this to have the correct concat semantics\n */\nfunction normalizeSpec(spec: { [key: string]: any }): void {\n  if (spec.env && typeof spec.env['exported-variables'] === 'string') {\n    spec.env['exported-variables'] = [spec.env['exported-variables']];\n  }\n  for (const key in spec.phases) {\n    if (Object.prototype.hasOwnProperty.call(spec.phases, key)) {\n      normalizeSpecPhase(spec.phases[key]);\n    }\n  }\n  if (spec.reports) {\n    for (const key in spec.reports) {\n      if (Object.prototype.hasOwnProperty.call(spec.reports, key)) {\n        const report = spec.reports[key];\n        if (typeof report.files === 'string') {\n          report.files = [report.files];\n        }\n      }\n    }\n  }\n  if (spec.artifacts) {\n    if (typeof spec.artifacts.files === 'string') {\n      spec.artifacts.files = [spec.artifacts.files];\n    }\n    for (const key in spec.artifacts['secondary-artifacts']) {\n      if (Object.prototype.hasOwnProperty.call(spec.artifacts['secondary-artifacts'], key)) {\n        const secArtifact = spec.artifacts['secondary-artifacts'][key];\n        if (typeof secArtifact.files === 'string') {\n          secArtifact.files = [secArtifact.files];\n        }\n      }\n    }\n  }\n  if (spec.cache && typeof spec.cache.paths === 'string') {\n    spec.cache.paths = [spec.cache.paths];\n  }\n}\n\nfunction normalizeSpecPhase(phase: { [key: string]: any }): void {\n  if (phase.commands && typeof phase.commands === 'string') {\n    phase.commands = [phase.commands];\n  }\n  if (phase.finally && typeof phase.finally === 'string') {\n    phase.finally = [phase.finally];\n  }\n}\n\nfunction mergeDeep(lhs: any, rhs: any): any {\n  if (Array.isArray(lhs) && Array.isArray(rhs)) {\n    return [...lhs, ...rhs];\n  }\n  if (Array.isArray(lhs) || Array.isArray(rhs)) {\n    return rhs;\n  }\n\n  const isObject = (obj: any) => obj && typeof obj === 'object';\n\n  if (isObject(lhs) && isObject(rhs)) {\n    const ret: any = { ...lhs };\n    for (const k of Object.keys(rhs)) {\n      ret[k] = k in lhs ? mergeDeep(lhs[k], rhs[k]) : rhs[k];\n    }\n    return ret;\n  }\n\n  return rhs;\n};"]}
\No newline at end of file