gitlab-ci-local 4.65.1 → 4.66.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -2,26 +2,26 @@
2
2
  "type": "module",
3
3
  "name": "gitlab-ci-local",
4
4
  "description": "Tired of pushing to test your .gitlab-ci.yml?",
5
- "main": "src/index.js",
6
- "bin": "src/index.js",
7
- "version": "4.65.1",
5
+ "main": "dist/index.js",
6
+ "bin": "dist/index.js",
7
+ "version": "4.66.0",
8
8
  "scripts": {
9
- "esbuild": "esbuild src/index.ts --outfile=bin/index.cjs --bundle --platform=node --format=cjs --minify --external:yargs --sourcemap=inline",
10
- "pkg-linux": "pkg bin/index.cjs --public --options=enable-source-maps --no-bytecode -t linux-x64 -o bin/linux/gitlab-ci-local && chmod +x bin/linux/gitlab-ci-local && gzip -c bin/linux/gitlab-ci-local > bin/linux.gz",
11
- "pkg-macos": "pkg bin/index.cjs --public --options=enable-source-maps --no-bytecode -t macos-x64 -o bin/macos/gitlab-ci-local && gzip -c bin/macos/gitlab-ci-local > bin/macos.gz",
12
- "pkg-win": "pkg bin/index.cjs --public --options=enable-source-maps --no-bytecode -t win-x64 -o bin/win/gitlab-ci-local && gzip -c bin/win/gitlab-ci-local.exe > bin/win.gz",
13
- "pkg-all": "npm run pkg-linux && npm run pkg-macos && npm run pkg-win && npm ci",
14
- "cleanup": "git clean -fX tests/ && git clean -fX src/",
15
- "build": " npm run cleanup && tsc",
16
- "check-all": "npm run build && npm run lint && npm run coverage && npm run audit",
17
- "audit": "npm audit --parseable",
18
- "lint": "npx eslint .",
19
- "test": "NODE_OPTIONS=\"$NODE_OPTIONS --experimental-vm-modules\" jest --testTimeout 60000",
20
- "test-except-dind": "npm run test -- --testPathIgnorePatterns=\"/dind.*/|/node_modules/|/.gitlab-ci-local/\"",
21
- "coverage": "npm run build && FORCE_COLOR=1 npm run test -- --coverage --verbose",
22
- "start": "tsx src/index.ts --cwd examples/docker-compose-nodejs",
23
- "dev": "touch .gitlab-ci.yml && nodemon -e ts --watch src --watch .gitlab-ci.yml --exec \"tsc --noEmit && tsx src/index.ts\"",
24
- "fetch-and-patch-schema": "curl https://gitlab.com/gitlab-org/gitlab/-/raw/master/app/assets/javascripts/editor/schema/ci.json -sf | jq 'del(.. | .pattern?)' > src/schema/schema.json"
9
+ "build": "bun build --compile --minify --sourcemap src/index.ts --outfile bin/gitlab-ci-local",
10
+ "build:linux-amd64": "bun build --compile --minify --sourcemap --target=bun-linux-x64 src/index.ts --outfile bin/linux-amd64/gitlab-ci-local",
11
+ "build:linux-arm64": "bun build --compile --minify --sourcemap --target=bun-linux-arm64 src/index.ts --outfile bin/linux-arm64/gitlab-ci-local",
12
+ "build:macos-x64": "bun build --compile --minify --sourcemap --target=bun-darwin-x64 src/index.ts --outfile bin/macos-x64/gitlab-ci-local",
13
+ "build:macos-arm64": "bun build --compile --minify --sourcemap --target=bun-darwin-arm64 src/index.ts --outfile bin/macos-arm64/gitlab-ci-local",
14
+ "build:win": "bun build --compile --minify --sourcemap --target=bun-windows-x64 src/index.ts --outfile bin/win/gitlab-ci-local.exe",
15
+ "build:node": "bun build --minify --sourcemap=linked src/index.ts --outdir dist --target=node",
16
+ "build-all": "bun run build:linux-amd64 && bun run build:linux-arm64 && bun run build:win && bun run build:macos-x64 && bun run build:macos-arm64",
17
+ "check-all": "bun run lint && bun run coverage",
18
+ "lint": "eslint .",
19
+ "test": "FORCE_COLOR=1 bun test --timeout 60000",
20
+ "test-except-dind": "bun test --timeout 60000 --filter '(?!.*dind)'",
21
+ "coverage": "FORCE_COLOR=1 bun test --timeout 60000 --coverage --coverage-reporter=text --coverage-reporter=lcov --verbose",
22
+ "start": "bun src/index.ts --cwd examples/docker-compose-nodejs",
23
+ "typecheck": "tsc --noEmit",
24
+ "fetch-schema": "curl https://gitlab.com/gitlab-org/gitlab/-/raw/master/app/assets/javascripts/editor/schema/ci.json -sf > src/schema.json"
25
25
  },
