aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
227 lines • 40.9 kB
JavaScript
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.AutoPullRequest = void 0;
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
const build_env_1 = require("../build-env");
const permissions = __importStar(require("../permissions"));
/**
* Creates a CodeBuild job that, when triggered, opens a GitHub Pull Request.
*/
class AutoPullRequest extends constructs_1.Construct {
constructor(parent, id, props) {
super(parent, id);
this.props = props;
this.baseBranch = props.base?.name ?? 'master';
this.headSource = props.head.source ?? this.baseBranch;
this.exports = props.exports ?? {};
for (const ex of Object.keys(this.exports)) {
if (this.headSource.includes(`\${${ex}}`) || this.headSource.includes(`\$${ex}`)) {
throw new Error(`head source (${this.headSource}) cannot contain dynamic exports: ${ex}`);
}
}
const sshKeySecret = props.repo.sshKeySecret;
const commitEmail = props.repo.commitEmail;
const commitUsername = props.repo.commitUsername;
const cloneDepth = props.cloneDepth === undefined ? 0 : props.cloneDepth;
const needsGitHubTokenSecret = !this.props.pushOnly || !!this.props.skipIfOpenPrsWithLabels;
let commands = [
...this.configureSshAccess(),
// when the job is triggered as a CodePipeline action, the working directory
// is populated with the output artifact of the CodeCommitSourceAction, which doesn't include
// the .git directory in the zipped s3 archive. (Yeah, fun stuff).
// see https://itnext.io/how-to-access-git-metadata-in-codebuild-when-using-codepipeline-codecommit-ceacf2c5c1dc
...this.cloneIfNeeded(),
];
if (this.props.condition) {
// there's no way to stop a BuildSpec execution halfway through without throwing an error. Believe me, I
// checked the code. Instead we define a variable that we will switch all other lines on/off.
commands.push(`${this.props.condition} ` +
'&& { echo \'Skip condition is met, skipping...\' && export SKIP=true; } ' +
'|| { echo \'Skip condition is not met, continuing...\' && export SKIP=false; }');
}
// read the token
if (needsGitHubTokenSecret) {
commands.push(`export GITHUB_TOKEN=$(aws secretsmanager get-secret-value --secret-id "${this.props.repo.tokenSecretArn}" --output=text --query=SecretString)`);
}
if (this.props.skipIfOpenPrsWithLabels) {
commands.push(...this.skipIfOpenPrs(this.props.skipIfOpenPrsWithLabels));
}
commands.push(...this.createHead(), ...this.pushHead());
if (!this.props.pushOnly) {
commands.push(...this.createPullRequest());
}
// toggle all commands according to the SKIP variable.
commands = commands.map((command) => `$SKIP || { ${command} ; }`);
// intially all commands are enabled.
commands.unshift('export SKIP=false');
this.project = new aws_cdk_lib_1.aws_codebuild.Project(this, 'PullRequest', {
source: props.repo.createBuildSource(this, false, { cloneDepth }),
description: props.projectDescription,
environment: (0, build_env_1.createBuildEnvironment)(props.build ?? {}),
buildSpec: aws_cdk_lib_1.aws_codebuild.BuildSpec.fromObject({
version: '0.2',
phases: {
pre_build: {
commands: [
`git config --global user.email "${commitEmail}"`,
`git config --global user.name "${commitUsername}"`,
],
},
build: { commands },
},
}),
ssmSessionPermissions: true,
});
// Always exists as the project is not a reference
const projectRole = this.project.role;
projectRole.addManagedPolicy(aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly'));
permissions.grantSecretRead(sshKeySecret, projectRole);
if (needsGitHubTokenSecret) {
permissions.grantSecretRead({ secretArn: props.repo.tokenSecretArn }, projectRole);
}
if (props.scheduleExpression) {
const schedule = aws_cdk_lib_1.aws_events.Schedule.expression(props.scheduleExpression);
new aws_cdk_lib_1.aws_events.Rule(this, 'Scheduler', {
description: 'Schedules an automatic Pull Request for this repository',
schedule,
targets: [new aws_cdk_lib_1.aws_events_targets.CodeBuildProject(this.project)],
});
}
this.alarm = this.project.metricFailedBuilds({ period: aws_cdk_lib_1.Duration.seconds(300) }).createAlarm(this, 'AutoPullRequestFailedAlarm', {
threshold: 1,
evaluationPeriods: 1,
treatMissingData: aws_cdk_lib_1.aws_cloudwatch.TreatMissingData.IGNORE,
});
}
createHead() {
return [
// check if head branch exists
`git rev-parse --verify origin/${this.props.head.name} ` +
// checkout and merge if it does (this might fail due to merge conflicts)
`&& { git checkout ${this.props.head.name} && git merge ${this.headSource} && ${this.runCommands()}; } ` +
// create if it doesnt. we initially use 'temp' to allow using exports in the head branch name. (e.g bump/$VERSION)
`|| { git checkout ${this.headSource} && git checkout -b temp && ${this.runCommands()} && git branch -M ${this.props.head.name}; }`,
];
}
cloneIfNeeded() {
return [
// check if .git exist
'ls .git ' +
// all good
'&& { echo ".git directory exists"; } ' +
// clone if it doesn't
`|| { echo ".git directory doesnot exist - cloning..." && git init . && git remote add origin git@github.com:${this.props.repo.owner}/${this.props.repo.repo}.git && git fetch && git reset --hard origin/${this.baseBranch} && git branch -M ${this.baseBranch} && git clean -fqdx; }`,
];
}
runCommands() {
const userCommands = this.props.commands ?? [];
const exports = Object.entries(this.exports).map(entry => `export ${entry[0]}=$(${entry[1]})`);
return [
...userCommands,
// exports should be executed immediately after the user commands (not before)
// because they might need access to artifacts produced by them (e.g version file).
...exports,
'echo Finished running user commands',
].join(' && ');
}
configureSshAccess() {
return [
'aws secretsmanager get-secret-value '
+ `--secret-id "${this.props.repo.sshKeySecret.secretArn}" `
+ '--output=text --query=SecretString > ~/.ssh/id_rsa',
'mkdir -p ~/.ssh',
'chmod 0600 ~/.ssh/id_rsa ~/.ssh/config',
'ssh-keyscan -t rsa github.com >> ~/.ssh/known_hosts',
];
}
pushHead() {
// We will do nothing and set `SKIP=true` if the head ref is an ancestor of the base branch (no PR could be created)
return [
`git merge-base --is-ancestor ${this.props.head.name} origin/${this.baseBranch}`
+ ` && { echo "Skipping: ${this.props.head.name} is an ancestor of origin/${this.baseBranch}"; export SKIP=true; }`
+ ` || { echo "Pushing: ${this.props.head.name} is ahead of origin/${this.baseBranch}"; export SKIP=false; }`,
`git remote add origin_ssh ${this.props.repo.repositoryUrlSsh}`,
// Need `--atomic`, otherwise `git push` might successfully push the tags but not to `main`.
`git push --atomic --follow-tags origin_ssh ${this.props.head.name}:${this.props.head.name}`,
];
}
skipIfOpenPrs(labels) {
const filters = [
`repo:${this.props.repo.owner}/${this.props.repo.repo}`,
'is:pr',
'is:open',
...labels.map(l => `label:${l}`),
];
return [
`${this.githubCurlGet(`/search/issues?q=${encodeURIComponent(filters.join(' '))}`, '-o search.json')}`,
'node -e \'process.exitCode = require("./search.json").total_count\''
+ ` || { echo "Found open PRs with label ${labels}, skipping PR."; export SKIP=true; }`,
];
}
createPullRequest() {
const head = this.props.head.name;
const base = this.baseBranch;
if (head === base) {
throw new Error(`Head branch ("${base}") is the same as the base branch ("${head}")`);
}
const props = this.props;
const title = props.title ?? `Merge ${head} to ${base}`;
const body = this.props.body ?? '';
const createRequest = { title, base, head };
const commands = [];
// create the PR
commands.push(`${this.githubCurl('/pulls', '-X POST -o pr.json', createRequest)} && export PR_NUMBER=$(node -p 'require("./pr.json").number')`);
// update the body
commands.push(this.githubCurl('/pulls/$PR_NUMBER', '-X PATCH', { body: body }));
if (this.props.labels && this.props.labels.length > 0) {
// apply labels.
commands.push(this.githubCurl('/issues/$PR_NUMBER/labels', '-X POST', { labels: this.props.labels }));
}
return commands;
}
githubCurl(uri, command, request) {
return [
'curl --fail',
command,
'--header "Authorization: token $GITHUB_TOKEN"',
'--header "Content-Type: application/json"',
`-d ${JSON.stringify(JSON.stringify(request))}`,
`https://api.github.com/repos/${this.props.repo.owner}/${this.props.repo.repo}${uri}`,
].join(' ');
}
githubCurlGet(uri, command) {
return [
'curl --fail',
command,
'--header "Authorization: token $GITHUB_TOKEN"',
'--header "Content-Type: application/json"',
`'https://api.github.com${uri}'`,
].join(' ');
}
}
exports.AutoPullRequest = AutoPullRequest;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pr.js","sourceRoot":"","sources":["pr.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,6CAOqB;AACrB,2CAAuC;AACvC,4CAA6E;AAC7E,4DAA8C;AA2L9C;;GAEG;AACH,MAAa,eAAgB,SAAQ,sBAAS;IAkB5C,YAAY,MAAiB,EAAE,EAAU,EAAE,KAA2B;QACpE,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QAElB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,IAAI,EAAE,IAAI,IAAI,QAAQ,CAAC;QAC/C,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,UAAU,CAAC;QACvD,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,IAAI,EAAE,CAAC;QAEnC,KAAK,MAAM,EAAE,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YAC1C,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,EAAE,EAAE,CAAC,EAAE;gBAChF,MAAM,IAAI,KAAK,CAAC,gBAAgB,IAAI,CAAC,UAAU,qCAAqC,EAAE,EAAE,CAAC,CAAC;aAC3F;SACF;QAED,MAAM,YAAY,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC;QAC7C,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC;QAC3C,MAAM,cAAc,GAAG,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC;QACjD,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC;QAEzE,MAAM,sBAAsB,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,CAAC;QAE5F,IAAI,QAAQ,GAAa;YAEvB,GAAG,IAAI,CAAC,kBAAkB,EAAE;YAE5B,4EAA4E;YAC5E,6FAA6F;YAC7F,kEAAkE;YAClE,gHAAgH;YAChH,GAAG,IAAI,CAAC,aAAa,EAAE;SACxB,CAAC;QAEF,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE;YACxB,wGAAwG;YACxG,6FAA6F;YAC7F,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG;gBACxC,0EAA0E;gBAC1E,gFAAgF,CAAC,CAAC;SACnF;QAED,iBAAiB;QACjB,IAAI,sBAAsB,EAAE;YAC1B,QAAQ,CAAC,IAAI,CAAC,0EAA0E,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,cAAc,uCAAuC,CAAC,CAAC;SAChK;QAED,IAAI,IAAI,CAAC,KAAK,CAAC,uBAAuB,EAAE;YACtC,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC,CAAC;SAC1E;QAED,QAAQ,CAAC,IAAI,CACX,GAAG,IAAI,CAAC,UAAU,EAAE,EACpB,GAAG,IAAI,CAAC,QAAQ,EAAE,CACnB,CAAC;QAEF,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE;YACxB,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;SAC5C;QAED,sDAAsD;QACtD,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAe,EAAE,EAAE,CAAC,cAAc,OAAO,MAAM,CAAC,CAAC;QAE1E,qCAAqC;QACrC,QAAQ,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC;QAEtC,IAAI,CAAC,OAAO,GAAG,IAAI,2BAAM,CAAC,OAAO,CAAC,IAAI,EAAE,aAAa,EAAE;YACrD,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,UAAU,EAAE,CAAC;YACjE,WAAW,EAAE,KAAK,CAAC,kBAAkB;YACrC,WAAW,EAAE,IAAA,kCAAsB,EAAC,KAAK,CAAC,KAAK,IAAI,EAAE,CAAC;YACtD,SAAS,EAAE,2BAAM,CAAC,SAAS,CAAC,UAAU,CAAC;gBACrC,OAAO,EAAE,KAAK;gBACd,MAAM,EAAE;oBACN,SAAS,EAAE;wBACT,QAAQ,EAAE;4BACR,mCAAmC,WAAW,GAAG;4BACjD,kCAAkC,cAAc,GAAG;yBACpD;qBACF;oBACD,KAAK,EAAE,EAAE,QAAQ,EAAE;iBACpB;aACF,CAAC;YACF,qBAAqB,EAAE,IAAI;SAC5B,CAAC,CAAC;QAEH,kDAAkD;QAClD,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,IAAK,CAAC;QACvC,WAAW,CAAC,gBAAgB,CAAC,qBAAG,CAAC,aAAa,CAAC,wBAAwB,CAAC,8CAA8C,CAAC,CAAC,CAAC;QACzH,WAAW,CAAC,eAAe,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;QACvD,IAAI,sBAAsB,EAAE;YAC1B,WAAW,CAAC,eAAe,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,WAAW,CAAC,CAAC;SACpF;QAED,IAAI,KAAK,CAAC,kBAAkB,EAAE;YAC5B,MAAM,QAAQ,GAAG,wBAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;YACtE,IAAI,wBAAM,CAAC,IAAI,CAAC,IAAI,EAAE,WAAW,EAAE;gBACjC,WAAW,EAAE,yDAAyD;gBACtE,QAAQ;gBACR,OAAO,EAAE,CAAC,IAAI,gCAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;aAC7D,CAAC,CAAC;SACJ;QAED,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,kBAAkB,CAAC,EAAE,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,4BAA4B,EAAE;YAC9H,SAAS,EAAE,CAAC;YACZ,iBAAiB,EAAE,CAAC;YACpB,gBAAgB,EAAE,4BAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC,CAAC;IACL,CAAC;IACO,UAAU;QAEhB,OAAO;YACL,8BAA8B;YAC9B,iCAAiC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG;gBAExD,yEAAyE;gBACzE,qBAAqB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,iBAAiB,IAAI,CAAC,UAAU,OAAO,IAAI,CAAC,WAAW,EAAE,OAAO;gBAEzG,mHAAmH;gBACnH,qBAAqB,IAAI,CAAC,UAAU,+BAA+B,IAAI,CAAC,WAAW,EAAE,qBAAqB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK;SAEpI,CAAC;IAEJ,CAAC;IAEO,aAAa;QAEnB,OAAO;YACL,sBAAsB;YACtB,UAAU;gBAEV,WAAW;gBACX,wCAAwC;gBAExC,sBAAsB;gBACtB,+GAA+G,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,gDAAgD,IAAI,CAAC,UAAU,qBAAqB,IAAI,CAAC,UAAU,wBAAwB;SAExR,CAAC;IAEJ,CAAC;IAEO,WAAW;QAEjB,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,EAAE,CAAC;QAC/C,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,UAAU,KAAK,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QAE/F,OAAO;YAEL,GAAG,YAAY;YAEf,8EAA8E;YAC9E,mFAAmF;YACnF,GAAG,OAAO;YAEV,qCAAqC;SACtC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAEjB,CAAC;IAEO,kBAAkB;QAExB,OAAO;YACL,sCAAsC;kBAClC,gBAAgB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,IAAI;kBAC1D,oDAAoD;YACxD,iBAAiB;YACjB,wCAAwC;YACxC,qDAAqD;SACtD,CAAC;IAEJ,CAAC;IAEO,QAAQ;QACd,oHAAoH;QACpH,OAAO;YACL,gCAAgC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,WAAW,IAAI,CAAC,UAAU,EAAE;kBAC5E,yBAAyB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,6BAA6B,IAAI,CAAC,UAAU,wBAAwB;kBACjH,wBAAwB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,uBAAuB,IAAI,CAAC,UAAU,yBAAyB;YAC/G,6BAA6B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,gBAAgB,EAAE;YAC/D,4FAA4F;YAC5F,8CAA8C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE;SAC7F,CAAC;IACJ,CAAC;IAEO,aAAa,CAAC,MAAgB;QACpC,MAAM,OAAO,GAAG;YACd,QAAQ,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE;YACvD,OAAO;YACP,SAAS;YACT,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC;SACjC,CAAC;QAEF,OAAO;YACL,GAAG,IAAI,CAAC,aAAa,CAAC,oBAAoB,kBAAkB,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,gBAAgB,CAAC,EAAE;YACtG,qEAAqE;kBACjE,yCAAyC,MAAM,sCAAsC;SAC1F,CAAC;IACJ,CAAC;IAGO,iBAAiB;QAEvB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC;QAClC,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC;QAE7B,IAAI,IAAI,KAAK,IAAI,EAAE;YACjB,MAAM,IAAI,KAAK,CAAC,iBAAiB,IAAI,uCAAuC,IAAI,IAAI,CAAC,CAAC;SACvF;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;QACzB,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,IAAI,SAAS,IAAI,OAAO,IAAI,EAAE,CAAC;QACxD,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,EAAE,CAAC;QAEnC,MAAM,aAAa,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC;QAE5C,MAAM,QAAQ,GAAG,EAAE,CAAC;QAEpB,gBAAgB;QAChB,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,oBAAoB,EAAE,aAAa,CAAC,+DAA+D,CAAC,CAAC;QAEhJ,kBAAkB;QAClB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,mBAAmB,EAAE,UAAU,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QAEhF,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE;YACvD,gBAAgB;YACd,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,2BAA2B,EAAE,SAAS,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;SACvG;QAED,OAAO,QAAQ,CAAC;IAElB,CAAC;IAEO,UAAU,CAAC,GAAW,EAAE,OAAe,EAAE,OAAY;QAC3D,OAAO;YACL,aAAa;YACb,OAAO;YACP,+CAA+C;YAC/C,2CAA2C;YAC3C,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,EAAE;YAC/C,gCAAgC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,GAAG,EAAE;SACtF,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACd,CAAC;IAEO,aAAa,CAAC,GAAW,EAAE,OAAe;QAChD,OAAO;YACL,aAAa;YACb,OAAO;YACP,+CAA+C;YAC/C,2CAA2C;YAC3C,0BAA0B,GAAG,GAAG;SACjC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACd,CAAC;CAEF;AA7QD,0CA6QC","sourcesContent":["import {\n  Duration,\n  aws_cloudwatch as cloudwatch,\n  aws_codebuild as cbuild,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { BuildEnvironmentProps, createBuildEnvironment } from '../build-env';\nimport * as permissions from '../permissions';\nimport { WritableGitHubRepo } from '../repo';\n\n/**\n * Properties for creating a Pull Request Job.\n */\nexport interface AutoPullRequestOptions {\n  /**\n   * The base branch of the PR.\n   *\n   * @default 'master'\n   */\n  base?: Base;\n\n  /**\n   * True if you only want to push the head branch without creating a PR.\n   * Useful when used along with 'commits' to execute a commit-and-push automatically.\n   *\n   * // TODO: Consider moving this functionality to a separate construct.\n   *\n   * @default false\n   */\n  readonly pushOnly?: boolean;\n\n  /**\n   * Title of the PR.\n   *\n   * @default `Merge ${head} to ${base}`\n   */\n  title?: string;\n\n  /**\n   * Body the PR. Note that the body is updated post PR creation,\n   * this means you can use the $PR_NUMBER env variable to refer to the PR itself.\n   *\n   * @default - no body.\n   */\n  body?: string;\n\n  /**\n   * Labels applied to the PR.\n   *\n   * @default - no labels.\n   */\n  labels?: string[];\n\n  /**\n   * Build environment for the CodeBuild job.\n   *\n   * @default - default configuration.\n   */\n  build?: BuildEnvironmentProps;\n\n  /**\n   * Git clone depth.\n   *\n   * @default 0 (clones the entire repository revisions)\n   */\n  cloneDepth?: number;\n\n  /**\n   * Key value pairs of variables to export. These variables will be available for dynamic evaluation in any\n   * subsequent command.\n   *\n   * Key - Variable name (e.g VERSION)\n   * Value - Command that evaluates to the value of the variable (e.g 'git describe')\n   *\n   * Example:\n   *\n   * Configure an export in the form of:\n   *\n   * { 'VERSION': 'git describe' }\n   *\n   * Use the $VERSION variable in the PR title: 'chore(release): $VERSION'\n   *\n   * Note that these exports are executed after the `commands` execution,\n   * so they have access to the artifacts said commands produce (e.g version bump).\n   *\n   * @default - no exports\n   */\n  exports?: { [key: string]: string };\n\n  /**\n   * The schedule to produce an automatic PR.\n   *\n   * The expression can be one of:\n   *\n   *  - cron expression, such as \"cron(0 12 * * ? *)\" will trigger every day at 12pm UTC\n   *  - rate expression, such as \"rate(1 day)\" will trigger every 24 hours from the time of deployment\n   *\n   * @see https://docs.aws.amazon.com/AmazonCloudWatch/latest/events/ScheduledEvents.html\n   *\n   * @default - no schedule, should be triggered manually.\n   */\n  scheduleExpression?: string;\n\n}\n\nexport interface AutoPullRequestProps extends AutoPullRequestOptions {\n  /**\n   * The repository to create a PR in.\n   */\n  repo: WritableGitHubRepo;\n\n  /**\n   * A set of commands to run against the head branch.\n   * Useful for things like version bumps or any auto-generated commits.\n   *\n   * Note that you cannot use export keys in these commands (See `exports` property)\n   *\n   * @default - no commands.\n   */\n  commands?: string[];\n\n  /**\n   * The head branch of the PR.\n   */\n  head: Head;\n\n  /**\n   * The exit code of this command determines whether or not to proceed with the\n   * PR creation. If configured, this command is the first one to run, and if it fails, all\n   * other commands will be skipped.\n   *\n   * This command is the first to execute, and should not assume any pre-existing state.\n   *\n   * @default - no condition\n   */\n  condition?: string;\n\n  /**\n   * If any PR labeled with the given labels is still open, no new PR will be created\n   *\n   * @default - don't look at open PRs\n   */\n  readonly skipIfOpenPrsWithLabels?: string[];\n\n  /**\n   * Description string for the CodeBuild project\n   *\n   * @default - No description\n   */\n  readonly projectDescription?: string;\n}\n\n/**\n * Properties for configuring the base branch of the PR.\n */\nexport interface Base {\n\n  /**\n   * Branch name.\n   *\n   * This branch must exist.\n   *\n   * @default 'master'\n   */\n  readonly name?: string;\n}\n\n/**\n * Properties for configuring the head branch of the PR.\n */\nexport interface Head {\n\n  /**\n   * Branch name.\n   *\n   * This branch will be created if it doesn't exist.\n   */\n  readonly name: string;\n\n  /**\n   * The source sha of the branch.\n   *\n   * If the given branch already exists, this sha will be auto-merged onto it. Note that in such a case,\n   * the PR creation might fail in case there are merge conflicts.\n   *\n   * If the given branch doesn't exist, the newly created branch will be based of this hash.\n   *\n   * Note that dynamic exports are not allowed for this property.\n   *\n   * @default - the base branch of the pr.\n   */\n  readonly source?: string;\n}\n\n/**\n * Creates a CodeBuild job that, when triggered, opens a GitHub Pull Request.\n */\nexport class AutoPullRequest extends Construct {\n\n  /**\n   * CloudWatch alarm that will be triggered if the job fails.\n   */\n  public readonly alarm: cloudwatch.Alarm;\n\n  /**\n   * The CodeBuild project this construct creates.\n   */\n  public readonly project: cbuild.IProject;\n\n  private readonly props: AutoPullRequestProps;\n\n  private readonly baseBranch: string;\n  private readonly headSource: string;\n  private readonly exports: { [key: string]: string };\n\n  constructor(parent: Construct, id: string, props: AutoPullRequestProps) {\n    super(parent, id);\n\n    this.props = props;\n\n    this.baseBranch = props.base?.name ?? 'master';\n    this.headSource = props.head.source ?? this.baseBranch;\n    this.exports = props.exports ?? {};\n\n    for (const ex of Object.keys(this.exports)) {\n      if (this.headSource.includes(`\\${${ex}}`) || this.headSource.includes(`\\$${ex}`)) {\n        throw new Error(`head source (${this.headSource}) cannot contain dynamic exports: ${ex}`);\n      }\n    }\n\n    const sshKeySecret = props.repo.sshKeySecret;\n    const commitEmail = props.repo.commitEmail;\n    const commitUsername = props.repo.commitUsername;\n    const cloneDepth = props.cloneDepth === undefined ? 0 : props.cloneDepth;\n\n    const needsGitHubTokenSecret = !this.props.pushOnly || !!this.props.skipIfOpenPrsWithLabels;\n\n    let commands: string[] = [\n\n      ...this.configureSshAccess(),\n\n      // when the job is triggered as a CodePipeline action, the working directory\n      // is populated with the output artifact of the CodeCommitSourceAction, which doesn't include\n      // the .git directory in the zipped s3 archive. (Yeah, fun stuff).\n      // see https://itnext.io/how-to-access-git-metadata-in-codebuild-when-using-codepipeline-codecommit-ceacf2c5c1dc\n      ...this.cloneIfNeeded(),\n    ];\n\n    if (this.props.condition) {\n      // there's no way to stop a BuildSpec execution halfway through without throwing an error. Believe me, I\n      // checked the code. Instead we define a variable that we will switch all other lines on/off.\n      commands.push(`${this.props.condition} ` +\n      '&& { echo \\'Skip condition is met, skipping...\\' && export SKIP=true; } ' +\n      '|| { echo \\'Skip condition is not met, continuing...\\' && export SKIP=false; }');\n    }\n\n    // read the token\n    if (needsGitHubTokenSecret) {\n      commands.push(`export GITHUB_TOKEN=$(aws secretsmanager get-secret-value --secret-id \"${this.props.repo.tokenSecretArn}\" --output=text --query=SecretString)`);\n    }\n\n    if (this.props.skipIfOpenPrsWithLabels) {\n      commands.push(...this.skipIfOpenPrs(this.props.skipIfOpenPrsWithLabels));\n    }\n\n    commands.push(\n      ...this.createHead(),\n      ...this.pushHead(),\n    );\n\n    if (!this.props.pushOnly) {\n      commands.push(...this.createPullRequest());\n    }\n\n    // toggle all commands according to the SKIP variable.\n    commands = commands.map((command: string) => `$SKIP || { ${command} ; }`);\n\n    // intially all commands are enabled.\n    commands.unshift('export SKIP=false');\n\n    this.project = new cbuild.Project(this, 'PullRequest', {\n      source: props.repo.createBuildSource(this, false, { cloneDepth }),\n      description: props.projectDescription,\n      environment: createBuildEnvironment(props.build ?? {}),\n      buildSpec: cbuild.BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          pre_build: {\n            commands: [\n              `git config --global user.email \"${commitEmail}\"`,\n              `git config --global user.name \"${commitUsername}\"`,\n            ],\n          },\n          build: { commands },\n        },\n      }),\n      ssmSessionPermissions: true,\n    });\n\n    // Always exists as the project is not a reference\n    const projectRole = this.project.role!;\n    projectRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly'));\n    permissions.grantSecretRead(sshKeySecret, projectRole);\n    if (needsGitHubTokenSecret) {\n      permissions.grantSecretRead({ secretArn: props.repo.tokenSecretArn }, projectRole);\n    }\n\n    if (props.scheduleExpression) {\n      const schedule = events.Schedule.expression(props.scheduleExpression);\n      new events.Rule(this, 'Scheduler', {\n        description: 'Schedules an automatic Pull Request for this repository',\n        schedule,\n        targets: [new events_targets.CodeBuildProject(this.project)],\n      });\n    }\n\n    this.alarm = this.project.metricFailedBuilds({ period: Duration.seconds(300) }).createAlarm(this, 'AutoPullRequestFailedAlarm', {\n      threshold: 1,\n      evaluationPeriods: 1,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    });\n  }\n  private createHead(): string[] {\n\n    return [\n      // check if head branch exists\n      `git rev-parse --verify origin/${this.props.head.name} ` +\n\n      // checkout and merge if it does (this might fail due to merge conflicts)\n      `&& { git checkout ${this.props.head.name} && git merge ${this.headSource} && ${this.runCommands()};  } ` +\n\n      // create if it doesnt. we initially use 'temp' to allow using exports in the head branch name. (e.g bump/$VERSION)\n      `|| { git checkout ${this.headSource} && git checkout -b temp && ${this.runCommands()} && git branch -M ${this.props.head.name}; }`,\n\n    ];\n\n  }\n\n  private cloneIfNeeded(): string[] {\n\n    return [\n      // check if .git exist\n      'ls .git ' +\n\n      // all good\n      '&& { echo \".git directory exists\";  } ' +\n\n      // clone if it doesn't\n      `|| { echo \".git directory doesnot exist - cloning...\" && git init . && git remote add origin git@github.com:${this.props.repo.owner}/${this.props.repo.repo}.git && git fetch && git reset --hard origin/${this.baseBranch} && git branch -M ${this.baseBranch} && git clean -fqdx; }`,\n\n    ];\n\n  }\n\n  private runCommands(): string {\n\n    const userCommands = this.props.commands ?? [];\n    const exports = Object.entries(this.exports).map(entry => `export ${entry[0]}=$(${entry[1]})`);\n\n    return [\n\n      ...userCommands,\n\n      // exports should be executed immediately after the user commands (not before)\n      // because they might need access to artifacts produced by them (e.g version file).\n      ...exports,\n\n      'echo Finished running user commands',\n    ].join(' && ');\n\n  }\n\n  private configureSshAccess(): string[] {\n\n    return [\n      'aws secretsmanager get-secret-value '\n        + `--secret-id \"${this.props.repo.sshKeySecret.secretArn}\" `\n        + '--output=text --query=SecretString > ~/.ssh/id_rsa',\n      'mkdir -p ~/.ssh',\n      'chmod 0600 ~/.ssh/id_rsa ~/.ssh/config',\n      'ssh-keyscan -t rsa github.com >> ~/.ssh/known_hosts',\n    ];\n\n  }\n\n  private pushHead(): string[] {\n    // We will do nothing and set `SKIP=true` if the head ref is an ancestor of the base branch (no PR could be created)\n    return [\n      `git merge-base --is-ancestor ${this.props.head.name} origin/${this.baseBranch}`\n        + ` && { echo \"Skipping: ${this.props.head.name} is an ancestor of origin/${this.baseBranch}\"; export SKIP=true; }`\n        + ` || { echo \"Pushing: ${this.props.head.name} is ahead of origin/${this.baseBranch}\"; export SKIP=false; }`,\n      `git remote add origin_ssh ${this.props.repo.repositoryUrlSsh}`,\n      // Need `--atomic`, otherwise `git push` might successfully push the tags but not to `main`.\n      `git push --atomic --follow-tags origin_ssh ${this.props.head.name}:${this.props.head.name}`,\n    ];\n  }\n\n  private skipIfOpenPrs(labels: string[]): string[] {\n    const filters = [\n      `repo:${this.props.repo.owner}/${this.props.repo.repo}`,\n      'is:pr',\n      'is:open',\n      ...labels.map(l => `label:${l}`),\n    ];\n\n    return [\n      `${this.githubCurlGet(`/search/issues?q=${encodeURIComponent(filters.join(' '))}`, '-o search.json')}`,\n      'node -e \\'process.exitCode = require(\"./search.json\").total_count\\''\n        + ` || { echo \"Found open PRs with label ${labels}, skipping PR.\"; export SKIP=true; }`,\n    ];\n  }\n\n\n  private createPullRequest(): string[] {\n\n    const head = this.props.head.name;\n    const base = this.baseBranch;\n\n    if (head === base) {\n      throw new Error(`Head branch (\"${base}\") is the same as the base branch (\"${head}\")`);\n    }\n\n    const props = this.props;\n    const title = props.title ?? `Merge ${head} to ${base}`;\n    const body = this.props.body ?? '';\n\n    const createRequest = { title, base, head };\n\n    const commands = [];\n\n    // create the PR\n    commands.push(`${this.githubCurl('/pulls', '-X POST -o pr.json', createRequest)} && export PR_NUMBER=$(node -p 'require(\"./pr.json\").number')`);\n\n    // update the body\n    commands.push(this.githubCurl('/pulls/$PR_NUMBER', '-X PATCH', { body: body }));\n\n    if (this.props.labels && this.props.labels.length > 0) {\n    // apply labels.\n      commands.push(this.githubCurl('/issues/$PR_NUMBER/labels', '-X POST', { labels: this.props.labels }));\n    }\n\n    return commands;\n\n  }\n\n  private githubCurl(uri: string, command: string, request: any): string {\n    return [\n      'curl --fail',\n      command,\n      '--header \"Authorization: token $GITHUB_TOKEN\"',\n      '--header \"Content-Type: application/json\"',\n      `-d ${JSON.stringify(JSON.stringify(request))}`,\n      `https://api.github.com/repos/${this.props.repo.owner}/${this.props.repo.repo}${uri}`,\n    ].join(' ');\n  }\n\n  private githubCurlGet(uri: string, command: string): string {\n    return [\n      'curl --fail',\n      command,\n      '--header \"Authorization: token $GITHUB_TOKEN\"',\n      '--header \"Content-Type: application/json\"',\n      `'https://api.github.com${uri}'`,\n    ].join(' ');\n  }\n\n}\n\n"]}