2020-10-25 14:13:43 +00:00
|
|
|
import * as handlebars from 'handlebars';
|
2020-12-24 03:13:41 +00:00
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as path from 'path';
|
2020-12-23 21:09:38 +00:00
|
|
|
import moment from 'moment';
|
2020-11-17 22:31:03 +00:00
|
|
|
import * as semver from 'semver';
|
2020-12-24 03:13:41 +00:00
|
|
|
import {Inputs, tmpDir} from './context';
|
2021-03-29 11:04:53 +00:00
|
|
|
import * as tcl from './tag';
|
|
|
|
import * as fcl from './flavor';
|
2020-12-01 04:50:39 +00:00
|
|
|
import * as core from '@actions/core';
|
2020-10-25 01:25:23 +00:00
|
|
|
import {Context} from '@actions/github/lib/context';
|
|
|
|
import {ReposGetResponseData} from '@octokit/types';
|
|
|
|
|
2020-10-26 00:39:21 +00:00
|
|
|
export interface Version {
|
2020-11-17 22:31:03 +00:00
|
|
|
main: string | undefined;
|
|
|
|
partial: string[];
|
2021-03-29 11:04:53 +00:00
|
|
|
latest: boolean | undefined;
|
2020-10-26 00:39:21 +00:00
|
|
|
}
|
|
|
|
|
2020-10-25 01:25:23 +00:00
|
|
|
export class Meta {
|
2020-12-01 04:38:08 +00:00
|
|
|
public readonly version: Version;
|
|
|
|
|
2020-10-25 01:25:23 +00:00
|
|
|
private readonly inputs: Inputs;
|
|
|
|
private readonly context: Context;
|
|
|
|
private readonly repo: ReposGetResponseData;
|
2021-03-29 11:04:53 +00:00
|
|
|
private readonly tags: tcl.Tag[];
|
|
|
|
private readonly flavor: fcl.Flavor;
|
2020-10-25 14:13:43 +00:00
|
|
|
private readonly date: Date;
|
2020-10-25 01:25:23 +00:00
|
|
|
|
|
|
|
constructor(inputs: Inputs, context: Context, repo: ReposGetResponseData) {
|
|
|
|
this.inputs = inputs;
|
|
|
|
this.context = context;
|
|
|
|
this.repo = repo;
|
2021-03-29 11:04:53 +00:00
|
|
|
this.tags = tcl.Transform(inputs.tags);
|
|
|
|
this.flavor = fcl.Transform(inputs.flavor);
|
2020-10-25 14:13:43 +00:00
|
|
|
this.date = new Date();
|
2020-12-01 04:38:08 +00:00
|
|
|
this.version = this.getVersion();
|
2020-10-25 01:25:23 +00:00
|
|
|
}
|
|
|
|
|
2020-12-01 04:38:08 +00:00
|
|
|
private getVersion(): Version {
|
2020-12-04 17:12:39 +00:00
|
|
|
let version: Version = {
|
2020-11-17 22:31:03 +00:00
|
|
|
main: undefined,
|
|
|
|
partial: [],
|
2021-03-29 11:04:53 +00:00
|
|
|
latest: undefined
|
2020-10-26 00:39:21 +00:00
|
|
|
};
|
|
|
|
|
2021-03-29 11:04:53 +00:00
|
|
|
for (const tag of this.tags) {
|
2021-03-30 11:11:51 +00:00
|
|
|
if (tag.attrs['enable'] == 'false') {
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
switch (tag.type) {
|
|
|
|
case tcl.Type.Schedule: {
|
|
|
|
version = this.procSchedule(version, tag);
|
|
|
|
break;
|
2020-10-26 00:39:21 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
case tcl.Type.Semver: {
|
|
|
|
version = this.procSemver(version, tag);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case tcl.Type.Match: {
|
|
|
|
version = this.procMatch(version, tag);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case tcl.Type.Ref: {
|
|
|
|
if (tag.attrs['event'] == tcl.RefEvent.Branch) {
|
|
|
|
version = this.procRefBranch(version, tag);
|
|
|
|
} else if (tag.attrs['event'] == tcl.RefEvent.Tag) {
|
|
|
|
version = this.procRefTag(version, tag);
|
|
|
|
} else if (tag.attrs['event'] == tcl.RefEvent.PR) {
|
|
|
|
version = this.procRefPr(version, tag);
|
2020-11-17 22:31:03 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case tcl.Type.Edge: {
|
|
|
|
version = this.procEdge(version, tag);
|
|
|
|
break;
|
2020-11-17 22:31:03 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
case tcl.Type.Raw: {
|
|
|
|
version = this.procRaw(version, tag);
|
|
|
|
break;
|
2020-10-27 01:32:26 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
case tcl.Type.Sha: {
|
|
|
|
version = this.procSha(version, tag);
|
|
|
|
break;
|
2020-10-26 00:39:21 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
version.partial = version.partial.filter((item, index) => version.partial.indexOf(item) === index);
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procSchedule(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/schedule/.test(this.context.eventName)) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const currentDate = this.date;
|
|
|
|
const vraw = handlebars.compile(tag.attrs['pattern'])({
|
|
|
|
date: function (format) {
|
|
|
|
return moment(currentDate).utc().format(format);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procSemver(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/tags\//.test(this.context.ref) && tag.attrs['value'].length == 0) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
let vraw: string;
|
|
|
|
if (tag.attrs['value'].length > 0) {
|
|
|
|
vraw = tag.attrs['value'];
|
|
|
|
} else {
|
|
|
|
vraw = this.context.ref.replace(/^refs\/tags\//g, '').replace(/\//g, '-');
|
|
|
|
}
|
|
|
|
if (!semver.valid(vraw)) {
|
|
|
|
core.warning(`${vraw} is not a valid semver. More info: https://semver.org/`);
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
let latest: boolean = false;
|
|
|
|
const sver = semver.parse(vraw, {
|
|
|
|
includePrerelease: true
|
|
|
|
});
|
|
|
|
if (semver.prerelease(vraw)) {
|
|
|
|
vraw = handlebars.compile('{{version}}')(sver);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
2020-10-26 00:39:21 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
} else {
|
|
|
|
vraw = handlebars.compile(tag.attrs['pattern'])(sver);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
2020-12-04 17:12:39 +00:00
|
|
|
}
|
2021-03-29 11:04:53 +00:00
|
|
|
latest = true;
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? latest : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procMatch(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/tags\//.test(this.context.ref) && tag.attrs['value'].length == 0) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
let vraw: string;
|
|
|
|
if (tag.attrs['value'].length > 0) {
|
|
|
|
vraw = tag.attrs['value'];
|
|
|
|
} else {
|
|
|
|
vraw = this.context.ref.replace(/^refs\/tags\//g, '').replace(/\//g, '-');
|
|
|
|
}
|
|
|
|
|
|
|
|
let latest: boolean = false;
|
|
|
|
let tmatch;
|
|
|
|
const isRegEx = tag.attrs['pattern'].match(/^\/(.+)\/(.*)$/);
|
|
|
|
if (isRegEx) {
|
|
|
|
tmatch = vraw.match(new RegExp(isRegEx[1], isRegEx[2]));
|
|
|
|
} else {
|
|
|
|
tmatch = vraw.match(tag.attrs['pattern']);
|
|
|
|
}
|
2021-04-05 19:19:05 +00:00
|
|
|
if (!tmatch) {
|
|
|
|
core.warning(`${tag.attrs['pattern']} does not match ${vraw}.`);
|
|
|
|
return version;
|
2021-03-29 11:04:53 +00:00
|
|
|
}
|
2021-04-05 19:19:05 +00:00
|
|
|
if (typeof tmatch[tag.attrs['group']] === 'undefined') {
|
|
|
|
core.warning(`Group ${tag.attrs['group']} does not exist for ${tag.attrs['pattern']} pattern.`);
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
vraw = tmatch[tag.attrs['group']];
|
|
|
|
latest = true;
|
2021-03-29 11:04:53 +00:00
|
|
|
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? latest : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procRefBranch(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/heads\//.test(this.context.ref)) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const vraw = this.setFlavor(this.context.ref.replace(/^refs\/heads\//g, '').replace(/[^a-zA-Z0-9._-]+/g, '-'), tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procRefTag(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/tags\//.test(this.context.ref)) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const vraw = this.setFlavor(this.context.ref.replace(/^refs\/tags\//g, '').replace(/\//g, '-'), tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? true : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procRefPr(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/pull\//.test(this.context.ref)) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const vraw = this.setFlavor(this.context.ref.replace(/^refs\/pull\//g, '').replace(/\/merge$/g, ''), tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procEdge(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!/^refs\/heads\//.test(this.context.ref)) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
let val = this.context.ref.replace(/^refs\/heads\//g, '').replace(/[^a-zA-Z0-9._-]+/g, '-');
|
|
|
|
if (tag.attrs['branch'].length == 0) {
|
|
|
|
tag.attrs['branch'] = this.repo.default_branch;
|
|
|
|
}
|
|
|
|
if (tag.attrs['branch'] === val) {
|
|
|
|
val = 'edge';
|
|
|
|
}
|
|
|
|
|
|
|
|
const vraw = this.setFlavor(val, tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
2020-12-04 17:12:39 +00:00
|
|
|
}
|
|
|
|
|
2020-10-26 00:39:21 +00:00
|
|
|
return version;
|
2020-10-25 01:40:42 +00:00
|
|
|
}
|
|
|
|
|
2021-03-29 11:04:53 +00:00
|
|
|
private procRaw(version: Version, tag: tcl.Tag): Version {
|
|
|
|
const vraw = this.setFlavor(tag.attrs['value'], tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private procSha(version: Version, tag: tcl.Tag): Version {
|
|
|
|
if (!this.context.sha) {
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
const vraw = this.setFlavor(this.context.sha.substr(0, 7), tag);
|
|
|
|
if (version.main == undefined) {
|
|
|
|
version.main = vraw;
|
|
|
|
} else if (vraw !== version.main) {
|
|
|
|
version.partial.push(vraw);
|
|
|
|
}
|
|
|
|
if (version.latest == undefined) {
|
|
|
|
version.latest = this.flavor.latest == 'auto' ? false : this.flavor.latest == 'true';
|
|
|
|
}
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
private setFlavor(val: string, tag: tcl.Tag): string {
|
|
|
|
if (tag.attrs['prefix'].length > 0) {
|
|
|
|
val = `${tag.attrs['prefix']}${val}`;
|
|
|
|
} else if (this.flavor.prefix.length > 0) {
|
|
|
|
val = `${this.flavor.prefix}${val}`;
|
|
|
|
}
|
|
|
|
if (tag.attrs['suffix'].length > 0) {
|
|
|
|
val = `${val}${tag.attrs['suffix']}`;
|
|
|
|
} else if (this.flavor.suffix.length > 0) {
|
|
|
|
val = `${val}${this.flavor.suffix}`;
|
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
public getTags(): Array<string> {
|
2020-12-01 04:38:08 +00:00
|
|
|
if (!this.version.main) {
|
2020-10-26 00:39:21 +00:00
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2020-10-25 01:25:23 +00:00
|
|
|
let tags: Array<string> = [];
|
|
|
|
for (const image of this.inputs.images) {
|
2020-11-20 15:19:08 +00:00
|
|
|
const imageLc = image.toLowerCase();
|
2020-12-01 04:38:08 +00:00
|
|
|
tags.push(`${imageLc}:${this.version.main}`);
|
|
|
|
for (const partial of this.version.partial) {
|
2020-11-20 15:19:08 +00:00
|
|
|
tags.push(`${imageLc}:${partial}`);
|
2020-11-17 22:31:03 +00:00
|
|
|
}
|
2020-12-01 04:38:08 +00:00
|
|
|
if (this.version.latest) {
|
2020-11-20 15:19:08 +00:00
|
|
|
tags.push(`${imageLc}:latest`);
|
2020-10-25 01:25:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tags;
|
|
|
|
}
|
|
|
|
|
2021-03-29 11:04:53 +00:00
|
|
|
public getLabels(): Array<string> {
|
2020-12-23 21:09:38 +00:00
|
|
|
let labels: Array<string> = [
|
2020-10-25 01:25:23 +00:00
|
|
|
`org.opencontainers.image.title=${this.repo.name || ''}`,
|
|
|
|
`org.opencontainers.image.description=${this.repo.description || ''}`,
|
|
|
|
`org.opencontainers.image.url=${this.repo.html_url || ''}`,
|
2020-10-31 19:16:51 +00:00
|
|
|
`org.opencontainers.image.source=${this.repo.html_url || ''}`,
|
2020-12-01 04:38:08 +00:00
|
|
|
`org.opencontainers.image.version=${this.version.main || ''}`,
|
2020-10-25 14:13:43 +00:00
|
|
|
`org.opencontainers.image.created=${this.date.toISOString()}`,
|
2020-10-25 01:25:23 +00:00
|
|
|
`org.opencontainers.image.revision=${this.context.sha || ''}`,
|
|
|
|
`org.opencontainers.image.licenses=${this.repo.license?.spdx_id || ''}`
|
|
|
|
];
|
2021-03-29 11:04:53 +00:00
|
|
|
labels.push(...this.inputs.labels);
|
2020-12-23 21:09:38 +00:00
|
|
|
return labels;
|
2020-10-25 01:25:23 +00:00
|
|
|
}
|
2020-12-24 03:13:41 +00:00
|
|
|
|
2021-03-29 11:04:53 +00:00
|
|
|
public getBakeFile(): string {
|
2020-12-24 03:13:41 +00:00
|
|
|
let jsonLabels = {};
|
2021-03-29 11:04:53 +00:00
|
|
|
for (let label of this.getLabels()) {
|
2020-12-24 03:13:41 +00:00
|
|
|
const matches = label.match(/([^=]*)=(.*)/);
|
|
|
|
if (!matches) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
jsonLabels[matches[1]] = matches[2];
|
|
|
|
}
|
|
|
|
|
|
|
|
const bakeFile = path.join(tmpDir(), 'ghaction-docker-meta-bake.json').split(path.sep).join(path.posix.sep);
|
|
|
|
fs.writeFileSync(
|
|
|
|
bakeFile,
|
|
|
|
JSON.stringify(
|
|
|
|
{
|
|
|
|
target: {
|
|
|
|
'ghaction-docker-meta': {
|
2021-03-29 11:04:53 +00:00
|
|
|
tags: this.getTags(),
|
2020-12-24 15:45:28 +00:00
|
|
|
labels: jsonLabels,
|
|
|
|
args: {
|
|
|
|
DOCKER_META_IMAGES: this.inputs.images.join(','),
|
|
|
|
DOCKER_META_VERSION: this.version.main
|
|
|
|
}
|
2020-12-24 03:13:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
null,
|
|
|
|
2
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
return bakeFile;
|
|
|
|
}
|
2020-10-25 01:25:23 +00:00
|
|
|
}
|