26
26
  "dependencies": {
27
27
  "ajv": "8.x.x",
@@ -46,82 +46,28 @@
46
46
  "semver": "7.x.x",
47
47
  "split2": "4.x.x",
48
48
  "terminal-link": "5.x.x",
49
- "yargs": "17.x.x"
49
+ "yargs": "18.x"
50
50
  },
51
51
  "devDependencies": {
52
- "@eslint/js": "9.x.x",
53
- "@stylistic/eslint-plugin-js": "4.x.x",
54
- "@stylistic/eslint-plugin-ts": "4.x.x",
52
+ "@eslint/js": "10.x",
53
+ "@stylistic/eslint-plugin": "5.x",
55
54
  "@types/base64url": "2.x.x",
55
+ "@types/bun": "1.x.x",
56
56
  "@types/checksum": "0.1.x",
57
57
  "@types/deep-extend": "0.6.x",
58
58
  "@types/fs-extra": "11.x.x",
59
- "@types/jest-when": "3.x.x",
60
59
  "@types/js-yaml": "4.x.x",
61
60
  "@types/micromatch": "4.x.x",
62
- "@types/node": "24.x.x",
63
61
  "@types/pretty-hrtime": "1.x.x",
64
62
  "@types/semver": "7.x.x",
65
63
  "@types/split2": "4.x.x",
66
64
  "@types/yargs": "17.x.x",
67
- "@yao-pkg/pkg": "6.x.x",
68
65
  "axios-mock-adapter": "2.x",
69
66
  "depcheck": "1.x.x",
70
- "esbuild": "0.27.x",
71
- "eslint": "9.x.x",
72
- "jest": "30.x.x",
73
- "jest-when": "3.x.x",
74
- "nodemon": "3.x.x",
75
- "ts-jest": "29.x.x",
76
- "tsx": "4.x.x",
67
+ "eslint": "10.x",
77
68
  "typescript": "5.x.x",
78
69
  "typescript-eslint": "8.x.x"
79
70
  },
80
- "engines": {
81
- "node": ">=18"
82
- },
83
- "pkg": {
84
- "assets": [
85
- "package.json",
86
- "**/src/schema/schema.json"
87
- ],
88
- "scripts": [
89
- "src/**/*.js"
90
- ]
91
- },
92
- "jest": {
93
- "extensionsToTreatAsEsm": [
94
- ".ts"
95
- ],
96
- "transform": {
97
- "^.+\\.tsx?$": [
98
- "ts-jest",
99
- {
100
- "useESM": true
101
- }
102
- ],
103
- "^.+\\.jsx?$": "babel-jest"
104
- },
105
- "preset": "ts-jest",
106
- "testMatch": [
107
- "**/*.test.ts"
108
- ],
109
- "coveragePathIgnorePatterns": [
110
- "<rootDir>/scripts"
111
- ],
112
- "testPathIgnorePatterns": [
113
- "/node_modules/",
114
- "/.gitlab-ci-local/"
115
- ],
116
- "moduleNameMapper": {
117
- "(.+)\\.js": "$1"
118
- },
119
- "coverageReporters": [
120
- "lcov",
121
- "json-summary",
122
- "text-summary"
123
- ]
124
- },
125
71
  "repository": {
126
72
  "type": "git",
127
73
  "url": "https://github.com/firecow/gitlab-ci-local.git"
@@ -145,7 +91,7 @@
145
91
  "gitlab-ci"
146
92
  ],
