unity-builder/src/model/remote-builder/kubernetes-build-platform.ts

484 lines
14 KiB
TypeScript
Raw Normal View History

2021-05-23 13:31:02 +00:00
import * as k8s from '@kubernetes/client-node';
2021-06-18 19:52:07 +00:00
import { BuildParameters } from '..';
2021-05-28 17:54:54 +00:00
import * as core from '@actions/core';
import { KubeConfig, Log } from '@kubernetes/client-node';
import { Writable } from 'stream';
2021-06-18 19:52:07 +00:00
import { RemoteBuilderProviderInterface } from './remote-builder-provider-interface';
import RemoteBuilderSecret from './remote-builder-secret';
import { waitUntil } from 'async-wait-until';
2021-06-18 19:52:07 +00:00
import KubernetesStorage from './kubernetes-storage';
import RemoteBuilderEnvironmentVariable from './remote-builder-environment-variable';
const base64 = require('base-64');
2021-06-06 19:39:06 +00:00
class Kubernetes implements RemoteBuilderProviderInterface {
private kubeConfig: KubeConfig;
private kubeClient: k8s.CoreV1Api;
private kubeClientBatch: k8s.BatchV1Api;
2021-06-06 21:22:22 +00:00
private buildId: string = '';
2021-06-19 03:31:29 +00:00
private buildCorrelationId: string = '';
2021-06-06 19:39:06 +00:00
private buildParameters: BuildParameters;
private baseImage: any;
2021-06-06 21:22:22 +00:00
private pvcName: string = '';
private secretName: string = '';
private jobName: string = '';
2021-06-06 19:39:06 +00:00
private namespace: string;
2021-06-06 20:14:12 +00:00
private podName: string = '';
private containerName: string = '';
2021-06-06 19:39:06 +00:00
constructor(buildParameters: BuildParameters, baseImage) {
2021-05-23 13:11:06 +00:00
const kc = new k8s.KubeConfig();
kc.loadFromDefault();
2021-05-23 13:31:02 +00:00
const k8sApi = kc.makeApiClient(k8s.CoreV1Api);
const k8sBatchApi = kc.makeApiClient(k8s.BatchV1Api);
2021-06-06 20:10:01 +00:00
core.info('Loaded default Kubernetes configuration for this environment');
2021-05-23 13:31:02 +00:00
2021-06-06 21:22:22 +00:00
this.kubeConfig = kc;
this.kubeClient = k8sApi;
this.kubeClientBatch = k8sBatchApi;
2021-06-19 03:31:29 +00:00
this.buildCorrelationId = Kubernetes.uuidv4();
2021-06-06 21:22:22 +00:00
this.namespace = 'default';
this.buildParameters = buildParameters;
this.baseImage = baseImage;
}
2021-06-19 04:49:32 +00:00
async runBuildTask(
buildId: string,
stackName: string,
image: string,
commands: string[],
mountdir: string,
workingdir: string,
environment: RemoteBuilderEnvironmentVariable[],
secrets: RemoteBuilderSecret[],
): Promise<void> {
try {
this.setUniqueBuildId();
const defaultSecretsArray: RemoteBuilderSecret[] = [
{
ParameterKey: 'GithubToken',
EnvironmentVariable: 'GITHUB_TOKEN',
ParameterValue: this.buildParameters.githubToken,
},
{
ParameterKey: 'UNITY_LICENSE',
EnvironmentVariable: 'UNITY_LICENSE',
ParameterValue: process.env.UNITY_LICENSE || '',
},
{
ParameterKey: 'ANDROID_KEYSTORE_BASE64',
EnvironmentVariable: 'ANDROID_KEYSTORE_BASE64',
ParameterValue: this.buildParameters.androidKeystoreBase64,
},
{
ParameterKey: 'ANDROID_KEYSTORE_PASS',
EnvironmentVariable: 'ANDROID_KEYSTORE_PASS',
ParameterValue: this.buildParameters.androidKeystorePass,
},
{
ParameterKey: 'ANDROID_KEYALIAS_PASS',
EnvironmentVariable: 'ANDROID_KEYALIAS_PASS',
ParameterValue: this.buildParameters.androidKeyaliasPass,
},
];
defaultSecretsArray.push(...secrets);
// setup
await this.createSecret(defaultSecretsArray);
await KubernetesStorage.createPersistentVolumeClaim(
this.buildParameters,
this.pvcName,
this.kubeClient,
this.namespace,
);
//run
const jobSpec = this.getJobSpec(commands, image);
core.info('Creating build job');
await this.kubeClientBatch.createNamespacedJob(this.namespace, jobSpec);
core.info('Job created');
await KubernetesStorage.watchUntilPVCNotPending(this.kubeClient, this.pvcName, this.namespace);
core.info('PVC Bound');
this.setPodNameAndContainerName(await this.findPod());
core.info('Watching pod until running');
await this.watchUntilPodRunning();
core.info('Pod running, streaming logs');
await this.streamLogs();
await this.cleanup();
} catch (error) {
core.info('Running job failed');
await this.cleanup();
throw error;
}
}
2021-06-06 21:22:22 +00:00
setUniqueBuildId() {
const buildId = Kubernetes.uuidv4();
const pvcName = `unity-builder-pvc-${buildId}`;
const secretName = `build-credentials-${buildId}`;
const jobName = `unity-builder-job-${buildId}`;
this.buildId = buildId;
this.pvcName = pvcName;
this.secretName = secretName;
this.jobName = jobName;
2021-06-06 19:39:06 +00:00
}
2021-06-19 04:49:32 +00:00
async runFullBuildFlow() {
2021-06-06 20:10:01 +00:00
core.info('Running Remote Builder on Kubernetes');
2021-06-06 20:14:12 +00:00
try {
await this.runCloneJob();
await this.runBuildJob();
} catch (error) {
2021-06-18 22:42:03 +00:00
core.error(error);
2021-06-18 18:57:50 +00:00
core.error(JSON.stringify(error.response, undefined, 4));
2021-06-06 20:19:24 +00:00
throw error;
2021-06-06 20:14:12 +00:00
}
2021-06-06 19:39:06 +00:00
core.setOutput('volume', this.pvcName);
}
2021-06-06 19:59:34 +00:00
async createSecret(secrets: RemoteBuilderSecret[]) {
const secret = new k8s.V1Secret();
secret.apiVersion = 'v1';
secret.kind = 'Secret';
secret.type = 'Opaque';
secret.metadata = {
name: this.secretName,
};
secret.data = {};
2021-06-06 19:59:34 +00:00
for (const buildSecret of secrets) {
secret.data[buildSecret.EnvironmentVariable] = base64.encode(buildSecret.ParameterValue);
2021-06-06 20:18:30 +00:00
secret.data[`${buildSecret.EnvironmentVariable}_NAME`] = base64.encode(buildSecret.ParameterKey);
2021-06-06 19:59:34 +00:00
}
2021-06-06 20:10:01 +00:00
try {
await this.kubeClient.createNamespacedSecret(this.namespace, secret);
} catch (error) {
throw error;
}
}
2021-06-06 20:32:24 +00:00
getJobSpec(command: string[], image: string) {
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,
},
},
],
containers: [
{
name: 'main',
2021-05-28 19:27:39 +00:00
image,
2021-05-28 20:03:41 +00:00
command,
resources: {
requests: {
memory: this.buildParameters.remoteBuildMemory,
cpu: this.buildParameters.remoteBuildCpu,
},
},
env: [
2021-05-28 19:27:39 +00:00
{
name: 'GITHUB_SHA',
value: this.buildId,
},
{
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-06-06 20:32:24 +00:00
return job;
}
2021-05-28 19:27:39 +00:00
2021-06-19 03:31:29 +00:00
async runJobInKubernetesPod(command: string[], image: string) {
2021-05-28 19:38:12 +00:00
try {
2021-06-06 21:22:22 +00:00
this.setUniqueBuildId();
2021-06-19 03:31:29 +00:00
const defaultSecretsArray: RemoteBuilderSecret[] = [
{
ParameterKey: 'GithubToken',
EnvironmentVariable: 'GITHUB_TOKEN',
ParameterValue: this.buildParameters.githubToken,
},
{
ParameterKey: 'UNITY_LICENSE',
EnvironmentVariable: 'UNITY_LICENSE',
ParameterValue: process.env.UNITY_LICENSE || '',
},
{
ParameterKey: 'ANDROID_KEYSTORE_BASE64',
EnvironmentVariable: 'ANDROID_KEYSTORE_BASE64',
ParameterValue: this.buildParameters.androidKeystoreBase64,
},
{
ParameterKey: 'ANDROID_KEYSTORE_PASS',
EnvironmentVariable: 'ANDROID_KEYSTORE_PASS',
ParameterValue: this.buildParameters.androidKeystorePass,
},
{
ParameterKey: 'ANDROID_KEYALIAS_PASS',
EnvironmentVariable: 'ANDROID_KEYALIAS_PASS',
ParameterValue: this.buildParameters.androidKeyaliasPass,
},
];
// setup
await this.createSecret(defaultSecretsArray);
await KubernetesStorage.createPersistentVolumeClaim(
this.buildParameters,
this.pvcName,
this.kubeClient,
this.namespace,
);
//run
2021-06-06 21:22:22 +00:00
const jobSpec = this.getJobSpec(command, image);
2021-06-06 20:32:24 +00:00
core.info('Creating build job');
await this.kubeClientBatch.createNamespacedJob(this.namespace, jobSpec);
core.info('Job created');
2021-06-18 22:55:48 +00:00
await KubernetesStorage.watchUntilPVCNotPending(this.kubeClient, this.pvcName, this.namespace);
2021-06-18 19:02:16 +00:00
core.info('PVC Bound');
2021-06-19 04:07:18 +00:00
this.setPodNameAndContainerName(await this.findPod());
2021-06-19 03:31:29 +00:00
core.info('Watching pod until running');
2021-06-06 19:39:06 +00:00
await this.watchUntilPodRunning();
2021-06-19 03:31:29 +00:00
core.info('Pod running, streaming logs');
2021-06-06 19:39:06 +00:00
await this.streamLogs();
2021-06-18 18:57:50 +00:00
await this.cleanup();
2021-05-28 19:38:12 +00:00
} catch (error) {
2021-06-18 22:20:04 +00:00
core.info('Running job failed');
2021-06-06 19:39:06 +00:00
await this.cleanup();
2021-06-06 22:44:50 +00:00
throw error;
2021-05-28 19:38:12 +00:00
}
2021-05-28 19:27:39 +00:00
}
2021-06-19 04:07:18 +00:00
async findPod() {
const pod = (await this.kubeClient.listNamespacedPod(this.namespace)).body.items.find(
(x) => x.metadata?.labels?.['job-name'] === this.jobName,
);
if (pod === undefined) {
throw new Error("pod with job-name label doesn't exist");
2021-06-06 19:39:06 +00:00
}
2021-06-19 04:07:18 +00:00
return pod;
2021-06-06 19:39:06 +00:00
}
async runCloneJob() {
2021-06-19 04:49:32 +00:00
await this.runBuildTask(
this.buildCorrelationId,
'',
'alpine/git',
2021-06-06 21:22:22 +00:00
[
'/bin/ash',
'-c',
`apk update;
apk add unzip;
2021-05-28 19:27:39 +00:00
apk add git-lfs;
apk add jq;
2021-06-06 19:59:34 +00:00
ls /credentials/
2021-05-28 19:27:39 +00:00
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;
2021-06-06 19:43:26 +00:00
ls
echo "end"`,
2021-06-06 21:22:22 +00:00
],
'',
'',
[],
[],
2021-05-28 19:27:39 +00:00
);
}
2021-06-06 19:39:06 +00:00
async runBuildJob() {
2021-06-19 04:49:32 +00:00
await this.runBuildTask(
this.buildCorrelationId,
'',
this.baseImage.toString(),
2021-06-06 21:22:22 +00:00
[
'bin/bash',
'-c',
`ls
2021-05-28 19:27:39 +00:00
for f in ./credentials/*; do export $(basename $f)="$(cat $f)"; done
ls /data
ls /data/builder
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
chmod -R +x /entrypoint.sh
chmod -R +x /steps
/entrypoint.sh
`,
2021-06-06 21:22:22 +00:00
],
'',
'',
[],
[],
2021-05-28 19:27:39 +00:00
);
}
async watchUntilPodRunning() {
2021-06-19 04:49:32 +00:00
let success: boolean = false;
core.info(`Watching ${this.podName} ${this.namespace}`);
2021-06-19 03:39:01 +00:00
await waitUntil(
async () => {
2021-06-19 04:49:32 +00:00
const phase = (await this.kubeClient.readNamespacedPodStatus(this.podName, this.namespace))?.body.status?.phase;
success = phase === 'Running';
if (success || phase !== 'Pending') return true;
return false;
2021-06-19 03:39:01 +00:00
},
{
timeout: 500000,
intervalBetweenAttempts: 15000,
},
);
2021-06-19 04:49:32 +00:00
return success;
2021-05-23 21:00:50 +00:00
}
2021-06-06 21:22:22 +00:00
setPodNameAndContainerName(pod: k8s.V1Pod) {
this.podName = pod.metadata?.name || '';
this.containerName = pod.status?.containerStatuses?.[0].name || '';
2021-05-28 17:37:30 +00:00
}
2021-06-06 19:39:06 +00:00
async streamLogs() {
2021-06-06 20:14:12 +00:00
core.info(`Streaming logs from pod: ${this.podName} container: ${this.containerName} namespace: ${this.namespace}`);
const stream = new Writable();
stream._write = (chunk, encoding, next) => {
core.info(chunk.toString());
next();
};
const logOptions = {
follow: true,
pretty: true,
previous: true,
};
2021-05-28 17:54:54 +00:00
try {
2021-06-19 16:07:04 +00:00
const resultError = await new Promise((resolve) =>
new Log(this.kubeConfig).log(this.namespace, this.podName, this.containerName, stream, resolve, logOptions),
2021-06-06 19:09:56 +00:00
);
2021-06-19 16:07:04 +00:00
if (resultError) {
throw resultError;
}
2021-05-28 17:54:54 +00:00
} catch (error) {
throw error;
2021-05-28 17:37:30 +00:00
}
core.info('end of log stream');
}
2021-06-06 19:39:06 +00:00
async cleanup() {
2021-05-28 20:20:04 +00:00
core.info('cleaning up');
2021-06-18 22:20:04 +00:00
try {
await this.kubeClientBatch.deleteNamespacedJob(this.jobName, this.namespace);
await this.kubeClient.deleteNamespacedPersistentVolumeClaim(this.pvcName, this.namespace);
await this.kubeClient.deleteNamespacedSecret(this.secretName, this.namespace);
} catch (error) {
2021-06-18 22:23:30 +00:00
core.info('Failed to cleanup, error:');
2021-06-18 22:20:04 +00:00
core.error(JSON.stringify(error, undefined, 4));
2021-06-18 22:23:30 +00:00
core.info('Abandoning cleanup, build error:');
2021-06-18 22:20:04 +00:00
}
}
static uuidv4() {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
const r = Math.trunc(Math.random() * 16);
const v = c === 'x' ? r : (r & 0x3) | 0x8;
return v.toString(16);
});
}
}
export default Kubernetes;