195 lines
7.1 KiB
TypeScript
195 lines
7.1 KiB
TypeScript
import BuildParameters from '../../../build-parameters';
|
|
import CloudRunnerEnvironmentVariable from '../../options/cloud-runner-environment-variable';
|
|
import CloudRunnerLogger from '../../services/core/cloud-runner-logger';
|
|
import { ProviderInterface } from '../provider-interface';
|
|
import CloudRunnerSecret from '../../options/cloud-runner-secret';
|
|
import Docker from '../../../docker';
|
|
import { Action } from '../../..';
|
|
import { writeFileSync } from 'node:fs';
|
|
import CloudRunner from '../../cloud-runner';
|
|
import { ProviderResource } from '../provider-resource';
|
|
import { ProviderWorkflow } from '../provider-workflow';
|
|
import { CloudRunnerSystem } from '../../services/core/cloud-runner-system';
|
|
import * as fs from 'node:fs';
|
|
import { CommandHookService } from '../../services/hooks/command-hook-service';
|
|
import { StringKeyValuePair } from '../../../shared-types';
|
|
|
|
class LocalDockerCloudRunner implements ProviderInterface {
|
|
public buildParameters!: BuildParameters;
|
|
|
|
listResources(): Promise<ProviderResource[]> {
|
|
return new Promise((resolve) => resolve([]));
|
|
}
|
|
listWorkflow(): Promise<ProviderWorkflow[]> {
|
|
throw new Error('Method not implemented.');
|
|
}
|
|
watchWorkflow(): Promise<string> {
|
|
throw new Error('Method not implemented.');
|
|
}
|
|
garbageCollect(
|
|
// eslint-disable-next-line no-unused-vars
|
|
filter: string,
|
|
// eslint-disable-next-line no-unused-vars
|
|
previewOnly: boolean,
|
|
// eslint-disable-next-line no-unused-vars
|
|
olderThan: Number,
|
|
// eslint-disable-next-line no-unused-vars
|
|
fullCache: boolean,
|
|
// eslint-disable-next-line no-unused-vars
|
|
baseDependencies: boolean,
|
|
): Promise<string> {
|
|
return new Promise((result) => result(``));
|
|
}
|
|
async cleanupWorkflow(
|
|
buildParameters: BuildParameters,
|
|
// eslint-disable-next-line no-unused-vars
|
|
branchName: string,
|
|
// eslint-disable-next-line no-unused-vars
|
|
defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[],
|
|
) {
|
|
const { workspace } = Action;
|
|
if (
|
|
fs.existsSync(
|
|
`${workspace}/cloud-runner-cache/cache/build/build-${buildParameters.buildGuid}.tar${
|
|
CloudRunner.buildParameters.useCompressionStrategy ? '.lz4' : ''
|
|
}`,
|
|
)
|
|
) {
|
|
await CloudRunnerSystem.Run(`ls ${workspace}/cloud-runner-cache/cache/build/`);
|
|
await CloudRunnerSystem.Run(
|
|
`rm -r ${workspace}/cloud-runner-cache/cache/build/build-${buildParameters.buildGuid}.tar${
|
|
CloudRunner.buildParameters.useCompressionStrategy ? '.lz4' : ''
|
|
}`,
|
|
);
|
|
}
|
|
}
|
|
setupWorkflow(
|
|
buildGuid: string,
|
|
buildParameters: BuildParameters,
|
|
// eslint-disable-next-line no-unused-vars
|
|
branchName: string,
|
|
// eslint-disable-next-line no-unused-vars
|
|
defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[],
|
|
) {
|
|
this.buildParameters = buildParameters;
|
|
}
|
|
|
|
public async runTaskInWorkflow(
|
|
buildGuid: string,
|
|
image: string,
|
|
commands: string,
|
|
mountdir: string,
|
|
workingdir: string,
|
|
environment: CloudRunnerEnvironmentVariable[],
|
|
secrets: CloudRunnerSecret[],
|
|
): Promise<string> {
|
|
CloudRunnerLogger.log(buildGuid);
|
|
CloudRunnerLogger.log(commands);
|
|
|
|
const { workspace, actionFolder } = Action;
|
|
const content: StringKeyValuePair[] = [];
|
|
for (const x of secrets) {
|
|
content.push({ name: x.EnvironmentVariable, value: x.ParameterValue });
|
|
}
|
|
|
|
// Replace localhost with host.docker.internal for LocalStack endpoints (similar to K8s)
|
|
// This allows Docker containers to access LocalStack running on the host
|
|
const endpointEnvironmentNames = new Set([
|
|
'AWS_S3_ENDPOINT',
|
|
'AWS_ENDPOINT',
|
|
'AWS_CLOUD_FORMATION_ENDPOINT',
|
|
'AWS_ECS_ENDPOINT',
|
|
'AWS_KINESIS_ENDPOINT',
|
|
'AWS_CLOUD_WATCH_LOGS_ENDPOINT',
|
|
'INPUT_AWSS3ENDPOINT',
|
|
'INPUT_AWSENDPOINT',
|
|
]);
|
|
for (const x of environment) {
|
|
let value = x.value;
|
|
if (
|
|
typeof value === 'string' &&
|
|
endpointEnvironmentNames.has(x.name) &&
|
|
(value.startsWith('http://localhost') || value.startsWith('http://127.0.0.1'))
|
|
) {
|
|
// Replace localhost with host.docker.internal so containers can access host services
|
|
value = value
|
|
.replace('http://localhost', 'http://host.docker.internal')
|
|
.replace('http://127.0.0.1', 'http://host.docker.internal');
|
|
CloudRunnerLogger.log(`Replaced localhost with host.docker.internal for ${x.name}: ${value}`);
|
|
}
|
|
content.push({ name: x.name, value });
|
|
}
|
|
|
|
// if (this.buildParameters?.cloudRunnerIntegrationTests) {
|
|
// core.info(JSON.stringify(content, undefined, 4));
|
|
// core.info(JSON.stringify(secrets, undefined, 4));
|
|
// core.info(JSON.stringify(environment, undefined, 4));
|
|
// }
|
|
|
|
// eslint-disable-next-line unicorn/no-for-loop
|
|
for (let index = 0; index < content.length; index++) {
|
|
if (content[index] === undefined) {
|
|
delete content[index];
|
|
}
|
|
}
|
|
let myOutput = '';
|
|
const sharedFolder = `/data/`;
|
|
|
|
// core.info(JSON.stringify({ workspace, actionFolder, ...this.buildParameters, ...content }, undefined, 4));
|
|
const entrypointFilePath = `start.sh`;
|
|
const fileContents = `#!/bin/bash
|
|
set -e
|
|
|
|
mkdir -p /github/workspace/cloud-runner-cache
|
|
mkdir -p /data/cache
|
|
cp -a /github/workspace/cloud-runner-cache/. ${sharedFolder}
|
|
${CommandHookService.ApplyHooksToCommands(commands, this.buildParameters)}
|
|
# Only copy cache directory, exclude retained workspaces to avoid running out of disk space
|
|
if [ -d "${sharedFolder}cache" ]; then
|
|
cp -a ${sharedFolder}cache/. /github/workspace/cloud-runner-cache/cache/ || true
|
|
fi
|
|
# Copy test files from /data/ root to workspace for test assertions
|
|
# This allows tests to write files to /data/ and have them available in the workspace
|
|
find ${sharedFolder} -maxdepth 1 -type f -name "test-*" -exec cp -a {} /github/workspace/cloud-runner-cache/ \\; || true
|
|
`;
|
|
writeFileSync(`${workspace}/${entrypointFilePath}`, fileContents, {
|
|
flag: 'w',
|
|
});
|
|
|
|
if (CloudRunner.buildParameters.cloudRunnerDebug) {
|
|
CloudRunnerLogger.log(`Running local-docker: \n ${fileContents}`);
|
|
}
|
|
|
|
if (fs.existsSync(`${workspace}/cloud-runner-cache`)) {
|
|
await CloudRunnerSystem.Run(`ls ${workspace}/cloud-runner-cache && du -sh ${workspace}/cloud-runner-cache`);
|
|
}
|
|
const exitCode = await Docker.run(
|
|
image,
|
|
{ workspace, actionFolder, ...this.buildParameters },
|
|
false,
|
|
`chmod +x /github/workspace/${entrypointFilePath} && /github/workspace/${entrypointFilePath}`,
|
|
content,
|
|
{
|
|
listeners: {
|
|
stdout: (data: Buffer) => {
|
|
myOutput += data.toString();
|
|
},
|
|
stderr: (data: Buffer) => {
|
|
myOutput += `[LOCAL-DOCKER-ERROR]${data.toString()}`;
|
|
},
|
|
},
|
|
},
|
|
true,
|
|
);
|
|
|
|
// Docker doesn't exit on fail now so adding this to ensure behavior is unchanged
|
|
// TODO: Is there a helpful way to consume the exit code or is it best to except
|
|
if (exitCode !== 0) {
|
|
throw new Error(`Build failed with exit code ${exitCode}`);
|
|
}
|
|
|
|
return myOutput;
|
|
}
|
|
}
|
|
export default LocalDockerCloudRunner;
|