147
93
  "files": [
148
- "src/**/*.js",
149
- "src/**/*.json"
94
+ "dist/index.js",
95
+ "dist/index.js.map"
150
96
  ]
151
97
  }
package/src/argv.js DELETED
@@ -1,307 +0,0 @@
1
- import assert from "assert";
2
- import fs from "fs-extra";
3
- import * as dotenv from "dotenv";
4
- import * as path from "path";
5
- import camelCase from "camelcase";
6
- import { Utils } from "./utils.js";
7
- import chalkBase from "chalk";
8
- import chalk from "chalk-template";
9
- async function isInGitRepository() {
10
- try {
11
- await Utils.spawn(["git", "rev-parse", "--is-inside-work-tree"]);
12
- return true;
13
- }
14
- catch {
15
- return false;
16
- }
17
- }
18
- async function gitRootPath() {
19
- const { stdout } = await Utils.spawn(["git", "rev-parse", "--show-toplevel"]);
20
- return stdout;
21
- }
22
- export class Argv {
23
- static default = {
24
- "variablesFile": ".gitlab-ci-local-variables.yml",
25
- "evaluateRuleChanges": true,
26
- "ignoreSchemaPaths": [],
27
- "ignorePredefinedVars": "",
28
- };
29
- map = new Map();
30
- writeStreams;
31
- async fallbackCwd(args) {
32
- if (args.cwd !== undefined || args.file !== undefined)
33
- return;
34
- if (fs.existsSync(`${process.cwd()}/.gitlab-ci.yml`))
35
- return;
36
- if (!(await isInGitRepository()))
37
- return;
38
- this.writeStreams?.stderr(chalk `{yellow .gitlab-ci.yml not found in cwd, falling back to git root directory}\n`);
39
- this.map.set("cwd", path.relative(process.cwd(), await gitRootPath()));
40
- }
41
- static async build(args, writeStreams) {
42
- const argv = new Argv(args, writeStreams);
43
- await argv.fallbackCwd(args);
44
- argv.injectDotenv(`${argv.home}/.gitlab-ci-local/.env`, args);
45
- argv.injectDotenv(`${argv.cwd}/.gitlab-ci-local-env`, args);
46
- if (!argv.shellExecutorNoImage && argv.shellIsolation) {
47
- writeStreams?.stderr(chalk `{black.bgYellowBright WARN } --shell-isolation does not work with --no-shell-executor-no-image\n`);
48
- }
49
- if (argv.defaultImageExplicitlySet && argv.shellIsolation) {
50
- writeStreams?.stderr(chalk `{black.bgYellowBright WARN } --default-image does not work with --shell-isolation=true\n`);
51
- }
52
- if (argv.defaultImageExplicitlySet && argv.shellExecutorNoImage) {
53
- writeStreams?.stderr(chalk `{black.bgYellowBright WARN } --default-image does not work with --shell-executor-no-image=true\n`);
54
- }
55
- if (argv.defaultImageExplicitlySet && argv.forceShellExecutor) {
56
- writeStreams?.stderr(chalk `{black.bgYellowBright WARN } --default-image does not work with --force-shell-executor=true\n`);
57
- }
58
- return argv;
59
- }
60
- constructor(argv, writeStreams) {
61
- if (argv.noColor) {
62
- chalkBase.level = 0;
63
- }
64
- this.writeStreams = writeStreams;
65
- for (const [key, value] of Object.entries(argv)) {
66
- this.map.set(key, value);
67
- }
68
- }
69
- injectDotenv(potentialDotenvFilepath, argv) {
70
- if (fs.existsSync(potentialDotenvFilepath)) {
71
- const config = dotenv.parse(fs.readFileSync(potentialDotenvFilepath));
72
- for (const [key, value] of Object.entries(config)) {
73
- const argKey = camelCase(key);
74
- // Special handle KEY=VALUE variable keys
75
- if (argKey === "variable") {
76
- let currentVal = argv[argKey];
77
- if (currentVal == null) {
78
- currentVal = [];
79
- this.map.set(argKey, currentVal);
80
- }
81
- if (!Array.isArray(currentVal)) {
82
- continue;
83
- }
84
- for (const pair of value.split(" ")) {
85
- currentVal.unshift(pair);
86
- }
87
- }
88
- else if (argv[argKey] == null) {
89
- // Work around `dotenv.parse` limitation https://github.com/motdotla/dotenv/issues/51#issuecomment-552559070
90
- if (value === "true")
91
- this.map.set(argKey, true);
92
- else if (value === "false")
93
- this.map.set(argKey, false);
94
- else if (value === "null")
95
- this.map.set(argKey, null);
96
- else if (!isNaN(Number(value)))
97
- this.map.set(argKey, Number(value));
98
- else
99
- this.map.set(argKey, value);
100
- }
101
- }
102
- }
103
- }
104
- get cwd() {
105
- let cwd = this.map.get("cwd") ?? ".";
106
- assert(typeof cwd != "object", "--cwd option cannot be an array");
107
- assert(!path.isAbsolute(cwd), "Please use relative path for the --cwd option");
108
- cwd = path.normalize(`${process.cwd()}/${cwd}`);
109
- cwd = cwd.replace(/\/$/, "");
110
- assert(fs.pathExistsSync(cwd), `${cwd} is not a directory`);
111
- return cwd;
112
- }
113
- get variablesFile() {
114
- return this.map.get("variablesFile") ?? Argv.default.variablesFile;
115
- }
116
- get evaluateRuleChanges() {
117
- return this.map.get("evaluateRuleChanges") ?? Argv.default.evaluateRuleChanges;
118
- }
119
- get file() {
120
- return this.map.get("file") ?? ".gitlab-ci.yml";
121
- }
122
- get stateDir() {
123
- return (this.map.get("stateDir") ?? ".gitlab-ci-local").replace(/\/$/, "");
124
- }
125
- get home() {
126
- return (this.map.get("home") ?? process.env.HOME ?? "").replace(/\/$/, "");
127
- }
128
- get volume() {
129
- const val = this.map.get("volume") ?? [];
130
- return typeof val == "string" ? val.split(" ") : val;
131
- }
132
- get network() {
133
- const val = this.map.get("network") ?? [];
134
- return typeof val == "string" ? val.split(" ") : val;
135
- }
136
- get extraHost() {
137
- const val = this.map.get("extraHost") ?? [];
138
- return typeof val == "string" ? val.split(" ") : val;
139
- }
140
- get caFile() {
141
- return this.map.get("caFile") ?? null;
142
- }
143
- get ignoreSchemaPaths() {
144
- return this.map.get("ignoreSchemaPaths") ?? Argv.default.ignoreSchemaPaths;
145
- }
146
- get ignorePredefinedVars() {
147
- return this.map.get("ignorePredefinedVars") ?? Argv.default.ignorePredefinedVars;
148
- }
149
- get pullPolicy() {
150
- return this.map.get("pullPolicy") ?? "if-not-present";
151
- }
152
- get remoteVariables() {
153
- const val = this.map.get("remoteVariables") ?? [];
154
- return typeof val == "string" ? val.split(" ") : val;
155
- }
156
- get variable() {
157
- const val = this.map.get("variable");
158
- const variables = {};
159
- const pairs = typeof val == "string" ? val.split(" ") : val;
160
- (pairs ?? []).forEach((variablePair) => {
161
- const exec = /(?<key>\w*?)(=)(?<value>(.|\n|\r)*)/.exec(variablePair);
162
- if (exec?.groups?.key) {
163
- variables[exec.groups.key] = exec?.groups?.value;
164
- }
165
- });
166
- return variables;
167
- }
168
- get unsetVariables() {
169
- return this.map.get("unsetVariable") ?? [];
170
- }
171
- get manual() {
172
- const val = this.map.get("manual") ?? [];
173
- return typeof val == "string" ? val.split(" ") : val;
174
- }
175
- get job() {
176
- return this.map.get("job") ?? [];
177
- }
178
- get autoCompleting() {
179
- return this.map.get("autoCompleting") ?? false;
180
- }
181
- get cleanup() {
182
- return this.map.get("cleanup") ?? true;
183
- }
184
- get quiet() {
185
- return this.map.get("quiet") ?? false;
186
- }
187
- get umask() {
188
- // TODO: default to false in 5.x.x
189
- return this.map.get("umask") ?? true;
190
- }
191
- get userns() {
192
- return this.map.get("userns");
193
- }
194
- get privileged() {
195
- return this.map.get("privileged") ?? false;
196
- }
197
- get device() {
198
- const val = this.map.get("device") ?? [];
199
- return typeof val == "string" ? val.split(" ") : val;
200
- }
201
- get ulimit() {
202
- const ulimit = this.map.get("ulimit");
203
- if (!ulimit)
204
- return null;
205
- return ulimit;
206
- }
207
- get needs() {
208
- return this.map.get("needs") ?? false;
209
- }
210
- get onlyNeeds() {
211
- return this.map.get("onlyNeeds") ?? false;
212
- }
213
- get stage() {
214
- return this.map.get("stage") ?? null;
215
- }
216
- get completion() {
217
- return this.map.get("completion") ?? false;
218
- }
219
- get list() {
220
- return this.map.get("list") ?? false;
221
- }
222
- get listAll() {
223
- return this.map.get("listAll") ?? false;
224
- }
225
- get listJson() {
226
- return this.map.get("listJson") ?? false;
227
- }
228
- get listCsv() {
229
- return this.map.get("listCsv") ?? false;
230
- }
231
- get listCsvAll() {
232
- return this.map.get("listCsvAll") ?? false;
233
- }
234
- get preview() {
235
- return this.map.get("preview") ?? false;
236
- }
237
- get validateDependencyChain() {
238
- return this.map.get("validateDependencyChain") ?? false;
239
- }
240
- get shellIsolation() {
241
- // TODO: default to true in 5.x.x
242
- return this.map.get("shellIsolation") ?? false;
243
- }
244
- get fetchIncludes() {
245
- return this.map.get("fetchIncludes") ?? false;
246
- }
247
- get mountCache() {
248
- return this.map.get("mountCache") ?? false;
249
- }
250
- get artifactsToSource() {
251
- // TODO: default to false in 5.x.x
252
- return this.map.get("artifactsToSource") ?? true;
253
- }
254
- get showTimestamps() {
255
- return this.map.get("timestamps") ?? false;
256
- }
257
- get maxJobNamePadding() {
258
- return this.map.get("maxJobNamePadding") ?? null;
259
- }
260
- get containerMacAddress() {
261
- return this.map.get("containerMacAddress") ?? null;
262
- }
263
- get containerEmulate() {
264
- return this.map.get("containerEmulate") ?? null;
265
- }
266
- get concurrency() {
267
- const concurrency = this.map.get("concurrency");
268
- if (!concurrency)
269
- return null;
270
- return Number(concurrency);
271
- }
272
- get containerExecutable() {
273
- return this.map.get("containerExecutable") ?? "docker";
274
- }
275
- get jsonSchemaValidation() {
276
- return this.map.get("jsonSchemaValidation") ?? true;
277
- }
278
- get shellExecutorNoImage() {
279
- // TODO: default to false in 5.x.x
280
- return this.map.get("shellExecutorNoImage") ?? true;
281
- }
282
- get forceShellExecutor() {
283
- return this.map.get("forceShellExecutor") ?? false;
284
- }
285
- get defaultImage() {
286
- return this.map.get("defaultImage") ?? "docker.io/ruby:3.1";
287
- }
288
- get waitImage() {
289
- return this.map.get("waitImage") ?? "docker.io/sumina46/wait-for-it:latest";
290
- }
291
- get helperImage() {
292
- return this.map.get("helperImage") ?? "docker.io/firecow/gitlab-ci-local-util:latest";
293
- }
294
- get defaultImageExplicitlySet() {
295
- return this.map.get("defaultImage") ?? false;
296
- }
297
- get maximumIncludes() {
298
- return this.map.get("maximumIncludes") ?? 150; // https://docs.gitlab.com/ee/administration/settings/continuous_integration.html#maximum-includes
299
- }
300
- get childPipelineDepth() {
301
- return this.map.get("childPipelineDepth");
302
- }
303
- get registry() {
304
- return this.map.get("registry") ?? false;
305
- }
306
- }
307
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"argv.js","sourceRoot":"","sources":["argv.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from \"assert\";\nimport fs from \"fs-extra\";\nimport * as dotenv from \"dotenv\";\nimport * as path from \"path\";\nimport camelCase from \"camelcase\";\nimport {Utils} from \"./utils.js\";\nimport {WriteStreams} from \"./write-streams.js\";\nimport chalkBase from \"chalk\";\nimport chalk from \"chalk-template\";\n\nasync function isInGitRepository () {\n    try {\n        await Utils.spawn([\"git\", \"rev-parse\", \"--is-inside-work-tree\"]);\n        return true;\n    } catch {\n        return false;\n    }\n}\n\nasync function gitRootPath () {\n    const {stdout} = await Utils.spawn([\"git\", \"rev-parse\", \"--show-toplevel\"]);\n    return stdout;\n}\n\nexport class Argv {\n    static readonly default = {\n        \"variablesFile\": \".gitlab-ci-local-variables.yml\",\n        \"evaluateRuleChanges\": true,\n        \"ignoreSchemaPaths\": [],\n        \"ignorePredefinedVars\": \"\",\n    };\n\n    map: Map<string, any> = new Map<string, any>();\n    private readonly writeStreams: WriteStreams | undefined;\n\n    private async fallbackCwd (args: any) {\n        if (args.cwd !== undefined || args.file !== undefined) return;\n        if (fs.existsSync(`${process.cwd()}/.gitlab-ci.yml`)) return;\n        if (!(await isInGitRepository())) return;\n\n        this.writeStreams?.stderr(chalk`{yellow .gitlab-ci.yml not found in cwd, falling back to git root directory}\\n`);\n        this.map.set(\"cwd\", path.relative(process.cwd(), await gitRootPath()));\n    }\n\n    static async build (args: any, writeStreams?: WriteStreams) {\n        const argv = new Argv(args, writeStreams);\n        await argv.fallbackCwd(args);\n\n        argv.injectDotenv(`${argv.home}/.gitlab-ci-local/.env`, args);\n        argv.injectDotenv(`${argv.cwd}/.gitlab-ci-local-env`, args);\n\n        if (!argv.shellExecutorNoImage && argv.shellIsolation) {\n            writeStreams?.stderr(chalk`{black.bgYellowBright  WARN } --shell-isolation does not work with --no-shell-executor-no-image\\n`);\n        }\n\n        if (argv.defaultImageExplicitlySet && argv.shellIsolation) {\n            writeStreams?.stderr(chalk`{black.bgYellowBright  WARN } --default-image does not work with --shell-isolation=true\\n`);\n        }\n\n        if (argv.defaultImageExplicitlySet && argv.shellExecutorNoImage) {\n            writeStreams?.stderr(chalk`{black.bgYellowBright  WARN } --default-image does not work with --shell-executor-no-image=true\\n`);\n        }\n\n        if (argv.defaultImageExplicitlySet && argv.forceShellExecutor) {\n            writeStreams?.stderr(chalk`{black.bgYellowBright  WARN } --default-image does not work with --force-shell-executor=true\\n`);\n        }\n\n        return argv;\n    }\n\n    private constructor (argv: any, writeStreams?: WriteStreams) {\n        if (argv.noColor) {\n            chalkBase.level = 0;\n        }\n        this.writeStreams = writeStreams;\n        for (const [key, value] of Object.entries(argv)) {\n            this.map.set(key, value);\n        }\n    }\n\n    private injectDotenv (potentialDotenvFilepath: string, argv: any) {\n        if (fs.existsSync(potentialDotenvFilepath)) {\n            const config = dotenv.parse(fs.readFileSync(potentialDotenvFilepath));\n            for (const [key, value] of Object.entries(config)) {\n                const argKey = camelCase(key);\n\n                // Special handle KEY=VALUE variable keys\n                if (argKey === \"variable\") {\n                    let currentVal = argv[argKey];\n                    if (currentVal == null) {\n                        currentVal = [];\n                        this.map.set(argKey, currentVal);\n                    }\n                    if (!Array.isArray(currentVal)) {\n                        continue;\n                    }\n                    for (const pair of value.split(\" \")) {\n                        currentVal.unshift(pair);\n                    }\n                } else if (argv[argKey] == null) {\n                    // Work around `dotenv.parse` limitation https://github.com/motdotla/dotenv/issues/51#issuecomment-552559070\n                    if (value === \"true\") this.map.set(argKey, true);\n                    else if (value === \"false\") this.map.set(argKey, false);\n                    else if (value === \"null\") this.map.set(argKey, null);\n                    else if (!isNaN(Number(value))) this.map.set(argKey, Number(value));\n                    else this.map.set(argKey, value);\n                }\n            }\n        }\n    }\n\n    get cwd (): string {\n        let cwd = this.map.get(\"cwd\") ?? \".\";\n        assert(typeof cwd != \"object\", \"--cwd option cannot be an array\");\n        assert(!path.isAbsolute(cwd), \"Please use relative path for the --cwd option\");\n        cwd = path.normalize(`${process.cwd()}/${cwd}`);\n        cwd = cwd.replace(/\\/$/, \"\");\n        assert(fs.pathExistsSync(cwd), `${cwd} is not a directory`);\n        return cwd;\n    }\n\n    get variablesFile (): string {\n        return this.map.get(\"variablesFile\") ?? Argv.default.variablesFile;\n    }\n\n    get evaluateRuleChanges (): boolean {\n        return this.map.get(\"evaluateRuleChanges\") ?? Argv.default.evaluateRuleChanges;\n    }\n\n    get file (): string {\n        return this.map.get(\"file\") ?? \".gitlab-ci.yml\";\n    }\n\n    get stateDir (): string {\n        return (this.map.get(\"stateDir\") ?? \".gitlab-ci-local\").replace(/\\/$/, \"\");\n    }\n\n    get home (): string {\n        return (this.map.get(\"home\") ?? process.env.HOME ?? \"\").replace(/\\/$/, \"\");\n    }\n\n    get volume (): string[] {\n        const val = this.map.get(\"volume\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get network (): string[] {\n        const val = this.map.get(\"network\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get extraHost (): string[] {\n        const val = this.map.get(\"extraHost\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get caFile (): string | null {\n        return this.map.get(\"caFile\") ?? null;\n    }\n\n    get ignoreSchemaPaths (): string[] {\n        return this.map.get(\"ignoreSchemaPaths\") ?? Argv.default.ignoreSchemaPaths;\n    }\n\n    get ignorePredefinedVars (): string[] {\n        return this.map.get(\"ignorePredefinedVars\") ?? Argv.default.ignorePredefinedVars;\n    }\n\n    get pullPolicy (): string {\n        return this.map.get(\"pullPolicy\") ?? \"if-not-present\";\n    }\n\n    get remoteVariables (): string[] {\n        const val = this.map.get(\"remoteVariables\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get variable (): {[key: string]: string} {\n        const val = this.map.get(\"variable\");\n        const variables: {[key: string]: string} = {};\n        const pairs = typeof val == \"string\" ? val.split(\" \") : val;\n        (pairs ?? []).forEach((variablePair: string) => {\n            const exec = /(?<key>\\w*?)(=)(?<value>(.|\\n|\\r)*)/.exec(variablePair);\n            if (exec?.groups?.key) {\n                variables[exec.groups.key] = exec?.groups?.value;\n            }\n        });\n        return variables;\n    }\n\n    get unsetVariables (): string[] {\n        return this.map.get(\"unsetVariable\") ?? [];\n    }\n\n    get manual (): string[] {\n        const val = this.map.get(\"manual\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get job (): string[] {\n        return this.map.get(\"job\") ?? [];\n    }\n\n    get autoCompleting (): boolean {\n        return this.map.get(\"autoCompleting\") ?? false;\n    }\n\n    get cleanup (): boolean {\n        return this.map.get(\"cleanup\") ?? true;\n    }\n\n    get quiet (): boolean {\n        return this.map.get(\"quiet\") ?? false;\n    }\n\n    get umask (): boolean {\n        // TODO: default to false in 5.x.x\n        return this.map.get(\"umask\") ?? true;\n    }\n\n    get userns (): string | undefined {\n        return this.map.get(\"userns\");\n    }\n\n    get privileged (): boolean {\n        return this.map.get(\"privileged\") ?? false;\n    }\n\n    get device (): string[] {\n        const val = this.map.get(\"device\") ?? [];\n        return typeof val == \"string\" ? val.split(\" \") : val;\n    }\n\n    get ulimit (): string | null {\n        const ulimit = this.map.get(\"ulimit\");\n        if (!ulimit) return null;\n        return ulimit;\n    }\n\n    get needs (): boolean {\n        return this.map.get(\"needs\") ?? false;\n    }\n\n    get onlyNeeds (): boolean {\n        return this.map.get(\"onlyNeeds\") ?? false;\n    }\n\n    get stage (): string | null {\n        return this.map.get(\"stage\") ?? null;\n    }\n\n    get completion (): boolean {\n        return this.map.get(\"completion\") ?? false;\n    }\n\n    get list (): boolean {\n        return this.map.get(\"list\") ?? false;\n    }\n\n    get listAll (): boolean {\n        return this.map.get(\"listAll\") ?? false;\n    }\n\n    get listJson (): boolean {\n        return this.map.get(\"listJson\") ?? false;\n    }\n\n    get listCsv (): boolean {\n        return this.map.get(\"listCsv\") ?? false;\n    }\n\n    get listCsvAll (): boolean {\n        return this.map.get(\"listCsvAll\") ?? false;\n    }\n\n    get preview (): boolean {\n        return this.map.get(\"preview\") ?? false;\n    }\n\n    get validateDependencyChain (): boolean {\n        return this.map.get(\"validateDependencyChain\") ?? false;\n    }\n\n    get shellIsolation (): boolean {\n        // TODO: default to true in 5.x.x\n        return this.map.get(\"shellIsolation\") ?? false;\n    }\n\n    get fetchIncludes (): boolean {\n        return this.map.get(\"fetchIncludes\") ?? false;\n    }\n\n    get mountCache (): boolean {\n        return this.map.get(\"mountCache\") ?? false;\n    }\n\n    get artifactsToSource (): boolean {\n        // TODO: default to false in 5.x.x\n        return this.map.get(\"artifactsToSource\") ?? true;\n    }\n\n    get showTimestamps (): boolean {\n        return this.map.get(\"timestamps\") ?? false;\n    }\n\n    get maxJobNamePadding (): number | null {\n        return this.map.get(\"maxJobNamePadding\") ?? null;\n    }\n\n    get containerMacAddress (): string | null {\n        return this.map.get(\"containerMacAddress\") ?? null;\n    }\n\n    get containerEmulate (): string | null {\n        return this.map.get(\"containerEmulate\") ?? null;\n    }\n\n    get concurrency (): number | null {\n        const concurrency = this.map.get(\"concurrency\");\n        if (!concurrency) return null;\n        return Number(concurrency);\n    }\n\n    get containerExecutable (): string {\n        return this.map.get(\"containerExecutable\") ?? \"docker\";\n    }\n\n    get jsonSchemaValidation (): boolean {\n        return this.map.get(\"jsonSchemaValidation\") ?? true;\n    }\n\n    get shellExecutorNoImage (): boolean {\n        // TODO: default to false in 5.x.x\n        return this.map.get(\"shellExecutorNoImage\") ?? true;\n    }\n\n    get forceShellExecutor (): boolean {\n        return this.map.get(\"forceShellExecutor\") ?? false;\n    }\n\n    get defaultImage (): string {\n        return this.map.get(\"defaultImage\") ?? \"docker.io/ruby:3.1\";\n    }\n\n    get waitImage (): string {\n        return this.map.get(\"waitImage\") ?? \"docker.io/sumina46/wait-for-it:latest\";\n    }\n\n    get helperImage (): string {\n        return this.map.get(\"helperImage\") ?? \"docker.io/firecow/gitlab-ci-local-util:latest\";\n    }\n\n    get defaultImageExplicitlySet (): boolean {\n        return this.map.get(\"defaultImage\") ?? false;\n    }\n\n    get maximumIncludes (): number {\n        return this.map.get(\"maximumIncludes\") ?? 150; // https://docs.gitlab.com/ee/administration/settings/continuous_integration.html#maximum-includes\n    }\n\n    get childPipelineDepth (): number {\n        return this.map.get(\"childPipelineDepth\");\n    }\n\n    get registry (): boolean {\n        return this.map.get(\"registry\") ?? false;\n    }\n}\n"]}