| 
									
										
										
										
											2023-12-10 11:50:15 +00:00
										 |  |  | import ImageEnvironmentFactory from './image-environment-factory'; | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  | import { existsSync, mkdirSync, readFileSync, rmSync } from 'fs'; | 
					
						
							| 
									
										
										
										
											2022-11-04 11:35:06 +00:00
										 |  |  | import LicensingServerSetup from './licensing-server-setup'; | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  | import type { RunnerContext } from './action'; | 
					
						
							| 
									
										
										
										
											2022-01-11 11:52:29 +00:00
										 |  |  | import { exec } from '@actions/exec'; | 
					
						
							| 
									
										
										
										
											2022-03-11 17:46:28 +00:00
										 |  |  | import path from 'path'; | 
					
						
							| 
									
										
										
										
											2020-01-29 21:22:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Build a path for a docker --cidfile parameter. Docker will store the the created container. | 
					
						
							|  |  |  |  * This path is stable for the whole execution of the action, so it can be executed with the same parameters | 
					
						
							|  |  |  |  * multiple times and get the same result. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const containerIdFilePath = parameters => { | 
					
						
							|  |  |  |   const { runnerTemporaryPath, githubAction } = parameters; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return path.join(runnerTemporaryPath, `container_${githubAction}`); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 11:52:29 +00:00
										 |  |  | const Docker = { | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  |   /** | 
					
						
							|  |  |  |    *  Remove a possible leftover container created by `Docker.run`. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   async ensureContainerRemoval(parameters: RunnerContext) { | 
					
						
							|  |  |  |     const cidfile = containerIdFilePath(parameters); | 
					
						
							|  |  |  |     if (!existsSync(cidfile)) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const container = readFileSync(cidfile, 'ascii').trim(); | 
					
						
							|  |  |  |     await exec(`docker`, ['rm', '--force', '--volumes', container], { silent: true }); | 
					
						
							|  |  |  |     rmSync(cidfile); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 11:52:29 +00:00
										 |  |  |   async run(image, parameters, silent = false) { | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     let runCommand = ''; | 
					
						
							| 
									
										
										
										
											2022-11-04 11:35:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parameters.unityLicensingServer !== '') { | 
					
						
							|  |  |  |       LicensingServerSetup.Setup(parameters.unityLicensingServer, parameters.actionFolder); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     switch (process.platform) { | 
					
						
							|  |  |  |       case 'linux': | 
					
						
							|  |  |  |         runCommand = this.getLinuxCommand(image, parameters); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case 'win32': | 
					
						
							|  |  |  |         runCommand = this.getWindowsCommand(image, parameters); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         throw new Error(`Operation system, ${process.platform}, is not supported yet.`); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     await exec(runCommand, undefined, { silent }); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getLinuxCommand(image, parameters): string { | 
					
						
							| 
									
										
										
										
											2020-01-30 23:57:08 +00:00
										 |  |  |     const { | 
					
						
							| 
									
										
										
										
											2022-04-03 10:14:39 +00:00
										 |  |  |       actionFolder, | 
					
						
							| 
									
										
										
										
											2020-01-30 23:57:08 +00:00
										 |  |  |       workspace, | 
					
						
							|  |  |  |       testMode, | 
					
						
							| 
									
										
										
										
											2020-04-01 20:24:13 +00:00
										 |  |  |       useHostNetwork, | 
					
						
							| 
									
										
										
										
											2021-05-28 21:55:58 +00:00
										 |  |  |       sshAgent, | 
					
						
							| 
									
										
										
										
											2023-09-06 21:35:36 +00:00
										 |  |  |       sshPublicKeysDirectoryPath, | 
					
						
							| 
									
										
										
										
											2022-01-01 20:38:47 +00:00
										 |  |  |       githubToken, | 
					
						
							| 
									
										
										
										
											2022-04-03 15:59:11 +00:00
										 |  |  |       runnerTemporaryPath, | 
					
						
							| 
									
										
										
										
											2023-11-05 03:50:09 +00:00
										 |  |  |       dockerCpuLimit, | 
					
						
							|  |  |  |       dockerMemoryLimit, | 
					
						
							| 
									
										
										
										
											2020-01-30 23:57:08 +00:00
										 |  |  |     } = parameters; | 
					
						
							| 
									
										
										
										
											2020-01-29 21:22:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 15:59:11 +00:00
										 |  |  |     const githubHome = path.join(runnerTemporaryPath, '_github_home'); | 
					
						
							| 
									
										
										
										
											2022-03-11 17:46:28 +00:00
										 |  |  |     if (!existsSync(githubHome)) mkdirSync(githubHome); | 
					
						
							| 
									
										
										
										
											2022-04-03 15:59:11 +00:00
										 |  |  |     const githubWorkflow = path.join(runnerTemporaryPath, '_github_workflow'); | 
					
						
							| 
									
										
										
										
											2022-03-11 17:46:28 +00:00
										 |  |  |     if (!existsSync(githubWorkflow)) mkdirSync(githubWorkflow); | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  |     const cidfile = containerIdFilePath(parameters); | 
					
						
							| 
									
										
										
										
											2022-04-21 08:50:37 +00:00
										 |  |  |     const testPlatforms = ( | 
					
						
							|  |  |  |       testMode === 'all' ? ['playmode', 'editmode', 'COMBINE_RESULTS'] : [testMode] | 
					
						
							|  |  |  |     ).join(';'); | 
					
						
							| 
									
										
										
										
											2022-03-11 17:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     return `docker run \
 | 
					
						
							| 
									
										
										
										
											2023-12-10 11:50:15 +00:00
										 |  |  |             --workdir /github/workspace \ | 
					
						
							|  |  |  |             --cidfile "${cidfile}" \ | 
					
						
							|  |  |  |             --rm \ | 
					
						
							|  |  |  |             ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ | 
					
						
							|  |  |  |             --env GIT_CONFIG_EXTENSIONS \ | 
					
						
							|  |  |  |             --env TEST_PLATFORMS="${testPlatforms}" \ | 
					
						
							|  |  |  |             ${sshAgent ? '--env SSH_AUTH_SOCK=/ssh-agent' : ''} \ | 
					
						
							|  |  |  |             --volume "${githubHome}:/root:z" \ | 
					
						
							|  |  |  |             --volume "${githubWorkflow}:/github/workflow:z" \ | 
					
						
							|  |  |  |             --volume "${workspace}:/github/workspace:z" \ | 
					
						
							|  |  |  |             --volume "${actionFolder}/test-standalone-scripts:/UnityStandaloneScripts:z" \ | 
					
						
							| 
									
										
										
										
											2023-12-11 00:33:51 +00:00
										 |  |  |             --volume "${actionFolder}/platforms/ubuntu:/steps:z" \ | 
					
						
							| 
									
										
										
										
											2023-12-10 11:50:15 +00:00
										 |  |  |             --volume "${actionFolder}/unity-config:/usr/share/unity3d/config/:z" \ | 
					
						
							|  |  |  |             --volume "${actionFolder}/BlankProject":"/BlankProject:z" \ | 
					
						
							|  |  |  |             --cpus=${dockerCpuLimit} \ | 
					
						
							|  |  |  |             --memory=${dockerMemoryLimit} \ | 
					
						
							|  |  |  |             ${sshAgent ? `--volume ${sshAgent}:/ssh-agent` : ''} \ | 
					
						
							|  |  |  |             ${ | 
					
						
							|  |  |  |               sshAgent && !sshPublicKeysDirectoryPath | 
					
						
							|  |  |  |                 ? `--volume /home/runner/.ssh/known_hosts:/root/.ssh/known_hosts:ro` | 
					
						
							|  |  |  |                 : '' | 
					
						
							|  |  |  |             } \ | 
					
						
							|  |  |  |             ${ | 
					
						
							|  |  |  |               sshPublicKeysDirectoryPath | 
					
						
							|  |  |  |                 ? `--volume ${sshPublicKeysDirectoryPath}:/root/.ssh:ro` | 
					
						
							|  |  |  |                 : '' | 
					
						
							|  |  |  |             } \ | 
					
						
							|  |  |  |             ${useHostNetwork ? '--net=host' : ''} \ | 
					
						
							|  |  |  |             ${githubToken ? '--env USE_EXIT_CODE=false' : '--env USE_EXIT_CODE=true'} \ | 
					
						
							|  |  |  |             ${image} \ | 
					
						
							| 
									
										
										
										
											2023-12-11 00:42:21 +00:00
										 |  |  |             /bin/bash -c "ls -lashi /steps`; | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getWindowsCommand(image, parameters): string { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       actionFolder, | 
					
						
							|  |  |  |       workspace, | 
					
						
							|  |  |  |       testMode, | 
					
						
							|  |  |  |       useHostNetwork, | 
					
						
							|  |  |  |       sshAgent, | 
					
						
							|  |  |  |       githubToken, | 
					
						
							|  |  |  |       runnerTemporaryPath, | 
					
						
							| 
									
										
										
										
											2023-11-05 03:50:09 +00:00
										 |  |  |       dockerCpuLimit, | 
					
						
							|  |  |  |       dockerMemoryLimit, | 
					
						
							|  |  |  |       dockerIsolationMode, | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     } = parameters; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const githubHome = path.join(runnerTemporaryPath, '_github_home'); | 
					
						
							|  |  |  |     if (!existsSync(githubHome)) mkdirSync(githubHome); | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  |     const cidfile = containerIdFilePath(parameters); | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     const githubWorkflow = path.join(runnerTemporaryPath, '_github_workflow'); | 
					
						
							|  |  |  |     if (!existsSync(githubWorkflow)) mkdirSync(githubWorkflow); | 
					
						
							|  |  |  |     const testPlatforms = ( | 
					
						
							|  |  |  |       testMode === 'all' ? ['playmode', 'editmode', 'COMBINE_RESULTS'] : [testMode] | 
					
						
							|  |  |  |     ).join(';'); | 
					
						
							| 
									
										
										
										
											2020-01-29 21:22:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |     return `docker run \
 | 
					
						
							| 
									
										
										
										
											2023-12-10 10:51:37 +00:00
										 |  |  |                 --workdir c:/github/workspace \ | 
					
						
							| 
									
										
											  
											
												feat: ensure cleanup of docker containers (#198)
Cancelled or timeouted workflow would keep the docker container running.
Closes game-ci/unity-test-runner#197
This has two parts:
Part one. The entrypoints.
`runs.post`: GitHub Action metadata allow running something after the
action (regardless of a failure, crash, timeout, ...).
https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions#runspost
However, it needs to be a `.js` file and it can't be configured to pass
any arguments.
There already was `index.js` used as the main entrypoint.
The build process of this file uses typescript compiler and ncc to pack
all dependencies into one .js file. And ncc has no way of generating
multiple files in one go, so the only solution would be to run ncc twice
and generate two independent files.
That would be quite unfortunate, wasting time and storage. So I rather
came up with a new entrypoint that symlinked from two locations.
And this new entrypoint understands how it was executed, so it can run
the correct behaviour. This makes it easy to add `runs.pre` if needed.
This new entrypoint is in `index.ts`. The original `index.ts` is now in
`main.ts`.
Part two.
The signals. I've tried:
* try/catch/finally around the `await Docker.run`. Catch and finally are
  not executed when process receives SIGINT. See the discussion in: https://github.com/nodejs/node/discussions/29480
* New AbortController and AbortSignal. Great concept, but the
  action.exec does not support it. So it can't be aborted.
* Doing cleanup on `process.on('exit')`. Unfortunately you can't really
  do async stuff from there, so can't really run the docker rm command
to delete the container.
* Using `process.on('SIGINT')`. For some reason that wasn't really
  executing for me. I'd not put my hand in fire for this, but I assume
because it was in the signal handler it does something special, or would
heed to be scheduled for later with `setTimeout(0)`.
Evaluating all these I came to a conclusion that it is fragile and just
relying on a `runs.post` is much better and safer.
`
											
										 
											2022-11-03 18:14:51 +00:00
										 |  |  |                 --cidfile "${cidfile}" \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 --rm \ | 
					
						
							| 
									
										
										
										
											2023-12-10 11:50:15 +00:00
										 |  |  |                 ${ImageEnvironmentFactory.getEnvVarString(parameters)} \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 --env TEST_PLATFORMS="${testPlatforms}" \ | 
					
						
							| 
									
										
										
										
											2023-12-10 10:51:37 +00:00
										 |  |  |                 --env GITHUB_WORKSPACE="c:/github/workspace" \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 ${sshAgent ? '--env SSH_AUTH_SOCK=c:/ssh-agent' : ''} \ | 
					
						
							| 
									
										
										
										
											2023-04-21 03:23:15 +00:00
										 |  |  |                 --volume "${actionFolder}/test-standalone-scripts":"c:/UnityStandaloneScripts" \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 --volume "${githubHome}":"c:/root" \ | 
					
						
							|  |  |  |                 --volume "${githubWorkflow}":"c:/github/workflow" \ | 
					
						
							|  |  |  |                 --volume "${workspace}":"c:/github/workspace" \ | 
					
						
							| 
									
										
										
										
											2023-12-10 10:51:37 +00:00
										 |  |  |                 --volume "${actionFolder}/platforms/windows":"c:/steps" \ | 
					
						
							|  |  |  |                 --volume "${actionFolder}/BlankProject":"c:/BlankProject" \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 ${sshAgent ? `--volume ${sshAgent}:c:/ssh-agent` : ''} \ | 
					
						
							|  |  |  |                 ${ | 
					
						
							|  |  |  |                   sshAgent | 
					
						
							|  |  |  |                     ? `--volume c:/Users/Administrator/.ssh/known_hosts:c:/root/.ssh/known_hosts` | 
					
						
							|  |  |  |                     : '' | 
					
						
							|  |  |  |                 } \ | 
					
						
							| 
									
										
										
										
											2023-11-05 03:50:09 +00:00
										 |  |  |                 --cpus=${dockerCpuLimit} \ | 
					
						
							|  |  |  |                 --memory=${dockerMemoryLimit} \ | 
					
						
							|  |  |  |                 --isolation=${dockerIsolationMode} \ | 
					
						
							| 
									
										
										
										
											2022-05-23 23:41:36 +00:00
										 |  |  |                 ${useHostNetwork ? '--net=host' : ''} \ | 
					
						
							|  |  |  |                 ${githubToken ? '--env USE_EXIT_CODE=false' : '--env USE_EXIT_CODE=true'} \ | 
					
						
							|  |  |  |                 ${image} \ | 
					
						
							| 
									
										
										
										
											2023-12-10 10:51:37 +00:00
										 |  |  |                 powershell c:/steps/entrypoint.ps1`; | 
					
						
							| 
									
										
										
										
											2022-01-11 11:52:29 +00:00
										 |  |  |   }, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-01-29 21:22:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | export default Docker; |