unity-builder/src/model/cli/shared-workspace-locking.ts

188 lines
6.6 KiB
TypeScript
Raw Normal View History

2022-09-16 22:51:19 +00:00
import { CloudRunnerSystem } from '../cloud-runner/services/cloud-runner-system';
import * as fs from 'fs';
2022-09-17 03:38:00 +00:00
import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger';
2022-09-17 03:41:41 +00:00
import CloudRunnerOptions from '../cloud-runner/cloud-runner-options';
2022-10-05 17:00:51 +00:00
import BuildParameters from '../build-parameters';
2022-09-16 21:24:23 +00:00
export class SharedWorkspaceLocking {
2022-10-05 00:37:32 +00:00
private static readonly workspaceRoot = `s3://game-ci-test-storage/locks/`;
2022-10-05 17:00:51 +00:00
public static async GetAllWorkspaces(buildParametersContext: BuildParameters): Promise<string[]> {
return (
await SharedWorkspaceLocking.ReadLines(
`aws s3 ls ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/`,
)
).map((x) => x.replace(`/`, ``));
}
2022-10-05 17:00:51 +00:00
public static async GetAllLocks(workspace: string, buildParametersContext: BuildParameters): Promise<string[]> {
if (!(await SharedWorkspaceLocking.DoesWorkspaceExist(workspace, buildParametersContext))) {
throw new Error("Workspace doesn't exist, can't call get all locks");
}
return (
2022-10-05 17:00:51 +00:00
await SharedWorkspaceLocking.ReadLines(
`aws s3 ls ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/`,
)
).map((x) => x.replace(`/`, ``));
}
2022-10-05 17:00:51 +00:00
public static async GetOrCreateLockedWorkspace(
workspaceIfCreated: string,
runId: string,
buildParametersContext: BuildParameters,
) {
2022-09-17 03:41:41 +00:00
if (!CloudRunnerOptions.retainWorkspaces) {
return;
}
CloudRunnerLogger.log(`run agent ${runId} is trying to access a workspace`);
2022-10-05 17:00:51 +00:00
const workspaces = await SharedWorkspaceLocking.GetFreeWorkspaces(buildParametersContext);
2022-09-16 21:24:23 +00:00
for (const element of workspaces) {
2022-10-05 17:00:51 +00:00
if (await SharedWorkspaceLocking.LockWorkspace(element, runId, buildParametersContext)) {
CloudRunnerLogger.log(`run agent ${runId} locked workspace: ${element}`);
2022-09-16 21:24:23 +00:00
return element;
}
}
2022-10-05 17:00:51 +00:00
const workspace = await SharedWorkspaceLocking.CreateWorkspace(workspaceIfCreated, buildParametersContext, runId);
CloudRunnerLogger.log(`run agent ${runId} didn't find a free workspace so created: ${workspace}`);
return workspace;
2022-09-16 21:24:23 +00:00
}
2022-10-05 17:00:51 +00:00
public static async DoesWorkspaceExist(workspace: string, buildParametersContext: BuildParameters) {
return (await SharedWorkspaceLocking.GetAllWorkspaces(buildParametersContext)).includes(workspace);
2022-09-29 21:03:16 +00:00
}
2022-10-05 17:00:51 +00:00
public static async HasWorkspaceLock(
workspace: string,
runId: string,
buildParametersContext: BuildParameters,
): Promise<boolean> {
if (!(await SharedWorkspaceLocking.DoesWorkspaceExist(workspace, buildParametersContext))) {
return false;
}
2022-09-29 21:03:16 +00:00
2022-10-05 17:00:51 +00:00
return (
(await SharedWorkspaceLocking.GetAllLocks(workspace, buildParametersContext)).filter((x) => x.includes(runId))
.length > 0
);
2022-10-05 00:20:48 +00:00
}
2022-10-05 17:00:51 +00:00
public static async GetFreeWorkspaces(buildParametersContext: BuildParameters): Promise<string[]> {
2022-09-29 20:30:34 +00:00
const result: string[] = [];
2022-10-05 17:00:51 +00:00
const workspaces = await SharedWorkspaceLocking.GetAllWorkspaces(buildParametersContext);
2022-09-29 20:30:34 +00:00
for (const element of workspaces) {
2022-10-05 17:00:51 +00:00
if (!(await SharedWorkspaceLocking.IsWorkspaceLocked(element, buildParametersContext))) {
2022-09-29 20:30:34 +00:00
result.push(element);
}
}
return result;
}
2022-10-05 17:00:51 +00:00
public static async IsWorkspaceLocked(workspace: string, buildParametersContext: BuildParameters): Promise<boolean> {
if (!(await SharedWorkspaceLocking.DoesWorkspaceExist(workspace, buildParametersContext))) {
return false;
2022-09-29 21:03:16 +00:00
}
const files = await SharedWorkspaceLocking.ReadLines(
2022-10-05 17:00:51 +00:00
`aws s3 ls ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/`,
);
2022-09-29 21:03:16 +00:00
const workspaceFileDoesNotExists =
files.filter((x) => {
return x.includes(`_workspace`);
}).length === 0;
const lockFilesExist =
files.filter((x) => {
return x.includes(`_lock`);
}).length > 0;
return workspaceFileDoesNotExists || lockFilesExist;
2022-09-29 20:30:34 +00:00
}
2022-10-05 17:00:51 +00:00
public static async CreateWorkspace(workspace: string, buildParametersContext: BuildParameters, lockId: string = ``) {
if (lockId !== ``) {
2022-10-05 17:00:51 +00:00
await SharedWorkspaceLocking.LockWorkspace(workspace, lockId, buildParametersContext);
}
const file = `${Date.now()}_workspace`;
fs.writeFileSync(file, '');
await CloudRunnerSystem.Run(
2022-10-05 17:00:51 +00:00
`aws s3 cp ./${file} ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/${file}`,
false,
true,
);
fs.rmSync(file);
2022-09-29 20:30:34 +00:00
return workspace;
2022-09-16 21:24:23 +00:00
}
2022-09-29 20:30:34 +00:00
2022-10-05 17:00:51 +00:00
public static async LockWorkspace(
workspace: string,
runId: string,
buildParametersContext: BuildParameters,
): Promise<boolean> {
2022-09-29 20:30:34 +00:00
const file = `${Date.now()}_${runId}_lock`;
2022-09-16 22:51:19 +00:00
fs.writeFileSync(file, '');
2022-09-29 20:30:34 +00:00
await CloudRunnerSystem.Run(
2022-10-05 17:00:51 +00:00
`aws s3 cp ./${file} ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/${file}`,
2022-09-29 20:30:34 +00:00
false,
true,
);
2022-09-16 22:51:19 +00:00
fs.rmSync(file);
2022-10-05 17:00:51 +00:00
return SharedWorkspaceLocking.HasWorkspaceLock(workspace, runId, buildParametersContext);
2022-09-16 22:51:19 +00:00
}
2022-09-29 20:30:34 +00:00
2022-10-05 17:00:51 +00:00
public static async ReleaseWorkspace(
workspace: string,
runId: string,
buildParametersContext: BuildParameters,
): Promise<boolean> {
if (!(await SharedWorkspaceLocking.DoesWorkspaceExist(workspace, buildParametersContext))) {
2022-09-29 21:25:43 +00:00
return true;
}
2022-10-05 17:00:51 +00:00
const file = (await SharedWorkspaceLocking.GetAllLocks(workspace, buildParametersContext)).filter((x) =>
x.includes(`_${runId}_lock`),
);
CloudRunnerLogger.log(
`${JSON.stringify(await SharedWorkspaceLocking.GetAllLocks(workspace, buildParametersContext))}`,
);
2022-09-29 20:30:34 +00:00
CloudRunnerLogger.log(`Deleting file ${file}`);
2022-10-05 17:00:51 +00:00
CloudRunnerLogger.log(
`aws s3 rm ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/${file}`,
);
await CloudRunnerSystem.Run(
`aws s3 rm ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace}/${file}`,
false,
true,
);
2022-09-29 20:30:34 +00:00
2022-10-05 17:00:51 +00:00
return !SharedWorkspaceLocking.HasWorkspaceLock(workspace, runId, buildParametersContext);
2022-09-29 20:30:34 +00:00
}
2022-10-05 17:00:51 +00:00
public static async CleanupWorkspace(workspace: string, buildParametersContext: BuildParameters) {
2022-09-29 20:30:34 +00:00
await CloudRunnerSystem.Run(
2022-10-05 17:00:51 +00:00
`aws s3 rm ${SharedWorkspaceLocking.workspaceRoot}${buildParametersContext.cacheKey}/${workspace} --recursive`,
2022-09-29 20:30:34 +00:00
false,
true,
);
}
2022-09-17 04:44:07 +00:00
private static async ReadLines(command: string): Promise<string[]> {
const result = await CloudRunnerSystem.Run(command, false, true);
return result
.split(`\n`)
.map((x) => x.replace(`\r`, ``))
.filter((x) => x !== ``)
.map((x) => {
const lineValues = x.split(` `);
return lineValues[lineValues.length - 1];
});
2022-09-17 04:09:36 +00:00
}
2022-09-16 21:24:23 +00:00
}
2022-09-16 18:48:40 +00:00
export default SharedWorkspaceLocking;