2021-03-13 23:44:01 +00:00
|
|
|
// @ts-ignore
|
2021-05-23 13:31:02 +00:00
|
|
|
import * as k8s from '@kubernetes/client-node';
|
|
|
|
|
import { BuildParameters } from '.';
|
2020-08-09 19:27:47 +00:00
|
|
|
const core = require('@actions/core');
|
2021-05-23 14:26:57 +00:00
|
|
|
const base64 = require('base-64');
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-28 17:46:22 +00:00
|
|
|
const pollInterval = 50000;
|
2020-08-09 19:27:47 +00:00
|
|
|
|
|
|
|
|
class Kubernetes {
|
2021-05-23 13:31:02 +00:00
|
|
|
private static kubeClient: k8s.CoreV1Api;
|
2021-05-23 14:26:57 +00:00
|
|
|
private static kubeClientBatch: k8s.BatchV1Api;
|
2021-03-13 23:44:01 +00:00
|
|
|
private static buildId: string;
|
2021-05-23 13:31:02 +00:00
|
|
|
private static buildParameters: BuildParameters;
|
2021-03-13 23:44:01 +00:00
|
|
|
private static baseImage: any;
|
|
|
|
|
private static pvcName: string;
|
|
|
|
|
private static secretName: string;
|
|
|
|
|
private static jobName: string;
|
|
|
|
|
private static namespace: string;
|
|
|
|
|
|
2021-05-23 13:31:02 +00:00
|
|
|
static async runBuildJob(buildParameters: BuildParameters, baseImage) {
|
2021-05-23 13:11:06 +00:00
|
|
|
core.info('Starting up k8s');
|
|
|
|
|
const kc = new k8s.KubeConfig();
|
|
|
|
|
kc.loadFromDefault();
|
2021-05-23 13:31:02 +00:00
|
|
|
const k8sApi = kc.makeApiClient(k8s.CoreV1Api);
|
2021-05-23 14:26:57 +00:00
|
|
|
const k8sBatchApi = kc.makeApiClient(k8s.BatchV1Api);
|
2021-05-23 13:11:06 +00:00
|
|
|
core.info('loaded from default');
|
2021-05-23 13:31:02 +00:00
|
|
|
|
|
|
|
|
// const kubeconfig = new KubeConfig();
|
|
|
|
|
// kubeconfig.loadFromString(base64.decode(buildParameters.kubeConfig));
|
|
|
|
|
// const backend = new Request({ kubeconfig });
|
|
|
|
|
// const kubeClient = new Client(backend);
|
|
|
|
|
// await kubeClient.loadSpec();
|
2020-08-09 19:27:47 +00:00
|
|
|
|
|
|
|
|
const buildId = Kubernetes.uuidv4();
|
|
|
|
|
const pvcName = `unity-builder-pvc-${buildId}`;
|
|
|
|
|
const secretName = `build-credentials-${buildId}`;
|
|
|
|
|
const jobName = `unity-builder-job-${buildId}`;
|
|
|
|
|
const namespace = 'default';
|
|
|
|
|
|
2021-05-23 13:31:02 +00:00
|
|
|
this.kubeClient = k8sApi;
|
2021-05-23 14:26:57 +00:00
|
|
|
this.kubeClientBatch = k8sBatchApi;
|
2021-03-13 23:44:01 +00:00
|
|
|
this.buildId = buildId;
|
|
|
|
|
this.buildParameters = buildParameters;
|
|
|
|
|
this.baseImage = baseImage;
|
|
|
|
|
this.pvcName = pvcName;
|
|
|
|
|
this.secretName = secretName;
|
|
|
|
|
this.jobName = jobName;
|
|
|
|
|
this.namespace = namespace;
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-23 15:08:32 +00:00
|
|
|
// setup
|
2021-05-23 14:26:57 +00:00
|
|
|
await Kubernetes.createSecret();
|
|
|
|
|
await Kubernetes.createPersistentVolumeClaim();
|
2021-05-23 15:08:32 +00:00
|
|
|
|
|
|
|
|
// start
|
|
|
|
|
await Kubernetes.scheduleBuildJob();
|
|
|
|
|
|
|
|
|
|
// watch
|
|
|
|
|
await Kubernetes.watchPersistentVolumeClaimUntilReady();
|
|
|
|
|
await Kubernetes.watchBuildJobUntilFinished();
|
|
|
|
|
|
|
|
|
|
// cleanup
|
|
|
|
|
await Kubernetes.cleanup();
|
2020-08-09 19:27:47 +00:00
|
|
|
|
|
|
|
|
core.setOutput('volume', pvcName);
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-23 14:26:57 +00:00
|
|
|
static async createSecret() {
|
|
|
|
|
const secret = new k8s.V1Secret();
|
|
|
|
|
secret.apiVersion = 'v1';
|
|
|
|
|
secret.kind = 'Secret';
|
|
|
|
|
secret.type = 'Opaque';
|
|
|
|
|
secret.metadata = {
|
|
|
|
|
name: this.secretName,
|
|
|
|
|
};
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-23 14:26:57 +00:00
|
|
|
secret.data = {
|
|
|
|
|
GITHUB_TOKEN: base64.encode(this.buildParameters.githubToken),
|
|
|
|
|
UNITY_LICENSE: base64.encode(process.env.UNITY_LICENSE),
|
|
|
|
|
ANDROID_KEYSTORE_BASE64: base64.encode(this.buildParameters.androidKeystoreBase64),
|
|
|
|
|
ANDROID_KEYSTORE_PASS: base64.encode(this.buildParameters.androidKeystorePass),
|
|
|
|
|
ANDROID_KEYALIAS_PASS: base64.encode(this.buildParameters.androidKeyaliasPass),
|
|
|
|
|
};
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-23 14:26:57 +00:00
|
|
|
await this.kubeClient.createNamespacedSecret(this.namespace, secret);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async createPersistentVolumeClaim() {
|
|
|
|
|
if (this.buildParameters.kubeVolume) {
|
|
|
|
|
core.info(this.buildParameters.kubeVolume);
|
|
|
|
|
this.pvcName = this.buildParameters.kubeVolume;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const pvc = new k8s.V1PersistentVolumeClaim();
|
|
|
|
|
pvc.apiVersion = 'v1';
|
|
|
|
|
pvc.kind = 'PersistentVolumeClaim';
|
|
|
|
|
pvc.metadata = {
|
|
|
|
|
name: this.pvcName,
|
|
|
|
|
};
|
|
|
|
|
pvc.spec = {
|
|
|
|
|
accessModes: ['ReadWriteOnce'],
|
|
|
|
|
volumeMode: 'Filesystem',
|
|
|
|
|
resources: {
|
|
|
|
|
requests: {
|
|
|
|
|
storage: this.buildParameters.kubeVolumeSize,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
await this.kubeClient.createNamespacedPersistentVolumeClaim(this.namespace, pvc);
|
|
|
|
|
core.info('Persistent Volume created, waiting for ready state...');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async watchPersistentVolumeClaimUntilReady() {
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, pollInterval));
|
|
|
|
|
const queryResult = await this.kubeClient.readNamespacedPersistentVolumeClaim(this.pvcName, this.namespace);
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-23 14:26:57 +00:00
|
|
|
if (queryResult.body.status?.phase === 'Pending') {
|
|
|
|
|
await Kubernetes.watchPersistentVolumeClaimUntilReady();
|
2021-05-23 15:08:32 +00:00
|
|
|
} else {
|
|
|
|
|
core.info('Persistent Volume ready for claims');
|
2021-05-23 14:26:57 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async scheduleBuildJob() {
|
|
|
|
|
core.info('Creating build job');
|
|
|
|
|
const job = new k8s.V1Job();
|
|
|
|
|
job.apiVersion = 'batch/v1';
|
|
|
|
|
job.kind = 'Job';
|
|
|
|
|
job.metadata = {
|
|
|
|
|
name: this.jobName,
|
|
|
|
|
labels: {
|
|
|
|
|
app: 'unity-builder',
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
job.spec = {
|
|
|
|
|
template: {
|
|
|
|
|
spec: {
|
|
|
|
|
volumes: [
|
|
|
|
|
{
|
|
|
|
|
name: 'data',
|
|
|
|
|
persistentVolumeClaim: {
|
|
|
|
|
claimName: this.pvcName,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'credentials',
|
|
|
|
|
secret: {
|
|
|
|
|
secretName: this.secretName,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
initContainers: [
|
|
|
|
|
{
|
|
|
|
|
name: 'clone',
|
|
|
|
|
image: 'alpine/git',
|
|
|
|
|
command: [
|
|
|
|
|
'/bin/sh',
|
|
|
|
|
'-c',
|
|
|
|
|
`apk update;
|
|
|
|
|
apk add git-lfs;
|
|
|
|
|
export GITHUB_TOKEN=$(cat /credentials/GITHUB_TOKEN);
|
|
|
|
|
cd /data;
|
|
|
|
|
git clone https://github.com/${process.env.GITHUB_REPOSITORY}.git repo;
|
|
|
|
|
git clone https://github.com/webbertakken/unity-builder.git builder;
|
|
|
|
|
cd repo;
|
|
|
|
|
git checkout $GITHUB_SHA;
|
|
|
|
|
ls`,
|
|
|
|
|
],
|
|
|
|
|
volumeMounts: [
|
|
|
|
|
{
|
|
|
|
|
name: 'data',
|
|
|
|
|
mountPath: '/data',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'credentials',
|
|
|
|
|
mountPath: '/credentials',
|
|
|
|
|
readOnly: true,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
env: [
|
|
|
|
|
{
|
|
|
|
|
name: 'GITHUB_SHA',
|
|
|
|
|
value: this.buildId,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
containers: [
|
|
|
|
|
{
|
|
|
|
|
name: 'main',
|
|
|
|
|
image: `${this.baseImage.toString()}`,
|
|
|
|
|
command: [
|
|
|
|
|
'bin/bash',
|
|
|
|
|
'-c',
|
2021-05-23 16:07:30 +00:00
|
|
|
`ls
|
|
|
|
|
for f in ./credentials/*; do export $(basename $f)="$(cat $f)"; done
|
2021-05-23 16:15:21 +00:00
|
|
|
ls /data
|
|
|
|
|
ls /data/builder
|
2021-05-23 16:22:58 +00:00
|
|
|
ls /data/builder/dist
|
|
|
|
|
cp -r /data/builder/dist/default-build-script /UnityBuilderAction
|
|
|
|
|
cp -r /data/builder/dist/entrypoint.sh /entrypoint.sh
|
|
|
|
|
cp -r /data/builder/dist/steps /steps
|
2021-05-23 16:38:44 +00:00
|
|
|
chmod -R +x /entrypoint.sh
|
|
|
|
|
chmod -R +x /steps
|
|
|
|
|
/entrypoint.sh
|
2021-05-23 14:26:57 +00:00
|
|
|
`,
|
|
|
|
|
],
|
|
|
|
|
resources: {
|
|
|
|
|
requests: {
|
|
|
|
|
memory: this.buildParameters.remoteBuildMemory,
|
|
|
|
|
cpu: this.buildParameters.remoteBuildCpu,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
env: [
|
|
|
|
|
{
|
|
|
|
|
name: 'GITHUB_WORKSPACE',
|
|
|
|
|
value: '/data/repo',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'PROJECT_PATH',
|
|
|
|
|
value: this.buildParameters.projectPath,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'BUILD_PATH',
|
|
|
|
|
value: this.buildParameters.buildPath,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'BUILD_FILE',
|
|
|
|
|
value: this.buildParameters.buildFile,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'BUILD_NAME',
|
|
|
|
|
value: this.buildParameters.buildName,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'BUILD_METHOD',
|
|
|
|
|
value: this.buildParameters.buildMethod,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'CUSTOM_PARAMETERS',
|
|
|
|
|
value: this.buildParameters.customParameters,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'CHOWN_FILES_TO',
|
|
|
|
|
value: this.buildParameters.chownFilesTo,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'BUILD_TARGET',
|
|
|
|
|
value: this.buildParameters.platform,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'ANDROID_VERSION_CODE',
|
|
|
|
|
value: this.buildParameters.androidVersionCode.toString(),
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'ANDROID_KEYSTORE_NAME',
|
|
|
|
|
value: this.buildParameters.androidKeystoreName,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'ANDROID_KEYALIAS_NAME',
|
|
|
|
|
value: this.buildParameters.androidKeyaliasName,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
volumeMounts: [
|
|
|
|
|
{
|
|
|
|
|
name: 'data',
|
|
|
|
|
mountPath: '/data',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'credentials',
|
|
|
|
|
mountPath: '/credentials',
|
|
|
|
|
readOnly: true,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
lifecycle: {
|
|
|
|
|
preStop: {
|
|
|
|
|
exec: {
|
|
|
|
|
command: [
|
|
|
|
|
'bin/bash',
|
|
|
|
|
'-c',
|
|
|
|
|
`cd /data/builder/action/steps;
|
|
|
|
|
chmod +x /return_license.sh;
|
|
|
|
|
/return_license.sh;`,
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
restartPolicy: 'Never',
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
job.spec.backoffLimit = 1;
|
2021-05-23 21:58:03 +00:00
|
|
|
await this.kubeClientBatch.createNamespacedJob(this.namespace, job);
|
2021-05-23 14:26:57 +00:00
|
|
|
core.info('Job created');
|
|
|
|
|
}
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-28 17:37:30 +00:00
|
|
|
static async watchPodUntilReadyAndRead(statusFilter: string) {
|
2021-05-23 15:08:32 +00:00
|
|
|
let ready = false;
|
2021-05-23 21:00:50 +00:00
|
|
|
|
2021-05-23 15:08:32 +00:00
|
|
|
while (!ready) {
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, pollInterval));
|
|
|
|
|
const pods = await this.kubeClient.listNamespacedPod(this.namespace);
|
|
|
|
|
for (let index = 0; index < pods.body.items.length; index++) {
|
|
|
|
|
const element = pods.body.items[index];
|
|
|
|
|
const jobname = element.metadata?.labels?.['job-name'];
|
|
|
|
|
const phase = element.status?.phase;
|
2021-05-28 17:37:30 +00:00
|
|
|
if (jobname === this.jobName && phase !== statusFilter) {
|
2021-05-23 15:08:32 +00:00
|
|
|
core.info('Pod no longer pending');
|
|
|
|
|
if (phase === 'Failure') {
|
|
|
|
|
core.error('Kubernetes job failed');
|
|
|
|
|
} else {
|
|
|
|
|
ready = true;
|
2021-05-23 21:00:50 +00:00
|
|
|
return element;
|
2021-05-23 15:08:32 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-05-23 21:00:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async watchBuildJobUntilFinished() {
|
2021-05-28 17:37:30 +00:00
|
|
|
const pod = (await Kubernetes.watchPodUntilReadyAndRead('Pending')) || {};
|
2020-08-09 19:27:47 +00:00
|
|
|
|
2021-05-23 21:31:59 +00:00
|
|
|
core.info(
|
|
|
|
|
`Watching build job ${pod.metadata?.name} ${JSON.stringify(
|
|
|
|
|
pod.status?.containerStatuses?.[0].state,
|
|
|
|
|
undefined,
|
|
|
|
|
4,
|
|
|
|
|
)}`,
|
|
|
|
|
);
|
2021-05-28 17:37:30 +00:00
|
|
|
await Kubernetes.streamLogs(pod.metadata?.name || '', this.namespace);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async streamLogs(name: string, namespace: string) {
|
|
|
|
|
let running = true;
|
|
|
|
|
let logQueryTime;
|
|
|
|
|
while (running) {
|
2021-05-28 17:39:20 +00:00
|
|
|
await new Promise((resolve) => setTimeout(resolve, pollInterval));
|
2021-05-28 17:37:30 +00:00
|
|
|
const logs = await this.kubeClient.readNamespacedPodLog(
|
|
|
|
|
name,
|
|
|
|
|
namespace,
|
2021-05-23 21:15:16 +00:00
|
|
|
undefined,
|
2021-05-28 17:37:30 +00:00
|
|
|
undefined,
|
|
|
|
|
undefined,
|
|
|
|
|
undefined,
|
|
|
|
|
undefined,
|
|
|
|
|
undefined,
|
|
|
|
|
logQueryTime,
|
|
|
|
|
undefined,
|
|
|
|
|
true,
|
|
|
|
|
);
|
|
|
|
|
core.info(logs.body);
|
|
|
|
|
const arrayOfLines = logs.body.match(/[^\n\r]+/g)?.reverse();
|
|
|
|
|
if (arrayOfLines) {
|
|
|
|
|
for (const element of arrayOfLines) {
|
|
|
|
|
const [time, ...line] = element.split(' ');
|
|
|
|
|
if (time !== logQueryTime) {
|
|
|
|
|
core.info(line.join(' '));
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
logQueryTime = arrayOfLines[0].split(' ')[0];
|
|
|
|
|
}
|
|
|
|
|
const pod = await this.kubeClient.readNamespacedPod(name, namespace);
|
|
|
|
|
running = pod.body.status?.phase === 'Running';
|
|
|
|
|
}
|
2021-05-23 15:08:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static async cleanup() {
|
|
|
|
|
await this.kubeClientBatch.deleteNamespacedJob(this.jobName, this.namespace);
|
|
|
|
|
await this.kubeClient.deleteNamespacedSecret(this.secretName, this.namespace);
|
|
|
|
|
}
|
2020-08-09 19:27:47 +00:00
|
|
|
|
|
|
|
|
static uuidv4() {
|
|
|
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
|
2021-03-13 23:44:01 +00:00
|
|
|
const r = Math.trunc(Math.random() * 16);
|
2020-08-09 19:27:47 +00:00
|
|
|
const v = c === 'x' ? r : (r & 0x3) | 0x8;
|
|
|
|
|
return v.toString(16);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
export default Kubernetes;
|