-
Notifications
You must be signed in to change notification settings - Fork 63
/
Copy pathdocker.js
163 lines (137 loc) · 5.64 KB
/
docker.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
const cp = require('child_process');
const core = require('@actions/core');
const fs = require('fs');
const { context } = require('@actions/github');
const { isGitHubTag, isBranch, isPullRequest, branchRefToSlug, prRefToSlug, tagRefToSlug } = require('./github');
const { timestamp, cpOptions, asBool } = require('./utils');
const GITHUB_REGISTRY_URLS = ['docker.pkg.github.com', 'ghcr.io'];
// Create the full Docker image name with registry prefix (without tags)
const createFullImageName = (registry, image, githubOwner) => {
if (GITHUB_REGISTRY_URLS.includes(registry)) {
return `${registry}/${githubOwner.toLowerCase()}/${image}`;
}
return `${registry}/${image}`;
};
// Create Docker tags based on input flags & Git branch
const createTags = (addLatest, addTimestamp) => {
core.info('Creating Docker image tags...');
const { sha } = context;
const ref = context.ref.toLowerCase();
const shortSha = sha.substring(0, 7);
const dockerTags = [];
if (isGitHubTag(ref)) {
// If GitHub tag exists, use it as the Docker tag
dockerTags.push(tagRefToSlug(ref));
} else if (isBranch(ref)) {
// If we're not building a tag, use branch-prefix-{GIT_SHORT_SHA) as the Docker tag
// refs/heads/jira-123/feature/something
const safeBranchName = branchRefToSlug(ref);
const baseTag = `${safeBranchName}-${shortSha}`;
const tag = addTimestamp ? `${baseTag}-${timestamp()}` : baseTag;
dockerTags.push(tag);
} else if (isPullRequest(ref)) {
// pull_request event, use pr-{PR_NUMBER)-{GIT_SHORT_SHA) as the Docker tag
const safePrRefName = prRefToSlug(ref);
const baseTag = `pr-${safePrRefName}-${shortSha}`;
const tag = addTimestamp ? `${baseTag}-${timestamp()}` : baseTag;
dockerTags.push(tag);
} else {
core.setFailed(
'Unsupported GitHub event - only supports push https://help.github.com/en/articles/events-that-trigger-workflows#push or' +
' pull https://docs.github.com/en/actions/using-workflows/events-that-trigger-workflows#pull_request'
);
}
if (asBool(addLatest)) {
dockerTags.push('latest');
}
core.info(`Docker tags created: ${dockerTags}`);
return dockerTags;
};
// Dynamically create 'docker build' command based on inputs provided
const createBuildCommand = (imageName, dockerfile, buildOpts) => {
const tagsSuffix = buildOpts.tags.map(tag => `-t ${imageName}:${tag}`).join(' ');
const builder = buildOpts.multiPlatform ? 'buildx build' : 'build';
let buildCommandPrefix = `docker ${builder} -f ${dockerfile} ${tagsSuffix}`;
if (buildOpts.buildArgs) {
const argsSuffix = buildOpts.buildArgs.map(arg => `--build-arg ${arg}`).join(' ');
buildCommandPrefix = `${buildCommandPrefix} ${argsSuffix}`;
}
if (buildOpts.labels) {
const labelsSuffix = buildOpts.labels.map(label => `--label ${label}`).join(' ');
buildCommandPrefix = `${buildCommandPrefix} ${labelsSuffix}`;
}
if (buildOpts.target) {
buildCommandPrefix = `${buildCommandPrefix} --target ${buildOpts.target}`;
}
if (buildOpts.platform) {
buildCommandPrefix = `${buildCommandPrefix} --platform ${buildOpts.platform}`;
}
if (buildOpts.multiPlatform && !buildOpts.skipPush) {
buildCommandPrefix = `${buildCommandPrefix} --push`;
}
if (buildOpts.ssh) {
const sshSuffix = buildOpts.ssh.map(ssh => `--ssh ${ssh}`).join(' ');
buildCommandPrefix = `${buildCommandPrefix} ${sshSuffix}`;
}
if (buildOpts.enableBuildKit) {
buildCommandPrefix = `DOCKER_BUILDKIT=1 ${buildCommandPrefix}`;
}
core.info(`BuildCommand ${buildCommandPrefix} ${buildOpts.buildDir}`);
return `${buildCommandPrefix} ${buildOpts.buildDir}`;
};
// Perform 'docker build' command
const build = (imageName, dockerfile, buildOpts) => {
if (!fs.existsSync(dockerfile)) {
core.setFailed(`Dockerfile does not exist in location ${dockerfile}`);
}
// Setup buildx driver
if (buildOpts.multiPlatform && !buildOpts.overrideDriver) {
cp.execSync('docker buildx create --use');
}
core.info(`Building Docker image ${imageName} with tags ${buildOpts.tags}...`);
cp.execSync(createBuildCommand(imageName, dockerfile, buildOpts), cpOptions);
};
const isEcr = registry => registry && registry.includes('amazonaws');
const getRegion = registry => registry.substring(registry.indexOf('ecr.') + 4, registry.indexOf('.amazonaws'));
// Log in to provided Docker registry
const login = (username, password, registry, skipPush) => {
if (skipPush) {
core.info('Input skipPush is set to true, skipping Docker log in step...');
return;
}
// If using ECR, use the AWS CLI login command in favor of docker login
if (isEcr(registry)) {
const region = getRegion(registry);
core.info(`Logging into ECR region ${region}...`);
cp.execSync(
`aws ecr get-login-password --region ${region} | docker login --username AWS --password-stdin ${registry}`
);
} else if (username && password) {
core.info(`Logging into Docker registry ${registry}...`);
cp.execSync(`docker login -u ${username} --password-stdin ${registry}`, {
input: password
});
} else {
core.setFailed('Must supply Docker registry credentials to push image!');
}
};
// Push Docker image & all tags
const push = (imageName, tags, buildOpts) => {
if (buildOpts?.multiPlatform) {
core.info('Input multiPlatform is set to true, skipping Docker push step...');
return;
}
if (buildOpts?.skipPush) {
core.info('Input skipPush is set to true, skipping Docker push step...');
return;
}
core.info(`Pushing tags ${tags} for Docker image ${imageName}...`);
cp.execSync(`docker push ${imageName} --all-tags`, cpOptions);
};
module.exports = {
createFullImageName,
createTags,
build,
login,
push
};