mirror of
https://github.com/actions/runner.git
synced 2025-12-11 21:06:55 +00:00
Compare commits
1 Commits
users/tihu
...
users/pje/
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b3ab3f746f |
74
job.yml
74
job.yml
@@ -1,74 +0,0 @@
|
|||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: Role
|
|
||||||
metadata:
|
|
||||||
name: pod-admin
|
|
||||||
namespace: default
|
|
||||||
rules:
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["pods", "pods/log", "pods/attach", "pods/exec"]
|
|
||||||
verbs: ["get", "list", "watch", "create", "update", "patch", "delete"]
|
|
||||||
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
metadata:
|
|
||||||
name: default-pod-admin
|
|
||||||
namespace: default
|
|
||||||
roleRef:
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
||||||
kind: Role
|
|
||||||
name: pod-admin
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: default
|
|
||||||
namespace: default
|
|
||||||
|
|
||||||
---
|
|
||||||
apiVersion: batch/v1
|
|
||||||
kind: Job
|
|
||||||
metadata:
|
|
||||||
namespace: default
|
|
||||||
name: actions-runners
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
# hostNetwork: true
|
|
||||||
volumes:
|
|
||||||
- name: runner-working
|
|
||||||
emptyDir: {}
|
|
||||||
containers:
|
|
||||||
- name: k8srunner
|
|
||||||
image: huangtingluo/kube-runner:v0
|
|
||||||
imagePullPolicy: Always
|
|
||||||
volumeMounts:
|
|
||||||
- mountPath: /actions-runner/_work
|
|
||||||
name: runner-working
|
|
||||||
env:
|
|
||||||
- name: GITHUB_PAT
|
|
||||||
value: ghp_
|
|
||||||
- name: RUNNER_CONFIG_URL
|
|
||||||
value: https://github.com/bbq-beets/ting-test
|
|
||||||
- name: K8S_NODE_NAME
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: spec.nodeName
|
|
||||||
- name: K8S_POD_NAME
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: metadata.name
|
|
||||||
- name: K8S_POD_NAMESPACE
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: metadata.namespace
|
|
||||||
- name: K8S_POD_IP
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: status.podIP
|
|
||||||
- name: K8S_POD_SERVICE_ACCOUNT
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: spec.serviceAccountName
|
|
||||||
restartPolicy: Never
|
|
||||||
backoffLimit: 1
|
|
||||||
completions: 1
|
|
||||||
parallelism: 1
|
|
||||||
@@ -1,11 +1,21 @@
|
|||||||
## Features
|
## Features
|
||||||
|
|
||||||
|
- Support the `--ephemeral` flag (#660)
|
||||||
|
- This optional flag will configure the runner to only take one job, and let the service un-configure the runner after that job finishes.
|
||||||
|
- Expect to see more info in the Github API documentation soon. We'll link to those docs directly as they become generally available!
|
||||||
|
|
||||||
## Bugs
|
## Bugs
|
||||||
|
|
||||||
- Fixed an issue where ephemeral runners did not restart after upgrading (#1396)
|
- Fix a bug in `script/delete` wherein a repo with multiple runners would be unable to find the correct runner (#1268) (#1269)
|
||||||
|
- Mitigate a race condition when requesting an OIDC `Id_token` (#1320)
|
||||||
|
- Make client retries more resilient in JobServer (#1316)
|
||||||
|
|
||||||
## Misc
|
## Misc
|
||||||
|
|
||||||
|
- Increase readability of colored console output (#1295) (#1319)
|
||||||
|
- Add more network troubleshooting to the docs (#1325)
|
||||||
|
- Bump [path-parse](https://github.com/jbgutierrez/path-parse) from 1.0.6 to 1.0.7 (#1256)
|
||||||
|
|
||||||
## Windows x64
|
## Windows x64
|
||||||
We recommend configuring the runner in a root folder of the Windows drive (e.g. "C:\actions-runner"). This will help avoid issues related to service identity folder permissions and long file path restrictions on Windows.
|
We recommend configuring the runner in a root folder of the Windows drive (e.g. "C:\actions-runner"). This will help avoid issues related to service identity folder permissions and long file path restrictions on Windows.
|
||||||
|
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
FROM mcr.microsoft.com/dotnet/sdk:3.1 AS Build
|
|
||||||
|
|
||||||
# ENV RUNNER_CONFIG_URL=""
|
|
||||||
# ENV GITHUB_PAT=""
|
|
||||||
# ENV RUNNER_NAME=""
|
|
||||||
# ENV RUNNER_GROUP=""
|
|
||||||
# ENV RUNNER_LABELS=""
|
|
||||||
# ENV GITHUB_RUNNER_SCOPE=""
|
|
||||||
# ENV GITHUB_SERVER_URL=""
|
|
||||||
# ENV GITHUB_API_URL=""
|
|
||||||
# ENV K8S_HOST_IP=""
|
|
||||||
|
|
||||||
RUN apt-get update --fix-missing \
|
|
||||||
&& apt-get install -y --no-install-recommends \
|
|
||||||
curl \
|
|
||||||
# jq \
|
|
||||||
# git \
|
|
||||||
apt-utils \
|
|
||||||
apt-transport-https \
|
|
||||||
unzip \
|
|
||||||
net-tools\
|
|
||||||
gnupg2\
|
|
||||||
&& apt-get clean \
|
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
# Install kubectl
|
|
||||||
# RUN curl -s https://packages.cloud.google.com/apt/doc/apt-key.gpg | apt-key add - && \
|
|
||||||
# echo "deb https://apt.kubernetes.io/ kubernetes-xenial main" | tee -a /etc/apt/sources.list.d/kubernetes.list && \
|
|
||||||
# apt-get update && apt-get -y install --no-install-recommends kubectl
|
|
||||||
|
|
||||||
# Install docker
|
|
||||||
# RUN curl -fsSL https://get.docker.com -o get-docker.sh
|
|
||||||
# RUN sh get-docker.sh
|
|
||||||
|
|
||||||
# Allow runner to run as root
|
|
||||||
# ENV RUNNER_ALLOW_RUNASROOT=1
|
|
||||||
|
|
||||||
# Directory for runner to operate in
|
|
||||||
RUN mkdir /actions-runner
|
|
||||||
RUN mkdir /actions-runner/src
|
|
||||||
WORKDIR /actions-runner/src
|
|
||||||
|
|
||||||
COPY ./ /actions-runner/src
|
|
||||||
|
|
||||||
RUN /actions-runner/src/dev.sh l
|
|
||||||
|
|
||||||
FROM mcr.microsoft.com/dotnet/core/runtime-deps:3.1
|
|
||||||
|
|
||||||
ENV RUNNER_CONFIG_URL=""
|
|
||||||
ENV GITHUB_PAT=""
|
|
||||||
|
|
||||||
RUN apt-get update --fix-missing \
|
|
||||||
&& apt-get install -y --no-install-recommends \
|
|
||||||
curl \
|
|
||||||
# jq \
|
|
||||||
# git \
|
|
||||||
# apt-utils \
|
|
||||||
# apt-transport-https \
|
|
||||||
# unzip \
|
|
||||||
# net-tools\
|
|
||||||
gnupg2\
|
|
||||||
&& apt-get clean \
|
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
# Install kubectl
|
|
||||||
RUN curl -s https://packages.cloud.google.com/apt/doc/apt-key.gpg | apt-key add - && \
|
|
||||||
echo "deb https://apt.kubernetes.io/ kubernetes-xenial main" | tee -a /etc/apt/sources.list.d/kubernetes.list && \
|
|
||||||
apt-get update && apt-get -y install --no-install-recommends kubectl
|
|
||||||
|
|
||||||
|
|
||||||
# Allow runner to run as root
|
|
||||||
ENV RUNNER_ALLOW_RUNASROOT=1
|
|
||||||
|
|
||||||
# Directory for runner to operate in
|
|
||||||
RUN mkdir /actions-runner
|
|
||||||
WORKDIR /actions-runner
|
|
||||||
COPY --from=Build /actions-runner/_layout /actions-runner
|
|
||||||
ENTRYPOINT ["./entrypoint.sh"]
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
{
|
|
||||||
"plugins": ["jest", "@typescript-eslint"],
|
|
||||||
"extends": ["plugin:github/es6"],
|
|
||||||
"parser": "@typescript-eslint/parser",
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 9,
|
|
||||||
"sourceType": "module",
|
|
||||||
"project": "./tsconfig.json"
|
|
||||||
},
|
|
||||||
"rules": {
|
|
||||||
"eslint-comments/no-use": "off",
|
|
||||||
"import/no-namespace": "off",
|
|
||||||
"no-console": "off",
|
|
||||||
"no-unused-vars": "off",
|
|
||||||
"@typescript-eslint/no-unused-vars": "error",
|
|
||||||
"@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}],
|
|
||||||
"@typescript-eslint/no-require-imports": "error",
|
|
||||||
"@typescript-eslint/array-type": "error",
|
|
||||||
"@typescript-eslint/await-thenable": "error",
|
|
||||||
"@typescript-eslint/ban-ts-ignore": "error",
|
|
||||||
"camelcase": "off",
|
|
||||||
"@typescript-eslint/camelcase": "error",
|
|
||||||
"@typescript-eslint/class-name-casing": "error",
|
|
||||||
"@typescript-eslint/explicit-function-return-type": ["error", {"allowExpressions": true}],
|
|
||||||
"@typescript-eslint/func-call-spacing": ["error", "never"],
|
|
||||||
"@typescript-eslint/generic-type-naming": ["error", "^[A-Z][A-Za-z]*$"],
|
|
||||||
"@typescript-eslint/no-array-constructor": "error",
|
|
||||||
"@typescript-eslint/no-empty-interface": "error",
|
|
||||||
"@typescript-eslint/no-explicit-any": "error",
|
|
||||||
"@typescript-eslint/no-extraneous-class": "error",
|
|
||||||
"@typescript-eslint/no-for-in-array": "error",
|
|
||||||
"@typescript-eslint/no-inferrable-types": "error",
|
|
||||||
"@typescript-eslint/no-misused-new": "error",
|
|
||||||
"@typescript-eslint/no-namespace": "error",
|
|
||||||
"@typescript-eslint/no-non-null-assertion": "warn",
|
|
||||||
"@typescript-eslint/no-object-literal-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-qualifier": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-useless-constructor": "error",
|
|
||||||
"@typescript-eslint/no-var-requires": "error",
|
|
||||||
"@typescript-eslint/prefer-for-of": "warn",
|
|
||||||
"@typescript-eslint/prefer-function-type": "warn",
|
|
||||||
"@typescript-eslint/prefer-includes": "error",
|
|
||||||
"@typescript-eslint/prefer-interface": "error",
|
|
||||||
"@typescript-eslint/prefer-string-starts-ends-with": "error",
|
|
||||||
"@typescript-eslint/promise-function-async": "error",
|
|
||||||
"@typescript-eslint/require-array-sort-compare": "error",
|
|
||||||
"@typescript-eslint/restrict-plus-operands": "error",
|
|
||||||
"semi": "off",
|
|
||||||
"@typescript-eslint/semi": ["error", "never"],
|
|
||||||
"@typescript-eslint/type-annotation-spacing": "error",
|
|
||||||
"@typescript-eslint/unbound-method": "error"
|
|
||||||
},
|
|
||||||
"env": {
|
|
||||||
"node": true,
|
|
||||||
"es6": true,
|
|
||||||
"jest/globals": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"printWidth": 80,
|
|
||||||
"tabWidth": 2,
|
|
||||||
"useTabs": false,
|
|
||||||
"semi": false,
|
|
||||||
"singleQuote": true,
|
|
||||||
"trailingComma": "none",
|
|
||||||
"bracketSpacing": false,
|
|
||||||
"arrowParens": "avoid",
|
|
||||||
"parser": "typescript"
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
To update kubeInnerHandler under `Misc/layoutbin` run `npm install && npm run all`
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,36 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "kubeInnerHandler",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "GitHub Actions",
|
|
||||||
"main": "lib/kubeInnerHandler.js",
|
|
||||||
"scripts": {
|
|
||||||
"build": "tsc",
|
|
||||||
"format": "prettier --write **/*.ts",
|
|
||||||
"format-check": "prettier --check **/*.ts",
|
|
||||||
"lint": "eslint src/**/*.ts",
|
|
||||||
"pack": "ncc build -o ../../layoutbin/kubeInnerHandler",
|
|
||||||
"all": "npm run build && npm run format && npm run lint && npm run pack"
|
|
||||||
},
|
|
||||||
"repository": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "git+https://github.com/actions/runner.git"
|
|
||||||
},
|
|
||||||
"keywords": [
|
|
||||||
"actions"
|
|
||||||
],
|
|
||||||
"author": "GitHub Actions",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"@actions/exec": "^1.1.0",
|
|
||||||
"@actions/core": "^1.6.0"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/node": "^12.7.12",
|
|
||||||
"@typescript-eslint/parser": "^2.8.0",
|
|
||||||
"@zeit/ncc": "^0.20.5",
|
|
||||||
"eslint": "^6.8.0",
|
|
||||||
"eslint-plugin-github": "^2.0.0",
|
|
||||||
"prettier": "^1.19.1",
|
|
||||||
"typescript": "^3.6.4"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
import * as exec from '@actions/exec'
|
|
||||||
import * as core from '@actions/core'
|
|
||||||
import * as events from 'events'
|
|
||||||
import * as readline from 'readline'
|
|
||||||
|
|
||||||
async function run(): Promise<void> {
|
|
||||||
let input = ''
|
|
||||||
|
|
||||||
const rl = readline.createInterface({
|
|
||||||
input: process.stdin
|
|
||||||
})
|
|
||||||
|
|
||||||
rl.on('line', line => {
|
|
||||||
core.debug(`Line from STDIN: ${line}`)
|
|
||||||
input = line
|
|
||||||
})
|
|
||||||
|
|
||||||
await events.once(rl, 'close')
|
|
||||||
|
|
||||||
core.debug(input)
|
|
||||||
|
|
||||||
const execInput = JSON.parse(input)
|
|
||||||
core.debug(JSON.stringify(execInput))
|
|
||||||
|
|
||||||
// podman exec -i --workdir /__w/canary/canary
|
|
||||||
// -e GITHUB_JOB -e GITHUB_REF -e GITHUB_SHA -e GITHUB_REPOSITORY
|
|
||||||
// -e GITHUB_REPOSITORY_OWNER -e GITHUB_RUN_ID -e GITHUB_RUN_NUMBER
|
|
||||||
// -e GITHUB_RETENTION_DAYS -e GITHUB_RUN_ATTEMPT -e GITHUB_ACTOR
|
|
||||||
// -e GITHUB_WORKFLOW -e GITHUB_HEAD_REF -e GITHUB_BASE_REF -e GITHUB_EVENT_NAME
|
|
||||||
// -e GITHUB_SERVER_URL -e GITHUB_API_URL -e GITHUB_GRAPHQL_URL
|
|
||||||
// -e GITHUB_WORKSPACE -e GITHUB_ACTION -e GITHUB_EVENT_PATH -e GITHUB_ACTION_REPOSITORY
|
|
||||||
// -e GITHUB_ACTION_REF -e GITHUB_PATH -e GITHUB_ENV -e RUNNER_DEBUG
|
|
||||||
// -e RUNNER_OS -e RUNNER_NAME -e RUNNER_TOOL_CACHE
|
|
||||||
// -e RUNNER_TEMP -e RUNNER_WORKSPACE
|
|
||||||
// eccdf520697a035599d6e8c8dc801f004fdd3797cdce88f590aba3669a88d9bc sh -e /__w/_temp/d3b30383-719c-4e76-a16f-8f85443352be.sh
|
|
||||||
|
|
||||||
const execArgs = []
|
|
||||||
const args = (<string>execInput.arguments).split(' ')
|
|
||||||
core.debug(JSON.stringify(args))
|
|
||||||
execArgs.push(...args)
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(execArgs))
|
|
||||||
|
|
||||||
await exec.exec(execInput.fileName, execArgs, {
|
|
||||||
env: execInput.environmentVariables
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
run()
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"target": "es6", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019' or 'ESNEXT'. */
|
|
||||||
"module": "commonjs", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */
|
|
||||||
"outDir": "./lib", /* Redirect output structure to the directory. */
|
|
||||||
"rootDir": "./src", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */
|
|
||||||
"strict": true, /* Enable all strict type-checking options. */
|
|
||||||
"noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */
|
|
||||||
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */
|
|
||||||
},
|
|
||||||
"exclude": ["node_modules", "**/*.test.ts"]
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
{
|
|
||||||
"plugins": ["jest", "@typescript-eslint"],
|
|
||||||
"extends": ["plugin:github/es6"],
|
|
||||||
"parser": "@typescript-eslint/parser",
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 9,
|
|
||||||
"sourceType": "module",
|
|
||||||
"project": "./tsconfig.json"
|
|
||||||
},
|
|
||||||
"rules": {
|
|
||||||
"eslint-comments/no-use": "off",
|
|
||||||
"import/no-namespace": "off",
|
|
||||||
"no-console": "off",
|
|
||||||
"no-unused-vars": "off",
|
|
||||||
"@typescript-eslint/no-unused-vars": "error",
|
|
||||||
"@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}],
|
|
||||||
"@typescript-eslint/no-require-imports": "error",
|
|
||||||
"@typescript-eslint/array-type": "error",
|
|
||||||
"@typescript-eslint/await-thenable": "error",
|
|
||||||
"@typescript-eslint/ban-ts-ignore": "error",
|
|
||||||
"camelcase": "off",
|
|
||||||
"@typescript-eslint/camelcase": "error",
|
|
||||||
"@typescript-eslint/class-name-casing": "error",
|
|
||||||
"@typescript-eslint/explicit-function-return-type": ["error", {"allowExpressions": true}],
|
|
||||||
"@typescript-eslint/func-call-spacing": ["error", "never"],
|
|
||||||
"@typescript-eslint/generic-type-naming": ["error", "^[A-Z][A-Za-z]*$"],
|
|
||||||
"@typescript-eslint/no-array-constructor": "error",
|
|
||||||
"@typescript-eslint/no-empty-interface": "error",
|
|
||||||
"@typescript-eslint/no-explicit-any": "error",
|
|
||||||
"@typescript-eslint/no-extraneous-class": "error",
|
|
||||||
"@typescript-eslint/no-for-in-array": "error",
|
|
||||||
"@typescript-eslint/no-inferrable-types": "error",
|
|
||||||
"@typescript-eslint/no-misused-new": "error",
|
|
||||||
"@typescript-eslint/no-namespace": "error",
|
|
||||||
"@typescript-eslint/no-non-null-assertion": "warn",
|
|
||||||
"@typescript-eslint/no-object-literal-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-qualifier": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-useless-constructor": "error",
|
|
||||||
"@typescript-eslint/no-var-requires": "error",
|
|
||||||
"@typescript-eslint/prefer-for-of": "warn",
|
|
||||||
"@typescript-eslint/prefer-function-type": "warn",
|
|
||||||
"@typescript-eslint/prefer-includes": "error",
|
|
||||||
"@typescript-eslint/prefer-interface": "error",
|
|
||||||
"@typescript-eslint/prefer-string-starts-ends-with": "error",
|
|
||||||
"@typescript-eslint/promise-function-async": "error",
|
|
||||||
"@typescript-eslint/require-array-sort-compare": "error",
|
|
||||||
"@typescript-eslint/restrict-plus-operands": "error",
|
|
||||||
"semi": "off",
|
|
||||||
"@typescript-eslint/semi": ["error", "never"],
|
|
||||||
"@typescript-eslint/type-annotation-spacing": "error",
|
|
||||||
"@typescript-eslint/unbound-method": "error"
|
|
||||||
},
|
|
||||||
"env": {
|
|
||||||
"node": true,
|
|
||||||
"es6": true,
|
|
||||||
"jest/globals": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"printWidth": 80,
|
|
||||||
"tabWidth": 2,
|
|
||||||
"useTabs": false,
|
|
||||||
"semi": false,
|
|
||||||
"singleQuote": true,
|
|
||||||
"trailingComma": "none",
|
|
||||||
"bracketSpacing": false,
|
|
||||||
"arrowParens": "avoid",
|
|
||||||
"parser": "typescript"
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
To update kubectlHandler under `Misc/layoutbin` run `npm install && npm run all`
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,36 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "kubectlHandler",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "GitHub Actions",
|
|
||||||
"main": "lib/kubectlHandler.js",
|
|
||||||
"scripts": {
|
|
||||||
"build": "tsc",
|
|
||||||
"format": "prettier --write **/*.ts",
|
|
||||||
"format-check": "prettier --check **/*.ts",
|
|
||||||
"lint": "eslint src/**/*.ts",
|
|
||||||
"pack": "ncc build -o ../../layoutbin/kubectlHandler",
|
|
||||||
"all": "npm run build && npm run format && npm run lint && npm run pack"
|
|
||||||
},
|
|
||||||
"repository": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "git+https://github.com/actions/runner.git"
|
|
||||||
},
|
|
||||||
"keywords": [
|
|
||||||
"actions"
|
|
||||||
],
|
|
||||||
"author": "GitHub Actions",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"@actions/exec": "^1.1.0",
|
|
||||||
"@actions/core": "^1.6.0"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/node": "^12.7.12",
|
|
||||||
"@typescript-eslint/parser": "^2.8.0",
|
|
||||||
"@zeit/ncc": "^0.20.5",
|
|
||||||
"eslint": "^6.8.0",
|
|
||||||
"eslint-plugin-github": "^2.0.0",
|
|
||||||
"prettier": "^1.19.1",
|
|
||||||
"typescript": "^3.6.4"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,156 +0,0 @@
|
|||||||
import * as exec from '@actions/exec'
|
|
||||||
import * as core from '@actions/core'
|
|
||||||
import * as events from 'events'
|
|
||||||
import * as readline from 'readline'
|
|
||||||
|
|
||||||
async function run(): Promise<void> {
|
|
||||||
let input = ''
|
|
||||||
|
|
||||||
const rl = readline.createInterface({
|
|
||||||
input: process.stdin
|
|
||||||
})
|
|
||||||
|
|
||||||
rl.on('line', line => {
|
|
||||||
core.debug(`Line from STDIN: ${line}`)
|
|
||||||
input = line
|
|
||||||
})
|
|
||||||
|
|
||||||
await events.once(rl, 'close')
|
|
||||||
|
|
||||||
core.debug(input)
|
|
||||||
|
|
||||||
const inputJson = JSON.parse(input)
|
|
||||||
core.debug(JSON.stringify(inputJson))
|
|
||||||
|
|
||||||
const command = inputJson.command
|
|
||||||
if (command === 'Create') {
|
|
||||||
const creationInput = inputJson.creationInput
|
|
||||||
core.debug(JSON.stringify(creationInput))
|
|
||||||
const containers = creationInput.containers
|
|
||||||
const jobContainer = containers[0]
|
|
||||||
|
|
||||||
// const networkName = 'actions_podman_network'
|
|
||||||
// // podman network create {network} -> track and return `network` for ${{job.container.network}}
|
|
||||||
// await exec.exec('podman', ['network', 'create', networkName])
|
|
||||||
|
|
||||||
const containerImage = `${jobContainer.containerImage}`
|
|
||||||
// podman pull docker.io/library/{image}
|
|
||||||
// await exec.exec('podman', ['pull', containerImage])
|
|
||||||
|
|
||||||
// kubectl run e088c842be1f46b394212618408aaba0_node1016jessie_6196c9
|
|
||||||
// --image=node:10.16-jessie
|
|
||||||
// -- tail -f /dev/null
|
|
||||||
const runArgs = ['run', 'job-container']
|
|
||||||
// runArgs.push(`--workdir=${jobContainer.containerWorkDirectory}`)
|
|
||||||
// runArgs.push(`--network=${networkName}`)
|
|
||||||
|
|
||||||
// for (const mountVolume of jobContainer.mountVolumes) {
|
|
||||||
// runArgs.push(
|
|
||||||
// `-v=${mountVolume.sourceVolumePath}:${mountVolume.targetVolumePath}`
|
|
||||||
// )
|
|
||||||
// }
|
|
||||||
runArgs.push(`--image=${containerImage}`)
|
|
||||||
runArgs.push(`--`)
|
|
||||||
runArgs.push(`tail`)
|
|
||||||
runArgs.push(`-f`)
|
|
||||||
runArgs.push(`/dev/null`)
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(runArgs))
|
|
||||||
|
|
||||||
// const containerId = await exec.getExecOutput('podman', [
|
|
||||||
// 'create',
|
|
||||||
// // `--workdir ${jobContainer.containerWorkDirectory}`,
|
|
||||||
// `--network=${networkName}`,
|
|
||||||
// // `-v=/Users/ting/Desktop/runner/_layout/_work:/__w`,
|
|
||||||
// `--entrypoint=${jobContainer.containerEntryPoint}`,
|
|
||||||
// `${containerImage}`,
|
|
||||||
// `${jobContainer.containerEntryPointArgs}`
|
|
||||||
// ])
|
|
||||||
|
|
||||||
await exec.exec('kubectl', runArgs)
|
|
||||||
|
|
||||||
// get PATH inside the container
|
|
||||||
|
|
||||||
const waitArgs = ['wait', '--for=condition=Ready', 'pod/job-container']
|
|
||||||
await exec.exec('kubectl', waitArgs)
|
|
||||||
|
|
||||||
// output containerId for ${{job.container.id}}
|
|
||||||
|
|
||||||
// copy over node.js
|
|
||||||
const cpNodeArgs = [
|
|
||||||
'cp',
|
|
||||||
'/actions-runner/externals/node12/bin',
|
|
||||||
'job-container:/__runner_util/'
|
|
||||||
]
|
|
||||||
await exec.exec('kubectl', cpNodeArgs)
|
|
||||||
|
|
||||||
// copy over innerhandler
|
|
||||||
const cpKubeInnerArgs = [
|
|
||||||
'cp',
|
|
||||||
'/actions-runner/bin/kubeInnerHandler',
|
|
||||||
'job-container:/__runner_util/kubeInnerHandler'
|
|
||||||
]
|
|
||||||
await exec.exec('kubectl', cpKubeInnerArgs)
|
|
||||||
|
|
||||||
// copy over _work
|
|
||||||
const cpWorkArgs = ['cp', '/actions-runner/_work', 'job-container:/__w/']
|
|
||||||
await exec.exec('kubectl', cpWorkArgs)
|
|
||||||
|
|
||||||
const creationOutput = {
|
|
||||||
JobContainerId: 'job-container',
|
|
||||||
Network: 'job-container'
|
|
||||||
}
|
|
||||||
|
|
||||||
const output = JSON.stringify({CreationOutput: creationOutput})
|
|
||||||
core.debug(output)
|
|
||||||
|
|
||||||
process.stderr.write(
|
|
||||||
`___CONTAINER_ENGINE_HANDLER_OUTPUT___${output}___CONTAINER_ENGINE_HANDLER_OUTPUT___`
|
|
||||||
)
|
|
||||||
} else if (command === 'Remove') {
|
|
||||||
const removeInput = inputJson.removeInput
|
|
||||||
core.debug(JSON.stringify(removeInput))
|
|
||||||
// const jobContainerId = removeInput.jobContainerId
|
|
||||||
|
|
||||||
// await exec.exec('kubectl', ['delete', 'pod', jobContainerId, '--force'])
|
|
||||||
// await exec.exec('podman', ['network', 'rm', '-f', network])
|
|
||||||
} else if (command === 'Exec') {
|
|
||||||
const execInput = inputJson.execInput
|
|
||||||
core.debug(JSON.stringify(execInput))
|
|
||||||
|
|
||||||
// podman exec -i --workdir /__w/canary/canary
|
|
||||||
// -e GITHUB_JOB -e GITHUB_REF -e GITHUB_SHA -e GITHUB_REPOSITORY
|
|
||||||
// -e GITHUB_REPOSITORY_OWNER -e GITHUB_RUN_ID -e GITHUB_RUN_NUMBER
|
|
||||||
// -e GITHUB_RETENTION_DAYS -e GITHUB_RUN_ATTEMPT -e GITHUB_ACTOR
|
|
||||||
// -e GITHUB_WORKFLOW -e GITHUB_HEAD_REF -e GITHUB_BASE_REF -e GITHUB_EVENT_NAME
|
|
||||||
// -e GITHUB_SERVER_URL -e GITHUB_API_URL -e GITHUB_GRAPHQL_URL
|
|
||||||
// -e GITHUB_WORKSPACE -e GITHUB_ACTION -e GITHUB_EVENT_PATH -e GITHUB_ACTION_REPOSITORY
|
|
||||||
// -e GITHUB_ACTION_REF -e GITHUB_PATH -e GITHUB_ENV -e RUNNER_DEBUG
|
|
||||||
// -e RUNNER_OS -e RUNNER_NAME -e RUNNER_TOOL_CACHE
|
|
||||||
// -e RUNNER_TEMP -e RUNNER_WORKSPACE
|
|
||||||
// eccdf520697a035599d6e8c8dc801f004fdd3797cdce88f590aba3669a88d9bc sh -e /__w/_temp/d3b30383-719c-4e76-a16f-8f85443352be.sh
|
|
||||||
|
|
||||||
const cpTempArgs = [
|
|
||||||
'cp',
|
|
||||||
'/actions-runner/_work/_temp',
|
|
||||||
'job-container:/__w/'
|
|
||||||
]
|
|
||||||
await exec.exec('kubectl', cpTempArgs)
|
|
||||||
|
|
||||||
const execArgs = ['exec']
|
|
||||||
execArgs.push(execInput.jobContainer.containerId)
|
|
||||||
execArgs.push('-i')
|
|
||||||
execArgs.push('-t')
|
|
||||||
execArgs.push('--')
|
|
||||||
execArgs.push('/__runner_util/node')
|
|
||||||
execArgs.push('/__runner_util/kubeInnerHandler')
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(execArgs))
|
|
||||||
|
|
||||||
await exec.exec('kubectl', execArgs, {
|
|
||||||
input: Buffer.from(JSON.stringify(execInput))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
run()
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"target": "es6", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019' or 'ESNEXT'. */
|
|
||||||
"module": "commonjs", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */
|
|
||||||
"outDir": "./lib", /* Redirect output structure to the directory. */
|
|
||||||
"rootDir": "./src", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */
|
|
||||||
"strict": true, /* Enable all strict type-checking options. */
|
|
||||||
"noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */
|
|
||||||
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */
|
|
||||||
},
|
|
||||||
"exclude": ["node_modules", "**/*.test.ts"]
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
{
|
|
||||||
"plugins": ["jest", "@typescript-eslint"],
|
|
||||||
"extends": ["plugin:github/es6"],
|
|
||||||
"parser": "@typescript-eslint/parser",
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 9,
|
|
||||||
"sourceType": "module",
|
|
||||||
"project": "./tsconfig.json"
|
|
||||||
},
|
|
||||||
"rules": {
|
|
||||||
"eslint-comments/no-use": "off",
|
|
||||||
"import/no-namespace": "off",
|
|
||||||
"no-console": "off",
|
|
||||||
"no-unused-vars": "off",
|
|
||||||
"@typescript-eslint/no-unused-vars": "error",
|
|
||||||
"@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}],
|
|
||||||
"@typescript-eslint/no-require-imports": "error",
|
|
||||||
"@typescript-eslint/array-type": "error",
|
|
||||||
"@typescript-eslint/await-thenable": "error",
|
|
||||||
"@typescript-eslint/ban-ts-ignore": "error",
|
|
||||||
"camelcase": "off",
|
|
||||||
"@typescript-eslint/camelcase": "error",
|
|
||||||
"@typescript-eslint/class-name-casing": "error",
|
|
||||||
"@typescript-eslint/explicit-function-return-type": ["error", {"allowExpressions": true}],
|
|
||||||
"@typescript-eslint/func-call-spacing": ["error", "never"],
|
|
||||||
"@typescript-eslint/generic-type-naming": ["error", "^[A-Z][A-Za-z]*$"],
|
|
||||||
"@typescript-eslint/no-array-constructor": "error",
|
|
||||||
"@typescript-eslint/no-empty-interface": "error",
|
|
||||||
"@typescript-eslint/no-explicit-any": "error",
|
|
||||||
"@typescript-eslint/no-extraneous-class": "error",
|
|
||||||
"@typescript-eslint/no-for-in-array": "error",
|
|
||||||
"@typescript-eslint/no-inferrable-types": "error",
|
|
||||||
"@typescript-eslint/no-misused-new": "error",
|
|
||||||
"@typescript-eslint/no-namespace": "error",
|
|
||||||
"@typescript-eslint/no-non-null-assertion": "warn",
|
|
||||||
"@typescript-eslint/no-object-literal-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-qualifier": "error",
|
|
||||||
"@typescript-eslint/no-unnecessary-type-assertion": "error",
|
|
||||||
"@typescript-eslint/no-useless-constructor": "error",
|
|
||||||
"@typescript-eslint/no-var-requires": "error",
|
|
||||||
"@typescript-eslint/prefer-for-of": "warn",
|
|
||||||
"@typescript-eslint/prefer-function-type": "warn",
|
|
||||||
"@typescript-eslint/prefer-includes": "error",
|
|
||||||
"@typescript-eslint/prefer-interface": "error",
|
|
||||||
"@typescript-eslint/prefer-string-starts-ends-with": "error",
|
|
||||||
"@typescript-eslint/promise-function-async": "error",
|
|
||||||
"@typescript-eslint/require-array-sort-compare": "error",
|
|
||||||
"@typescript-eslint/restrict-plus-operands": "error",
|
|
||||||
"semi": "off",
|
|
||||||
"@typescript-eslint/semi": ["error", "never"],
|
|
||||||
"@typescript-eslint/type-annotation-spacing": "error",
|
|
||||||
"@typescript-eslint/unbound-method": "error"
|
|
||||||
},
|
|
||||||
"env": {
|
|
||||||
"node": true,
|
|
||||||
"es6": true,
|
|
||||||
"jest/globals": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
dist/
|
|
||||||
lib/
|
|
||||||
node_modules/
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"printWidth": 80,
|
|
||||||
"tabWidth": 2,
|
|
||||||
"useTabs": false,
|
|
||||||
"semi": false,
|
|
||||||
"singleQuote": true,
|
|
||||||
"trailingComma": "none",
|
|
||||||
"bracketSpacing": false,
|
|
||||||
"arrowParens": "avoid",
|
|
||||||
"parser": "typescript"
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
To update podmanHandler under `Misc/layoutbin` run `npm install && npm run all`
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,36 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "podmanHandler",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "GitHub Actions",
|
|
||||||
"main": "lib/podmanHandler.js",
|
|
||||||
"scripts": {
|
|
||||||
"build": "tsc",
|
|
||||||
"format": "prettier --write **/*.ts",
|
|
||||||
"format-check": "prettier --check **/*.ts",
|
|
||||||
"lint": "eslint src/**/*.ts",
|
|
||||||
"pack": "ncc build -o ../../layoutbin/podmanHandler",
|
|
||||||
"all": "npm run build && npm run format && npm run lint && npm run pack"
|
|
||||||
},
|
|
||||||
"repository": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "git+https://github.com/actions/runner.git"
|
|
||||||
},
|
|
||||||
"keywords": [
|
|
||||||
"actions"
|
|
||||||
],
|
|
||||||
"author": "GitHub Actions",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"@actions/exec": "^1.1.0",
|
|
||||||
"@actions/core": "^1.6.0"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/node": "^12.7.12",
|
|
||||||
"@typescript-eslint/parser": "^2.8.0",
|
|
||||||
"@zeit/ncc": "^0.20.5",
|
|
||||||
"eslint": "^6.8.0",
|
|
||||||
"eslint-plugin-github": "^2.0.0",
|
|
||||||
"prettier": "^1.19.1",
|
|
||||||
"typescript": "^3.6.4"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,150 +0,0 @@
|
|||||||
import * as exec from '@actions/exec'
|
|
||||||
import * as core from '@actions/core'
|
|
||||||
import * as events from 'events'
|
|
||||||
import * as readline from 'readline'
|
|
||||||
|
|
||||||
async function run(): Promise<void> {
|
|
||||||
let input = ''
|
|
||||||
|
|
||||||
const rl = readline.createInterface({
|
|
||||||
input: process.stdin
|
|
||||||
})
|
|
||||||
|
|
||||||
rl.on('line', line => {
|
|
||||||
core.debug(`Line from STDIN: ${line}`)
|
|
||||||
input = line
|
|
||||||
})
|
|
||||||
|
|
||||||
await events.once(rl, 'close')
|
|
||||||
|
|
||||||
core.debug(input)
|
|
||||||
|
|
||||||
const inputJson = JSON.parse(input)
|
|
||||||
core.debug(JSON.stringify(inputJson))
|
|
||||||
|
|
||||||
const command = inputJson.command
|
|
||||||
if (command === 'Create') {
|
|
||||||
const creationInput = inputJson.creationInput
|
|
||||||
core.debug(JSON.stringify(creationInput))
|
|
||||||
const containers = creationInput.containers
|
|
||||||
const jobContainer = containers[0]
|
|
||||||
|
|
||||||
const networkName = 'actions_podman_network'
|
|
||||||
// podman network create {network} -> track and return `network` for ${{job.container.network}}
|
|
||||||
await exec.exec('podman', ['network', 'create', networkName])
|
|
||||||
|
|
||||||
const containerImage = `docker.io/library/${jobContainer.containerImage}`
|
|
||||||
// podman pull docker.io/library/{image}
|
|
||||||
await exec.exec('podman', ['pull', containerImage])
|
|
||||||
|
|
||||||
// podman create --name e088c842be1f46b394212618408aaba0_node1016jessie_6196c9
|
|
||||||
// --label fa4e14
|
|
||||||
// --workdir /__w/canary/canary
|
|
||||||
// --network github_network_f98a6e1e96e74d919d814c165641cba3
|
|
||||||
// -e "HOME=/github/home" -e GITHUB_ACTIONS=true -e CI=true
|
|
||||||
// -v "/var/run/docker.sock":"/var/run/docker.sock"
|
|
||||||
// -v "/home/runner/work":"/__w"
|
|
||||||
// -v "/home/runner/runners/2.283.2/externals":"/__e":ro
|
|
||||||
// -v "/home/runner/work/_temp":"/__w/_temp"
|
|
||||||
// -v "/home/runner/work/_actions":"/__w/_actions"
|
|
||||||
// -v "/opt/hostedtoolcache":"/__t"
|
|
||||||
// -v "/home/runner/work/_temp/_github_home":"/github/home"
|
|
||||||
// -v "/home/runner/work/_temp/_github_workflow":"/github/workflow"
|
|
||||||
// --entrypoint "tail" node:10.16-jessie "-f" "/dev/null"
|
|
||||||
const creatArgs = ['create']
|
|
||||||
creatArgs.push(`--workdir=${jobContainer.containerWorkDirectory}`)
|
|
||||||
creatArgs.push(`--network=${networkName}`)
|
|
||||||
|
|
||||||
for (const mountVolume of jobContainer.mountVolumes) {
|
|
||||||
creatArgs.push(
|
|
||||||
`-v=${mountVolume.sourceVolumePath}:${mountVolume.targetVolumePath}`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
creatArgs.push(`--entrypoint=tail`)
|
|
||||||
creatArgs.push(containerImage)
|
|
||||||
creatArgs.push(`-f`)
|
|
||||||
creatArgs.push(`/dev/null`)
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(creatArgs))
|
|
||||||
|
|
||||||
// const containerId = await exec.getExecOutput('podman', [
|
|
||||||
// 'create',
|
|
||||||
// // `--workdir ${jobContainer.containerWorkDirectory}`,
|
|
||||||
// `--network=${networkName}`,
|
|
||||||
// // `-v=/Users/ting/Desktop/runner/_layout/_work:/__w`,
|
|
||||||
// `--entrypoint=${jobContainer.containerEntryPoint}`,
|
|
||||||
// `${containerImage}`,
|
|
||||||
// `${jobContainer.containerEntryPointArgs}`
|
|
||||||
// ])
|
|
||||||
|
|
||||||
const containerId = await exec.getExecOutput('podman', creatArgs)
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(containerId))
|
|
||||||
|
|
||||||
// podman start {containerId}
|
|
||||||
await exec.exec('podman', ['start', containerId.stdout.trim()])
|
|
||||||
|
|
||||||
// get PATH inside the container
|
|
||||||
|
|
||||||
// output containerId for ${{job.container.id}}
|
|
||||||
|
|
||||||
const creationOutput = {
|
|
||||||
JobContainerId: containerId.stdout.trim(),
|
|
||||||
Network: networkName
|
|
||||||
}
|
|
||||||
|
|
||||||
const output = JSON.stringify({CreationOutput: creationOutput})
|
|
||||||
core.debug(output)
|
|
||||||
|
|
||||||
process.stderr.write(
|
|
||||||
`___CONTAINER_ENGINE_HANDLER_OUTPUT___${output}___CONTAINER_ENGINE_HANDLER_OUTPUT___`
|
|
||||||
)
|
|
||||||
} else if (command === 'Remove') {
|
|
||||||
const removeInput = inputJson.removeInput
|
|
||||||
core.debug(JSON.stringify(removeInput))
|
|
||||||
const jobContainerId = removeInput.jobContainerId
|
|
||||||
const network = removeInput.network
|
|
||||||
|
|
||||||
await exec.exec('podman', ['rm', '-f', jobContainerId])
|
|
||||||
await exec.exec('podman', ['network', 'rm', '-f', network])
|
|
||||||
} else if (command === 'Exec') {
|
|
||||||
const execInput = inputJson.execInput
|
|
||||||
core.debug(JSON.stringify(execInput))
|
|
||||||
|
|
||||||
// podman exec -i --workdir /__w/canary/canary
|
|
||||||
// -e GITHUB_JOB -e GITHUB_REF -e GITHUB_SHA -e GITHUB_REPOSITORY
|
|
||||||
// -e GITHUB_REPOSITORY_OWNER -e GITHUB_RUN_ID -e GITHUB_RUN_NUMBER
|
|
||||||
// -e GITHUB_RETENTION_DAYS -e GITHUB_RUN_ATTEMPT -e GITHUB_ACTOR
|
|
||||||
// -e GITHUB_WORKFLOW -e GITHUB_HEAD_REF -e GITHUB_BASE_REF -e GITHUB_EVENT_NAME
|
|
||||||
// -e GITHUB_SERVER_URL -e GITHUB_API_URL -e GITHUB_GRAPHQL_URL
|
|
||||||
// -e GITHUB_WORKSPACE -e GITHUB_ACTION -e GITHUB_EVENT_PATH -e GITHUB_ACTION_REPOSITORY
|
|
||||||
// -e GITHUB_ACTION_REF -e GITHUB_PATH -e GITHUB_ENV -e RUNNER_DEBUG
|
|
||||||
// -e RUNNER_OS -e RUNNER_NAME -e RUNNER_TOOL_CACHE
|
|
||||||
// -e RUNNER_TEMP -e RUNNER_WORKSPACE
|
|
||||||
// eccdf520697a035599d6e8c8dc801f004fdd3797cdce88f590aba3669a88d9bc sh -e /__w/_temp/d3b30383-719c-4e76-a16f-8f85443352be.sh
|
|
||||||
|
|
||||||
const execArgs = ['exec']
|
|
||||||
execArgs.push('-i')
|
|
||||||
execArgs.push(`--workdir=${execInput.workingDirectory}`)
|
|
||||||
for (const envKey of execInput.environmentKeys) {
|
|
||||||
execArgs.push(`-e=${envKey}`)
|
|
||||||
}
|
|
||||||
execArgs.push(execInput.jobContainer.containerId)
|
|
||||||
execArgs.push(execInput.fileName)
|
|
||||||
|
|
||||||
const args = (<string>execInput.arguments).split(' ')
|
|
||||||
core.debug(JSON.stringify(args))
|
|
||||||
|
|
||||||
execArgs.push(...args)
|
|
||||||
|
|
||||||
core.debug(JSON.stringify(execArgs))
|
|
||||||
|
|
||||||
await exec.exec('podman', execArgs)
|
|
||||||
}
|
|
||||||
|
|
||||||
await exec.exec('podman', ['network', 'ls'])
|
|
||||||
await exec.exec('podman', ['ps', '-a'])
|
|
||||||
}
|
|
||||||
|
|
||||||
run()
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"target": "es6", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019' or 'ESNEXT'. */
|
|
||||||
"module": "commonjs", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */
|
|
||||||
"outDir": "./lib", /* Redirect output structure to the directory. */
|
|
||||||
"rootDir": "./src", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */
|
|
||||||
"strict": true, /* Enable all strict type-checking options. */
|
|
||||||
"noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */
|
|
||||||
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */
|
|
||||||
},
|
|
||||||
"exclude": ["node_modules", "**/*.test.ts"]
|
|
||||||
}
|
|
||||||
@@ -25,7 +25,5 @@
|
|||||||
</dict>
|
</dict>
|
||||||
<key>ProcessType</key>
|
<key>ProcessType</key>
|
||||||
<string>Interactive</string>
|
<string>Interactive</string>
|
||||||
<key>SessionCreate</key>
|
|
||||||
<true/>
|
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,49 +0,0 @@
|
|||||||
// Job container creation
|
|
||||||
|
|
||||||
// podman network create {network} -> track and return `network` for ${{job.container.network}}
|
|
||||||
|
|
||||||
// podman pull docker.io/library/{image}
|
|
||||||
|
|
||||||
// podman create --name e088c842be1f46b394212618408aaba0_node1016jessie_6196c9
|
|
||||||
// --label fa4e14
|
|
||||||
// --workdir /__w/canary/canary
|
|
||||||
// --network github_network_f98a6e1e96e74d919d814c165641cba3
|
|
||||||
// -e "HOME=/github/home" -e GITHUB_ACTIONS=true -e CI=true
|
|
||||||
// -v "/var/run/docker.sock":"/var/run/docker.sock"
|
|
||||||
// -v "/home/runner/work":"/__w"
|
|
||||||
// -v "/home/runner/runners/2.283.2/externals":"/__e":ro
|
|
||||||
// -v "/home/runner/work/_temp":"/__w/_temp"
|
|
||||||
// -v "/home/runner/work/_actions":"/__w/_actions"
|
|
||||||
// -v "/opt/hostedtoolcache":"/__t"
|
|
||||||
// -v "/home/runner/work/_temp/_github_home":"/github/home"
|
|
||||||
// -v "/home/runner/work/_temp/_github_workflow":"/github/workflow"
|
|
||||||
// --entrypoint "tail" node:10.16-jessie "-f" "/dev/null"
|
|
||||||
|
|
||||||
// podman start {containerId}
|
|
||||||
|
|
||||||
// get PATH inside the container
|
|
||||||
|
|
||||||
// output containerId for ${{job.container.id}}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Job container stop
|
|
||||||
|
|
||||||
// podman rm --force {containerId}
|
|
||||||
|
|
||||||
// podman network rm {network}
|
|
||||||
|
|
||||||
|
|
||||||
// Run step
|
|
||||||
|
|
||||||
// podman exec -i --workdir /__w/canary/canary
|
|
||||||
// -e GITHUB_JOB -e GITHUB_REF -e GITHUB_SHA -e GITHUB_REPOSITORY
|
|
||||||
// -e GITHUB_REPOSITORY_OWNER -e GITHUB_RUN_ID -e GITHUB_RUN_NUMBER
|
|
||||||
// -e GITHUB_RETENTION_DAYS -e GITHUB_RUN_ATTEMPT -e GITHUB_ACTOR
|
|
||||||
// -e GITHUB_WORKFLOW -e GITHUB_HEAD_REF -e GITHUB_BASE_REF -e GITHUB_EVENT_NAME
|
|
||||||
// -e GITHUB_SERVER_URL -e GITHUB_API_URL -e GITHUB_GRAPHQL_URL
|
|
||||||
// -e GITHUB_WORKSPACE -e GITHUB_ACTION -e GITHUB_EVENT_PATH -e GITHUB_ACTION_REPOSITORY
|
|
||||||
// -e GITHUB_ACTION_REF -e GITHUB_PATH -e GITHUB_ENV -e RUNNER_DEBUG
|
|
||||||
// -e RUNNER_OS -e RUNNER_NAME -e RUNNER_TOOL_CACHE
|
|
||||||
// -e RUNNER_TEMP -e RUNNER_WORKSPACE
|
|
||||||
// eccdf520697a035599d6e8c8dc801f004fdd3797cdce88f590aba3669a88d9bc sh -e /__w/_temp/d3b30383-719c-4e76-a16f-8f85443352be.sh
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -18,8 +18,6 @@ downloadrunnerversion=_DOWNLOAD_RUNNER_VERSION_
|
|||||||
logfile="_UPDATE_LOG_"
|
logfile="_UPDATE_LOG_"
|
||||||
restartinteractiverunner=_RESTART_INTERACTIVE_RUNNER_
|
restartinteractiverunner=_RESTART_INTERACTIVE_RUNNER_
|
||||||
|
|
||||||
telemetryfile="$rootfolder/_diag/.telemetry"
|
|
||||||
|
|
||||||
# log user who run the script
|
# log user who run the script
|
||||||
date "+[%F %T-%4N] --------whoami--------" >> "$logfile" 2>&1
|
date "+[%F %T-%4N] --------whoami--------" >> "$logfile" 2>&1
|
||||||
whoami >> "$logfile" 2>&1
|
whoami >> "$logfile" 2>&1
|
||||||
@@ -120,56 +118,12 @@ then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# fix upgrade issue with macOS when running as a service
|
# fix upgrade issue with macOS
|
||||||
attemptedtargetedfix=0
|
|
||||||
currentplatform=$(uname | awk '{print tolower($0)}')
|
currentplatform=$(uname | awk '{print tolower($0)}')
|
||||||
if [[ "$currentplatform" == 'darwin' && restartinteractiverunner -eq 0 ]]; then
|
if [[ "$currentplatform" == 'darwin' ]]; then
|
||||||
# We needed a fix for https://github.com/actions/runner/issues/743
|
# need a short-term fix for https://github.com/actions/runner/issues/743
|
||||||
# We will recreate the ./externals/node12/bin/node of the past runner version that launched the runnerlistener service
|
# we will recreate all the ./externals/node12/bin/node of the past 5 versions
|
||||||
# Otherwise mac gatekeeper kills the processes we spawn on creation as we are running a process with no backing file
|
# v2.280.3 v2.280.2 v2.280.1 v2.279.0 v2.278.0
|
||||||
|
|
||||||
# We need the pid for the nodejs loop, get that here, its the parent of the runner C# pid
|
|
||||||
# assumption here is only one process is invoking rootfolder/runsvc.sh
|
|
||||||
procgroup=$(ps x -o pgid,command | grep "$rootfolder/runsvc.sh" | grep -v grep | awk '{print $1}')
|
|
||||||
if [[ $? -eq 0 && -n "$procgroup" ]]
|
|
||||||
then
|
|
||||||
# inspect the open file handles to find the node process
|
|
||||||
# we can't actually inspect the process using ps because it uses relative paths and doesn't follow symlinks
|
|
||||||
path=$(lsof -a -g "$procgroup" -F n | grep node12/bin/node | grep externals | tail -1 | cut -c2-)
|
|
||||||
if [[ $? -eq 0 && -n "$path" ]]
|
|
||||||
then
|
|
||||||
# trim the last 5 characters of the path '/node'
|
|
||||||
trimmedpath=$(dirname "$path")
|
|
||||||
if [[ $? -eq 0 && -n "$trimmedpath" ]]
|
|
||||||
then
|
|
||||||
attemptedtargetedfix=1
|
|
||||||
# Create the path if it does not exist
|
|
||||||
if [[ ! -e "$path" ]]
|
|
||||||
then
|
|
||||||
date "+[%F %T-%4N] Creating fallback node at path $path" >> "$logfile" 2>&1
|
|
||||||
mkdir -p "$trimmedpath"
|
|
||||||
cp "$rootfolder/externals/node12/bin/node" "$path"
|
|
||||||
else
|
|
||||||
date "+[%F %T-%4N] Path for fallback node exists, skipping creating $path" >> "$logfile" 2>&1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to trim runner path. TrimmedPath: $trimmedpath, path: $path, pgid: $procgroup, root: $rootfolder" >> "$logfile" 2>&1
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to trim runner path. TrimmedPath: $trimmedpath, path: $path, pgid: $procgroup, root: $rootfolder" >> "$telemetryfile" 2>&1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to find runner path. Path: $path, pgid: $procgroup, root: $rootfolder" >> "$logfile" 2>&1
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to find runner path. Path: $path, pgid: $procgroup, root: $rootfolder" >> "$telemetryfile" 2>&1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to find runner pgid. pgid: $procgroup, root: $rootfolder" >> "$logfile" 2>&1
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Failed to find runner pgid. pgid: $procgroup, root: $rootfolder" >> "$telemetryfile" 2>&1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ $attemptedtargetedfix -eq 0 ]
|
|
||||||
then
|
|
||||||
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Defaulting to old macOS service fix" >> "$logfile" 2>&1
|
|
||||||
date "+[%F %T-%4N] DarwinRunnerUpgrade: Defaulting to old macOS service fix" >> "$telemetryfile" 2>&1
|
|
||||||
if [[ ! -e "$rootfolder/externals.2.280.3/node12/bin/node" ]]
|
if [[ ! -e "$rootfolder/externals.2.280.3/node12/bin/node" ]]
|
||||||
then
|
then
|
||||||
mkdir -p "$rootfolder/externals.2.280.3/node12/bin"
|
mkdir -p "$rootfolder/externals.2.280.3/node12/bin"
|
||||||
@@ -188,34 +142,17 @@ if [[ "$currentplatform" == 'darwin' && restartinteractiverunner -eq 0 ]]; then
|
|||||||
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.280.1/node12/bin/node"
|
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.280.1/node12/bin/node"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# GHES 3.2
|
|
||||||
if [[ ! -e "$rootfolder/externals.2.279.0/node12/bin/node" ]]
|
if [[ ! -e "$rootfolder/externals.2.279.0/node12/bin/node" ]]
|
||||||
then
|
then
|
||||||
mkdir -p "$rootfolder/externals.2.279.0/node12/bin"
|
mkdir -p "$rootfolder/externals.2.279.0/node12/bin"
|
||||||
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.279.0/node12/bin/node"
|
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.279.0/node12/bin/node"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# GHES 3.1.2 or later
|
|
||||||
if [[ ! -e "$rootfolder/externals.2.278.0/node12/bin/node" ]]
|
if [[ ! -e "$rootfolder/externals.2.278.0/node12/bin/node" ]]
|
||||||
then
|
then
|
||||||
mkdir -p "$rootfolder/externals.2.278.0/node12/bin"
|
mkdir -p "$rootfolder/externals.2.278.0/node12/bin"
|
||||||
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.278.0/node12/bin/node"
|
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.278.0/node12/bin/node"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# GHES 3.1.0
|
|
||||||
if [[ ! -e "$rootfolder/externals.2.276.1/node12/bin/node" ]]
|
|
||||||
then
|
|
||||||
mkdir -p "$rootfolder/externals.2.276.1/node12/bin"
|
|
||||||
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.276.1/node12/bin/node"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# GHES 3.0
|
|
||||||
if [[ ! -e "$rootfolder/externals.2.273.5/node12/bin/node" ]]
|
|
||||||
then
|
|
||||||
mkdir -p "$rootfolder/externals.2.273.5/node12/bin"
|
|
||||||
cp "$rootfolder/externals/node12/bin/node" "$rootfolder/externals.2.273.5/node12/bin/node"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
date "+[%F %T-%4N] Update succeed" >> "$logfile"
|
date "+[%F %T-%4N] Update succeed" >> "$logfile"
|
||||||
|
|||||||
@@ -1,68 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
function fatal() {
|
|
||||||
echo "error: $1" >&2
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
|
|
||||||
[ -n "${GITHUB_PAT:-""}" ] || fatal "GITHUB_PAT variable must be set"
|
|
||||||
[ -n "${RUNNER_CONFIG_URL:-""}" ] || fatal "RUNNER_CONFIG_URL variable must be set"
|
|
||||||
# [ -n "${RUNNER_NAME:-""}" ] || fatal "RUNNER_NAME variable must be set"
|
|
||||||
|
|
||||||
# if [ -n "${RUNNER_NAME}" ]; then
|
|
||||||
# # Use container id to gen unique runner name if name not provide
|
|
||||||
# CONTAINER_ID=$(cat /proc/self/cgroup | head -n 1 | tr '/' '\n' | tail -1 | cut -c1-12)
|
|
||||||
# RUNNER_NAME="actions-runner-${CONTAINER_ID}"
|
|
||||||
# fi
|
|
||||||
|
|
||||||
# if the scope has a slash, it's a repo runner
|
|
||||||
# orgs_or_repos="orgs"
|
|
||||||
# if [[ "$GITHUB_RUNNER_SCOPE" == *\/* ]]; then
|
|
||||||
# orgs_or_repos="repos"
|
|
||||||
# fi
|
|
||||||
|
|
||||||
# RUNNER_REG_URL="${GITHUB_SERVER_URL:=https://github.com}/${GITHUB_RUNNER_SCOPE}"
|
|
||||||
|
|
||||||
# echo "Runner Name : ${RUNNER_NAME}"
|
|
||||||
echo "Registration URL : ${RUNNER_CONFIG_URL}"
|
|
||||||
# echo "GitHub API URL : ${GITHUB_API_URL:=https://api.github.com}"
|
|
||||||
# echo "Runner Labels : ${RUNNER_LABELS:=""}"
|
|
||||||
|
|
||||||
# TODO: if api url is not default, validate it ends in /api/v3
|
|
||||||
|
|
||||||
# RUNNER_LABELS_ARG=""
|
|
||||||
# if [ -n "${RUNNER_LABELS}" ]; then
|
|
||||||
# RUNNER_LABELS_ARG="--labels ${RUNNER_LABELS}"
|
|
||||||
# fi
|
|
||||||
|
|
||||||
# RUNNER_GROUP_ARG=""
|
|
||||||
# if [ -n "${RUNNER_GROUP}" ]; then
|
|
||||||
# RUNNER_GROUP_ARG="--runnergroup ${RUNNER_GROUP}"
|
|
||||||
# fi
|
|
||||||
|
|
||||||
# if [ -n "${K8S_HOST_IP}" ]; then
|
|
||||||
# export http_proxy=http://$K8S_HOST_IP:9090
|
|
||||||
# fi
|
|
||||||
|
|
||||||
# curl -v -s -X POST ${GITHUB_API_URL}/${orgs_or_repos}/${GITHUB_RUNNER_SCOPE}/actions/runners/registration-token -H "authorization: token $GITHUB_PAT" -H "accept: application/vnd.github.everest-preview+json"
|
|
||||||
|
|
||||||
# Generate registration token
|
|
||||||
# RUNNER_REG_TOKEN=$(curl -s -X POST ${GITHUB_API_URL}/${orgs_or_repos}/${GITHUB_RUNNER_SCOPE}/actions/runners/registration-token -H "authorization: token $GITHUB_PAT" -H "accept: application/vnd.github.everest-preview+json" | jq -r '.token')
|
|
||||||
|
|
||||||
# Create the runner and configure it
|
|
||||||
./config.sh --unattended --url $RUNNER_CONFIG_URL --pat $GITHUB_PAT --replace --ephemeral
|
|
||||||
|
|
||||||
# while (! docker version ); do
|
|
||||||
# # Docker takes a few seconds to initialize
|
|
||||||
# echo "Waiting for Docker to launch..."
|
|
||||||
# sleep 1
|
|
||||||
# done
|
|
||||||
|
|
||||||
# unset env
|
|
||||||
unset RUNNER_CONFIG_URL
|
|
||||||
unset GITHUB_PAT
|
|
||||||
|
|
||||||
# Run it
|
|
||||||
./run.sh
|
|
||||||
@@ -43,21 +43,6 @@ else
|
|||||||
else
|
else
|
||||||
sleep 5
|
sleep 5
|
||||||
fi
|
fi
|
||||||
elif [[ $returnCode == 4 ]]; then
|
|
||||||
if [ ! -x "$(command -v sleep)" ]; then
|
|
||||||
if [ ! -x "$(command -v ping)" ]; then
|
|
||||||
COUNT="0"
|
|
||||||
while [[ $COUNT != 5000 ]]; do
|
|
||||||
echo "SLEEP" > /dev/null
|
|
||||||
COUNT=$[$COUNT+1]
|
|
||||||
done
|
|
||||||
else
|
|
||||||
ping -c 5 127.0.0.1 > /dev/null
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
sleep 5
|
|
||||||
fi
|
|
||||||
"$DIR"/bin/Runner.Listener run $*
|
|
||||||
else
|
else
|
||||||
exit $returnCode
|
exit $returnCode
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -26,7 +26,6 @@ namespace GitHub.Runner.Common
|
|||||||
Certificates,
|
Certificates,
|
||||||
Options,
|
Options,
|
||||||
SetupInfo,
|
SetupInfo,
|
||||||
Telemetry
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class Constants
|
public static class Constants
|
||||||
@@ -129,7 +128,7 @@ namespace GitHub.Runner.Common
|
|||||||
public static readonly string Ephemeral = "ephemeral";
|
public static readonly string Ephemeral = "ephemeral";
|
||||||
public static readonly string Help = "help";
|
public static readonly string Help = "help";
|
||||||
public static readonly string Replace = "replace";
|
public static readonly string Replace = "replace";
|
||||||
public static readonly string Once = "once"; // Keep this around since customers still relies on it
|
public static readonly string Once = "once"; // TODO: Remove in 10/2021
|
||||||
public static readonly string RunAsService = "runasservice";
|
public static readonly string RunAsService = "runasservice";
|
||||||
public static readonly string Unattended = "unattended";
|
public static readonly string Unattended = "unattended";
|
||||||
public static readonly string Version = "version";
|
public static readonly string Version = "version";
|
||||||
@@ -155,7 +154,6 @@ namespace GitHub.Runner.Common
|
|||||||
public static readonly string LowDiskSpace = "LOW_DISK_SPACE";
|
public static readonly string LowDiskSpace = "LOW_DISK_SPACE";
|
||||||
public static readonly string UnsupportedCommand = "UNSUPPORTED_COMMAND";
|
public static readonly string UnsupportedCommand = "UNSUPPORTED_COMMAND";
|
||||||
public static readonly string UnsupportedCommandMessageDisabled = "The `{0}` command is disabled. Please upgrade to using Environment Files or opt into unsecure command execution by setting the `ACTIONS_ALLOW_UNSECURE_COMMANDS` environment variable to `true`. For more information see: https://github.blog/changelog/2020-10-01-github-actions-deprecating-set-env-and-add-path-commands/";
|
public static readonly string UnsupportedCommandMessageDisabled = "The `{0}` command is disabled. Please upgrade to using Environment Files or opt into unsecure command execution by setting the `ACTIONS_ALLOW_UNSECURE_COMMANDS` environment variable to `true`. For more information see: https://github.blog/changelog/2020-10-01-github-actions-deprecating-set-env-and-add-path-commands/";
|
||||||
public static readonly string UnsupportedStopCommandTokenDisabled = "You cannot use a endToken that is an empty string, the string 'pause-logging', or another workflow command. For more information see: https://docs.github.com/en/actions/learn-github-actions/workflow-commands-for-github-actions#example-stopping-and-starting-workflow-commands or opt into insecure command execution by setting the `ACTIONS_ALLOW_UNSECURE_STOPCOMMAND_TOKENS` environment variable to `true`.";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class RunnerEvent
|
public static class RunnerEvent
|
||||||
@@ -215,7 +213,6 @@ namespace GitHub.Runner.Common
|
|||||||
// Keep alphabetical
|
// Keep alphabetical
|
||||||
//
|
//
|
||||||
public static readonly string AllowUnsupportedCommands = "ACTIONS_ALLOW_UNSECURE_COMMANDS";
|
public static readonly string AllowUnsupportedCommands = "ACTIONS_ALLOW_UNSECURE_COMMANDS";
|
||||||
public static readonly string AllowUnsupportedStopCommandTokens = "ACTIONS_ALLOW_UNSECURE_STOPCOMMAND_TOKENS";
|
|
||||||
public static readonly string RunnerDebug = "ACTIONS_RUNNER_DEBUG";
|
public static readonly string RunnerDebug = "ACTIONS_RUNNER_DEBUG";
|
||||||
public static readonly string StepDebug = "ACTIONS_STEP_DEBUG";
|
public static readonly string StepDebug = "ACTIONS_STEP_DEBUG";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -343,12 +343,6 @@ namespace GitHub.Runner.Common
|
|||||||
".setup_info");
|
".setup_info");
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WellKnownConfigFile.Telemetry:
|
|
||||||
path = Path.Combine(
|
|
||||||
GetDirectory(WellKnownDirectory.Diag),
|
|
||||||
".telemetry");
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
throw new NotSupportedException($"Unexpected well known config file: '{configFile}'");
|
throw new NotSupportedException($"Unexpected well known config file: '{configFile}'");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,11 +2,8 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Net.Http;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using GitHub.Runner.Sdk;
|
|
||||||
using GitHub.Services.Common;
|
|
||||||
using GitHub.Services.WebApi;
|
using GitHub.Services.WebApi;
|
||||||
|
|
||||||
namespace GitHub.Runner.Common
|
namespace GitHub.Runner.Common
|
||||||
@@ -39,9 +36,6 @@ namespace GitHub.Runner.Common
|
|||||||
{
|
{
|
||||||
_connection = jobConnection;
|
_connection = jobConnection;
|
||||||
int attemptCount = 5;
|
int attemptCount = 5;
|
||||||
var configurationStore = HostContext.GetService<IConfigurationStore>();
|
|
||||||
var runnerSettings = configurationStore.GetSettings();
|
|
||||||
|
|
||||||
while (!_connection.HasAuthenticated && attemptCount-- > 0)
|
while (!_connection.HasAuthenticated && attemptCount-- > 0)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -51,13 +45,8 @@ namespace GitHub.Runner.Common
|
|||||||
}
|
}
|
||||||
catch (Exception ex) when (attemptCount > 0)
|
catch (Exception ex) when (attemptCount > 0)
|
||||||
{
|
{
|
||||||
Trace.Info($"Catch exception during connect. {attemptCount} attempts left.");
|
Trace.Info($"Catch exception during connect. {attemptCount} attemp left.");
|
||||||
Trace.Error(ex);
|
Trace.Error(ex);
|
||||||
|
|
||||||
if (runnerSettings.IsHostedServer)
|
|
||||||
{
|
|
||||||
await CheckNetworkEndpointsAsync();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await Task.Delay(100);
|
await Task.Delay(100);
|
||||||
@@ -67,52 +56,6 @@ namespace GitHub.Runner.Common
|
|||||||
_hasConnection = true;
|
_hasConnection = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task CheckNetworkEndpointsAsync()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Trace.Info("Requesting Actions Service health endpoint status");
|
|
||||||
using (var httpClientHandler = HostContext.CreateHttpClientHandler())
|
|
||||||
using (var actionsClient = new HttpClient(httpClientHandler))
|
|
||||||
{
|
|
||||||
var baseUri = new Uri(_connection.Uri.GetLeftPart(UriPartial.Authority));
|
|
||||||
|
|
||||||
actionsClient.DefaultRequestHeaders.UserAgent.AddRange(HostContext.UserAgents);
|
|
||||||
|
|
||||||
// Call the _apis/health endpoint
|
|
||||||
var response = await actionsClient.GetAsync(new Uri(baseUri, "_apis/health"));
|
|
||||||
Trace.Info($"Actions health status code: {response.StatusCode}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
// Log error, but continue as this call is best-effort
|
|
||||||
Trace.Info($"Actions Service health endpoint failed due to {ex.GetType().Name}");
|
|
||||||
Trace.Error(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Trace.Info("Requesting Github API endpoint status");
|
|
||||||
// This is a dotcom public API... just call it directly
|
|
||||||
using (var httpClientHandler = HostContext.CreateHttpClientHandler())
|
|
||||||
using (var gitHubClient = new HttpClient(httpClientHandler))
|
|
||||||
{
|
|
||||||
gitHubClient.DefaultRequestHeaders.UserAgent.AddRange(HostContext.UserAgents);
|
|
||||||
|
|
||||||
// Call the api.github.com endpoint
|
|
||||||
var response = await gitHubClient.GetAsync("https://api.github.com");
|
|
||||||
Trace.Info($"api.github.com status code: {response.StatusCode}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
// Log error, but continue as this call is best-effort
|
|
||||||
Trace.Info($"Github API endpoint failed due to {ex.GetType().Name}");
|
|
||||||
Trace.Error(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void CheckConnection()
|
private void CheckConnection()
|
||||||
{
|
{
|
||||||
if (!_hasConnection)
|
if (!_hasConnection)
|
||||||
|
|||||||
@@ -29,10 +29,8 @@ namespace GitHub.Runner.Common
|
|||||||
// Configuration
|
// Configuration
|
||||||
Task<TaskAgent> AddAgentAsync(Int32 agentPoolId, TaskAgent agent);
|
Task<TaskAgent> AddAgentAsync(Int32 agentPoolId, TaskAgent agent);
|
||||||
Task DeleteAgentAsync(int agentPoolId, int agentId);
|
Task DeleteAgentAsync(int agentPoolId, int agentId);
|
||||||
Task DeleteAgentAsync(int agentId);
|
|
||||||
Task<List<TaskAgentPool>> GetAgentPoolsAsync(string agentPoolName = null, TaskAgentPoolType poolType = TaskAgentPoolType.Automation);
|
Task<List<TaskAgentPool>> GetAgentPoolsAsync(string agentPoolName = null, TaskAgentPoolType poolType = TaskAgentPoolType.Automation);
|
||||||
Task<List<TaskAgent>> GetAgentsAsync(int agentPoolId, string agentName = null);
|
Task<List<TaskAgent>> GetAgentsAsync(int agentPoolId, string agentName = null);
|
||||||
Task<List<TaskAgent>> GetAgentsAsync(string agentName);
|
|
||||||
Task<TaskAgent> ReplaceAgentAsync(int agentPoolId, TaskAgent agent);
|
Task<TaskAgent> ReplaceAgentAsync(int agentPoolId, TaskAgent agent);
|
||||||
|
|
||||||
// messagequeue
|
// messagequeue
|
||||||
@@ -254,11 +252,6 @@ namespace GitHub.Runner.Common
|
|||||||
return _genericTaskAgentClient.GetAgentsAsync(agentPoolId, agentName, false);
|
return _genericTaskAgentClient.GetAgentsAsync(agentPoolId, agentName, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task<List<TaskAgent>> GetAgentsAsync(string agentName)
|
|
||||||
{
|
|
||||||
return GetAgentsAsync(0, agentName); // search in all all agentPools
|
|
||||||
}
|
|
||||||
|
|
||||||
public Task<TaskAgent> ReplaceAgentAsync(int agentPoolId, TaskAgent agent)
|
public Task<TaskAgent> ReplaceAgentAsync(int agentPoolId, TaskAgent agent)
|
||||||
{
|
{
|
||||||
CheckConnection(RunnerConnectionType.Generic);
|
CheckConnection(RunnerConnectionType.Generic);
|
||||||
@@ -271,11 +264,6 @@ namespace GitHub.Runner.Common
|
|||||||
return _genericTaskAgentClient.DeleteAgentAsync(agentPoolId, agentId);
|
return _genericTaskAgentClient.DeleteAgentAsync(agentPoolId, agentId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task DeleteAgentAsync(int agentId)
|
|
||||||
{
|
|
||||||
return DeleteAgentAsync(0, agentId); // agentPool is ignored server side
|
|
||||||
}
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------
|
//-----------------------------------------------------------------
|
||||||
// MessageQueue
|
// MessageQueue
|
||||||
//-----------------------------------------------------------------
|
//-----------------------------------------------------------------
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ namespace GitHub.Runner.Listener
|
|||||||
Constants.Runner.CommandLine.Flags.Commit,
|
Constants.Runner.CommandLine.Flags.Commit,
|
||||||
Constants.Runner.CommandLine.Flags.Ephemeral,
|
Constants.Runner.CommandLine.Flags.Ephemeral,
|
||||||
Constants.Runner.CommandLine.Flags.Help,
|
Constants.Runner.CommandLine.Flags.Help,
|
||||||
Constants.Runner.CommandLine.Flags.Once,
|
|
||||||
Constants.Runner.CommandLine.Flags.Replace,
|
Constants.Runner.CommandLine.Flags.Replace,
|
||||||
Constants.Runner.CommandLine.Flags.RunAsService,
|
Constants.Runner.CommandLine.Flags.RunAsService,
|
||||||
Constants.Runner.CommandLine.Flags.Unattended,
|
Constants.Runner.CommandLine.Flags.Unattended,
|
||||||
@@ -69,7 +68,7 @@ namespace GitHub.Runner.Listener
|
|||||||
public bool Version => TestFlag(Constants.Runner.CommandLine.Flags.Version);
|
public bool Version => TestFlag(Constants.Runner.CommandLine.Flags.Version);
|
||||||
public bool Ephemeral => TestFlag(Constants.Runner.CommandLine.Flags.Ephemeral);
|
public bool Ephemeral => TestFlag(Constants.Runner.CommandLine.Flags.Ephemeral);
|
||||||
|
|
||||||
// Keep this around since customers still relies on it
|
// TODO: Remove in 10/2021
|
||||||
public bool RunOnce => TestFlag(Constants.Runner.CommandLine.Flags.Once);
|
public bool RunOnce => TestFlag(Constants.Runner.CommandLine.Flags.Once);
|
||||||
|
|
||||||
// Constructor.
|
// Constructor.
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ namespace GitHub.Runner.Listener.Configuration
|
|||||||
bool IsConfigured();
|
bool IsConfigured();
|
||||||
Task ConfigureAsync(CommandSettings command);
|
Task ConfigureAsync(CommandSettings command);
|
||||||
Task UnconfigureAsync(CommandSettings command);
|
Task UnconfigureAsync(CommandSettings command);
|
||||||
void DeleteLocalRunnerConfig();
|
|
||||||
RunnerSettings LoadSettings();
|
RunnerSettings LoadSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -330,38 +329,6 @@ namespace GitHub.Runner.Listener.Configuration
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete .runner and .credentials files
|
|
||||||
public void DeleteLocalRunnerConfig()
|
|
||||||
{
|
|
||||||
bool isConfigured = _store.IsConfigured();
|
|
||||||
bool hasCredentials = _store.HasCredentials();
|
|
||||||
//delete credential config files
|
|
||||||
var currentAction = "Removing .credentials";
|
|
||||||
if (hasCredentials)
|
|
||||||
{
|
|
||||||
_store.DeleteCredential();
|
|
||||||
var keyManager = HostContext.GetService<IRSAKeyManager>();
|
|
||||||
keyManager.DeleteKey();
|
|
||||||
_term.WriteSuccessMessage("Removed .credentials");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_term.WriteLine("Does not exist. Skipping " + currentAction);
|
|
||||||
}
|
|
||||||
|
|
||||||
//delete settings config file
|
|
||||||
currentAction = "Removing .runner";
|
|
||||||
if (isConfigured)
|
|
||||||
{
|
|
||||||
_store.DeleteSettings();
|
|
||||||
_term.WriteSuccessMessage("Removed .runner");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_term.WriteLine("Does not exist. Skipping " + currentAction);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task UnconfigureAsync(CommandSettings command)
|
public async Task UnconfigureAsync(CommandSettings command)
|
||||||
{
|
{
|
||||||
string currentAction = string.Empty;
|
string currentAction = string.Empty;
|
||||||
@@ -415,7 +382,7 @@ namespace GitHub.Runner.Listener.Configuration
|
|||||||
// Determine the service deployment type based on connection data. (Hosted/OnPremises)
|
// Determine the service deployment type based on connection data. (Hosted/OnPremises)
|
||||||
await _runnerServer.ConnectAsync(new Uri(settings.ServerUrl), creds);
|
await _runnerServer.ConnectAsync(new Uri(settings.ServerUrl), creds);
|
||||||
|
|
||||||
var agents = await _runnerServer.GetAgentsAsync(settings.AgentName);
|
var agents = await _runnerServer.GetAgentsAsync(settings.PoolId, settings.AgentName);
|
||||||
Trace.Verbose("Returns {0} agents", agents.Count);
|
Trace.Verbose("Returns {0} agents", agents.Count);
|
||||||
TaskAgent agent = agents.FirstOrDefault();
|
TaskAgent agent = agents.FirstOrDefault();
|
||||||
if (agent == null)
|
if (agent == null)
|
||||||
@@ -424,7 +391,7 @@ namespace GitHub.Runner.Listener.Configuration
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
await _runnerServer.DeleteAgentAsync(settings.AgentId);
|
await _runnerServer.DeleteAgentAsync(settings.PoolId, settings.AgentId);
|
||||||
|
|
||||||
_term.WriteLine();
|
_term.WriteLine();
|
||||||
_term.WriteSuccessMessage("Runner removed successfully");
|
_term.WriteSuccessMessage("Runner removed successfully");
|
||||||
@@ -435,7 +402,31 @@ namespace GitHub.Runner.Listener.Configuration
|
|||||||
_term.WriteLine("Cannot connect to server, because config files are missing. Skipping removing runner from the server.");
|
_term.WriteLine("Cannot connect to server, because config files are missing. Skipping removing runner from the server.");
|
||||||
}
|
}
|
||||||
|
|
||||||
DeleteLocalRunnerConfig();
|
//delete credential config files
|
||||||
|
currentAction = "Removing .credentials";
|
||||||
|
if (hasCredentials)
|
||||||
|
{
|
||||||
|
_store.DeleteCredential();
|
||||||
|
var keyManager = HostContext.GetService<IRSAKeyManager>();
|
||||||
|
keyManager.DeleteKey();
|
||||||
|
_term.WriteSuccessMessage("Removed .credentials");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_term.WriteLine("Does not exist. Skipping " + currentAction);
|
||||||
|
}
|
||||||
|
|
||||||
|
//delete settings config file
|
||||||
|
currentAction = "Removing .runner";
|
||||||
|
if (isConfigured)
|
||||||
|
{
|
||||||
|
_store.DeleteSettings();
|
||||||
|
_term.WriteSuccessMessage("Removed .runner");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_term.WriteLine("Does not exist. Skipping " + currentAction);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception)
|
catch (Exception)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -36,10 +36,7 @@ namespace GitHub.Runner.Listener
|
|||||||
{
|
{
|
||||||
private readonly Lazy<Dictionary<long, TaskResult>> _localRunJobResult = new Lazy<Dictionary<long, TaskResult>>();
|
private readonly Lazy<Dictionary<long, TaskResult>> _localRunJobResult = new Lazy<Dictionary<long, TaskResult>>();
|
||||||
private int _poolId;
|
private int _poolId;
|
||||||
|
RunnerSettings _runnerSetting;
|
||||||
IConfigurationStore _configurationStore;
|
|
||||||
|
|
||||||
RunnerSettings _runnerSettings;
|
|
||||||
private static readonly string _workerProcessName = $"Runner.Worker{IOUtil.ExeExtension}";
|
private static readonly string _workerProcessName = $"Runner.Worker{IOUtil.ExeExtension}";
|
||||||
|
|
||||||
// this is not thread-safe
|
// this is not thread-safe
|
||||||
@@ -57,9 +54,9 @@ namespace GitHub.Runner.Listener
|
|||||||
base.Initialize(hostContext);
|
base.Initialize(hostContext);
|
||||||
|
|
||||||
// get pool id from config
|
// get pool id from config
|
||||||
_configurationStore = hostContext.GetService<IConfigurationStore>();
|
var configurationStore = hostContext.GetService<IConfigurationStore>();
|
||||||
_runnerSettings = _configurationStore.GetSettings();
|
_runnerSetting = configurationStore.GetSettings();
|
||||||
_poolId = _runnerSettings.PoolId;
|
_poolId = _runnerSetting.PoolId;
|
||||||
|
|
||||||
int channelTimeoutSeconds;
|
int channelTimeoutSeconds;
|
||||||
if (!int.TryParse(Environment.GetEnvironmentVariable("GITHUB_ACTIONS_RUNNER_CHANNEL_TIMEOUT") ?? string.Empty, out channelTimeoutSeconds))
|
if (!int.TryParse(Environment.GetEnvironmentVariable("GITHUB_ACTIONS_RUNNER_CHANNEL_TIMEOUT") ?? string.Empty, out channelTimeoutSeconds))
|
||||||
@@ -664,15 +661,13 @@ namespace GitHub.Runner.Listener
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
request = await runnerServer.RenewAgentRequestAsync(poolId, requestId, lockToken, orchestrationId, token);
|
request = await runnerServer.RenewAgentRequestAsync(poolId, requestId, lockToken, orchestrationId, token);
|
||||||
|
|
||||||
Trace.Info($"Successfully renew job request {requestId}, job is valid till {request.LockedUntil.Value}");
|
Trace.Info($"Successfully renew job request {requestId}, job is valid till {request.LockedUntil.Value}");
|
||||||
|
|
||||||
if (!firstJobRequestRenewed.Task.IsCompleted)
|
if (!firstJobRequestRenewed.Task.IsCompleted)
|
||||||
{
|
{
|
||||||
// fire first renew succeed event.
|
// fire first renew succeed event.
|
||||||
firstJobRequestRenewed.TrySetResult(0);
|
firstJobRequestRenewed.TrySetResult(0);
|
||||||
|
|
||||||
// Update settings if the runner name has been changed server-side
|
|
||||||
UpdateAgentNameIfNeeded(request.ReservedAgent?.Name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (encounteringError > 0)
|
if (encounteringError > 0)
|
||||||
@@ -772,27 +767,6 @@ namespace GitHub.Runner.Listener
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UpdateAgentNameIfNeeded(string agentName)
|
|
||||||
{
|
|
||||||
var isNewAgentName = !string.Equals(_runnerSettings.AgentName, agentName, StringComparison.Ordinal);
|
|
||||||
if (!isNewAgentName || string.IsNullOrEmpty(agentName))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_runnerSettings.AgentName = agentName;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_configurationStore.SaveSettings(_runnerSettings);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Trace.Error("Cannot update the settings file:");
|
|
||||||
Trace.Error(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Best effort upload any logs for this job.
|
// Best effort upload any logs for this job.
|
||||||
private async Task TryUploadUnfinishedLogs(Pipelines.AgentJobRequestMessage message)
|
private async Task TryUploadUnfinishedLogs(Pipelines.AgentJobRequestMessage message)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -95,7 +95,16 @@ namespace GitHub.Runner.Listener
|
|||||||
var unknownCommandlines = command.Validate();
|
var unknownCommandlines = command.Validate();
|
||||||
if (unknownCommandlines.Count > 0)
|
if (unknownCommandlines.Count > 0)
|
||||||
{
|
{
|
||||||
terminal.WriteError($"Unrecognized command-line input arguments: '{string.Join(", ", unknownCommandlines)}'. For usage refer to: .\\config.cmd --help or ./config.sh --help");
|
string separator;
|
||||||
|
string ext;
|
||||||
|
#if OS_WINDOWS
|
||||||
|
separator = "\\";
|
||||||
|
ext = "cmd";
|
||||||
|
#else
|
||||||
|
separator = "/";
|
||||||
|
ext = "sh";
|
||||||
|
#endif
|
||||||
|
terminal.WriteError($"Unrecognized command-line input arguments: '{string.Join(", ", unknownCommandlines)}'. For usage refer to `.{separator}config.{ext} --help`");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Defer to the Runner class to execute the command.
|
// Defer to the Runner class to execute the command.
|
||||||
|
|||||||
@@ -233,14 +233,8 @@ namespace GitHub.Runner.Listener
|
|||||||
Trace.Info($"Set runner startup type - {startType}");
|
Trace.Info($"Set runner startup type - {startType}");
|
||||||
HostContext.StartupType = startType;
|
HostContext.StartupType = startType;
|
||||||
|
|
||||||
if (command.RunOnce)
|
|
||||||
{
|
|
||||||
_term.WriteLine("Warning: '--once' is going to be deprecated in the future, please consider using '--ephemeral' during runner registration.", ConsoleColor.Yellow);
|
|
||||||
_term.WriteLine("https://docs.github.com/en/actions/hosting-your-own-runners/autoscaling-with-self-hosted-runners#using-ephemeral-runners-for-autoscaling", ConsoleColor.Yellow);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the runner interactively or as service
|
// Run the runner interactively or as service
|
||||||
return await RunAsync(settings, command.RunOnce || settings.Ephemeral);
|
return await RunAsync(settings, command.RunOnce || settings.Ephemeral); // TODO: Remove RunOnce later.
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -316,9 +310,6 @@ namespace GitHub.Runner.Listener
|
|||||||
|
|
||||||
IJobDispatcher jobDispatcher = null;
|
IJobDispatcher jobDispatcher = null;
|
||||||
CancellationTokenSource messageQueueLoopTokenSource = CancellationTokenSource.CreateLinkedTokenSource(HostContext.RunnerShutdownToken);
|
CancellationTokenSource messageQueueLoopTokenSource = CancellationTokenSource.CreateLinkedTokenSource(HostContext.RunnerShutdownToken);
|
||||||
|
|
||||||
// Should we try to cleanup ephemeral runners
|
|
||||||
bool runOnceJobCompleted = false;
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var notification = HostContext.GetService<IJobNotification>();
|
var notification = HostContext.GetService<IJobNotification>();
|
||||||
@@ -380,7 +371,6 @@ namespace GitHub.Runner.Listener
|
|||||||
Task completeTask = await Task.WhenAny(getNextMessage, jobDispatcher.RunOnceJobCompleted.Task);
|
Task completeTask = await Task.WhenAny(getNextMessage, jobDispatcher.RunOnceJobCompleted.Task);
|
||||||
if (completeTask == jobDispatcher.RunOnceJobCompleted.Task)
|
if (completeTask == jobDispatcher.RunOnceJobCompleted.Task)
|
||||||
{
|
{
|
||||||
runOnceJobCompleted = true;
|
|
||||||
Trace.Info("Job has finished at backend, the runner will exit since it is running under onetime use mode.");
|
Trace.Info("Job has finished at backend, the runner will exit since it is running under onetime use mode.");
|
||||||
Trace.Info("Stop message queue looping.");
|
Trace.Info("Stop message queue looping.");
|
||||||
messageQueueLoopTokenSource.Cancel();
|
messageQueueLoopTokenSource.Cancel();
|
||||||
@@ -488,12 +478,6 @@ namespace GitHub.Runner.Listener
|
|||||||
}
|
}
|
||||||
|
|
||||||
messageQueueLoopTokenSource.Dispose();
|
messageQueueLoopTokenSource.Dispose();
|
||||||
|
|
||||||
if (settings.Ephemeral && runOnceJobCompleted)
|
|
||||||
{
|
|
||||||
var configManager = HostContext.GetService<IConfigurationManager>();
|
|
||||||
configManager.DeleteLocalRunnerConfig();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (TaskAgentAccessTokenExpiredException)
|
catch (TaskAgentAccessTokenExpiredException)
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
using GitHub.DistributedTask.Pipelines.ContextData;
|
using GitHub.DistributedTask.Pipelines;
|
||||||
|
using GitHub.DistributedTask.Pipelines.ContextData;
|
||||||
using GitHub.DistributedTask.WebApi;
|
using GitHub.DistributedTask.WebApi;
|
||||||
|
using GitHub.Runner.Common.Util;
|
||||||
using GitHub.Runner.Worker.Container;
|
using GitHub.Runner.Worker.Container;
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
@@ -108,18 +110,11 @@ namespace GitHub.Runner.Worker
|
|||||||
// Stop command
|
// Stop command
|
||||||
if (string.Equals(actionCommand.Command, _stopCommand, StringComparison.OrdinalIgnoreCase))
|
if (string.Equals(actionCommand.Command, _stopCommand, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
ValidateStopToken(context, actionCommand.Data);
|
context.Output(input);
|
||||||
|
context.Debug("Paused processing commands until '##[{actionCommand.Data}]' is received");
|
||||||
_stopToken = actionCommand.Data;
|
_stopToken = actionCommand.Data;
|
||||||
_stopProcessCommand = true;
|
_stopProcessCommand = true;
|
||||||
_registeredCommands.Add(_stopToken);
|
_registeredCommands.Add(_stopToken);
|
||||||
if (_stopToken.Length > 6)
|
|
||||||
{
|
|
||||||
HostContext.SecretMasker.AddValue(_stopToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
context.Output(input);
|
|
||||||
context.Debug("Paused processing commands until the token you called ::stopCommands:: with is received");
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// Found command
|
// Found command
|
||||||
@@ -153,42 +148,7 @@ namespace GitHub.Runner.Worker
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ValidateStopToken(IExecutionContext context, string stopToken)
|
internal static bool EnhancedAnnotationsEnabled(IExecutionContext context) {
|
||||||
{
|
|
||||||
#if OS_WINDOWS
|
|
||||||
var envContext = context.ExpressionValues["env"] as DictionaryContextData;
|
|
||||||
#else
|
|
||||||
var envContext = context.ExpressionValues["env"] as CaseSensitiveDictionaryContextData;
|
|
||||||
#endif
|
|
||||||
var allowUnsecureStopCommandTokens = false;
|
|
||||||
allowUnsecureStopCommandTokens = StringUtil.ConvertToBoolean(Environment.GetEnvironmentVariable(Constants.Variables.Actions.AllowUnsupportedStopCommandTokens));
|
|
||||||
if (!allowUnsecureStopCommandTokens && envContext.ContainsKey(Constants.Variables.Actions.AllowUnsupportedStopCommandTokens))
|
|
||||||
{
|
|
||||||
allowUnsecureStopCommandTokens = StringUtil.ConvertToBoolean(envContext[Constants.Variables.Actions.AllowUnsupportedStopCommandTokens].ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isTokenInvalid = _registeredCommands.Contains(stopToken)
|
|
||||||
|| string.IsNullOrEmpty(stopToken)
|
|
||||||
|| string.Equals(stopToken, "pause-logging", StringComparison.OrdinalIgnoreCase);
|
|
||||||
|
|
||||||
if (isTokenInvalid)
|
|
||||||
{
|
|
||||||
var telemetry = new JobTelemetry
|
|
||||||
{
|
|
||||||
Message = $"Invoked ::stopCommand:: with token: [{stopToken}]",
|
|
||||||
Type = JobTelemetryType.ActionCommand
|
|
||||||
};
|
|
||||||
context.JobTelemetry.Add(telemetry);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isTokenInvalid && !allowUnsecureStopCommandTokens)
|
|
||||||
{
|
|
||||||
throw new Exception(Constants.Runner.UnsupportedStopCommandTokenDisabled);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static bool EnhancedAnnotationsEnabled(IExecutionContext context)
|
|
||||||
{
|
|
||||||
return context.Global.Variables.GetBoolean("DistributedTask.EnhancedAnnotations") ?? false;
|
return context.Global.Variables.GetBoolean("DistributedTask.EnhancedAnnotations") ?? false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -633,12 +633,7 @@ namespace GitHub.Runner.Worker
|
|||||||
}
|
}
|
||||||
catch (Exception ex) when (!executionContext.CancellationToken.IsCancellationRequested) // Do not retry if the run is canceled.
|
catch (Exception ex) when (!executionContext.CancellationToken.IsCancellationRequested) // Do not retry if the run is canceled.
|
||||||
{
|
{
|
||||||
// UnresolvableActionDownloadInfoException is a 422 client error, don't retry
|
if (attempt < 3)
|
||||||
// Some possible cases are:
|
|
||||||
// * Repo is rate limited
|
|
||||||
// * Repo or tag doesn't exist, or isn't public
|
|
||||||
// * Policy validation failed
|
|
||||||
if (attempt < 3 && !(ex is WebApi.UnresolvableActionDownloadInfoException))
|
|
||||||
{
|
{
|
||||||
executionContext.Output($"Failed to resolve action download info. Error: {ex.Message}");
|
executionContext.Output($"Failed to resolve action download info. Error: {ex.Message}");
|
||||||
executionContext.Debug(ex.ToString());
|
executionContext.Debug(ex.ToString());
|
||||||
@@ -654,7 +649,6 @@ namespace GitHub.Runner.Worker
|
|||||||
// Some possible cases are:
|
// Some possible cases are:
|
||||||
// * Repo is rate limited
|
// * Repo is rate limited
|
||||||
// * Repo or tag doesn't exist, or isn't public
|
// * Repo or tag doesn't exist, or isn't public
|
||||||
// * Policy validation failed
|
|
||||||
if (ex is WebApi.UnresolvableActionDownloadInfoException)
|
if (ex is WebApi.UnresolvableActionDownloadInfoException)
|
||||||
{
|
{
|
||||||
throw;
|
throw;
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ namespace GitHub.Runner.Worker.Container
|
|||||||
_pathMappings.Add(new PathMapping(hostContext.GetDirectory(WellKnownDirectory.Externals), "/__e"));
|
_pathMappings.Add(new PathMapping(hostContext.GetDirectory(WellKnownDirectory.Externals), "/__e"));
|
||||||
if (this.IsJobContainer)
|
if (this.IsJobContainer)
|
||||||
{
|
{
|
||||||
// this.MountVolumes.Add(new MountVolume("/var/run/docker.sock", "/var/run/docker.sock"));
|
this.MountVolumes.Add(new MountVolume("/var/run/docker.sock", "/var/run/docker.sock"));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (container.Ports?.Count > 0)
|
if (container.Ports?.Count > 0)
|
||||||
|
|||||||
@@ -12,88 +12,9 @@ using GitHub.Runner.Sdk;
|
|||||||
using GitHub.DistributedTask.Pipelines.ContextData;
|
using GitHub.DistributedTask.Pipelines.ContextData;
|
||||||
using Microsoft.Win32;
|
using Microsoft.Win32;
|
||||||
using GitHub.DistributedTask.Pipelines.ObjectTemplating;
|
using GitHub.DistributedTask.Pipelines.ObjectTemplating;
|
||||||
using System.Threading.Channels;
|
|
||||||
using GitHub.Services.WebApi;
|
|
||||||
using System.Text;
|
|
||||||
using System.Runtime.Serialization;
|
|
||||||
|
|
||||||
namespace GitHub.Runner.Worker
|
namespace GitHub.Runner.Worker
|
||||||
{
|
{
|
||||||
[DataContract]
|
|
||||||
public class ContainerEngineHandlerInput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public string Command { get; set; }
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public ContainersCreationInput CreationInput { get; set; }
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public JobContainerExecInput ExecInput { get; set; }
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public ContainersRemoveInput RemoveInput { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[DataContract]
|
|
||||||
public class ContainersCreationInput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public List<ContainerInfo> Containers { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[DataContract]
|
|
||||||
public class JobContainerExecInput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public ContainerInfo JobContainer { get; set; }
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public string WorkingDirectory { get; set; }
|
|
||||||
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public string FileName { get; set; }
|
|
||||||
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public string Arguments { get; set; }
|
|
||||||
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public List<string> EnvironmentKeys { get; set; }
|
|
||||||
|
|
||||||
[DataMember]
|
|
||||||
public Dictionary<string, string> EnvironmentVariables { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
[DataContract]
|
|
||||||
public class ContainersRemoveInput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public string Network { get; set; }
|
|
||||||
[DataMember]
|
|
||||||
public string JobContainerId { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[DataContract]
|
|
||||||
public class ContainersCreationOutput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public string Network { get; set; }
|
|
||||||
[DataMember]
|
|
||||||
public string JobContainerId { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[DataContract]
|
|
||||||
public class ContainerEngineHandlerOutput
|
|
||||||
{
|
|
||||||
[DataMember]
|
|
||||||
public ContainersCreationOutput CreationOutput { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[ServiceLocator(Default = typeof(ContainerOperationProvider))]
|
[ServiceLocator(Default = typeof(ContainerOperationProvider))]
|
||||||
public interface IContainerOperationProvider : IRunnerService
|
public interface IContainerOperationProvider : IRunnerService
|
||||||
{
|
{
|
||||||
@@ -103,57 +24,25 @@ namespace GitHub.Runner.Worker
|
|||||||
|
|
||||||
public class ContainerOperationProvider : RunnerService, IContainerOperationProvider
|
public class ContainerOperationProvider : RunnerService, IContainerOperationProvider
|
||||||
{
|
{
|
||||||
private IDockerCommandManager _dockerManager = null;
|
private IDockerCommandManager _dockerManager;
|
||||||
|
|
||||||
public override void Initialize(IHostContext hostContext)
|
public override void Initialize(IHostContext hostContext)
|
||||||
{
|
{
|
||||||
base.Initialize(hostContext);
|
base.Initialize(hostContext);
|
||||||
// _dockerManager = HostContext.GetService<IDockerCommandManager>();
|
_dockerManager = HostContext.GetService<IDockerCommandManager>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task StartContainersAsync(IExecutionContext executionContext, object data)
|
public async Task StartContainersAsync(IExecutionContext executionContext, object data)
|
||||||
{
|
{
|
||||||
Trace.Entering();
|
Trace.Entering();
|
||||||
// if (!Constants.Runner.Platform.Equals(Constants.OSPlatform.Linux))
|
if (!Constants.Runner.Platform.Equals(Constants.OSPlatform.Linux))
|
||||||
// {
|
{
|
||||||
// throw new NotSupportedException("Container operations are only supported on Linux runners");
|
throw new NotSupportedException("Container operations are only supported on Linux runners");
|
||||||
// }
|
}
|
||||||
ArgUtil.NotNull(executionContext, nameof(executionContext));
|
ArgUtil.NotNull(executionContext, nameof(executionContext));
|
||||||
List<ContainerInfo> containers = data as List<ContainerInfo>;
|
List<ContainerInfo> containers = data as List<ContainerInfo>;
|
||||||
ArgUtil.NotNull(containers, nameof(containers));
|
ArgUtil.NotNull(containers, nameof(containers));
|
||||||
|
|
||||||
foreach (var container in containers)
|
|
||||||
{
|
|
||||||
if (container.IsJobContainer)
|
|
||||||
{
|
|
||||||
// Configure job container - Mount workspace and tools, set up environment, and start long running process
|
|
||||||
var githubContext = executionContext.ExpressionValues["github"] as GitHubContext;
|
|
||||||
ArgUtil.NotNull(githubContext, nameof(githubContext));
|
|
||||||
var workingDirectory = githubContext["workspace"] as StringContextData;
|
|
||||||
ArgUtil.NotNullOrEmpty(workingDirectory, nameof(workingDirectory));
|
|
||||||
container.MountVolumes.Add(new MountVolume(HostContext.GetDirectory(WellKnownDirectory.Work), container.TranslateToContainerPath(HostContext.GetDirectory(WellKnownDirectory.Work))));
|
|
||||||
container.MountVolumes.Add(new MountVolume(HostContext.GetDirectory(WellKnownDirectory.Externals), container.TranslateToContainerPath(HostContext.GetDirectory(WellKnownDirectory.Externals)), true));
|
|
||||||
container.MountVolumes.Add(new MountVolume(HostContext.GetDirectory(WellKnownDirectory.Temp), container.TranslateToContainerPath(HostContext.GetDirectory(WellKnownDirectory.Temp))));
|
|
||||||
// container.MountVolumes.Add(new MountVolume(HostContext.GetDirectory(WellKnownDirectory.Actions), container.TranslateToContainerPath(HostContext.GetDirectory(WellKnownDirectory.Actions))));
|
|
||||||
container.MountVolumes.Add(new MountVolume(HostContext.GetDirectory(WellKnownDirectory.Tools), container.TranslateToContainerPath(HostContext.GetDirectory(WellKnownDirectory.Tools))));
|
|
||||||
|
|
||||||
var tempHomeDirectory = Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Temp), "_github_home");
|
|
||||||
Directory.CreateDirectory(tempHomeDirectory);
|
|
||||||
container.MountVolumes.Add(new MountVolume(tempHomeDirectory, "/github/home"));
|
|
||||||
container.AddPathTranslateMapping(tempHomeDirectory, "/github/home");
|
|
||||||
container.ContainerEnvironmentVariables["HOME"] = container.TranslateToContainerPath(tempHomeDirectory);
|
|
||||||
|
|
||||||
var tempWorkflowDirectory = Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Temp), "_github_workflow");
|
|
||||||
Directory.CreateDirectory(tempWorkflowDirectory);
|
|
||||||
container.MountVolumes.Add(new MountVolume(tempWorkflowDirectory, "/github/workflow"));
|
|
||||||
container.AddPathTranslateMapping(tempWorkflowDirectory, "/github/workflow");
|
|
||||||
|
|
||||||
container.ContainerWorkDirectory = container.TranslateToContainerPath(workingDirectory);
|
|
||||||
container.ContainerEntryPoint = "tail";
|
|
||||||
container.ContainerEntryPointArgs = "-f /dev/null";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var postJobStep = new JobExtensionRunner(runAsync: this.StopContainersAsync,
|
var postJobStep = new JobExtensionRunner(runAsync: this.StopContainersAsync,
|
||||||
condition: $"{PipelineTemplateConstants.Always}()",
|
condition: $"{PipelineTemplateConstants.Always}()",
|
||||||
displayName: "Stop containers",
|
displayName: "Stop containers",
|
||||||
@@ -162,68 +51,6 @@ namespace GitHub.Runner.Worker
|
|||||||
executionContext.Debug($"Register post job cleanup for stopping/deleting containers.");
|
executionContext.Debug($"Register post job cleanup for stopping/deleting containers.");
|
||||||
executionContext.RegisterPostJobStep(postJobStep);
|
executionContext.RegisterPostJobStep(postJobStep);
|
||||||
|
|
||||||
var podManHandler = Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Bin), "kubectlHandler", "index.js");
|
|
||||||
if (File.Exists(podManHandler))
|
|
||||||
{
|
|
||||||
var podmanInput = new ContainerEngineHandlerInput()
|
|
||||||
{
|
|
||||||
Command = "Create",
|
|
||||||
CreationInput = new ContainersCreationInput()
|
|
||||||
{
|
|
||||||
Containers = containers
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
ContainerEngineHandlerOutput podmanOutput = null;
|
|
||||||
using (var processInvoker = HostContext.CreateService<IProcessInvoker>())
|
|
||||||
{
|
|
||||||
var redirectStandardIn = Channel.CreateUnbounded<string>(new UnboundedChannelOptions() { SingleReader = true, SingleWriter = true });
|
|
||||||
redirectStandardIn.Writer.TryWrite(JsonUtility.ToString(podmanInput));
|
|
||||||
|
|
||||||
processInvoker.OutputDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
{
|
|
||||||
executionContext.Output(message.Data);
|
|
||||||
};
|
|
||||||
|
|
||||||
processInvoker.ErrorDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
{
|
|
||||||
executionContext.Output(message.Data);
|
|
||||||
if (podmanOutput == null && message.Data.IndexOf("___CONTAINER_ENGINE_HANDLER_OUTPUT___") >= 0)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
podmanOutput = JsonUtility.FromString<ContainerEngineHandlerOutput>(message.Data.Replace("___CONTAINER_ENGINE_HANDLER_OUTPUT___", ""));
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
executionContext.Error(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Execute the process. Exit code 0 should always be returned.
|
|
||||||
// A non-zero exit code indicates infrastructural failure.
|
|
||||||
// Task failure should be communicated over STDOUT using ## commands.
|
|
||||||
await processInvoker.ExecuteAsync(workingDirectory: HostContext.GetDirectory(WellKnownDirectory.Bin),
|
|
||||||
fileName: Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Externals), "node12", "bin", $"node{IOUtil.ExeExtension}"),
|
|
||||||
arguments: podManHandler,
|
|
||||||
environment: null,
|
|
||||||
requireExitCodeZero: false,
|
|
||||||
outputEncoding: Encoding.UTF8,
|
|
||||||
killProcessOnCancel: false,
|
|
||||||
redirectStandardIn: redirectStandardIn,
|
|
||||||
cancellationToken: executionContext.CancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (podmanOutput != null)
|
|
||||||
{
|
|
||||||
executionContext.JobContext.Container["network"] = new StringContextData(podmanOutput.CreationOutput.Network);
|
|
||||||
executionContext.JobContext.Container["id"] = new StringContextData(podmanOutput.CreationOutput.JobContainerId);
|
|
||||||
executionContext.Global.Container.ContainerId = podmanOutput.CreationOutput.JobContainerId;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Check whether we are inside a container.
|
// Check whether we are inside a container.
|
||||||
// Our container feature requires to map working directory from host to the container.
|
// Our container feature requires to map working directory from host to the container.
|
||||||
// If we are already inside a container, we will not able to find out the real working direcotry path on the host.
|
// If we are already inside a container, we will not able to find out the real working direcotry path on the host.
|
||||||
@@ -326,7 +153,6 @@ namespace GitHub.Runner.Worker
|
|||||||
}
|
}
|
||||||
executionContext.Output("##[endgroup]");
|
executionContext.Output("##[endgroup]");
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
public async Task StopContainersAsync(IExecutionContext executionContext, object data)
|
public async Task StopContainersAsync(IExecutionContext executionContext, object data)
|
||||||
{
|
{
|
||||||
@@ -336,62 +162,6 @@ namespace GitHub.Runner.Worker
|
|||||||
List<ContainerInfo> containers = data as List<ContainerInfo>;
|
List<ContainerInfo> containers = data as List<ContainerInfo>;
|
||||||
ArgUtil.NotNull(containers, nameof(containers));
|
ArgUtil.NotNull(containers, nameof(containers));
|
||||||
|
|
||||||
var podManHandler = Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Bin), "kubectlHandler", "index.js");
|
|
||||||
if (File.Exists(podManHandler))
|
|
||||||
{
|
|
||||||
var podmanInput = new ContainerEngineHandlerInput()
|
|
||||||
{
|
|
||||||
Command = "Remove",
|
|
||||||
RemoveInput = new ContainersRemoveInput()
|
|
||||||
{
|
|
||||||
Network = executionContext.JobContext.Container["network"].ToString(),
|
|
||||||
JobContainerId = executionContext.JobContext.Container["id"].ToString()
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
ContainerEngineHandlerOutput podmanOutput = null;
|
|
||||||
using (var processInvoker = HostContext.CreateService<IProcessInvoker>())
|
|
||||||
{
|
|
||||||
var redirectStandardIn = Channel.CreateUnbounded<string>(new UnboundedChannelOptions() { SingleReader = true, SingleWriter = true });
|
|
||||||
redirectStandardIn.Writer.TryWrite(JsonUtility.ToString(podmanInput));
|
|
||||||
|
|
||||||
processInvoker.OutputDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
{
|
|
||||||
executionContext.Output(message.Data);
|
|
||||||
};
|
|
||||||
|
|
||||||
processInvoker.ErrorDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
{
|
|
||||||
executionContext.Output(message.Data);
|
|
||||||
if (podmanOutput == null && message.Data.IndexOf("___CONTAINER_ENGINE_HANDLER_OUTPUT___") >= 0)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
podmanOutput = JsonUtility.FromString<ContainerEngineHandlerOutput>(message.Data.Replace("___CONTAINER_ENGINE_HANDLER_OUTPUT___", ""));
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
executionContext.Error(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Execute the process. Exit code 0 should always be returned.
|
|
||||||
// A non-zero exit code indicates infrastructural failure.
|
|
||||||
// Task failure should be communicated over STDOUT using ## commands.
|
|
||||||
await processInvoker.ExecuteAsync(workingDirectory: HostContext.GetDirectory(WellKnownDirectory.Work),
|
|
||||||
fileName: Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Externals), "node12", "bin", $"node{IOUtil.ExeExtension}"),
|
|
||||||
arguments: podManHandler,
|
|
||||||
environment: null,
|
|
||||||
requireExitCodeZero: false,
|
|
||||||
outputEncoding: Encoding.UTF8,
|
|
||||||
killProcessOnCancel: false,
|
|
||||||
redirectStandardIn: redirectStandardIn,
|
|
||||||
cancellationToken: executionContext.CancellationToken);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
foreach (var container in containers)
|
foreach (var container in containers)
|
||||||
{
|
{
|
||||||
await StopContainerAsync(executionContext, container);
|
await StopContainerAsync(executionContext, container);
|
||||||
@@ -399,7 +169,6 @@ namespace GitHub.Runner.Worker
|
|||||||
// Remove the container network
|
// Remove the container network
|
||||||
await RemoveContainerNetworkAsync(executionContext, containers.First().ContainerNetwork);
|
await RemoveContainerNetworkAsync(executionContext, containers.First().ContainerNetwork);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private async Task StartContainerAsync(IExecutionContext executionContext, ContainerInfo container)
|
private async Task StartContainerAsync(IExecutionContext executionContext, ContainerInfo container)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -52,7 +52,6 @@ namespace GitHub.Runner.Worker
|
|||||||
Dictionary<string, VariableValue> JobOutputs { get; }
|
Dictionary<string, VariableValue> JobOutputs { get; }
|
||||||
ActionsEnvironmentReference ActionsEnvironment { get; }
|
ActionsEnvironmentReference ActionsEnvironment { get; }
|
||||||
List<ActionsStepTelemetry> ActionsStepsTelemetry { get; }
|
List<ActionsStepTelemetry> ActionsStepsTelemetry { get; }
|
||||||
List<JobTelemetry> JobTelemetry { get; }
|
|
||||||
DictionaryContextData ExpressionValues { get; }
|
DictionaryContextData ExpressionValues { get; }
|
||||||
IList<IFunctionInfo> ExpressionFunctions { get; }
|
IList<IFunctionInfo> ExpressionFunctions { get; }
|
||||||
JobContext JobContext { get; }
|
JobContext JobContext { get; }
|
||||||
@@ -151,7 +150,6 @@ namespace GitHub.Runner.Worker
|
|||||||
|
|
||||||
public ActionsEnvironmentReference ActionsEnvironment { get; private set; }
|
public ActionsEnvironmentReference ActionsEnvironment { get; private set; }
|
||||||
public List<ActionsStepTelemetry> ActionsStepsTelemetry { get; private set; }
|
public List<ActionsStepTelemetry> ActionsStepsTelemetry { get; private set; }
|
||||||
public List<JobTelemetry> JobTelemetry { get; private set; }
|
|
||||||
public DictionaryContextData ExpressionValues { get; } = new DictionaryContextData();
|
public DictionaryContextData ExpressionValues { get; } = new DictionaryContextData();
|
||||||
public IList<IFunctionInfo> ExpressionFunctions { get; } = new List<IFunctionInfo>();
|
public IList<IFunctionInfo> ExpressionFunctions { get; } = new List<IFunctionInfo>();
|
||||||
|
|
||||||
@@ -296,7 +294,6 @@ namespace GitHub.Runner.Worker
|
|||||||
child.ContextName = contextName;
|
child.ContextName = contextName;
|
||||||
child.EmbeddedId = embeddedId;
|
child.EmbeddedId = embeddedId;
|
||||||
child.SiblingScopeName = siblingScopeName;
|
child.SiblingScopeName = siblingScopeName;
|
||||||
child.JobTelemetry = JobTelemetry;
|
|
||||||
if (intraActionState == null)
|
if (intraActionState == null)
|
||||||
{
|
{
|
||||||
child.IntraActionState = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
child.IntraActionState = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
||||||
@@ -653,8 +650,6 @@ namespace GitHub.Runner.Worker
|
|||||||
// ActionsStepTelemetry
|
// ActionsStepTelemetry
|
||||||
ActionsStepsTelemetry = new List<ActionsStepTelemetry>();
|
ActionsStepsTelemetry = new List<ActionsStepTelemetry>();
|
||||||
|
|
||||||
JobTelemetry = new List<JobTelemetry>();
|
|
||||||
|
|
||||||
// Service container info
|
// Service container info
|
||||||
Global.ServiceContainers = new List<ContainerInfo>();
|
Global.ServiceContainers = new List<ContainerInfo>();
|
||||||
|
|
||||||
|
|||||||
@@ -23,9 +23,6 @@ namespace GitHub.Runner.Worker
|
|||||||
"job",
|
"job",
|
||||||
"path",
|
"path",
|
||||||
"ref",
|
"ref",
|
||||||
"ref_name",
|
|
||||||
"ref_protected",
|
|
||||||
"ref_type",
|
|
||||||
"repository",
|
"repository",
|
||||||
"repository_owner",
|
"repository_owner",
|
||||||
"retention_days",
|
"retention_days",
|
||||||
@@ -42,17 +39,10 @@ namespace GitHub.Runner.Worker
|
|||||||
{
|
{
|
||||||
foreach (var data in this)
|
foreach (var data in this)
|
||||||
{
|
{
|
||||||
if (_contextEnvAllowlist.Contains(data.Key))
|
if (_contextEnvAllowlist.Contains(data.Key) && data.Value is StringContextData value)
|
||||||
{
|
|
||||||
if (data.Value is StringContextData value)
|
|
||||||
{
|
{
|
||||||
yield return new KeyValuePair<string, string>($"GITHUB_{data.Key.ToUpperInvariant()}", value);
|
yield return new KeyValuePair<string, string>($"GITHUB_{data.Key.ToUpperInvariant()}", value);
|
||||||
}
|
}
|
||||||
else if (data.Value is BooleanContextData booleanValue)
|
|
||||||
{
|
|
||||||
yield return new KeyValuePair<string, string>($"GITHUB_{data.Key.ToUpperInvariant()}", booleanValue.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,6 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
||||||
event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
||||||
|
|
||||||
IExecutionContext ExecutionContext { get; set; }
|
|
||||||
|
|
||||||
string ResolvePathForStepHost(string path);
|
string ResolvePathForStepHost(string path);
|
||||||
|
|
||||||
Task<string> DetermineNodeRuntimeVersion(IExecutionContext executionContext);
|
Task<string> DetermineNodeRuntimeVersion(IExecutionContext executionContext);
|
||||||
@@ -55,8 +53,6 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
public event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
public event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
||||||
public event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
public event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
||||||
|
|
||||||
public IExecutionContext ExecutionContext { get; set; }
|
|
||||||
|
|
||||||
public string ResolvePathForStepHost(string path)
|
public string ResolvePathForStepHost(string path)
|
||||||
{
|
{
|
||||||
return path;
|
return path;
|
||||||
@@ -103,8 +99,6 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
public event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
public event EventHandler<ProcessDataReceivedEventArgs> OutputDataReceived;
|
||||||
public event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
public event EventHandler<ProcessDataReceivedEventArgs> ErrorDataReceived;
|
||||||
|
|
||||||
public IExecutionContext ExecutionContext { get; set; }
|
|
||||||
|
|
||||||
public string ResolvePathForStepHost(string path)
|
public string ResolvePathForStepHost(string path)
|
||||||
{
|
{
|
||||||
// make sure container exist.
|
// make sure container exist.
|
||||||
@@ -180,74 +174,6 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
ArgUtil.NotNull(Container, nameof(Container));
|
ArgUtil.NotNull(Container, nameof(Container));
|
||||||
ArgUtil.NotNullOrEmpty(Container.ContainerId, nameof(Container.ContainerId));
|
ArgUtil.NotNullOrEmpty(Container.ContainerId, nameof(Container.ContainerId));
|
||||||
|
|
||||||
var podManHandler = Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Bin), "kubectlHandler", "index.js");
|
|
||||||
if (File.Exists(podManHandler))
|
|
||||||
{
|
|
||||||
var podmanInput = new ContainerEngineHandlerInput()
|
|
||||||
{
|
|
||||||
Command = "Exec",
|
|
||||||
ExecInput = new JobContainerExecInput()
|
|
||||||
{
|
|
||||||
JobContainer = this.Container,
|
|
||||||
WorkingDirectory = workingDirectory,
|
|
||||||
FileName = fileName,
|
|
||||||
Arguments = arguments,
|
|
||||||
EnvironmentKeys = environment.Keys.ToList(),
|
|
||||||
EnvironmentVariables = environment.ToDictionary(x => x.Key, y => y.Value)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// make sure all env are using container path
|
|
||||||
foreach (var envKey in environment.Keys.ToList())
|
|
||||||
{
|
|
||||||
environment[envKey] = this.Container.TranslateToContainerPath(environment[envKey]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ContainerEngineHandlerOutput podmanOutput = null;
|
|
||||||
using (var processInvoker = HostContext.CreateService<IProcessInvoker>())
|
|
||||||
{
|
|
||||||
var redirectStandardIn = Channel.CreateUnbounded<string>(new UnboundedChannelOptions() { SingleReader = true, SingleWriter = true });
|
|
||||||
redirectStandardIn.Writer.TryWrite(JsonUtility.ToString(podmanInput));
|
|
||||||
|
|
||||||
// processInvoker.OutputDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
// {
|
|
||||||
// ExecutionContext.Output(message.Data);
|
|
||||||
// };
|
|
||||||
|
|
||||||
// processInvoker.ErrorDataReceived += delegate (object sender, ProcessDataReceivedEventArgs message)
|
|
||||||
// {
|
|
||||||
// executionContext.Output(message.Data);
|
|
||||||
// if (podmanOutput == null && message.Data.IndexOf("___CONTAINER_ENGINE_HANDLER_OUTPUT___") >= 0)
|
|
||||||
// {
|
|
||||||
// try
|
|
||||||
// {
|
|
||||||
// podmanOutput = JsonUtility.FromString<ContainerEngineHandlerOutput>(message.Data.Replace("___CONTAINER_ENGINE_HANDLER_OUTPUT___", ""));
|
|
||||||
// }
|
|
||||||
// catch (Exception ex)
|
|
||||||
// {
|
|
||||||
// executionContext.Error(ex);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// };
|
|
||||||
processInvoker.OutputDataReceived += OutputDataReceived;
|
|
||||||
processInvoker.ErrorDataReceived += ErrorDataReceived;
|
|
||||||
|
|
||||||
// Execute the process. Exit code 0 should always be returned.
|
|
||||||
// A non-zero exit code indicates infrastructural failure.
|
|
||||||
// Task failure should be communicated over STDOUT using ## commands.
|
|
||||||
return await processInvoker.ExecuteAsync(workingDirectory: HostContext.GetDirectory(WellKnownDirectory.Work),
|
|
||||||
fileName: Path.Combine(HostContext.GetDirectory(WellKnownDirectory.Externals), "node12", "bin", $"node{IOUtil.ExeExtension}"),
|
|
||||||
arguments: podManHandler,
|
|
||||||
environment: environment,
|
|
||||||
requireExitCodeZero: requireExitCodeZero,
|
|
||||||
outputEncoding: Encoding.UTF8,
|
|
||||||
killProcessOnCancel: killProcessOnCancel,
|
|
||||||
redirectStandardIn: redirectStandardIn,
|
|
||||||
cancellationToken: cancellationToken);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var dockerManager = HostContext.GetService<IDockerCommandManager>();
|
var dockerManager = HostContext.GetService<IDockerCommandManager>();
|
||||||
string dockerClientPath = dockerManager.DockerPath;
|
string dockerClientPath = dockerManager.DockerPath;
|
||||||
|
|
||||||
@@ -314,5 +240,4 @@ namespace GitHub.Runner.Worker.Handlers
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
@@ -107,9 +106,6 @@ namespace GitHub.Runner.Worker
|
|||||||
|
|
||||||
jobContext.SetRunnerContext("os", VarUtil.OS);
|
jobContext.SetRunnerContext("os", VarUtil.OS);
|
||||||
|
|
||||||
var runnerSettings = HostContext.GetService<IConfigurationStore>().GetSettings();
|
|
||||||
jobContext.SetRunnerContext("name", runnerSettings.AgentName);
|
|
||||||
|
|
||||||
string toolsDirectory = HostContext.GetDirectory(WellKnownDirectory.Tools);
|
string toolsDirectory = HostContext.GetDirectory(WellKnownDirectory.Tools);
|
||||||
Directory.CreateDirectory(toolsDirectory);
|
Directory.CreateDirectory(toolsDirectory);
|
||||||
jobContext.SetRunnerContext("tool_cache", toolsDirectory);
|
jobContext.SetRunnerContext("tool_cache", toolsDirectory);
|
||||||
@@ -229,15 +225,8 @@ namespace GitHub.Runner.Worker
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load any upgrade telemetry
|
|
||||||
LoadFromTelemetryFile(jobContext.JobTelemetry);
|
|
||||||
|
|
||||||
// Make sure we don't submit secrets as telemetry
|
|
||||||
MaskTelemetrySecrets(jobContext.JobTelemetry);
|
|
||||||
|
|
||||||
Trace.Info("Raising job completed event.");
|
Trace.Info("Raising job completed event.");
|
||||||
var jobCompletedEvent = new JobCompletedEvent(message.RequestId, message.JobId, result, jobContext.JobOutputs, jobContext.ActionsEnvironment, jobContext.ActionsStepsTelemetry, jobContext.JobTelemetry);
|
var jobCompletedEvent = new JobCompletedEvent(message.RequestId, message.JobId, result, jobContext.JobOutputs, jobContext.ActionsEnvironment, jobContext.ActionsStepsTelemetry);
|
||||||
|
|
||||||
|
|
||||||
var completeJobRetryLimit = 5;
|
var completeJobRetryLimit = 5;
|
||||||
var exceptions = new List<Exception>();
|
var exceptions = new List<Exception>();
|
||||||
@@ -281,38 +270,6 @@ namespace GitHub.Runner.Worker
|
|||||||
throw new AggregateException(exceptions);
|
throw new AggregateException(exceptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void MaskTelemetrySecrets(List<JobTelemetry> jobTelemetry)
|
|
||||||
{
|
|
||||||
foreach (var telemetryItem in jobTelemetry)
|
|
||||||
{
|
|
||||||
telemetryItem.Message = HostContext.SecretMasker.MaskSecrets(telemetryItem.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void LoadFromTelemetryFile(List<JobTelemetry> jobTelemetry)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var telemetryFilePath = HostContext.GetConfigFile(WellKnownConfigFile.Telemetry);
|
|
||||||
if (File.Exists(telemetryFilePath))
|
|
||||||
{
|
|
||||||
var telemetryData = File.ReadAllText(telemetryFilePath, Encoding.UTF8);
|
|
||||||
var telemetry = new JobTelemetry
|
|
||||||
{
|
|
||||||
Message = $"Runner File Telemetry:\n{telemetryData}",
|
|
||||||
Type = JobTelemetryType.General
|
|
||||||
};
|
|
||||||
jobTelemetry.Add(telemetry);
|
|
||||||
IOUtil.DeleteFile(telemetryFilePath);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Trace.Error("Error when trying to load telemetry from telemetry file");
|
|
||||||
Trace.Error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task ShutdownQueue(bool throwOnFailure)
|
private async Task ShutdownQueue(bool throwOnFailure)
|
||||||
{
|
{
|
||||||
if (_jobServerQueue != null)
|
if (_jobServerQueue != null)
|
||||||
|
|||||||
@@ -155,19 +155,6 @@ namespace GitHub.DistributedTask.WebApi
|
|||||||
this.ActionsStepsTelemetry = actionsStepsTelemetry;
|
this.ActionsStepsTelemetry = actionsStepsTelemetry;
|
||||||
}
|
}
|
||||||
|
|
||||||
public JobCompletedEvent(
|
|
||||||
Int64 requestId,
|
|
||||||
Guid jobId,
|
|
||||||
TaskResult result,
|
|
||||||
Dictionary<String, VariableValue> outputs,
|
|
||||||
ActionsEnvironmentReference actionsEnvironment,
|
|
||||||
List<ActionsStepTelemetry> actionsStepsTelemetry,
|
|
||||||
List<JobTelemetry> jobTelemetry)
|
|
||||||
: this(requestId, jobId, result, outputs, actionsEnvironment, actionsStepsTelemetry)
|
|
||||||
{
|
|
||||||
this.JobTelemetry = jobTelemetry;
|
|
||||||
}
|
|
||||||
|
|
||||||
[DataMember(EmitDefaultValue = false)]
|
[DataMember(EmitDefaultValue = false)]
|
||||||
public Int64 RequestId
|
public Int64 RequestId
|
||||||
{
|
{
|
||||||
@@ -202,13 +189,6 @@ namespace GitHub.DistributedTask.WebApi
|
|||||||
get;
|
get;
|
||||||
set;
|
set;
|
||||||
}
|
}
|
||||||
|
|
||||||
[DataMember(EmitDefaultValue = false)]
|
|
||||||
public List<JobTelemetry> JobTelemetry
|
|
||||||
{
|
|
||||||
get;
|
|
||||||
set;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[DataContract]
|
[DataContract]
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
using System.Runtime.Serialization;
|
|
||||||
|
|
||||||
namespace GitHub.DistributedTask.WebApi
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Information about a job run on the runner
|
|
||||||
/// </summary>
|
|
||||||
[DataContract]
|
|
||||||
public class JobTelemetry
|
|
||||||
{
|
|
||||||
[DataMember(EmitDefaultValue = false)]
|
|
||||||
public string Message { get; set; }
|
|
||||||
|
|
||||||
[DataMember(EmitDefaultValue = false)]
|
|
||||||
public JobTelemetryType Type { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
using System.Runtime.Serialization;
|
|
||||||
|
|
||||||
namespace GitHub.DistributedTask.WebApi
|
|
||||||
{
|
|
||||||
public enum JobTelemetryType
|
|
||||||
{
|
|
||||||
[EnumMember]
|
|
||||||
General = 0,
|
|
||||||
|
|
||||||
[EnumMember]
|
|
||||||
ActionCommand = 1,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -161,8 +161,7 @@ namespace GitHub.Runner.Common.Tests.Listener.Configuration
|
|||||||
"--work", _expectedWorkFolder,
|
"--work", _expectedWorkFolder,
|
||||||
"--auth", _expectedAuthType,
|
"--auth", _expectedAuthType,
|
||||||
"--token", _expectedToken,
|
"--token", _expectedToken,
|
||||||
"--labels", userLabels,
|
"--labels", userLabels
|
||||||
"--ephemeral",
|
|
||||||
});
|
});
|
||||||
trace.Info("Constructed.");
|
trace.Info("Constructed.");
|
||||||
_store.Setup(x => x.IsConfigured()).Returns(false);
|
_store.Setup(x => x.IsConfigured()).Returns(false);
|
||||||
@@ -180,7 +179,6 @@ namespace GitHub.Runner.Common.Tests.Listener.Configuration
|
|||||||
Assert.True(s.AgentName.Equals(_expectedAgentName));
|
Assert.True(s.AgentName.Equals(_expectedAgentName));
|
||||||
Assert.True(s.PoolId.Equals(_secondRunnerGroupId));
|
Assert.True(s.PoolId.Equals(_secondRunnerGroupId));
|
||||||
Assert.True(s.WorkFolder.Equals(_expectedWorkFolder));
|
Assert.True(s.WorkFolder.Equals(_expectedWorkFolder));
|
||||||
Assert.True(s.Ephemeral.Equals(true));
|
|
||||||
|
|
||||||
// validate GetAgentPoolsAsync gets called twice with automation pool type
|
// validate GetAgentPoolsAsync gets called twice with automation pool type
|
||||||
_runnerServer.Verify(x => x.GetAgentPoolsAsync(It.IsAny<string>(), It.Is<TaskAgentPoolType>(p => p == TaskAgentPoolType.Automation)), Times.Exactly(2));
|
_runnerServer.Verify(x => x.GetAgentPoolsAsync(It.IsAny<string>(), It.Is<TaskAgentPoolType>(p => p == TaskAgentPoolType.Automation)), Times.Exactly(2));
|
||||||
|
|||||||
@@ -264,170 +264,6 @@ namespace GitHub.Runner.Common.Tests.Listener
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Runner")]
|
|
||||||
public async void RenewJobRequestNewAgentNameUpdatesSettings()
|
|
||||||
{
|
|
||||||
//Arrange
|
|
||||||
using (var hc = new TestHostContext(this))
|
|
||||||
{
|
|
||||||
var count = 0;
|
|
||||||
var oldName = "OldName";
|
|
||||||
var newName = "NewName";
|
|
||||||
var oldSettings = new RunnerSettings { AgentName = oldName };
|
|
||||||
var reservedAgent = new TaskAgentReference { Name = newName };
|
|
||||||
|
|
||||||
var trace = hc.GetTrace(nameof(DispatcherRenewJobRequestStopOnJobTokenExpiredExceptions));
|
|
||||||
TaskCompletionSource<int> firstJobRequestRenewed = new TaskCompletionSource<int>();
|
|
||||||
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var request = new Mock<TaskAgentJobRequest>();
|
|
||||||
request.Object.ReservedAgent = reservedAgent;
|
|
||||||
PropertyInfo lockUntilProperty = request.Object.GetType().GetProperty("LockedUntil", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);
|
|
||||||
Assert.NotNull(lockUntilProperty);
|
|
||||||
lockUntilProperty.SetValue(request.Object, DateTime.UtcNow.AddMinutes(5));
|
|
||||||
hc.SetSingleton<IRunnerServer>(_runnerServer.Object);
|
|
||||||
hc.SetSingleton<IConfigurationStore>(_configurationStore.Object);
|
|
||||||
_configurationStore.Setup(x => x.GetSettings()).Returns(oldSettings);
|
|
||||||
_runnerServer.Setup(x => x.RenewAgentRequestAsync(It.IsAny<int>(), It.IsAny<long>(), It.IsAny<Guid>(), It.IsAny<string>(), It.IsAny<CancellationToken>()))
|
|
||||||
.Returns(() =>
|
|
||||||
{
|
|
||||||
count++;
|
|
||||||
if (count < 5)
|
|
||||||
{
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
else if (count == 5 || count == 6 || count == 7)
|
|
||||||
{
|
|
||||||
throw new TimeoutException("");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
cancellationTokenSource.Cancel();
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var jobDispatcher = new JobDispatcher();
|
|
||||||
jobDispatcher.Initialize(hc);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
await jobDispatcher.RenewJobRequestAsync(0, 0, Guid.Empty, Guid.NewGuid().ToString(), firstJobRequestRenewed, cancellationTokenSource.Token);
|
|
||||||
|
|
||||||
// Assert
|
|
||||||
_configurationStore.Verify(x => x.SaveSettings(It.Is<RunnerSettings>(settings => settings.AgentName == newName)), Times.Once);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Runner")]
|
|
||||||
public async void RenewJobRequestSameAgentNameIgnored()
|
|
||||||
{
|
|
||||||
//Arrange
|
|
||||||
using (var hc = new TestHostContext(this))
|
|
||||||
{
|
|
||||||
var count = 0;
|
|
||||||
var oldName = "OldName";
|
|
||||||
var newName = "OldName";
|
|
||||||
var oldSettings = new RunnerSettings { AgentName = oldName };
|
|
||||||
var reservedAgent = new TaskAgentReference { Name = newName };
|
|
||||||
|
|
||||||
var trace = hc.GetTrace(nameof(DispatcherRenewJobRequestStopOnJobTokenExpiredExceptions));
|
|
||||||
TaskCompletionSource<int> firstJobRequestRenewed = new TaskCompletionSource<int>();
|
|
||||||
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var request = new Mock<TaskAgentJobRequest>();
|
|
||||||
request.Object.ReservedAgent = reservedAgent;
|
|
||||||
PropertyInfo lockUntilProperty = request.Object.GetType().GetProperty("LockedUntil", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);
|
|
||||||
Assert.NotNull(lockUntilProperty);
|
|
||||||
lockUntilProperty.SetValue(request.Object, DateTime.UtcNow.AddMinutes(5));
|
|
||||||
hc.SetSingleton<IRunnerServer>(_runnerServer.Object);
|
|
||||||
hc.SetSingleton<IConfigurationStore>(_configurationStore.Object);
|
|
||||||
_configurationStore.Setup(x => x.GetSettings()).Returns(oldSettings);
|
|
||||||
_runnerServer.Setup(x => x.RenewAgentRequestAsync(It.IsAny<int>(), It.IsAny<long>(), It.IsAny<Guid>(), It.IsAny<string>(), It.IsAny<CancellationToken>()))
|
|
||||||
.Returns(() =>
|
|
||||||
{
|
|
||||||
count++;
|
|
||||||
if (count < 5)
|
|
||||||
{
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
else if (count == 5 || count == 6 || count == 7)
|
|
||||||
{
|
|
||||||
throw new TimeoutException("");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
cancellationTokenSource.Cancel();
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
var jobDispatcher = new JobDispatcher();
|
|
||||||
jobDispatcher.Initialize(hc);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
await jobDispatcher.RenewJobRequestAsync(0, 0, Guid.Empty, Guid.NewGuid().ToString(), firstJobRequestRenewed, cancellationTokenSource.Token);
|
|
||||||
|
|
||||||
// Assert
|
|
||||||
_configurationStore.Verify(x => x.SaveSettings(It.IsAny<RunnerSettings>()), Times.Never);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Runner")]
|
|
||||||
public async void RenewJobRequestNullAgentNameIgnored()
|
|
||||||
{
|
|
||||||
//Arrange
|
|
||||||
using (var hc = new TestHostContext(this))
|
|
||||||
{
|
|
||||||
var count = 0;
|
|
||||||
var oldName = "OldName";
|
|
||||||
var oldSettings = new RunnerSettings { AgentName = oldName };
|
|
||||||
|
|
||||||
var trace = hc.GetTrace(nameof(DispatcherRenewJobRequestStopOnJobTokenExpiredExceptions));
|
|
||||||
TaskCompletionSource<int> firstJobRequestRenewed = new TaskCompletionSource<int>();
|
|
||||||
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
|
|
||||||
|
|
||||||
var request = new Mock<TaskAgentJobRequest>();
|
|
||||||
PropertyInfo lockUntilProperty = request.Object.GetType().GetProperty("LockedUntil", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);
|
|
||||||
Assert.NotNull(lockUntilProperty);
|
|
||||||
lockUntilProperty.SetValue(request.Object, DateTime.UtcNow.AddMinutes(5));
|
|
||||||
hc.SetSingleton<IRunnerServer>(_runnerServer.Object);
|
|
||||||
hc.SetSingleton<IConfigurationStore>(_configurationStore.Object);
|
|
||||||
_configurationStore.Setup(x => x.GetSettings()).Returns(oldSettings);
|
|
||||||
_runnerServer.Setup(x => x.RenewAgentRequestAsync(It.IsAny<int>(), It.IsAny<long>(), It.IsAny<Guid>(), It.IsAny<string>(), It.IsAny<CancellationToken>()))
|
|
||||||
.Returns(() =>
|
|
||||||
{
|
|
||||||
count++;
|
|
||||||
if (count < 5)
|
|
||||||
{
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
else if (count == 5 || count == 6 || count == 7)
|
|
||||||
{
|
|
||||||
throw new TimeoutException("");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
cancellationTokenSource.Cancel();
|
|
||||||
return Task.FromResult<TaskAgentJobRequest>(request.Object);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var jobDispatcher = new JobDispatcher();
|
|
||||||
jobDispatcher.Initialize(hc);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
await jobDispatcher.RenewJobRequestAsync(0, 0, Guid.Empty, Guid.NewGuid().ToString(), firstJobRequestRenewed, cancellationTokenSource.Token);
|
|
||||||
|
|
||||||
// Assert
|
|
||||||
_configurationStore.Verify(x => x.SaveSettings(It.IsAny<RunnerSettings>()), Times.Never);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
[Trait("Level", "L0")]
|
[Trait("Level", "L0")]
|
||||||
[Trait("Category", "Runner")]
|
[Trait("Category", "Runner")]
|
||||||
|
|||||||
@@ -149,9 +149,6 @@ namespace GitHub.Runner.Common.Tests.Listener
|
|||||||
_messageListener.Verify(x => x.CreateSessionAsync(It.IsAny<CancellationToken>()), Times.Once());
|
_messageListener.Verify(x => x.CreateSessionAsync(It.IsAny<CancellationToken>()), Times.Once());
|
||||||
_messageListener.Verify(x => x.DeleteSessionAsync(), Times.Once());
|
_messageListener.Verify(x => x.DeleteSessionAsync(), Times.Once());
|
||||||
_messageListener.Verify(x => x.DeleteMessageAsync(It.IsAny<TaskAgentMessage>()), Times.AtLeastOnce());
|
_messageListener.Verify(x => x.DeleteMessageAsync(It.IsAny<TaskAgentMessage>()), Times.AtLeastOnce());
|
||||||
|
|
||||||
// verify that we didn't try to delete local settings file (since we're not ephemeral)
|
|
||||||
_configurationManager.Verify(x => x.DeleteLocalRunnerConfig(), Times.Never());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -315,9 +312,6 @@ namespace GitHub.Runner.Common.Tests.Listener
|
|||||||
_messageListener.Verify(x => x.CreateSessionAsync(It.IsAny<CancellationToken>()), Times.Once());
|
_messageListener.Verify(x => x.CreateSessionAsync(It.IsAny<CancellationToken>()), Times.Once());
|
||||||
_messageListener.Verify(x => x.DeleteSessionAsync(), Times.Once());
|
_messageListener.Verify(x => x.DeleteSessionAsync(), Times.Once());
|
||||||
_messageListener.Verify(x => x.DeleteMessageAsync(It.IsAny<TaskAgentMessage>()), Times.AtLeastOnce());
|
_messageListener.Verify(x => x.DeleteMessageAsync(It.IsAny<TaskAgentMessage>()), Times.AtLeastOnce());
|
||||||
|
|
||||||
// verify that we did try to delete local settings file (since we're ephemeral)
|
|
||||||
_configurationManager.Verify(x => x.DeleteLocalRunnerConfig(), Times.Once());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using GitHub.DistributedTask.Pipelines.ContextData;
|
|
||||||
using GitHub.DistributedTask.WebApi;
|
using GitHub.DistributedTask.WebApi;
|
||||||
using GitHub.Runner.Worker;
|
using GitHub.Runner.Worker;
|
||||||
using GitHub.Runner.Worker.Container;
|
using GitHub.Runner.Worker.Container;
|
||||||
@@ -84,7 +83,6 @@ namespace GitHub.Runner.Common.Tests.Worker
|
|||||||
{
|
{
|
||||||
using (TestHostContext hc = CreateTestContext())
|
using (TestHostContext hc = CreateTestContext())
|
||||||
{
|
{
|
||||||
_ec.Setup(x => x.ExpressionValues).Returns(GetExpressionValues());
|
|
||||||
_ec.Setup(x => x.Write(It.IsAny<string>(), It.IsAny<string>()))
|
_ec.Setup(x => x.Write(It.IsAny<string>(), It.IsAny<string>()))
|
||||||
.Returns((string tag, string line) =>
|
.Returns((string tag, string line) =>
|
||||||
{
|
{
|
||||||
@@ -107,88 +105,6 @@ namespace GitHub.Runner.Common.Tests.Worker
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
|
||||||
[InlineData("stop-commands", "1")]
|
|
||||||
[InlineData("", "1")]
|
|
||||||
[InlineData("set-env", "1")]
|
|
||||||
[InlineData("stop-commands", "true")]
|
|
||||||
[InlineData("", "true")]
|
|
||||||
[InlineData("set-env", "true")]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Worker")]
|
|
||||||
public void StopProcessCommand__AllowsInvalidStopTokens__IfEnvVarIsSet(string invalidToken, string allowUnsupportedStopCommandTokens)
|
|
||||||
{
|
|
||||||
using (TestHostContext hc = CreateTestContext())
|
|
||||||
{
|
|
||||||
_ec.Object.Global.EnvironmentVariables = new Dictionary<string, string>();
|
|
||||||
var expressionValues = new DictionaryContextData
|
|
||||||
{
|
|
||||||
["env"] =
|
|
||||||
#if OS_WINDOWS
|
|
||||||
new DictionaryContextData{ { Constants.Variables.Actions.AllowUnsupportedStopCommandTokens, new StringContextData(allowUnsupportedStopCommandTokens) }}
|
|
||||||
#else
|
|
||||||
new CaseSensitiveDictionaryContextData{ { Constants.Variables.Actions.AllowUnsupportedStopCommandTokens, new StringContextData(allowUnsupportedStopCommandTokens) }}
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
_ec.Setup(x => x.ExpressionValues).Returns(expressionValues);
|
|
||||||
_ec.Setup(x => x.JobTelemetry).Returns(new List<JobTelemetry>());
|
|
||||||
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, $"::stop-commands::{invalidToken}", null));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory]
|
|
||||||
[InlineData("stop-commands")]
|
|
||||||
[InlineData("")]
|
|
||||||
[InlineData("set-env")]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Worker")]
|
|
||||||
public void StopProcessCommand__FailOnInvalidStopTokens(string invalidToken)
|
|
||||||
{
|
|
||||||
using (TestHostContext hc = CreateTestContext())
|
|
||||||
{
|
|
||||||
_ec.Object.Global.EnvironmentVariables = new Dictionary<string, string>();
|
|
||||||
_ec.Setup(x => x.ExpressionValues).Returns(GetExpressionValues());
|
|
||||||
_ec.Setup(x => x.JobTelemetry).Returns(new List<JobTelemetry>());
|
|
||||||
Assert.Throws<Exception>(() => _commandManager.TryProcessCommand(_ec.Object, $"::stop-commands::{invalidToken}", null));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Worker")]
|
|
||||||
public void StopProcessCommandAcceptsValidToken()
|
|
||||||
{
|
|
||||||
var validToken = "randomToken";
|
|
||||||
using (TestHostContext hc = CreateTestContext())
|
|
||||||
{
|
|
||||||
_ec.Setup(x => x.ExpressionValues).Returns(GetExpressionValues());
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, $"::stop-commands::{validToken}", null));
|
|
||||||
Assert.False(_commandManager.TryProcessCommand(_ec.Object, "##[set-env name=foo]bar", null));
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, $"::{validToken}::", null));
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, "##[set-env name=foo]bar", null));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
[Trait("Level", "L0")]
|
|
||||||
[Trait("Category", "Worker")]
|
|
||||||
public void StopProcessCommandMasksValidTokenForEntireRun()
|
|
||||||
{
|
|
||||||
var validToken = "randomToken";
|
|
||||||
using (TestHostContext hc = CreateTestContext())
|
|
||||||
{
|
|
||||||
_ec.Setup(x => x.ExpressionValues).Returns(GetExpressionValues());
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, $"::stop-commands::{validToken}", null));
|
|
||||||
Assert.False(_commandManager.TryProcessCommand(_ec.Object, "##[set-env name=foo]bar", null));
|
|
||||||
Assert.Equal("***", hc.SecretMasker.MaskSecrets(validToken));
|
|
||||||
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, $"::{validToken}::", null));
|
|
||||||
Assert.True(_commandManager.TryProcessCommand(_ec.Object, "##[set-env name=foo]bar", null));
|
|
||||||
Assert.Equal("***", hc.SecretMasker.MaskSecrets(validToken));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
[Trait("Level", "L0")]
|
[Trait("Level", "L0")]
|
||||||
[Trait("Category", "Worker")]
|
[Trait("Category", "Worker")]
|
||||||
@@ -286,7 +202,7 @@ namespace GitHub.Runner.Common.Tests.Worker
|
|||||||
return 1;
|
return 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
var registeredCommands = new HashSet<string>(new string[1] { "warning" });
|
var registeredCommands = new HashSet<string>(new string[1]{ "warning" });
|
||||||
ActionCommand command;
|
ActionCommand command;
|
||||||
|
|
||||||
// Columns when lines are different
|
// Columns when lines are different
|
||||||
@@ -459,19 +375,5 @@ namespace GitHub.Runner.Common.Tests.Worker
|
|||||||
|
|
||||||
return hostContext;
|
return hostContext;
|
||||||
}
|
}
|
||||||
|
|
||||||
private DictionaryContextData GetExpressionValues()
|
|
||||||
{
|
|
||||||
return new DictionaryContextData
|
|
||||||
{
|
|
||||||
["env"] =
|
|
||||||
#if OS_WINDOWS
|
|
||||||
new DictionaryContextData()
|
|
||||||
#else
|
|
||||||
new CaseSensitiveDictionaryContextData()
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,10 +2,10 @@
|
|||||||
<Project ToolsVersion="14.0" DefaultTargets="Build"
|
<Project ToolsVersion="14.0" DefaultTargets="Build"
|
||||||
xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
<Target Name="GenerateConstant">
|
<Target Name="GenerateConstant">
|
||||||
<!-- <Exec Command="git rev-parse HEAD" ConsoleToMSBuild="true">
|
<Exec Command="git rev-parse HEAD" ConsoleToMSBuild="true">
|
||||||
<Output TaskParameter="ConsoleOutput" PropertyName="GitInfoCommitHash" />
|
<Output TaskParameter="ConsoleOutput" PropertyName="GitInfoCommitHash" />
|
||||||
</Exec> -->
|
</Exec>
|
||||||
<Message Text="Building $(Product): --- $(PackageRuntime)" Importance="high"/>
|
<Message Text="Building $(Product): $(GitInfoCommitHash) --- $(PackageRuntime)" Importance="high"/>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<BuildConstants Include="namespace GitHub.Runner.Sdk"/>
|
<BuildConstants Include="namespace GitHub.Runner.Sdk"/>
|
||||||
@@ -14,7 +14,7 @@
|
|||||||
<BuildConstants Include="%20%20%20%20{"/>
|
<BuildConstants Include="%20%20%20%20{"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20public static class Source"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20public static class Source"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20{"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20{"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20%20%20%20%20public static readonly string CommitHash = %22dfcfae49e59b6dc3c2bb5295c649b33c4b49c964%22%3B"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20%20%20%20%20public static readonly string CommitHash = %22$(GitInfoCommitHash)%22%3B"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20}%0A"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20}%0A"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20public static class RunnerPackage"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20public static class RunnerPackage"/>
|
||||||
<BuildConstants Include="%20%20%20%20%20%20%20%20{"/>
|
<BuildConstants Include="%20%20%20%20%20%20%20%20{"/>
|
||||||
@@ -27,6 +27,7 @@
|
|||||||
|
|
||||||
<WriteLinesToFile File="Runner.Sdk/BuildConstants.cs" Lines="@(BuildConstants)" Overwrite="true" />
|
<WriteLinesToFile File="Runner.Sdk/BuildConstants.cs" Lines="@(BuildConstants)" Overwrite="true" />
|
||||||
|
|
||||||
|
<Exec Command="git update-index --assume-unchanged ./Runner.Sdk/BuildConstants.cs" ConsoleToMSBuild="true" />
|
||||||
</Target>
|
</Target>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
2.283.3
|
2.282.0
|
||||||
|
|||||||
Reference in New Issue
Block a user