gitlab-ci-local 4.26.2 → 4.27.1

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/src/job.js CHANGED
@@ -32,6 +32,8 @@ const camelcase_1 = __importDefault(require("camelcase"));
32
32
  const exit_error_1 = require("./types/exit-error");
33
33
  const utils_1 = require("./utils");
34
34
  const asserts_1 = require("./asserts");
35
+ const cache_entry_1 = require("./cache-entry");
36
+ const mutex_1 = require("./mutex");
35
37
  class Job {
36
38
  constructor(opt) {
37
39
  this._prescriptsExitCode = null;
@@ -39,17 +41,17 @@ class Job {
39
41
  this._coveragePercent = null;
40
42
  this._running = false;
41
43
  this._containerId = null;
42
- this._serviceIds = [];
43
44
  this._serviceNetworkId = null;
44
- this._artifactsContainerId = null;
45
45
  this._containerVolumeNames = [];
46
46
  this._longRunningSilentTimeout = -1;
47
47
  this._producers = null;
48
+ this._containersToClean = [];
48
49
  const jobData = opt.data;
49
50
  const gitData = opt.gitData;
50
51
  const globals = opt.globals;
51
- const homeVariables = opt.homeVariables;
52
- const projectVariables = opt.projectVariables;
52
+ const variablesFromFiles = opt.variablesFromFiles;
53
+ const cliVariables = opt.cliVariables;
54
+ this.mountCache = opt.mountCache;
53
55
  this.extraHosts = opt.extraHosts;
54
56
  this.volumes = opt.volumes;
55
57
  this.writeStreams = opt.writeStreams;
@@ -63,11 +65,9 @@ class Job {
63
65
  this.shellIsolation = opt.shellIsolation;
64
66
  this.when = jobData.when || "on_success";
65
67
  this.allowFailure = jobData.allow_failure ?? false;
66
- this._needs = jobData.needs || null;
67
68
  this.dependencies = jobData.dependencies || null;
68
69
  this.rules = jobData.rules || null;
69
70
  this.environment = typeof jobData.environment === "string" ? { name: jobData.environment } : jobData.environment;
70
- this.cache = jobData.cache || [];
71
71
  let CI_PROJECT_DIR = `${this.cwd}`;
72
72
  if (this.imageName) {
73
73
  CI_PROJECT_DIR = `/builds/${this.safeJobName}`;
@@ -101,12 +101,14 @@ class Job {
101
101
  CI_JOB_ID: `${this.jobId}`,
102
102
  CI_PIPELINE_ID: `${this.pipelineIid + 1000}`,
103
103
  CI_PIPELINE_IID: `${this.pipelineIid}`,
104
- CI_SERVER_HOST: `${gitData.remote.domain}`,
105
- CI_SERVER_URL: `https://${gitData.remote.domain}:443`,
106
- CI_API_V4_URL: `https://${gitData.remote.domain}/api/v4`,
107
- CI_PROJECT_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}`,
108
- CI_JOB_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}/-/jobs/${this.jobId}`,
109
- CI_PIPELINE_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}/pipelines/${this.pipelineIid}`,
104
+ CI_SERVER_HOST: `${gitData.remote.host}`,
105
+ CI_SERVER_PORT: `${gitData.remote.port}`,
106
+ CI_SERVER_URL: `https://${gitData.remote.host}:443`,
107
+ CI_SERVER_PROTOCOL: "https",
108
+ CI_API_V4_URL: `https://${gitData.remote.host}/api/v4`,
109
+ CI_PROJECT_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}`,
110
+ CI_JOB_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}/-/jobs/${this.jobId}`,
111
+ CI_PIPELINE_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}/pipelines/${this.pipelineIid}`,
110
112
  CI_JOB_NAME: `${this.name}`,
111
113
  CI_JOB_STAGE: `${this.stage}`,
112
114
  CI_REGISTRY: gitData.CI_REGISTRY,
@@ -114,10 +116,10 @@ class Job {
114
116
  GITLAB_CI: "false",
115
117
  };
116
118
  // Create expanded variables
117
- const envs = { ...predefinedVariables, ...globals.variables || {}, ...jobData.variables || {}, ...homeVariables, ...projectVariables };
119
+ const envs = { ...predefinedVariables, ...globals.variables || {}, ...jobData.variables || {}, ...variablesFromFiles, ...cliVariables };
118
120
  const expandedGlobalVariables = utils_1.Utils.expandVariables(globals.variables || {}, envs);
119
121
  const expandedJobVariables = utils_1.Utils.expandVariables(jobData.variables || {}, envs);
120
- this.expandedVariables = { ...predefinedVariables, ...expandedGlobalVariables, ...expandedJobVariables, ...homeVariables, ...projectVariables };
122
+ this.expandedVariables = { ...predefinedVariables, ...expandedGlobalVariables, ...expandedJobVariables, ...variablesFromFiles, ...cliVariables };
121
123
  // Set {when, allowFailure} based on rules result
122
124
  if (this.rules) {
123
125
  const ruleResult = utils_1.Utils.getRulesResult(this.rules, this.expandedVariables);
@@ -130,14 +132,16 @@ class Job {
130
132
  if (this.injectSSHAgent && this.imageName === null) {
131
133
  throw new exit_error_1.ExitError(`${this.chalkJobName} @InjectSSHAgent can only be used with image:`);
132
134
  }
133
- }
134
- static async getUniqueCacheName(cwd, cacheEntry, expandedVariables) {
135
- if (typeof cacheEntry.key === "string") {
136
- return utils_1.Utils.expandText(cacheEntry.key);
135
+ if (this.imageName && this.mountCache) {
136
+ for (const c of this.cache) {
137
+ c.paths.forEach((p) => {
138
+ const path = utils_1.Utils.expandText(p, this.expandedVariables);
139
+ if (path.includes("*")) {
140
+ throw new exit_error_1.ExitError(`${this.name} cannot have * in cache paths, when --mount-cache is enabled`);
141
+ }
142
+ });
143
+ }
137
144
  }
138
- return "md-" + await utils_1.Utils.checksumFiles(cacheEntry.key.files.map(f => {
139
- return `${cwd}/${utils_1.Utils.expandText(f, expandedVariables)}`;
140
- }));
141
145
  }
142
146
  get artifactsToSource() {
143
147
  return this.jobData["artifactsToSource"] == null ? true : this.jobData["artifactsToSource"];
@@ -149,10 +153,11 @@ class Job {
149
153
  return utils_1.Utils.getSafeJobName(this.name);
150
154
  }
151
155
  get needs() {
152
- if (!this._needs)
156
+ const needs = this.jobData["needs"];
157
+ if (!needs)
153
158
  return null;
154
159
  const list = [];
155
- this._needs?.forEach((need) => {
160
+ needs.forEach((need) => {
156
161
  list.push({
157
162
  job: typeof need === "string" ? need : need.job,
158
163
  artifacts: typeof need === "string" ? true : need.artifacts,
@@ -160,6 +165,29 @@ class Job {
160
165
  });
161
166
  return list;
162
167
  }
168
+ get cache() {
169
+ let cacheData = this.jobData["cache"];
170
+ const cacheList = [];
171
+ if (!cacheData)
172
+ return [];
173
+ cacheData = Array.isArray(cacheData) ? cacheData : [cacheData];
174
+ cacheData.forEach((c) => {
175
+ const key = c["key"];
176
+ const policy = c["policy"] ?? "pull-push";
177
+ if (!["pull", "push", "pull-push"].includes(policy)) {
178
+ throw new exit_error_1.ExitError("cache policy is not 'pull', 'push' or 'pull-push'");
179
+ }
180
+ const paths = c["paths"] ?? [];
181
+ cacheList.push(new cache_entry_1.CacheEntry(key, paths, policy));
182
+ });
183
+ return cacheList;
184
+ }
185
+ get buildVolumeName() {
186
+ return `gcl-${this.safeJobName}-${this.jobId}-build`;
187
+ }
188
+ get tmpVolumeName() {
189
+ return `gcl-${this.safeJobName}-${this.jobId}-tmp`;
190
+ }
163
191
  get imageName() {
164
192
  const image = this.jobData["image"];
165
193
  if (!image) {
@@ -276,20 +304,9 @@ class Job {
276
304
  async cleanupResources() {
277
305
  const writeStreams = this.writeStreams;
278
306
  clearTimeout(this._longRunningSilentTimeout);
279
- if (this._containerId) {
307
+ for (const id of this._containersToClean) {
280
308
  try {
281
- await utils_1.Utils.spawn(`docker rm -f ${this._containerId}`);
282
- }
283
- catch (e) {
284
- (0, asserts_1.assert)(e instanceof Error, "e is not instanceof Error");
285
- writeStreams.stderr((0, chalk_1.default) `{yellow ${e.message}}`);
286
- }
287
- }
288
- if (this._serviceIds) {
289
- try {
290
- for (const serviceId of this._serviceIds) {
291
- await utils_1.Utils.spawn(`docker rm -f ${serviceId}`);
292
- }
309
+ await utils_1.Utils.spawn(`docker rm -f ${id}`);
293
310
  }
294
311
  catch (e) {
295
312
  (0, asserts_1.assert)(e instanceof Error, "e is not instanceof Error");
@@ -305,15 +322,6 @@ class Job {
305
322
  writeStreams.stderr((0, chalk_1.default) `{yellow ${e.message}}`);
306
323
  }
307
324
  }
308
- if (this._artifactsContainerId) {
309
- try {
310
- await utils_1.Utils.spawn(`docker rm -f ${this._artifactsContainerId}`);
311
- }
312
- catch (e) {
313
- (0, asserts_1.assert)(e instanceof Error, "e is not instanceof Error");
314
- writeStreams.stderr((0, chalk_1.default) `{yellow ${e.message}}`);
315
- }
316
- }
317
325
  if (this._containerVolumeNames.length > 0) {
318
326
  try {
319
327
  for (const containerVolume of this._containerVolumeNames) {
@@ -348,11 +356,26 @@ class Job {
348
356
  });
349
357
  return cmd;
350
358
  }
359
+ async mountCacheCmd(safeJobName, writeStreams) {
360
+ if (this.imageName && !this.mountCache)
361
+ return "";
362
+ let cmd = "";
363
+ for (const c of this.cache) {
364
+ const uniqueCacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);
365
+ c.paths.forEach((p) => {
366
+ const path = utils_1.Utils.expandText(p, this.expandedVariables);
367
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright mounting cache} for path ${path}\n`);
368
+ const cacheMount = `gcl-${this.expandedVariables.CI_PROJECT_PATH_SLUG}-${uniqueCacheName}`;
369
+ cmd += `-v ${cacheMount}:/builds/${safeJobName}/${path} `;
370
+ });
371
+ }
372
+ return cmd;
373
+ }
351
374
  async execScripts(scripts, privileged) {
352
375
  const safeJobName = this.safeJobName;
353
376
  const outputFilesPath = `${this.cwd}/.gitlab-ci-local/output/${safeJobName}.log`;
354
- const buildVolumeName = `gcl-${this.safeJobName}-${this.jobId}-build`;
355
- const tmpVolumeName = `gcl-${this.safeJobName}-${this.jobId}-tmp`;
377
+ const buildVolumeName = this.buildVolumeName;
378
+ const tmpVolumeName = this.tmpVolumeName;
356
379
  const writeStreams = this.writeStreams;
357
380
  const reportsDotenvVariables = await this.initProducerReportsDotenvVariables(writeStreams);
358
381
  let time;
@@ -393,6 +416,7 @@ class Job {
393
416
  for (const service of this.services) {
394
417
  await this.pullImage(writeStreams, service.getName(this.expandedVariables));
395
418
  await this.startService(writeStreams, service, privileged);
419
+ await this.serviceHealthCheck(writeStreams, service);
396
420
  }
397
421
  }
398
422
  const volumePromises = [];
@@ -421,7 +445,7 @@ class Job {
421
445
  for (const [key, value] of Object.entries(reportsDotenvVariables)) {
422
446
  dockerCmd += `-e ${key}='${String(value).trim()}' `;
423
447
  }
424
- dockerCmd += await this.createCacheDockerVolumeMounts(safeJobName, writeStreams);
448
+ dockerCmd += await this.mountCacheCmd(safeJobName, writeStreams);
425
449
  dockerCmd += `${this.imageName} sh -c "\n`;
426
450
  dockerCmd += "if [ -x /usr/local/bin/bash ]; then\n";
427
451
  dockerCmd += "\texec /usr/local/bin/bash \n";
@@ -443,6 +467,7 @@ class Job {
443
467
  dockerCmd += "fi\n\"";
444
468
  const { stdout: containerId } = await utils_1.Utils.spawn(dockerCmd, this.cwd);
445
469
  this._containerId = containerId.replace(/\r?\n/g, "");
470
+ this._containersToClean.push(this._containerId);
446
471
  time = process.hrtime();
447
472
  // Copy source files into container.
448
473
  await utils_1.Utils.spawn(`docker cp .gitlab-ci-local/builds/.docker/. ${this._containerId}:/builds/${safeJobName}`, this.cwd);
@@ -456,10 +481,11 @@ class Job {
456
481
  endTime = process.hrtime(time);
457
482
  writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright copied to container} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
458
483
  }
484
+ await this.copyCacheIn(writeStreams);
459
485
  await this.copyArtifactsIn(writeStreams);
460
486
  if (this.imageName) {
461
- // Make sure tracked files and artifacts are root owned in docker-executor jobs.
462
- await utils_1.Utils.spawn(`docker run --rm -w /app/ -v ${buildVolumeName}:/app/ firecow/gitlab-ci-local-util bash -c "chown 0:0 -R . && chmod a+rw -R ."`);
487
+ // Files in docker-executor build folder must be root owned.
488
+ await utils_1.Utils.spawn(`docker run --rm -v ${tmpVolumeName}:/tmp/ -v ${buildVolumeName}:/app/ firecow/gitlab-ci-local-util bash -c "chown 0:0 -R /app/ && chmod a+rw -R /app/ && chmod a+rw -R /tmp/"`);
463
489
  }
464
490
  let cmd = "set -eo pipefail\n";
465
491
  cmd += "exec 0< /dev/null\n";
@@ -505,7 +531,10 @@ class Job {
505
531
  cp.stdin.end(`./.gitlab-ci-local/scripts/${safeJobName}`);
506
532
  }
507
533
  });
508
- await this.copyArtifactsOut(writeStreams, buildVolumeName);
534
+ if (exitCode == 0) {
535
+ await this.copyCacheOut(writeStreams);
536
+ await this.copyArtifactsOut(writeStreams);
537
+ }
509
538
  return exitCode;
510
539
  }
511
540
  async pullImage(writeStreams, imageToPull) {
@@ -545,78 +574,112 @@ class Job {
545
574
  }
546
575
  return producerReportsEnvs;
547
576
  }
548
- async copyArtifactsIn(writeStreams) {
549
- if (!this.imageName && !this.shellIsolation) {
577
+ async copyCacheIn(writeStreams) {
578
+ if (this.mountCache && this.imageName)
550
579
  return;
551
- }
552
- if (!this.producers || this.producers.length === 0) {
580
+ if ((!this.imageName && !this.shellIsolation) || this.cache.length === 0)
553
581
  return;
554
- }
555
- const safeJobName = this.safeJobName;
556
- const cpFunc = async (folder) => {
557
- if (!this.imageName && this.shellIsolation) {
558
- return utils_1.Utils.spawn(`rsync -a ${folder}/. ${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);
582
+ for (const c of this.cache) {
583
+ if (!["pull", "pull-push"].includes(c.policy))
584
+ return;
585
+ const time = process.hrtime();
586
+ const cacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);
587
+ const cacheFolder = `${this.cwd}/.gitlab-ci-local/cache/${cacheName}`;
588
+ if (!await fs.pathExists(cacheFolder)) {
589
+ continue;
559
590
  }
560
- return utils_1.Utils.spawn(`docker cp ${folder}/. ${this._containerId}:/builds/${safeJobName}`);
561
- };
591
+ await mutex_1.Mutex.exclusive(cacheName, async () => {
592
+ await this.copyIn(cacheFolder);
593
+ });
594
+ const endTime = process.hrtime(time);
595
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright imported cache '${cacheName}'} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
596
+ }
597
+ }
598
+ async copyArtifactsIn(writeStreams) {
599
+ if ((!this.imageName && !this.shellIsolation) || (this.producers ?? []).length === 0)
600
+ return;
562
601
  const time = process.hrtime();
563
602
  const promises = [];
564
- for (const producer of this.producers) {
603
+ for (const producer of this.producers ?? []) {
565
604
  const producerSafeName = utils_1.Utils.getSafeJobName(producer.name);
566
605
  const artifactFolder = `${this.cwd}/.gitlab-ci-local/artifacts/${producerSafeName}`;
567
606
  if (!await fs.pathExists(artifactFolder)) {
568
607
  throw new exit_error_1.ExitError(`${artifactFolder} doesn't exist, did you forget --needs`);
569
608
  }
570
- promises.push(cpFunc(artifactFolder));
609
+ promises.push(this.copyIn(artifactFolder));
571
610
  }
572
611
  await Promise.all(promises);
573
612
  const endTime = process.hrtime(time);
574
- const targetText = this.imageName ? "container" : "isolated shell";
575
- writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright copied artifacts to ${targetText}} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
613
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright imported artifacts} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
576
614
  }
577
- async copyArtifactsOut(writeStreams, buildVolumeName) {
615
+ copyIn(source) {
578
616
  const safeJobName = this.safeJobName;
579
- if (!this.shellIsolation && !this.imageName || !this.artifacts) {
617
+ if (!this.imageName && this.shellIsolation) {
618
+ return utils_1.Utils.spawn(`rsync -a ${source}/. ${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);
619
+ }
620
+ return utils_1.Utils.spawn(`docker cp ${source}/. ${this._containerId}:/builds/${safeJobName}`);
621
+ }
622
+ async copyCacheOut(writeStreams) {
623
+ if (this.mountCache && this.imageName)
624
+ return;
625
+ if ((!this.imageName && !this.shellIsolation) || this.cache.length === 0)
580
626
  return;
627
+ let time, endTime;
628
+ for (const c of this.cache) {
629
+ if (!["push", "pull-push"].includes(c.policy))
630
+ return;
631
+ const cacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);
632
+ for (const path of c.paths) {
633
+ time = process.hrtime();
634
+ const expandedPath = utils_1.Utils.expandText(path, this.expandedVariables);
635
+ let cmd = "shopt -s globstar nullglob dotglob\n";
636
+ cmd += `mkdir -p ../../cache/${cacheName}\n`;
637
+ cmd += `rsync -Ra ${expandedPath} ../../cache/${cacheName}/. || true\n`;
638
+ await mutex_1.Mutex.exclusive(cacheName, async () => {
639
+ await this.copyOut(cmd, "cache", []);
640
+ });
641
+ endTime = process.hrtime(time);
642
+ const readdir = await fs.readdir(`${this.cwd}/.gitlab-ci-local/cache/${cacheName}`);
643
+ if (readdir.length === 0) {
644
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {yellow !! no cache was copied for ${path} !!}\n`);
645
+ }
646
+ else {
647
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright exported cache ${expandedPath} '${cacheName}'} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
648
+ }
649
+ }
581
650
  }
651
+ }
652
+ async copyArtifactsOut(writeStreams) {
653
+ const safeJobName = this.safeJobName;
654
+ if (!this.shellIsolation && !this.imageName || !this.artifacts)
655
+ return;
582
656
  let time, endTime;
583
657
  let cpCmd = "shopt -s globstar nullglob dotglob\n";
584
658
  cpCmd += `mkdir -p ../../artifacts/${safeJobName}\n`;
585
659
  for (const artifactPath of this.artifacts?.paths ?? []) {
586
660
  const expandedPath = utils_1.Utils.expandText(artifactPath, this.expandedVariables);
587
- cpCmd += `echo Started copying ${expandedPath} to ../../artifacts/${safeJobName}\n`;
588
- cpCmd += `rsync -Ra ${expandedPath} ../../artifacts/${safeJobName}/.\n`;
589
- cpCmd += `echo Done copying ${expandedPath} to ../../artifacts/${safeJobName}\n`;
661
+ cpCmd += `rsync -Ra ${expandedPath} ../../artifacts/${safeJobName}/. || true\n`;
590
662
  }
591
663
  for (const artifactExcludePath of this.artifacts?.exclude ?? []) {
592
664
  const expandedPath = utils_1.Utils.expandText(artifactExcludePath, this.expandedVariables);
593
- cpCmd += `echo Started removing exclude '${expandedPath}' from ../../artifacts/${safeJobName}\n`;
594
665
  cpCmd += `ls -1d '../../artifacts/${safeJobName}/${expandedPath}' | xargs -n1 rm -rf || true\n`;
595
- cpCmd += `echo Done removing exclude '${expandedPath}' from ../../artifacts/${safeJobName}\n`;
596
666
  }
597
667
  const reportDotenv = this.artifacts.reports?.dotenv ?? null;
598
668
  if (reportDotenv != null) {
599
669
  cpCmd += `mkdir -p ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\n`;
600
- cpCmd += `echo Started copying ${reportDotenv} to ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\n`;
601
670
  cpCmd += `rsync -Ra ${reportDotenv} ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv/.\n`;
602
- cpCmd += `echo Done copying ${reportDotenv} to ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\n`;
603
671
  }
604
672
  time = process.hrtime();
605
- if (this.imageName) {
606
- const cacheMountStr = await this.createCacheDockerVolumeMounts(safeJobName, writeStreams);
607
- const dockerCreateCmd = `docker create -i ${cacheMountStr} -v ${buildVolumeName}:/builds/${safeJobName}/ -w /builds/${safeJobName}/ firecow/gitlab-ci-local-util bash -c "${cpCmd}"`;
608
- const { stdout: artifactsContainerId } = await utils_1.Utils.spawn(dockerCreateCmd, this.cwd);
609
- this._artifactsContainerId = artifactsContainerId.replace(/\r?\n/g, "");
610
- await fs.mkdirp(`${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}`);
611
- await utils_1.Utils.spawn(`docker start ${this._artifactsContainerId} --attach`);
612
- await utils_1.Utils.spawn(`docker cp ${this._artifactsContainerId}:/artifacts/. .gitlab-ci-local/artifacts/.`, this.cwd);
673
+ const dockerCmdExtras = this.mountCache ? [await this.mountCacheCmd(this.safeJobName, writeStreams)] : [];
674
+ await this.copyOut(cpCmd, "artifacts", dockerCmdExtras);
675
+ endTime = process.hrtime(time);
676
+ const readdir = await fs.readdir(`${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}`);
677
+ if (readdir.length === 0) {
678
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {yellow !! no artifacts was copied !!}\n`);
613
679
  }
614
- else if (this.shellIsolation) {
615
- await utils_1.Utils.spawn(`mkdir -p ../../artifacts/${safeJobName}`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);
616
- await utils_1.Utils.spawn(`bash -e -c "${cpCmd}"`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);
680
+ else {
681
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright exported artifacts} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
617
682
  }
618
- endTime = process.hrtime(time);
619
- writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright saved artifacts} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
620
683
  if (this.artifactsToSource) {
621
684
  time = process.hrtime();
622
685
  await utils_1.Utils.spawn(`rsync --exclude=/.gitlab-ci-reports/ -a ${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}/. ${this.cwd}`);
@@ -627,6 +690,21 @@ class Job {
627
690
  writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright copied artifacts to cwd} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
628
691
  }
629
692
  }
693
+ async copyOut(cmd, type, dockerCmdExtras) {
694
+ const safeJobName = this.safeJobName;
695
+ const buildVolumeName = this.buildVolumeName;
696
+ await fs.mkdirp(`${this.cwd}/.gitlab-ci-local/${type}`);
697
+ if (this.imageName) {
698
+ const { stdout: cid } = await utils_1.Utils.spawn(`docker create -i ${dockerCmdExtras.join(" ")} -v ${buildVolumeName}:/builds/${safeJobName}/ -w /builds/${safeJobName}/ firecow/gitlab-ci-local-util bash -c "${cmd}"`, this.cwd);
699
+ const containerId = cid.replace(/\r?\n/g, "");
700
+ this._containersToClean.push(containerId);
701
+ await utils_1.Utils.spawn(`docker start ${containerId} --attach`);
702
+ await utils_1.Utils.spawn(`docker cp ${containerId}:/${type}/. .gitlab-ci-local/${type}/.`, this.cwd);
703
+ }
704
+ else if (this.shellIsolation) {
705
+ await utils_1.Utils.spawn(`bash -eo pipefail -c "${cmd}"`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);
706
+ }
707
+ }
630
708
  refreshLongRunningSilentTimeout(writeStreams) {
631
709
  clearTimeout(this._longRunningSilentTimeout);
632
710
  this._longRunningSilentTimeout = setTimeout(() => {
@@ -650,20 +728,6 @@ class Job {
650
728
  const { stdout: networkId } = await utils_1.Utils.spawn(`docker network create ${networkName}`);
651
729
  this._serviceNetworkId = networkId.replace(/\r?\n/g, "");
652
730
  }
653
- async createCacheDockerVolumeMounts(safeJobName, writeStreams) {
654
- let cmd = "";
655
- for (const entry of this.cache) {
656
- const uniqueCacheName = await Job.getUniqueCacheName(this.cwd, entry, this.expandedVariables);
657
- entry.paths.forEach((p) => {
658
- const path = utils_1.Utils.expandText(p, this.expandedVariables);
659
- writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright mounting cache} for path ${path}\n`);
660
- const cachedir = `/tmp/gitlab-ci-local/cache/${uniqueCacheName}/${path}`;
661
- fs.ensureDirSync(cachedir);
662
- cmd += `-v ${cachedir}:/builds/${safeJobName}/${path} `;
663
- });
664
- }
665
- return cmd;
666
- }
667
731
  async startService(writeStreams, service, privileged) {
668
732
  let dockerCmd = `docker run -d --network gitlab-ci-local-${this.jobId} `;
669
733
  if (privileged) {
@@ -692,11 +756,42 @@ class Job {
692
756
  }
693
757
  const time = process.hrtime();
694
758
  const { stdout: containerId } = await utils_1.Utils.spawn(dockerCmd, this.cwd);
695
- this._serviceIds.push(containerId.replace(/\r?\n/g, ""));
759
+ this._containersToClean.push(containerId.replace(/\r?\n/g, ""));
696
760
  this.refreshLongRunningSilentTimeout(writeStreams);
697
761
  const endTime = process.hrtime(time);
698
762
  writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {magentaBright started service image: ${serviceName} with aliases: ${aliases.join(", ")}} in {magenta ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
699
763
  }
764
+ async serviceHealthCheck(writeStreams, service) {
765
+ const dockerInspectCmd = `docker inspect ${service.getName(this.expandedVariables)}`;
766
+ const { stdout: imageDetails } = await utils_1.Utils.spawn(dockerInspectCmd, this.cwd);
767
+ const imageDetailObj = JSON.parse(imageDetails);
768
+ // Copied from the startService block. Important thing is that the aliases match
769
+ const serviceAlias = service.getAlias(this.expandedVariables);
770
+ const serviceName = service.getName(this.expandedVariables);
771
+ const serviceNameWithoutVersion = serviceName.replace(/(.*)(:.*)/, "$1");
772
+ const aliases = [serviceNameWithoutVersion.replace("/", "-"), serviceNameWithoutVersion.replace("/", "__")];
773
+ if (serviceAlias) {
774
+ aliases.push(serviceAlias);
775
+ }
776
+ // Iterate over each port defined in the image, and try to connect to the alias
777
+ for (const port of Object.keys(imageDetailObj[0].ContainerConfig.ExposedPorts)) {
778
+ if (port.endsWith("/tcp")) {
779
+ const portNum = parseInt(port.replace("/tcp", ""));
780
+ let dockerCmd = `docker run -d --network gitlab-ci-local-${this.jobId} `;
781
+ dockerCmd += ` willwill/wait-for-it "${aliases[0]}:${portNum}" -t 30`;
782
+ const time = process.hrtime();
783
+ const { status: result, stdout } = await utils_1.Utils.spawn(dockerCmd, this.cwd);
784
+ this._containersToClean.push(stdout.replace(/\r?\n/g, ""));
785
+ const endTime = process.hrtime(time);
786
+ if (result == 0) {
787
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {greenBright service image: ${serviceName} healthcheck passed: ${aliases[0]}:${portNum}} in {green ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
788
+ }
789
+ else {
790
+ writeStreams.stdout((0, chalk_1.default) `${this.chalkJobName} {redBright service image: ${serviceName} healthcheck failed: ${aliases[0]}:${portNum}} in {red ${(0, pretty_hrtime_1.default)(endTime)}}\n`);
791
+ }
792
+ }
793
+ }
794
+ }
700
795
  }
701
796
  exports.Job = Job;
702
797
  Job.illegalJobNames = [
@@ -704,4 +799,4 @@ Job.illegalJobNames = [
704
799
  "stages", "pages", "types", "before_script", "default",
705
800
  "after_script", "variables", "cache", "workflow",
706
801
  ];
707
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"job.js","sourceRoot":"","sources":["job.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;AAAA,kDAA0B;AAC1B,+CAAiC;AACjC,4DAA8C;AAC9C,6CAA+B;AAC/B,kEAAyC;AACzC,0DAAkC;AAClC,mDAA6C;AAC7C,mCAA8B;AAK9B,uCAAiC;AAEjC,MAAa,GAAG;IAyCZ,YAAY,GAAe;QAlBnB,wBAAmB,GAAkB,IAAI,CAAC;QAC1C,0BAAqB,GAAG,CAAC,CAAC;QAC1B,qBAAgB,GAAkB,IAAI,CAAC;QACvC,aAAQ,GAAG,KAAK,CAAC;QACjB,iBAAY,GAAkB,IAAI,CAAC;QACnC,gBAAW,GAAa,EAAE,CAAC;QAC3B,sBAAiB,GAAkB,IAAI,CAAC;QACxC,0BAAqB,GAAkB,IAAI,CAAC;QAC5C,0BAAqB,GAAa,EAAE,CAAC;QACrC,8BAAyB,GAAmB,CAAC,CAAQ,CAAC;QACtD,eAAU,GAAqD,IAAI,CAAC;QASxE,MAAM,OAAO,GAAG,GAAG,CAAC,IAAI,CAAC;QACzB,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;QAC5B,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;QAC5B,MAAM,aAAa,GAAG,GAAG,CAAC,aAAa,CAAC;QACxC,MAAM,gBAAgB,GAAG,GAAG,CAAC,gBAAgB,CAAC;QAE9C,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC,UAAU,CAAC;QACjC,IAAI,CAAC,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;QAC3B,IAAI,CAAC,YAAY,GAAG,GAAG,CAAC,YAAY,CAAC;QACrC,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC,OAAO,CAAC;QAC9B,IAAI,CAAC,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;QAC3B,IAAI,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;QACrB,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,CAAC;QACnB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,OAAO,CAAC,CAAC;QACjD,IAAI,CAAC,OAAO,GAAG,GAAG,CAAC,IAAI,CAAC;QACxB,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC,WAAW,CAAC;QACnC,IAAI,CAAC,cAAc,GAAG,GAAG,CAAC,cAAc,CAAC;QAEzC,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,YAAY,CAAC;QACzC,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,aAAa,IAAI,KAAK,CAAC;QACnD,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,KAAK,IAAI,IAAI,CAAC;QACpC,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC;QACjD,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,IAAI,CAAC;QACnC,IAAI,CAAC,WAAW,GAAG,OAAO,OAAO,CAAC,WAAW,KAAK,QAAQ,CAAC,CAAC,CAAC,EAAC,IAAI,EAAE,OAAO,CAAC,WAAW,EAAC,CAAC,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC;QAC/G,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;QAEjC,IAAI,cAAc,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACnC,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,cAAc,GAAG,WAAW,IAAI,CAAC,WAAW,EAAE,CAAC;SAClD;aAAM,IAAI,IAAI,CAAC,cAAc,EAAE;YAC5B,cAAc,GAAG,GAAG,IAAI,CAAC,GAAG,4BAA4B,IAAI,CAAC,WAAW,EAAE,CAAC;SAC9E;QAED,MAAM,mBAAmB,GAAG;YACxB,iBAAiB,EAAE,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC;YACpD,iBAAiB,EAAE,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC;YACpD,gBAAgB,EAAE,OAAO,CAAC,IAAI,CAAC,kBAAkB,CAAC;YAClD,cAAc,EAAE,OAAO,CAAC,IAAI,CAAC,gBAAgB,CAAC;YAC9C,mBAAmB,EAAE,OAAO,CAAC,MAAM,CAAC,SAAS;YAC7C,aAAa,EAAE,OAAO,CAAC,MAAM,CAAC,GAAG;YACjC,cAAc;YACd,eAAe,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO;YACvC,gBAAgB,EAAE,GAAG,IAAA,mBAAS,EAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE;YACxD,eAAe,EAAE,OAAO,CAAC,eAAe;YACxC,oBAAoB,EAAE,OAAO,CAAC,oBAAoB;YAClD,oBAAoB,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE;YAC/C,qBAAqB,EAAE,UAAU;YACjC,aAAa,EAAE,MAAM;YACrB,uBAAuB,EAAE,OAAO;YAChC,gBAAgB,EAAE,OAAO,CAAC,MAAM,CAAC,QAAQ;YACzC,kBAAkB,EAAE,OAAO,CAAC,MAAM,CAAC,QAAQ;YAC3C,kBAAkB,EAAE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,WAAW,EAAE;YACvI,eAAe,EAAE,cAAc;YAC/B,iBAAiB,EAAE,gCAAgC;YACnD,qBAAqB,EAAE,kBAAkB;YACzC,kBAAkB,EAAE,MAAM;YAC1B,SAAS,EAAE,GAAG,IAAI,CAAC,KAAK,EAAE;YAC1B,cAAc,EAAE,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,EAAE;YAC5C,eAAe,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE;YACtC,cAAc,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE;YAC1C,aAAa,EAAE,WAAW,OAAO,CAAC,MAAM,CAAC,MAAM,MAAM;YACrD,aAAa,EAAE,WAAW,OAAO,CAAC,MAAM,CAAC,MAAM,SAAS;YACxD,cAAc,EAAE,WAAW,OAAO,CAAC,MAAM,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,OAAO,CAAC,MAAM,CAAC,OAAO,EAAE;YACpG,UAAU,EAAE,WAAW,OAAO,CAAC,MAAM,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,OAAO,CAAC,MAAM,CAAC,OAAO,WAAW,IAAI,CAAC,KAAK,EAAE;YACrH,eAAe,EAAE,WAAW,OAAO,CAAC,MAAM,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,OAAO,CAAC,MAAM,CAAC,OAAO,cAAc,IAAI,CAAC,WAAW,EAAE;YACnI,WAAW,EAAE,GAAG,IAAI,CAAC,IAAI,EAAE;YAC3B,YAAY,EAAE,GAAG,IAAI,CAAC,KAAK,EAAE;YAC7B,WAAW,EAAE,OAAO,CAAC,WAAW;YAChC,iBAAiB,EAAE,OAAO,CAAC,iBAAiB;YAC5C,SAAS,EAAE,OAAO;SACrB,CAAC;QAEF,4BAA4B;QAC5B,MAAM,IAAI,GAAG,EAAC,GAAG,mBAAmB,EAAE,GAAG,OAAO,CAAC,SAAS,IAAI,EAAE,EAAE,GAAG,OAAO,CAAC,SAAS,IAAI,EAAE,EAAE,GAAG,aAAa,EAAE,GAAG,gBAAgB,EAAC,CAAC;QACrI,MAAM,uBAAuB,GAAG,aAAK,CAAC,eAAe,CAAC,OAAO,CAAC,SAAS,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;QACrF,MAAM,oBAAoB,GAAG,aAAK,CAAC,eAAe,CAAC,OAAO,CAAC,SAAS,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;QAClF,IAAI,CAAC,iBAAiB,GAAG,EAAC,GAAG,mBAAmB,EAAE,GAAG,uBAAuB,EAAE,GAAG,oBAAoB,EAAE,GAAG,aAAa,EAAE,GAAG,gBAAgB,EAAC,CAAC;QAE9I,iDAAiD;QACjD,IAAI,IAAI,CAAC,KAAK,EAAE;YACZ,MAAM,UAAU,GAAG,aAAK,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC5E,IAAI,CAAC,IAAI,GAAG,UAAU,CAAC,IAAI,CAAC;YAC5B,IAAI,CAAC,YAAY,GAAG,UAAU,CAAC,YAAY,CAAC;SAC/C;QAED,IAAI,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC,EAAE;YACzE,MAAM,IAAI,sBAAS,CAAC,GAAG,IAAI,CAAC,YAAY,oEAAoE,CAAC,CAAC;SACjH;QAED,IAAI,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE;YAChD,MAAM,IAAI,sBAAS,CAAC,GAAG,IAAI,CAAC,YAAY,+CAA+C,CAAC,CAAC;SAC5F;IACL,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,kBAAkB,CAAC,GAAW,EAAE,UAAkE,EAAE,iBAA4C;QACzJ,IAAI,OAAO,UAAU,CAAC,GAAG,KAAK,QAAQ,EAAE;YACpC,OAAO,aAAK,CAAC,UAAU,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;SAC3C;QACD,OAAO,KAAK,GAAG,MAAM,aAAK,CAAC,aAAa,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YAClE,OAAO,GAAG,GAAG,IAAI,aAAK,CAAC,UAAU,CAAC,CAAC,EAAE,iBAAiB,CAAC,EAAE,CAAC;QAC9D,CAAC,CAAC,CAAC,CAAC;IACR,CAAC;IAED,IAAI,iBAAiB;QACjB,OAAO,IAAI,CAAC,OAAO,CAAC,mBAAmB,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC;IAChG,CAAC;IAED,IAAI,YAAY;QACZ,OAAO,IAAA,eAAK,EAAA,eAAe,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;IACpE,CAAC;IAED,IAAI,WAAW;QACX,OAAO,aAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC3C,CAAC;IAED,IAAI,KAAK;QACL,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,OAAO,IAAI,CAAC;QAC9B,MAAM,IAAI,GAAwC,EAAE,CAAC;QACrD,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YAC1B,IAAI,CAAC,IAAI,CAAC;gBACN,GAAG,EAAE,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG;gBAC/C,SAAS,EAAE,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS;aAC9D,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QACH,OAAO,IAAI,CAAC;IAChB,CAAC;IAGD,IAAI,SAAS;QACT,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,CAAC,KAAK,EAAE;YACR,OAAO,IAAI,CAAC;SACf;QAED,MAAM,SAAS,GAAG,aAAK,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACvE,OAAO,SAAS,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,SAAS,CAAC;IACvE,CAAC;IAED,IAAI,eAAe;QACf,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAEpC,IAAI,CAAC,KAAK,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE;YAC7B,OAAO,IAAI,CAAC;SACf;QACD,IAAI,OAAO,KAAK,CAAC,UAAU,KAAK,QAAQ,EAAE;YACtC,MAAM,IAAI,sBAAS,CAAC,mCAAmC,CAAC,CAAC;SAC5D;QACD,OAAO,KAAK,CAAC,UAAU,CAAC;IAC5B,CAAC;IAED,IAAI,QAAQ;QACR,OAAO,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IACpC,CAAC;IAED,IAAI,SAAS;QACT,OAAO,IAAI,CAAC,UAAU,CAAC;IAC3B,CAAC;IAED,IAAI,SAAS,CAAC,SAA2D;QACrE,IAAA,gBAAM,EAAC,IAAI,CAAC,UAAU,IAAI,IAAI,EAAE,sCAAsC,CAAC,CAAC;QACxE,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;IAChC,CAAC;IAED,IAAI,KAAK;QACL,OAAO,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,MAAM,CAAC;IAC3C,CAAC;IAED,IAAI,WAAW;QACX,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,KAAK,CAAC;IAChD,CAAC;IAED,IAAI,cAAc;QACd,OAAO,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,KAAK,CAAC;IACnD,CAAC;IAED,IAAI,WAAW;QACX,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;IAC7C,CAAC;IAED,IAAI,SAAS;QACT,OAAO,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IACrC,CAAC;IAED,IAAI,aAAa;QACb,OAAO,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;IAC/C,CAAC;IAED,IAAI,YAAY;QACZ,OAAO,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;IAC9C,CAAC;IAED,IAAI,OAAO;QACP,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IAClC,CAAC;IAED,IAAI,kBAAkB;QAClB,OAAO,IAAI,CAAC,mBAAmB,CAAC;IACpC,CAAC;IAED,IAAI,oBAAoB;QACpB,OAAO,IAAI,CAAC,qBAAqB,CAAC;IACtC,CAAC;IAED,IAAI,OAAO;QACP,OAAO,IAAI,CAAC,QAAQ,CAAC;IACzB,CAAC;IAED,IAAI,OAAO;QACP,OAAO,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,mBAAmB,KAAK,IAAI,CAAC;IAC9D,CAAC;IAED,IAAI,QAAQ;QACR,OAAO,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,mBAAmB,KAAK,IAAI,CAAC;IAC/D,CAAC;IAED,IAAI,eAAe;QACf,OAAO,IAAI,CAAC,gBAAgB,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,UAAmB;QAC3B,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QACnC,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACvC,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QAErC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QAErB,MAAM,EAAE,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,GAAG,4BAA4B,WAAW,MAAM,CAAC,CAAC;QAC9E,MAAM,EAAE,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,GAAG,4BAA4B,WAAW,MAAM,CAAC,CAAC;QAE5E,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;YACnB,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,6BAA6B,IAAI,CAAC,SAAS,IAAI,OAAO,aAAa,IAAI,CAAC,KAAK,MAAM,CAAC,CAAC;SACrI;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC3D,IAAI,CAAC,mBAAmB,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC;QAC1E,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;YACtF,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;YAC7F,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,OAAO;SACV;QAED,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,IAAI,IAAI,CAAC,YAAY,EAAE;YACrF,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YAC5F,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,OAAO;SACV;QAED,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,IAAI,IAAI,CAAC,YAAY,EAAE;YACnD,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;SAC/F;QAED,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;YACpD,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;SAChG;QAED,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YAC9B,IAAI,CAAC,qBAAqB,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,YAAY,EAAE,UAAU,CAAC,CAAC;SACtF;QAED,IAAI,IAAI,CAAC,qBAAqB,GAAG,CAAC,EAAE;YAChC,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,qBAAqB,EAAE,IAAI,EAAE,eAAe,CAAC,IAAI,CAAC,CAAC;SAClH;QAED,YAAY,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAE9D,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE;YACvB,IAAI,CAAC,gBAAgB,GAAG,MAAM,aAAK,CAAC,kBAAkB,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;SACxG;QAED,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC9B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,gBAAgB;QAClB,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACvC,YAAY,CAAC,IAAI,CAAC,yBAAyB,CAAC,CAAC;QAE7C,IAAI,IAAI,CAAC,YAAY,EAAE;YACnB,IAAI;gBACA,MAAM,aAAK,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;aAC1D;YAAC,OAAO,CAAC,EAAE;gBACR,IAAA,gBAAM,EAAC,CAAC,YAAY,KAAK,EAAE,2BAA2B,CAAC,CAAC;gBACxD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,WAAW,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC;aACrD;SACJ;QAED,IAAI,IAAI,CAAC,WAAW,EAAE;YAClB,IAAI;gBACA,KAAK,MAAM,SAAS,IAAI,IAAI,CAAC,WAAW,EAAE;oBACtC,MAAM,aAAK,CAAC,KAAK,CAAC,gBAAgB,SAAS,EAAE,CAAC,CAAC;iBAClD;aACJ;YAAC,OAAO,CAAC,EAAE;gBACR,IAAA,gBAAM,EAAC,CAAC,YAAY,KAAK,EAAE,2BAA2B,CAAC,CAAC;gBACxD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,WAAW,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC;aACrD;SACJ;QAED,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI;gBACA,MAAM,aAAK,CAAC,KAAK,CAAC,qBAAqB,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;aACpE;YAAC,OAAO,CAAC,EAAE;gBACR,IAAA,gBAAM,EAAC,CAAC,YAAY,KAAK,EAAE,2BAA2B,CAAC,CAAC;gBACxD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,WAAW,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC;aACrD;SACJ;QAED,IAAI,IAAI,CAAC,qBAAqB,EAAE;YAC5B,IAAI;gBACA,MAAM,aAAK,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,qBAAqB,EAAE,CAAC,CAAC;aACnE;YAAC,OAAO,CAAC,EAAE;gBACR,IAAA,gBAAM,EAAC,CAAC,YAAY,KAAK,EAAE,2BAA2B,CAAC,CAAC;gBACxD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,WAAW,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC;aACrD;SACJ;QAED,IAAI,IAAI,CAAC,qBAAqB,CAAC,MAAM,GAAG,CAAC,EAAE;YACvC,IAAI;gBACA,KAAK,MAAM,eAAe,IAAI,IAAI,CAAC,qBAAqB,EAAE;oBACtD,MAAM,aAAK,CAAC,KAAK,CAAC,oBAAoB,eAAe,EAAE,CAAC,CAAC;iBAC5D;aACJ;YAAC,OAAO,CAAC,EAAE;gBACR,IAAA,gBAAM,EAAC,CAAC,YAAY,KAAK,EAAE,2BAA2B,CAAC,CAAC;gBACxD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,WAAW,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC;aACrD;SACJ;IACL,CAAC;IAEO,6BAA6B;QACjC,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACtB,OAAO,EAAE,CAAC;SACb;QACD,IAAI,OAAO,CAAC,QAAQ,KAAK,QAAQ,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,CAAC,SAAS,CAAC,IAAI,IAAI,CAAC,EAAE;YACjF,OAAO,2HAA2H,CAAC;SACtI;QACD,OAAO,uBAAuB,OAAO,CAAC,GAAG,CAAC,aAAa,OAAO,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,OAAO,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3H,CAAC;IAEO,sBAAsB,CAAC,OAAiB;QAC5C,IAAI,GAAG,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE;YACvB,iCAAiC;YACjC,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YACpC,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,iCAAiC,CAAC,CAAC,CAAC,EAAE,CAAC;YAChF,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,EAAE,OAAO,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YAC/F,GAAG,IAAI,IAAA,eAAK,EAAA,kBAAkB,IAAI,GAAG,aAAa,MAAM,CAAC;YAEzD,wBAAwB;YACxB,GAAG,IAAI,GAAG,MAAM,IAAI,CAAC;QACzB,CAAC,CAAC,CAAC;QACH,OAAO,GAAG,CAAC;IACf,CAAC;IAEO,KAAK,CAAC,WAAW,CAAC,OAAiB,EAAE,UAAmB;QAC5D,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACrC,MAAM,eAAe,GAAG,GAAG,IAAI,CAAC,GAAG,4BAA4B,WAAW,MAAM,CAAC;QACjF,MAAM,eAAe,GAAG,OAAO,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,KAAK,QAAQ,CAAC;QACtE,MAAM,aAAa,GAAG,OAAO,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,KAAK,MAAM,CAAC;QAClE,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACvC,MAAM,sBAAsB,GAAG,MAAM,IAAI,CAAC,kCAAkC,CAAC,YAAY,CAAC,CAAC;QAC3F,IAAI,IAAI,CAAC;QACT,IAAI,OAAO,CAAC;QAEZ,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,IAAI,OAAO,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE;YAC5C,OAAO,CAAC,CAAC;SACZ;QAED,qEAAqE;QACrE,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,EAAE;YACxC,MAAM,aAAK,CAAC,iBAAiB,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,WAAW,EAAE,CAAC,CAAC;SAC7D;QAED,IAAI,IAAI,CAAC,WAAW,EAAE;YAClB,MAAM,IAAI,GAAG,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;YAClD,MAAM,aAAa,GAAG,YAAY,CAAC,KAAK,CAAC,IAAI,EAAE;gBAC3C,KAAK,EAAE,MAAM;gBACb,KAAK,EAAE,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC;gBACxC,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,GAAG,EAAE,EAAC,GAAG,IAAI,CAAC,iBAAiB,EAAE,GAAG,OAAO,CAAC,GAAG,EAAC;aACnD,CAAC,CAAC;YACH,OAAO,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;gBAC3C,aAAa,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;gBACvD,aAAa,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;YACpD,CAAC,CAAC,CAAC;SACN;QAED,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;QAEnD,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,MAAM,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;YAEnD,IAAI,SAAS,GAAG,EAAE,CAAC;YACnB,IAAI,UAAU,EAAE;gBACZ,SAAS,IAAI,wCAAwC,IAAI,CAAC,6BAA6B,EAAE,GAAG,CAAC;aAChG;iBAAM;gBACH,SAAS,IAAI,2BAA2B,IAAI,CAAC,6BAA6B,EAAE,GAAG,CAAC;aACnF;YACD,IAAI,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE;gBACvB,MAAM,IAAI,CAAC,mBAAmB,CAAC,mBAAmB,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;gBAChE,SAAS,IAAI,6BAA6B,IAAI,CAAC,KAAK,GAAG,CAAC;gBACxD,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE;oBACjC,MAAM,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC;oBAC5E,MAAM,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;iBAC9D;aACJ;YAED,MAAM,cAAc,GAAG,EAAE,CAAC;YAC1B,cAAc,CAAC,IAAI,CAAC,aAAK,CAAC,KAAK,CAAC,wBAAwB,eAAe,EAAE,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;YACtF,cAAc,CAAC,IAAI,CAAC,aAAK,CAAC,KAAK,CAAC,wBAAwB,aAAa,EAAE,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;YACpF,SAAS,IAAI,YAAY,eAAe,YAAY,WAAW,GAAG,CAAC;YACnE,SAAS,IAAI,YAAY,aAAa,SAAS,CAAC;YAChD,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YACjD,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAC/C,MAAM,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;YAElC,SAAS,IAAI,qBAAqB,WAAW,GAAG,CAAC;YAEjD,KAAK,MAAM,MAAM,IAAI,IAAI,CAAC,OAAO,EAAE;gBAC/B,SAAS,IAAI,YAAY,MAAM,GAAG,CAAC;aACtC;YAED,KAAK,MAAM,SAAS,IAAI,IAAI,CAAC,UAAU,EAAE;gBACrC,SAAS,IAAI,cAAc,SAAS,GAAG,CAAC;aAC3C;YAED,IAAI,IAAI,CAAC,eAAe,EAAE;gBACtB,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;oBAC/B,SAAS,IAAI,iBAAiB,CAAC,IAAI,CAAC;gBACxC,CAAC,CAAC,CAAC;aACN;YAED,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE;gBAC/D,SAAS,IAAI,MAAM,GAAG,KAAK,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC;aACvD;YACD,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,sBAAsB,CAAC,EAAE;gBAC/D,SAAS,IAAI,MAAM,GAAG,KAAK,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC;aACvD;YAED,SAAS,IAAI,MAAM,IAAI,CAAC,6BAA6B,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YAEjF,SAAS,IAAI,GAAG,IAAI,CAAC,SAAS,YAAY,CAAC;YAC3C,SAAS,IAAI,uCAAuC,CAAC;YACrD,SAAS,IAAI,+BAA+B,CAAC;YAC7C,SAAS,IAAI,mCAAmC,CAAC;YACjD,SAAS,IAAI,yBAAyB,CAAC;YACvC,SAAS,IAAI,+BAA+B,CAAC;YAC7C,SAAS,IAAI,qBAAqB,CAAC;YACnC,SAAS,IAAI,uCAAuC,CAAC;YACrD,SAAS,IAAI,6BAA6B,CAAC;YAC3C,SAAS,IAAI,iCAAiC,CAAC;YAC/C,SAAS,IAAI,uBAAuB,CAAC;YACrC,SAAS,IAAI,6BAA6B,CAAC;YAC3C,SAAS,IAAI,mBAAmB,CAAC;YACjC,SAAS,IAAI,iCAAiC,CAAC;YAC/C,SAAS,IAAI,uBAAuB,CAAC;YACrC,SAAS,IAAI,QAAQ,CAAC;YACtB,SAAS,IAAI,0BAA0B,CAAC;YACxC,SAAS,IAAI,YAAY,CAAC;YAC1B,SAAS,IAAI,QAAQ,CAAC;YAEtB,MAAM,EAAC,MAAM,EAAE,WAAW,EAAC,GAAG,MAAM,aAAK,CAAC,KAAK,CAAC,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;YACrE,IAAI,CAAC,YAAY,GAAG,WAAW,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;YAEtD,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;YACxB,oCAAoC;YACpC,MAAM,aAAK,CAAC,KAAK,CAAC,+CAA+C,IAAI,CAAC,YAAY,YAAY,WAAW,EAAE,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;YACvH,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;YAEnD,sCAAsC;YACtC,MAAM,mBAAmB,GAAG,uCAAuC,IAAI,CAAC,OAAO,CAAC,oBAAoB,GAAG,CAAC;YACxG,IAAI,MAAM,EAAE,CAAC,UAAU,CAAC,mBAAmB,CAAC,EAAE;gBAC1C,MAAM,aAAK,CAAC,KAAK,CAAC,aAAa,mBAAmB,IAAI,IAAI,CAAC,YAAY,IAAI,mBAAmB,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;gBAC7G,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;aACtD;YAED,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC/B,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,oDAAoD,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;SAChI;QAED,MAAM,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC;QAEzC,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,gFAAgF;YAChF,MAAM,aAAK,CAAC,KAAK,CAAC,+BAA+B,eAAe,iFAAiF,CAAC,CAAC;SACtJ;QAED,IAAI,GAAG,GAAG,oBAAoB,CAAC;QAC/B,GAAG,IAAI,qBAAqB,CAAC;QAE7B,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,EAAE;YACxC,GAAG,IAAI,8BAA8B,WAAW,KAAK,CAAC;SACzD;QACD,GAAG,IAAI,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;QAE5C,GAAG,IAAI,UAAU,CAAC;QAElB,MAAM,EAAE,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,GAAG,6BAA6B,WAAW,EAAE,EAAE,GAAG,EAAE,OAAO,CAAC,CAAC;QACzF,MAAM,EAAE,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,GAAG,6BAA6B,WAAW,EAAE,EAAE,MAAM,CAAC,CAAC;QAE9E,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,MAAM,aAAK,CAAC,KAAK,CAAC,wCAAwC,IAAI,CAAC,YAAY,gBAAgB,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;SAC1G;QAED,MAAM,EAAE,GAAG,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,4BAA4B,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,MAAM,EAAE;YACxG,KAAK,EAAE,MAAM;YACb,KAAK,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC;YAC/B,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,GAAG,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,EAAC,GAAG,OAAO,CAAC,GAAG,EAAC,CAAC,CAAC,CAAC,EAAC,GAAG,IAAI,CAAC,iBAAiB,EAAE,GAAG,sBAAsB,EAAE,GAAG,OAAO,CAAC,GAAG,EAAC;SAClH,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,CAAC,CAAM,EAAE,MAA6B,EAAE,QAAiC,EAAE,EAAE;YACzF,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;YACnD,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE;gBACtC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;oBACnB,SAAS;iBACZ;gBAED,MAAM,CAAC,GAAG,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC;gBAChC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,EAAE;oBACjC,MAAM,CAAC,GAAG,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;iBAC/B;gBACD,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC;gBACpB,EAAE,CAAC,cAAc,CAAC,eAAe,EAAE,GAAG,IAAI,IAAI,CAAC,CAAC;aACnD;QACL,CAAC,CAAC;QAEF,MAAM,QAAQ,GAAG,MAAM,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3D,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,CAAC,EAAE,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,IAAA,eAAK,EAAA,gBAAgB,CAAC,GAAG,CAAC,CAAC,CAAC;YAClH,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,CAAC,EAAE,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,IAAA,eAAK,EAAA,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC;YAEhH,EAAE,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAClE,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAE3D,IAAI,IAAI,CAAC,SAAS,EAAE;gBAChB,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,gBAAgB,WAAW,EAAE,CAAC,CAAC;aAC/C;iBAAM;gBACH,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,8BAA8B,WAAW,EAAE,CAAC,CAAC;aAC7D;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,eAAe,CAAC,CAAC;QAE3D,OAAO,QAAQ,CAAC;IACpB,CAAC;IAEO,KAAK,CAAC,SAAS,CAAC,YAA0B,EAAE,WAAmB;QACnE,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QAC9B,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,OAAO,IAAI,mEAAmE,WAAW,MAAM,CAAC;QAChG,OAAO,IAAI,6BAA6B,CAAC;QACzC,OAAO,IAAI,mBAAmB,WAAW,KAAK,CAAC;QAC/C,OAAO,IAAI,iBAAiB,WAAW,IAAI,CAAC;QAC5C,OAAO,IAAI,MAAM,CAAC;QAClB,MAAM,aAAK,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;QACnD,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACrC,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,2BAA2B,WAAW,gBAAgB,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACnI,CAAC;IAEO,KAAK,CAAC,kCAAkC,CAAC,YAA0B;QACvE,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QACjC,IAAI,mBAAmB,GAAG,EAAE,CAAC;QAC7B,KAAK,MAAM,QAAQ,IAAI,SAAS,IAAI,EAAE,EAAE;YACpC,IAAI,QAAQ,CAAC,MAAM,KAAK,IAAI;gBAAE,SAAS;YAEvC,MAAM,gBAAgB,GAAG,aAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YAC7D,IAAI,UAAU,CAAC;YACf,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;gBACzC,UAAU,GAAG,GAAG,IAAI,CAAC,GAAG,IAAI,QAAQ,CAAC,MAAM,EAAE,CAAC;aACjD;iBAAM;gBACH,UAAU,GAAG,GAAG,IAAI,CAAC,GAAG,+BAA+B,gBAAgB,8BAA8B,QAAQ,CAAC,MAAM,EAAE,CAAC;aAC1H;YACD,IAAI,MAAM,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE;gBACjC,MAAM,iBAAiB,GAAG,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC;gBACtE,mBAAmB,GAAG,EAAC,GAAG,mBAAmB,EAAE,GAAG,iBAAiB,EAAC,CAAC;aACxE;iBAAM;gBACH,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,aAAa,QAAQ,CAAC,MAAM,kBAAkB,QAAQ,CAAC,IAAI,yBAAyB,CAAC,CAAC;aACtI;SAEJ;QACD,OAAO,mBAAmB,CAAC;IAC/B,CAAC;IAEO,KAAK,CAAC,eAAe,CAAC,YAA0B;QACpD,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACzC,OAAO;SACV;QAED,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE;YAChD,OAAO;SACV;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QAErC,MAAM,MAAM,GAAG,KAAK,EAAE,MAAc,EAAE,EAAE;YACpC,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,EAAE;gBACxC,OAAO,aAAK,CAAC,KAAK,CAAC,YAAY,MAAM,MAAM,IAAI,CAAC,GAAG,4BAA4B,WAAW,EAAE,CAAC,CAAC;aACjG;YACD,OAAO,aAAK,CAAC,KAAK,CAAC,aAAa,MAAM,MAAM,IAAI,CAAC,YAAY,YAAY,WAAW,EAAE,CAAC,CAAC;QAC5F,CAAC,CAAC;QAEF,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QAC9B,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,SAAS,EAAE;YACnC,MAAM,gBAAgB,GAAG,aAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YAC7D,MAAM,cAAc,GAAG,GAAG,IAAI,CAAC,GAAG,+BAA+B,gBAAgB,EAAE,CAAC;YACpF,IAAI,CAAC,MAAM,EAAE,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE;gBACtC,MAAM,IAAI,sBAAS,CAAC,GAAG,cAAc,wCAAwC,CAAC,CAAC;aAClF;YACD,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC;SACzC;QACD,MAAM,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC5B,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACrC,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,gBAAgB,CAAC;QACnE,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,uCAAuC,UAAU,iBAAiB,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IAC/I,CAAC;IAEO,KAAK,CAAC,gBAAgB,CAAC,YAA0B,EAAE,eAAuB;QAC9E,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QAErC,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YAC5D,OAAO;SACV;QAED,IAAI,IAAI,EAAE,OAAO,CAAC;QAClB,IAAI,KAAK,GAAG,sCAAsC,CAAC;QACnD,KAAK,IAAI,4BAA4B,WAAW,IAAI,CAAC;QACrD,KAAK,MAAM,YAAY,IAAI,IAAI,CAAC,SAAS,EAAE,KAAK,IAAI,EAAE,EAAE;YACpD,MAAM,YAAY,GAAG,aAAK,CAAC,UAAU,CAAC,YAAY,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC5E,KAAK,IAAI,wBAAwB,YAAY,uBAAuB,WAAW,IAAI,CAAC;YACpF,KAAK,IAAI,aAAa,YAAY,oBAAoB,WAAW,MAAM,CAAC;YACxE,KAAK,IAAI,qBAAqB,YAAY,uBAAuB,WAAW,IAAI,CAAC;SACpF;QAED,KAAK,MAAM,mBAAmB,IAAI,IAAI,CAAC,SAAS,EAAE,OAAO,IAAI,EAAE,EAAE;YAC7D,MAAM,YAAY,GAAG,aAAK,CAAC,UAAU,CAAC,mBAAmB,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;YACnF,KAAK,IAAI,kCAAkC,YAAY,0BAA0B,WAAW,IAAI,CAAC;YACjG,KAAK,IAAI,2BAA2B,WAAW,IAAI,YAAY,gCAAgC,CAAC;YAChG,KAAK,IAAI,+BAA+B,YAAY,0BAA0B,WAAW,IAAI,CAAC;SACjG;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,MAAM,IAAI,IAAI,CAAC;QAC5D,IAAI,YAAY,IAAI,IAAI,EAAE;YACtB,KAAK,IAAI,4BAA4B,WAAW,8BAA8B,CAAC;YAC/E,KAAK,IAAI,wBAAwB,YAAY,uBAAuB,WAAW,8BAA8B,CAAC;YAC9G,KAAK,IAAI,aAAa,YAAY,oBAAoB,WAAW,gCAAgC,CAAC;YAClG,KAAK,IAAI,qBAAqB,YAAY,uBAAuB,WAAW,8BAA8B,CAAC;SAC9G;QAED,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QACxB,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,6BAA6B,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YAC1F,MAAM,eAAe,GAAG,oBAAoB,aAAa,OAAO,eAAe,YAAY,WAAW,gBAAgB,WAAW,2CAA2C,KAAK,GAAG,CAAC;YACrL,MAAM,EAAC,MAAM,EAAE,oBAAoB,EAAC,GAAG,MAAM,aAAK,CAAC,KAAK,CAAC,eAAe,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;YACpF,IAAI,CAAC,qBAAqB,GAAG,oBAAoB,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;YACxE,MAAM,EAAE,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,+BAA+B,WAAW,EAAE,CAAC,CAAC;YACzE,MAAM,aAAK,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,qBAAqB,WAAW,CAAC,CAAC;YACzE,MAAM,aAAK,CAAC,KAAK,CAAC,aAAa,IAAI,CAAC,qBAAqB,4CAA4C,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;SACpH;aAAM,IAAI,IAAI,CAAC,cAAc,EAAE;YAC5B,MAAM,aAAK,CAAC,KAAK,CAAC,4BAA4B,WAAW,EAAE,EAAE,GAAG,IAAI,CAAC,GAAG,4BAA4B,WAAW,EAAE,CAAC,CAAC;YACnH,MAAM,aAAK,CAAC,KAAK,CAAC,eAAe,KAAK,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,4BAA4B,WAAW,EAAE,CAAC,CAAC;SACpG;QACD,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAC/B,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,gDAAgD,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAEzH,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACxB,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;YACxB,MAAM,aAAK,CAAC,KAAK,CAAC,2CAA2C,IAAI,CAAC,GAAG,+BAA+B,WAAW,MAAM,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;YACjI,IAAI,YAAY,IAAI,IAAI,EAAE;gBACtB,MAAM,aAAK,CAAC,KAAK,CAAC,YAAY,IAAI,CAAC,GAAG,+BAA+B,WAAW,gCAAgC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;aAC/H;YACD,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC/B,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,wDAAwD,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;SACpI;IACL,CAAC;IAEO,+BAA+B,CAAC,YAA0B;QAC9D,YAAY,CAAC,IAAI,CAAC,yBAAyB,CAAC,CAAC;QAC7C,IAAI,CAAC,yBAAyB,GAAG,UAAU,CAAC,GAAG,EAAE;YAC7C,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,8BAA8B,CAAC,CAAC;YAC7E,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;QACvD,CAAC,EAAE,KAAK,CAAC,CAAC;IACd,CAAC;IAEO,eAAe,CAAC,SAA2B,EAAE,IAAY,EAAE,OAAO,GAAG,KAAK,EAAE,aAAa,GAAG,EAAE;QAClG,MAAM,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;QACtD,IAAI,OAAO,EAAE;YACT,OAAO,IAAA,eAAK,EAAA,GAAG,WAAW,gCAAgC,IAAI,CAAC,QAAQ,EAAE,KAAK,aAAa,EAAE,CAAC;SACjG;QAED,OAAO,IAAA,eAAK,EAAA,GAAG,WAAW,uBAAuB,IAAI,CAAC,QAAQ,EAAE,MAAM,aAAa,EAAE,CAAC;IAC1F,CAAC;IAEO,iBAAiB,CAAC,SAA2B;QACjD,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC1C,MAAM,OAAO,GAAG,IAAA,uBAAY,EAAC,OAAO,CAAC,CAAC;QACtC,OAAO,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,yCAAyC,OAAO,GAAG,CAAC;IACxF,CAAC;IAEO,KAAK,CAAC,mBAAmB,CAAC,WAAmB;QACjD,MAAM,EAAC,MAAM,EAAE,SAAS,EAAC,GAAG,MAAM,aAAK,CAAC,KAAK,CAAC,yBAAyB,WAAW,EAAE,CAAC,CAAC;QACtF,IAAI,CAAC,iBAAiB,GAAG,SAAS,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IAC7D,CAAC;IAEO,KAAK,CAAC,6BAA6B,CAAC,WAAmB,EAAE,YAA0B;QACvF,IAAI,GAAG,GAAG,EAAE,CAAC;QACb,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,KAAK,EAAE;YAC5B,MAAM,eAAe,GAAG,MAAM,GAAG,CAAC,kBAAkB,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC9F,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;gBACtB,MAAM,IAAI,GAAG,aAAK,CAAC,UAAU,CAAC,CAAC,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACzD,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,4CAA4C,IAAI,IAAI,CAAC,CAAC;gBACnG,MAAM,QAAQ,GAAG,8BAA8B,eAAe,IAAI,IAAI,EAAE,CAAC;gBACzE,EAAE,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;gBAC3B,GAAG,IAAI,MAAM,QAAQ,YAAY,WAAW,IAAI,IAAI,GAAG,CAAC;YAC5D,CAAC,CAAC,CAAC;SACN;QACD,OAAO,GAAG,CAAC;IACf,CAAC;IAEO,KAAK,CAAC,YAAY,CAAC,YAA0B,EAAE,OAAgB,EAAE,UAAmB;QACxF,IAAI,SAAS,GAAG,2CAA2C,IAAI,CAAC,KAAK,GAAG,CAAC;QAEzE,IAAI,UAAU,EAAE;YACZ,SAAS,IAAI,eAAe,CAAC;SAChC;QAED,CAAC,OAAO,CAAC,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YAC1C,SAAS,IAAI,iBAAiB,CAAC,IAAI,CAAC;QACxC,CAAC,CAAC,CAAC;QACH,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAC9D,MAAM,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAC5D,MAAM,yBAAyB,GAAG,WAAW,CAAC,OAAO,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;QACzE,MAAM,OAAO,GAAG,CAAC,yBAAyB,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,yBAAyB,CAAC,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC,CAAC;QAC5G,IAAI,YAAY,EAAE;YACd,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SAC9B;QAED,KAAI,MAAM,KAAK,IAAI,OAAO,EAAE;YACxB,SAAS,IAAI,mBAAmB,KAAK,GAAG,CAAC;SAC5C;QAED,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE;YAC/D,SAAS,IAAI,MAAM,GAAG,KAAK,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC;SACvD;QAED,SAAS,IAAI,GAAG,WAAW,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAC3D,IAAI,OAAO,EAAE;YACT,SAAS,IAAI,IAAI,OAAO,EAAE,CAAC;SAC9B;QAED,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QAC9B,MAAM,EAAC,MAAM,EAAE,WAAW,EAAC,GAAG,MAAM,aAAK,CAAC,KAAK,CAAC,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QACrE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QACzD,IAAI,CAAC,+BAA+B,CAAC,YAAY,CAAC,CAAC;QACnD,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACrC,YAAY,CAAC,MAAM,CAAC,IAAA,eAAK,EAAA,GAAG,IAAI,CAAC,YAAY,0CAA0C,WAAW,kBAAkB,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,IAAA,uBAAY,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACvL,CAAC;;AAhyBL,kBAiyBC;AA/xBmB,mBAAe,GAAG;IAC9B,SAAS,EAAE,qBAAqB,EAAE,OAAO,EAAE,UAAU;IACrD,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE,SAAS;IACtD,cAAc,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU;CACnD,CAAC","sourcesContent":["import chalk from \"chalk\";\nimport * as dotenv from \"dotenv\";\nimport * as childProcess from \"child_process\";\nimport * as fs from \"fs-extra\";\nimport prettyHrtime from \"pretty-hrtime\";\nimport camelCase from \"camelcase\";\nimport {ExitError} from \"./types/exit-error\";\nimport {Utils} from \"./utils\";\nimport {JobOptions} from \"./types/job-options\";\nimport {WriteStreams} from \"./types/write-streams\";\nimport {Service} from \"./service\";\nimport {GitData} from \"./git-data\";\nimport {assert} from \"./asserts\";\n\nexport class Job {\n\n    static readonly illegalJobNames = [\n        \"include\", \"local_configuration\", \"image\", \"services\",\n        \"stages\", \"pages\", \"types\", \"before_script\", \"default\",\n        \"after_script\", \"variables\", \"cache\", \"workflow\",\n    ];\n\n    readonly name: string;\n    readonly jobNamePad: number;\n    readonly dependencies: string[] | null;\n    readonly environment?: { name: string; url: string | null };\n    readonly jobId: number;\n    readonly cwd: string;\n    readonly rules?: { if: string; when: string; allow_failure: boolean }[];\n    readonly expandedVariables: { [key: string]: string };\n    readonly allowFailure: boolean;\n    readonly when: string;\n    readonly pipelineIid: number;\n    readonly cache: { key: string | { files: string[] }; paths: string[] }[];\n    readonly gitData: GitData;\n    readonly shellIsolation: boolean;\n\n    private _prescriptsExitCode: number | null = null;\n    private _afterScriptsExitCode = 0;\n    private _coveragePercent: string | null = null;\n    private _running = false;\n    private _containerId: string | null = null;\n    private _serviceIds: string[] = [];\n    private _serviceNetworkId: string | null = null;\n    private _artifactsContainerId: string | null = null;\n    private _containerVolumeNames: string[] = [];\n    private _longRunningSilentTimeout: NodeJS.Timeout = -1 as any;\n    private _producers: { name: string; dotenv: string | null }[] | null = null;\n\n    private readonly _needs: { job: string; artifacts: true }[] | string[] | null;\n    private readonly jobData: any;\n    private readonly writeStreams: WriteStreams;\n    private readonly extraHosts: string[];\n    private readonly volumes: string[];\n\n    constructor(opt: JobOptions) {\n        const jobData = opt.data;\n        const gitData = opt.gitData;\n        const globals = opt.globals;\n        const homeVariables = opt.homeVariables;\n        const projectVariables = opt.projectVariables;\n\n        this.extraHosts = opt.extraHosts;\n        this.volumes = opt.volumes;\n        this.writeStreams = opt.writeStreams;\n        this.jobNamePad = opt.namePad;\n        this.gitData = opt.gitData;\n        this.name = opt.name;\n        this.cwd = opt.cwd;\n        this.jobId = Math.floor(Math.random() * 1000000);\n        this.jobData = opt.data;\n        this.pipelineIid = opt.pipelineIid;\n        this.shellIsolation = opt.shellIsolation;\n\n        this.when = jobData.when || \"on_success\";\n        this.allowFailure = jobData.allow_failure ?? false;\n        this._needs = jobData.needs || null;\n        this.dependencies = jobData.dependencies || null;\n        this.rules = jobData.rules || null;\n        this.environment = typeof jobData.environment === \"string\" ? {name: jobData.environment} : jobData.environment;\n        this.cache = jobData.cache || [];\n\n        let CI_PROJECT_DIR = `${this.cwd}`;\n        if (this.imageName) {\n            CI_PROJECT_DIR = `/builds/${this.safeJobName}`;\n        } else if (this.shellIsolation) {\n            CI_PROJECT_DIR = `${this.cwd}/.gitlab-ci-local/builds/${this.safeJobName}`;\n        }\n\n        const predefinedVariables = {\n            GITLAB_USER_LOGIN: gitData.user[\"GITLAB_USER_LOGIN\"],\n            GITLAB_USER_EMAIL: gitData.user[\"GITLAB_USER_EMAIL\"],\n            GITLAB_USER_NAME: gitData.user[\"GITLAB_USER_NAME\"],\n            GITLAB_USER_ID: gitData.user[\"GITLAB_USER_ID\"],\n            CI_COMMIT_SHORT_SHA: gitData.commit.SHORT_SHA, // Changes\n            CI_COMMIT_SHA: gitData.commit.SHA,\n            CI_PROJECT_DIR,\n            CI_PROJECT_NAME: gitData.remote.project,\n            CI_PROJECT_TITLE: `${camelCase(gitData.remote.project)}`,\n            CI_PROJECT_PATH: gitData.CI_PROJECT_PATH,\n            CI_PROJECT_PATH_SLUG: gitData.CI_PROJECT_PATH_SLUG,\n            CI_PROJECT_NAMESPACE: `${gitData.remote.group}`,\n            CI_PROJECT_VISIBILITY: \"internal\",\n            CI_PROJECT_ID: \"1217\",\n            CI_COMMIT_REF_PROTECTED: \"false\",\n            CI_COMMIT_BRANCH: gitData.commit.REF_NAME, // Not available in merge request or tag pipelines\n            CI_COMMIT_REF_NAME: gitData.commit.REF_NAME, // Tag or branch name\n            CI_COMMIT_REF_SLUG: gitData.commit.REF_NAME.replace(/[^a-z0-9]+/ig, \"-\").replace(/^-/, \"\").replace(/-$/, \"\").slice(0, 63).toLowerCase(),\n            CI_COMMIT_TITLE: \"Commit Title\", // First line of commit message.\n            CI_COMMIT_MESSAGE: \"Commit Title\\nMore commit text\", // Full commit message\n            CI_COMMIT_DESCRIPTION: \"More commit text\",\n            CI_PIPELINE_SOURCE: \"push\",\n            CI_JOB_ID: `${this.jobId}`,\n            CI_PIPELINE_ID: `${this.pipelineIid + 1000}`,\n            CI_PIPELINE_IID: `${this.pipelineIid}`,\n            CI_SERVER_HOST: `${gitData.remote.domain}`,\n            CI_SERVER_URL: `https://${gitData.remote.domain}:443`,\n            CI_API_V4_URL: `https://${gitData.remote.domain}/api/v4`,\n            CI_PROJECT_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}`,\n            CI_JOB_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}/-/jobs/${this.jobId}`, // Changes on rerun.\n            CI_PIPELINE_URL: `https://${gitData.remote.domain}/${gitData.remote.group}/${gitData.remote.project}/pipelines/${this.pipelineIid}`,\n            CI_JOB_NAME: `${this.name}`,\n            CI_JOB_STAGE: `${this.stage}`,\n            CI_REGISTRY: gitData.CI_REGISTRY,\n            CI_REGISTRY_IMAGE: gitData.CI_REGISTRY_IMAGE,\n            GITLAB_CI: \"false\",\n        };\n\n        // Create expanded variables\n        const envs = {...predefinedVariables, ...globals.variables || {}, ...jobData.variables || {}, ...homeVariables, ...projectVariables};\n        const expandedGlobalVariables = Utils.expandVariables(globals.variables || {}, envs);\n        const expandedJobVariables = Utils.expandVariables(jobData.variables || {}, envs);\n        this.expandedVariables = {...predefinedVariables, ...expandedGlobalVariables, ...expandedJobVariables, ...homeVariables, ...projectVariables};\n\n        // Set {when, allowFailure} based on rules result\n        if (this.rules) {\n            const ruleResult = Utils.getRulesResult(this.rules, this.expandedVariables);\n            this.when = ruleResult.when;\n            this.allowFailure = ruleResult.allowFailure;\n        }\n\n        if (this.interactive && (this.when !== \"manual\" || this.imageName !== null)) {\n            throw new ExitError(`${this.chalkJobName} @Interactive decorator cannot have image: and must be when:manual`);\n        }\n\n        if (this.injectSSHAgent && this.imageName === null) {\n            throw new ExitError(`${this.chalkJobName} @InjectSSHAgent can only be used with image:`);\n        }\n    }\n\n    static async getUniqueCacheName(cwd: string, cacheEntry: { key: string | { files: string[] }; paths: string[] }, expandedVariables: { [key: string]: string }) {\n        if (typeof cacheEntry.key === \"string\") {\n            return Utils.expandText(cacheEntry.key);\n        }\n        return \"md-\" + await Utils.checksumFiles(cacheEntry.key.files.map(f => {\n            return `${cwd}/${Utils.expandText(f, expandedVariables)}`;\n        }));\n    }\n\n    get artifactsToSource() {\n        return this.jobData[\"artifactsToSource\"] == null ? true : this.jobData[\"artifactsToSource\"];\n    }\n\n    get chalkJobName() {\n        return chalk`{blueBright ${this.name.padEnd(this.jobNamePad)}}`;\n    }\n\n    get safeJobName() {\n        return Utils.getSafeJobName(this.name);\n    }\n\n    get needs(): {job: string; artifacts: boolean}[] | null {\n        if (!this._needs) return null;\n        const list: {job: string; artifacts: boolean}[] = [];\n        this._needs?.forEach((need) => {\n            list.push({\n                job: typeof need === \"string\" ? need : need.job,\n                artifacts: typeof need === \"string\" ? true : need.artifacts,\n            });\n        });\n        return list;\n    }\n\n\n    get imageName(): string | null {\n        const image = this.jobData[\"image\"];\n        if (!image) {\n            return null;\n        }\n\n        const imageName = Utils.expandText(image.name, this.expandedVariables);\n        return imageName.includes(\":\") ? imageName : `${imageName}:latest`;\n    }\n\n    get imageEntrypoint(): string[] | null {\n        const image = this.jobData[\"image\"];\n\n        if (!image || !image.entrypoint) {\n            return null;\n        }\n        if (typeof image.entrypoint !== \"object\") {\n            throw new ExitError(\"image:entrypoint must be an array\");\n        }\n        return image.entrypoint;\n    }\n\n    get services(): Service[] {\n        return this.jobData[\"services\"];\n    }\n\n    get producers(): { name: string; dotenv: string | null }[] | null {\n        return this._producers;\n    }\n\n    set producers(producers: { name: string; dotenv: string | null }[] | null) {\n        assert(this._producers == null, \"this._producers can only be set once\");\n        this._producers = producers;\n    }\n\n    get stage(): string {\n        return this.jobData[\"stage\"] || \"test\";\n    }\n\n    get interactive(): boolean {\n        return this.jobData[\"interactive\"] || false;\n    }\n\n    get injectSSHAgent(): boolean {\n        return this.jobData[\"injectSSHAgent\"] || false;\n    }\n\n    get description(): string {\n        return this.jobData[\"description\"] ?? \"\";\n    }\n\n    get artifacts(): { paths?: string[]; exclude?: string[]; reports?: { dotenv?: string } }|null {\n        return this.jobData[\"artifacts\"];\n    }\n\n    get beforeScripts(): string[] {\n        return this.jobData[\"before_script\"] || [];\n    }\n\n    get afterScripts(): string[] {\n        return this.jobData[\"after_script\"] || [];\n    }\n\n    get scripts(): string[] {\n        return this.jobData[\"script\"];\n    }\n\n    get preScriptsExitCode() {\n        return this._prescriptsExitCode;\n    }\n\n    get afterScriptsExitCode() {\n        return this._afterScriptsExitCode;\n    }\n\n    get running() {\n        return this._running;\n    }\n\n    get started() {\n        return this._running || this._prescriptsExitCode !== null;\n    }\n\n    get finished() {\n        return !this._running && this._prescriptsExitCode !== null;\n    }\n\n    get coveragePercent(): string | null {\n        return this._coveragePercent;\n    }\n\n    async start(privileged: boolean): Promise<void> {\n        const startTime = process.hrtime();\n        const writeStreams = this.writeStreams;\n        const safeJobname = this.safeJobName;\n\n        this._running = true;\n\n        await fs.ensureFile(`${this.cwd}/.gitlab-ci-local/output/${safeJobname}.log`);\n        await fs.truncate(`${this.cwd}/.gitlab-ci-local/output/${safeJobname}.log`);\n\n        if (!this.interactive) {\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright starting} ${this.imageName ?? \"shell\"} ({yellow ${this.stage}})\\n`);\n        }\n\n        const prescripts = this.beforeScripts.concat(this.scripts);\n        this._prescriptsExitCode = await this.execScripts(prescripts, privileged);\n        if (this.afterScripts.length === 0 && this._prescriptsExitCode > 0 && !this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, false)}\\n`);\n            await this.cleanupResources();\n            this._running = false;\n            return;\n        }\n\n        if (this.afterScripts.length === 0 && this._prescriptsExitCode > 0 && this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, true)}\\n`);\n            await this.cleanupResources();\n            this._running = false;\n            return;\n        }\n\n        if (this._prescriptsExitCode > 0 && this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, true)}\\n`);\n        }\n\n        if (this._prescriptsExitCode > 0 && !this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, false)}\\n`);\n        }\n\n        if (this.afterScripts.length > 0) {\n            this._afterScriptsExitCode = await this.execScripts(this.afterScripts, privileged);\n        }\n\n        if (this._afterScriptsExitCode > 0) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._afterScriptsExitCode, true, \" after_script\")}\\n`);\n        }\n\n        writeStreams.stdout(`${this.getFinishedString(startTime)}\\n`);\n\n        if (this.jobData.coverage) {\n            this._coveragePercent = await Utils.getCoveragePercent(this.cwd, this.jobData.coverage, safeJobname);\n        }\n\n        await this.cleanupResources();\n        this._running = false;\n    }\n\n    async cleanupResources() {\n        const writeStreams = this.writeStreams;\n        clearTimeout(this._longRunningSilentTimeout);\n\n        if (this._containerId) {\n            try {\n                await Utils.spawn(`docker rm -f ${this._containerId}`);\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._serviceIds) {\n            try {\n                for (const serviceId of this._serviceIds) {\n                    await Utils.spawn(`docker rm -f ${serviceId}`);\n                }\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._serviceNetworkId) {\n            try {\n                await Utils.spawn(`docker network rm ${this._serviceNetworkId}`);\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._artifactsContainerId) {\n            try {\n                await Utils.spawn(`docker rm -f ${this._artifactsContainerId}`);\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._containerVolumeNames.length > 0) {\n            try {\n                for (const containerVolume of this._containerVolumeNames) {\n                    await Utils.spawn(`docker volume rm ${containerVolume}`);\n                }\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n    }\n\n    private generateInjectSSHAgentOptions() {\n        if (!this.injectSSHAgent) {\n            return \"\";\n        }\n        if (process.platform === \"darwin\" || (process.env.OSTYPE?.match(/^darwin/) ?? null)) {\n            return \"--env SSH_AUTH_SOCK=/run/host-services/ssh-auth.sock -v /run/host-services/ssh-auth.sock:/run/host-services/ssh-auth.sock\";\n        }\n        return `--env SSH_AUTH_SOCK=${process.env.SSH_AUTH_SOCK} -v ${process.env.SSH_AUTH_SOCK}:${process.env.SSH_AUTH_SOCK}`;\n    }\n\n    private generateScriptCommands(scripts: string[]) {\n        let cmd = \"\";\n        scripts.forEach((script) => {\n            // Print command echo'ed in color\n            const split = script.split(/\\r?\\n/);\n            const multilineText = split.length > 1 ? \" # collapsed multi-line command\" : \"\";\n            const text = split[0]?.replace(/[\\\\]/g, \"\\\\\\\\\").replace(/[\"]/g, \"\\\\\\\"\").replace(/[$]/g, \"\\\\$\");\n            cmd += chalk`echo \"{green $ ${text}${multilineText}}\"\\n`;\n\n            // Execute actual script\n            cmd += `${script}\\n`;\n        });\n        return cmd;\n    }\n\n    private async execScripts(scripts: string[], privileged: boolean): Promise<number> {\n        const safeJobName = this.safeJobName;\n        const outputFilesPath = `${this.cwd}/.gitlab-ci-local/output/${safeJobName}.log`;\n        const buildVolumeName = `gcl-${this.safeJobName}-${this.jobId}-build`;\n        const tmpVolumeName = `gcl-${this.safeJobName}-${this.jobId}-tmp`;\n        const writeStreams = this.writeStreams;\n        const reportsDotenvVariables = await this.initProducerReportsDotenvVariables(writeStreams);\n        let time;\n        let endTime;\n\n        if (scripts.length === 0 || scripts[0] == null) {\n            return 0;\n        }\n\n        // Copy git tracked files to build folder if shell isolation enabled.\n        if (!this.imageName && this.shellIsolation) {\n            await Utils.rsyncTrackedFiles(this.cwd, `${safeJobName}`);\n        }\n\n        if (this.interactive) {\n            const iCmd = this.generateScriptCommands(scripts);\n            const interactiveCp = childProcess.spawn(iCmd, {\n                shell: \"bash\",\n                stdio: [\"inherit\", \"inherit\", \"inherit\"],\n                cwd: this.cwd,\n                env: {...this.expandedVariables, ...process.env},\n            });\n            return new Promise<number>((resolve, reject) => {\n                interactiveCp.on(\"exit\", (code) => resolve(code ?? 0));\n                interactiveCp.on(\"error\", (err) => reject(err));\n            });\n        }\n\n        this.refreshLongRunningSilentTimeout(writeStreams);\n\n        if (this.imageName) {\n            await this.pullImage(writeStreams, this.imageName);\n\n            let dockerCmd = \"\";\n            if (privileged) {\n                dockerCmd += `docker create --privileged -u 0:0 -i ${this.generateInjectSSHAgentOptions()} `;\n            } else {\n                dockerCmd += `docker create -u 0:0 -i ${this.generateInjectSSHAgentOptions()} `;\n            }\n            if (this.services?.length) {\n                await this.createDockerNetwork(`gitlab-ci-local-${this.jobId}`);\n                dockerCmd += `--network gitlab-ci-local-${this.jobId} `;\n                for (const service of this.services) {\n                    await this.pullImage(writeStreams, service.getName(this.expandedVariables));\n                    await this.startService(writeStreams, service, privileged);\n                }\n            }\n\n            const volumePromises = [];\n            volumePromises.push(Utils.spawn(`docker volume create ${buildVolumeName}`, this.cwd));\n            volumePromises.push(Utils.spawn(`docker volume create ${tmpVolumeName}`, this.cwd));\n            dockerCmd += `--volume ${buildVolumeName}:/builds/${safeJobName} `;\n            dockerCmd += `--volume ${tmpVolumeName}:/tmp/ `;\n            this._containerVolumeNames.push(buildVolumeName);\n            this._containerVolumeNames.push(tmpVolumeName);\n            await Promise.all(volumePromises);\n\n            dockerCmd += `--workdir /builds/${safeJobName} `;\n\n            for (const volume of this.volumes) {\n                dockerCmd += `--volume ${volume} `;\n            }\n\n            for (const extraHost of this.extraHosts) {\n                dockerCmd += `--add-host=${extraHost} `;\n            }\n\n            if (this.imageEntrypoint) {\n                this.imageEntrypoint.forEach((e) => {\n                    dockerCmd += `--entrypoint \"${e}\" `;\n                });\n            }\n\n            for (const [key, value] of Object.entries(this.expandedVariables)) {\n                dockerCmd += `-e ${key}='${String(value).trim()}' `;\n            }\n            for (const [key, value] of Object.entries(reportsDotenvVariables)) {\n                dockerCmd += `-e ${key}='${String(value).trim()}' `;\n            }\n\n            dockerCmd += await this.createCacheDockerVolumeMounts(safeJobName, writeStreams);\n\n            dockerCmd += `${this.imageName} sh -c \"\\n`;\n            dockerCmd += \"if [ -x /usr/local/bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /usr/local/bin/bash \\n\";\n            dockerCmd += \"elif [ -x /usr/bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /usr/bin/bash \\n\";\n            dockerCmd += \"elif [ -x /bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /bin/bash \\n\";\n            dockerCmd += \"elif [ -x /usr/local/bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /usr/local/bin/sh \\n\";\n            dockerCmd += \"elif [ -x /usr/bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /usr/bin/sh \\n\";\n            dockerCmd += \"elif [ -x /bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /bin/sh \\n\";\n            dockerCmd += \"elif [ -x /busybox/sh ]; then\\n\";\n            dockerCmd += \"\\texec /busybox/sh \\n\";\n            dockerCmd += \"else\\n\";\n            dockerCmd += \"\\techo shell not found\\n\";\n            dockerCmd += \"\\texit 1\\n\";\n            dockerCmd += \"fi\\n\\\"\";\n\n            const {stdout: containerId} = await Utils.spawn(dockerCmd, this.cwd);\n            this._containerId = containerId.replace(/\\r?\\n/g, \"\");\n\n            time = process.hrtime();\n            // Copy source files into container.\n            await Utils.spawn(`docker cp .gitlab-ci-local/builds/.docker/. ${this._containerId}:/builds/${safeJobName}`, this.cwd);\n            this.refreshLongRunningSilentTimeout(writeStreams);\n\n            // Copy file variables into container.\n            const fileVariablesFolder = `/tmp/gitlab-ci-local-file-variables-${this.gitData.CI_PROJECT_PATH_SLUG}/`;\n            if (await fs.pathExists(fileVariablesFolder)) {\n                await Utils.spawn(`docker cp ${fileVariablesFolder} ${this._containerId}:${fileVariablesFolder}/`, this.cwd);\n                this.refreshLongRunningSilentTimeout(writeStreams);\n            }\n\n            endTime = process.hrtime(time);\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright copied to container} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n\n        await this.copyArtifactsIn(writeStreams);\n\n        if (this.imageName) {\n            // Make sure tracked files and artifacts are root owned in docker-executor jobs.\n            await Utils.spawn(`docker run --rm -w /app/ -v ${buildVolumeName}:/app/ firecow/gitlab-ci-local-util bash -c \"chown 0:0 -R . && chmod a+rw -R .\"`);\n        }\n\n        let cmd = \"set -eo pipefail\\n\";\n        cmd += \"exec 0< /dev/null\\n\";\n\n        if (!this.imageName && this.shellIsolation) {\n            cmd += `cd .gitlab-ci-local/builds/${safeJobName}/\\n`;\n        }\n        cmd += this.generateScriptCommands(scripts);\n\n        cmd += \"exit 0\\n\";\n\n        await fs.outputFile(`${this.cwd}/.gitlab-ci-local/scripts/${safeJobName}`, cmd, \"utf-8\");\n        await fs.chmod(`${this.cwd}/.gitlab-ci-local/scripts/${safeJobName}`, \"0755\");\n\n        if (this.imageName) {\n            await Utils.spawn(`docker cp .gitlab-ci-local/scripts/. ${this._containerId}:/gcl-scripts/`, this.cwd);\n        }\n\n        const cp = childProcess.spawn(this._containerId ? `docker start --attach -i ${this._containerId}` : \"bash\", {\n            shell: \"bash\",\n            stdio: [\"pipe\", \"pipe\", \"pipe\"],\n            cwd: this.cwd,\n            env: this.imageName ? {...process.env} : {...this.expandedVariables, ...reportsDotenvVariables, ...process.env},\n        });\n\n        const outFunc = (e: any, stream: (txt: string) => void, colorize: (str: string) => string) => {\n            this.refreshLongRunningSilentTimeout(writeStreams);\n            for (const line of `${e}`.split(/\\r?\\n/)) {\n                if (line.length === 0) {\n                    continue;\n                }\n\n                stream(`${this.chalkJobName} `);\n                if (!line.startsWith(\"\\u001b[32m$\")) {\n                    stream(`${colorize(\">\")} `);\n                }\n                stream(`${line}\\n`);\n                fs.appendFileSync(outputFilesPath, `${line}\\n`);\n            }\n        };\n\n        const exitCode = await new Promise<number>((resolve, reject) => {\n            cp.stdout.on(\"data\", (e) => outFunc(e, writeStreams.stdout.bind(writeStreams), (s) => chalk`{greenBright ${s}}`));\n            cp.stderr.on(\"data\", (e) => outFunc(e, writeStreams.stderr.bind(writeStreams), (s) => chalk`{redBright ${s}}`));\n\n            cp.on(\"exit\", (code) => setTimeout(() => resolve(code ?? 0), 10));\n            cp.on(\"error\", (err) => setTimeout(() => reject(err), 10));\n\n            if (this.imageName) {\n                cp.stdin.end(`/gcl-scripts/${safeJobName}`);\n            } else {\n                cp.stdin.end(`./.gitlab-ci-local/scripts/${safeJobName}`);\n            }\n        });\n\n        await this.copyArtifactsOut(writeStreams, buildVolumeName);\n\n        return exitCode;\n    }\n\n    private async pullImage(writeStreams: WriteStreams, imageToPull: string) {\n        const time = process.hrtime();\n        let pullCmd = \"\";\n        pullCmd += `docker image ls --format '{{.Repository}}:{{.Tag}}' | grep -E '^${imageToPull}$'\\n`;\n        pullCmd += \"if [ \\\"$?\\\" -ne 0 ]; then\\n\";\n        pullCmd += `\\techo \"Pulling ${imageToPull}\"\\n`;\n        pullCmd += `\\tdocker pull ${imageToPull}\\n`;\n        pullCmd += \"fi\\n\";\n        await Utils.spawn(pullCmd, this.cwd);\n        this.refreshLongRunningSilentTimeout(writeStreams);\n        const endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright pulled} ${imageToPull} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n\n    private async initProducerReportsDotenvVariables(writeStreams: WriteStreams) {\n        const producers = this.producers;\n        let producerReportsEnvs = {};\n        for (const producer of producers ?? []) {\n            if (producer.dotenv === null) continue;\n\n            const safeProducerName = Utils.getSafeJobName(producer.name);\n            let dotenvFile;\n            if (!this.shellIsolation && !this.imageName) {\n                dotenvFile = `${this.cwd}/${producer.dotenv}`;\n            } else {\n                dotenvFile = `${this.cwd}/.gitlab-ci-local/artifacts/${safeProducerName}/.gitlab-ci-reports/dotenv/${producer.dotenv}`;\n            }\n            if (await fs.pathExists(dotenvFile)) {\n                const producerReportEnv = dotenv.parse(await fs.readFile(dotenvFile));\n                producerReportsEnvs = {...producerReportsEnvs, ...producerReportEnv};\n            } else {\n                writeStreams.stderr(chalk`${this.chalkJobName} {yellow '${producer.dotenv}' produced by '${producer.name}' could not be found}\\n`);\n            }\n\n        }\n        return producerReportsEnvs;\n    }\n\n    private async copyArtifactsIn(writeStreams: WriteStreams) {\n        if (!this.imageName && !this.shellIsolation) {\n            return;\n        }\n\n        if (!this.producers || this.producers.length === 0) {\n            return;\n        }\n\n        const safeJobName = this.safeJobName;\n\n        const cpFunc = async (folder: string) => {\n            if (!this.imageName && this.shellIsolation) {\n                return Utils.spawn(`rsync -a ${folder}/. ${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);\n            }\n            return Utils.spawn(`docker cp ${folder}/. ${this._containerId}:/builds/${safeJobName}`);\n        };\n\n        const time = process.hrtime();\n        const promises = [];\n        for (const producer of this.producers) {\n            const producerSafeName = Utils.getSafeJobName(producer.name);\n            const artifactFolder = `${this.cwd}/.gitlab-ci-local/artifacts/${producerSafeName}`;\n            if (!await fs.pathExists(artifactFolder)) {\n                throw new ExitError(`${artifactFolder} doesn't exist, did you forget --needs`);\n            }\n            promises.push(cpFunc(artifactFolder));\n        }\n        await Promise.all(promises);\n        const endTime = process.hrtime(time);\n        const targetText = this.imageName ? \"container\" : \"isolated shell\";\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright copied artifacts to ${targetText}} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n\n    private async copyArtifactsOut(writeStreams: WriteStreams, buildVolumeName: string) {\n        const safeJobName = this.safeJobName;\n\n        if (!this.shellIsolation && !this.imageName || !this.artifacts) {\n            return;\n        }\n\n        let time, endTime;\n        let cpCmd = \"shopt -s globstar nullglob dotglob\\n\";\n        cpCmd += `mkdir -p ../../artifacts/${safeJobName}\\n`;\n        for (const artifactPath of this.artifacts?.paths ?? []) {\n            const expandedPath = Utils.expandText(artifactPath, this.expandedVariables);\n            cpCmd += `echo Started copying ${expandedPath} to ../../artifacts/${safeJobName}\\n`;\n            cpCmd += `rsync -Ra ${expandedPath} ../../artifacts/${safeJobName}/.\\n`;\n            cpCmd += `echo Done copying ${expandedPath} to ../../artifacts/${safeJobName}\\n`;\n        }\n\n        for (const artifactExcludePath of this.artifacts?.exclude ?? []) {\n            const expandedPath = Utils.expandText(artifactExcludePath, this.expandedVariables);\n            cpCmd += `echo Started removing exclude '${expandedPath}' from ../../artifacts/${safeJobName}\\n`;\n            cpCmd += `ls -1d '../../artifacts/${safeJobName}/${expandedPath}' | xargs -n1 rm -rf || true\\n`;\n            cpCmd += `echo Done removing exclude '${expandedPath}' from ../../artifacts/${safeJobName}\\n`;\n        }\n\n        const reportDotenv = this.artifacts.reports?.dotenv ?? null;\n        if (reportDotenv != null) {\n            cpCmd += `mkdir -p ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\\n`;\n            cpCmd += `echo Started copying ${reportDotenv} to ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\\n`;\n            cpCmd += `rsync -Ra ${reportDotenv} ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv/.\\n`;\n            cpCmd += `echo Done copying ${reportDotenv} to ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\\n`;\n        }\n\n        time = process.hrtime();\n        if (this.imageName) {\n            const cacheMountStr = await this.createCacheDockerVolumeMounts(safeJobName, writeStreams);\n            const dockerCreateCmd = `docker create -i ${cacheMountStr} -v ${buildVolumeName}:/builds/${safeJobName}/ -w /builds/${safeJobName}/ firecow/gitlab-ci-local-util bash -c \"${cpCmd}\"`;\n            const {stdout: artifactsContainerId} = await Utils.spawn(dockerCreateCmd, this.cwd);\n            this._artifactsContainerId = artifactsContainerId.replace(/\\r?\\n/g, \"\");\n            await fs.mkdirp(`${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}`);\n            await Utils.spawn(`docker start ${this._artifactsContainerId} --attach`);\n            await Utils.spawn(`docker cp ${this._artifactsContainerId}:/artifacts/. .gitlab-ci-local/artifacts/.`, this.cwd);\n        } else if (this.shellIsolation) {\n            await Utils.spawn(`mkdir -p ../../artifacts/${safeJobName}`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);\n            await Utils.spawn(`bash -e -c \"${cpCmd}\"`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);\n        }\n        endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright saved artifacts} in {magenta ${prettyHrtime(endTime)}}\\n`);\n\n        if (this.artifactsToSource) {\n            time = process.hrtime();\n            await Utils.spawn(`rsync --exclude=/.gitlab-ci-reports/ -a ${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}/. ${this.cwd}`);\n            if (reportDotenv != null) {\n                await Utils.spawn(`rsync -a ${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}/.gitlab-ci-reports/dotenv/. ${this.cwd}`);\n            }\n            endTime = process.hrtime(time);\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright copied artifacts to cwd} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n    }\n\n    private refreshLongRunningSilentTimeout(writeStreams: WriteStreams) {\n        clearTimeout(this._longRunningSilentTimeout);\n        this._longRunningSilentTimeout = setTimeout(() => {\n            writeStreams.stdout(chalk`${this.chalkJobName} {grey > still running...}\\n`);\n            this.refreshLongRunningSilentTimeout(writeStreams);\n        }, 10000);\n    }\n\n    private getExitedString(startTime: [number, number], code: number, warning = false, prependString = \"\") {\n        const finishedStr = this.getFinishedString(startTime);\n        if (warning) {\n            return chalk`${finishedStr} {black.bgYellowBright  WARN ${code.toString()} }${prependString}`;\n        }\n\n        return chalk`${finishedStr} {black.bgRed  FAIL ${code.toString()} } ${prependString}`;\n    }\n\n    private getFinishedString(startTime: [number, number]) {\n        const endTime = process.hrtime(startTime);\n        const timeStr = prettyHrtime(endTime);\n        return chalk`${this.chalkJobName} {magentaBright finished} in {magenta ${timeStr}}`;\n    }\n\n    private async createDockerNetwork(networkName: string) {\n        const {stdout: networkId} = await Utils.spawn(`docker network create ${networkName}`);\n        this._serviceNetworkId = networkId.replace(/\\r?\\n/g, \"\");\n    }\n\n    private async createCacheDockerVolumeMounts(safeJobName: string, writeStreams: WriteStreams) {\n        let cmd = \"\";\n        for (const entry of this.cache) {\n            const uniqueCacheName = await Job.getUniqueCacheName(this.cwd, entry, this.expandedVariables);\n            entry.paths.forEach((p) => {\n                const path = Utils.expandText(p, this.expandedVariables);\n                writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright mounting cache} for path ${path}\\n`);\n                const cachedir = `/tmp/gitlab-ci-local/cache/${uniqueCacheName}/${path}`;\n                fs.ensureDirSync(cachedir);\n                cmd += `-v ${cachedir}:/builds/${safeJobName}/${path} `;\n            });\n        }\n        return cmd;\n    }\n\n    private async startService(writeStreams: WriteStreams, service: Service, privileged: boolean) {\n        let dockerCmd = `docker run -d --network gitlab-ci-local-${this.jobId} `;\n\n        if (privileged) {\n            dockerCmd += \"--privileged \";\n        }\n\n        (service.getEntrypoint() ?? []).forEach((e) => {\n            dockerCmd += `--entrypoint \"${e}\" `;\n        });\n        const serviceAlias = service.getAlias(this.expandedVariables);\n        const serviceName = service.getName(this.expandedVariables);\n        const serviceNameWithoutVersion = serviceName.replace(/(.*)(:.*)/, \"$1\");\n        const aliases = [serviceNameWithoutVersion.replace(\"/\", \"-\"), serviceNameWithoutVersion.replace(\"/\", \"__\")];\n        if (serviceAlias) {\n            aliases.push(serviceAlias);\n        }\n\n        for(const alias of aliases) {\n            dockerCmd += `--network-alias=${alias} `;\n        }\n\n        for (const [key, value] of Object.entries(this.expandedVariables)) {\n            dockerCmd += `-e ${key}=\"${String(value).trim()}\" `;\n        }\n\n        dockerCmd += `${serviceName}`;\n        const command = service.getCommand(this.expandedVariables);\n        if (command) {\n            dockerCmd += ` ${command}`;\n        }\n\n        const time = process.hrtime();\n        const {stdout: containerId} = await Utils.spawn(dockerCmd, this.cwd);\n        this._serviceIds.push(containerId.replace(/\\r?\\n/g, \"\"));\n        this.refreshLongRunningSilentTimeout(writeStreams);\n        const endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright started service image: ${serviceName} with aliases: ${aliases.join(\", \")}} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n}\n"]}
802
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"job.js","sourceRoot":"","sources":["job.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport * as dotenv from \"dotenv\";\nimport * as childProcess from \"child_process\";\nimport * as fs from \"fs-extra\";\nimport prettyHrtime from \"pretty-hrtime\";\nimport camelCase from \"camelcase\";\nimport {ExitError} from \"./types/exit-error\";\nimport {Utils} from \"./utils\";\nimport {JobOptions} from \"./types/job-options\";\nimport {WriteStreams} from \"./types/write-streams\";\nimport {Service} from \"./service\";\nimport {GitData} from \"./git-data\";\nimport {assert} from \"./asserts\";\nimport {CacheEntry} from \"./cache-entry\";\nimport {Mutex} from \"./mutex\";\n\nexport class Job {\n\n    static readonly illegalJobNames = [\n        \"include\", \"local_configuration\", \"image\", \"services\",\n        \"stages\", \"pages\", \"types\", \"before_script\", \"default\",\n        \"after_script\", \"variables\", \"cache\", \"workflow\",\n    ];\n\n    readonly name: string;\n    readonly jobNamePad: number;\n    readonly dependencies: string[] | null;\n    readonly environment?: { name: string; url: string | null };\n    readonly jobId: number;\n    readonly cwd: string;\n    readonly rules?: { if: string; when: string; allow_failure: boolean }[];\n    readonly expandedVariables: { [key: string]: string };\n    readonly allowFailure: boolean;\n    readonly when: string;\n    readonly pipelineIid: number;\n    readonly gitData: GitData;\n    readonly shellIsolation: boolean;\n    readonly mountCache: boolean;\n\n    private _prescriptsExitCode: number | null = null;\n    private _afterScriptsExitCode = 0;\n    private _coveragePercent: string | null = null;\n    private _running = false;\n    private _containerId: string | null = null;\n    private _serviceNetworkId: string | null = null;\n    private _containerVolumeNames: string[] = [];\n    private _longRunningSilentTimeout: NodeJS.Timeout = -1 as any;\n    private _producers: { name: string; dotenv: string | null }[] | null = null;\n\n    private _containersToClean: string[] = [];\n\n    private readonly jobData: any;\n    private readonly writeStreams: WriteStreams;\n    private readonly extraHosts: string[];\n    private readonly volumes: string[];\n\n    constructor(opt: JobOptions) {\n        const jobData = opt.data;\n        const gitData = opt.gitData;\n        const globals = opt.globals;\n        const variablesFromFiles = opt.variablesFromFiles;\n        const cliVariables = opt.cliVariables;\n\n        this.mountCache = opt.mountCache;\n        this.extraHosts = opt.extraHosts;\n        this.volumes = opt.volumes;\n        this.writeStreams = opt.writeStreams;\n        this.jobNamePad = opt.namePad;\n        this.gitData = opt.gitData;\n        this.name = opt.name;\n        this.cwd = opt.cwd;\n        this.jobId = Math.floor(Math.random() * 1000000);\n        this.jobData = opt.data;\n        this.pipelineIid = opt.pipelineIid;\n        this.shellIsolation = opt.shellIsolation;\n\n        this.when = jobData.when || \"on_success\";\n        this.allowFailure = jobData.allow_failure ?? false;\n        this.dependencies = jobData.dependencies || null;\n        this.rules = jobData.rules || null;\n        this.environment = typeof jobData.environment === \"string\" ? {name: jobData.environment} : jobData.environment;\n\n        let CI_PROJECT_DIR = `${this.cwd}`;\n        if (this.imageName) {\n            CI_PROJECT_DIR = `/builds/${this.safeJobName}`;\n        } else if (this.shellIsolation) {\n            CI_PROJECT_DIR = `${this.cwd}/.gitlab-ci-local/builds/${this.safeJobName}`;\n        }\n\n        const predefinedVariables = {\n            GITLAB_USER_LOGIN: gitData.user[\"GITLAB_USER_LOGIN\"],\n            GITLAB_USER_EMAIL: gitData.user[\"GITLAB_USER_EMAIL\"],\n            GITLAB_USER_NAME: gitData.user[\"GITLAB_USER_NAME\"],\n            GITLAB_USER_ID: gitData.user[\"GITLAB_USER_ID\"],\n            CI_COMMIT_SHORT_SHA: gitData.commit.SHORT_SHA, // Changes\n            CI_COMMIT_SHA: gitData.commit.SHA,\n            CI_PROJECT_DIR,\n            CI_PROJECT_NAME: gitData.remote.project,\n            CI_PROJECT_TITLE: `${camelCase(gitData.remote.project)}`,\n            CI_PROJECT_PATH: gitData.CI_PROJECT_PATH,\n            CI_PROJECT_PATH_SLUG: gitData.CI_PROJECT_PATH_SLUG,\n            CI_PROJECT_NAMESPACE: `${gitData.remote.group}`,\n            CI_PROJECT_VISIBILITY: \"internal\",\n            CI_PROJECT_ID: \"1217\",\n            CI_COMMIT_REF_PROTECTED: \"false\",\n            CI_COMMIT_BRANCH: gitData.commit.REF_NAME, // Not available in merge request or tag pipelines\n            CI_COMMIT_REF_NAME: gitData.commit.REF_NAME, // Tag or branch name\n            CI_COMMIT_REF_SLUG: gitData.commit.REF_NAME.replace(/[^a-z0-9]+/ig, \"-\").replace(/^-/, \"\").replace(/-$/, \"\").slice(0, 63).toLowerCase(),\n            CI_COMMIT_TITLE: \"Commit Title\", // First line of commit message.\n            CI_COMMIT_MESSAGE: \"Commit Title\\nMore commit text\", // Full commit message\n            CI_COMMIT_DESCRIPTION: \"More commit text\",\n            CI_PIPELINE_SOURCE: \"push\",\n            CI_JOB_ID: `${this.jobId}`,\n            CI_PIPELINE_ID: `${this.pipelineIid + 1000}`,\n            CI_PIPELINE_IID: `${this.pipelineIid}`,\n            CI_SERVER_HOST: `${gitData.remote.host}`,\n            CI_SERVER_PORT: `${gitData.remote.port}`,\n            CI_SERVER_URL: `https://${gitData.remote.host}:443`,\n            CI_SERVER_PROTOCOL: \"https\",\n            CI_API_V4_URL: `https://${gitData.remote.host}/api/v4`,\n            CI_PROJECT_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}`,\n            CI_JOB_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}/-/jobs/${this.jobId}`, // Changes on rerun.\n            CI_PIPELINE_URL: `https://${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}/pipelines/${this.pipelineIid}`,\n            CI_JOB_NAME: `${this.name}`,\n            CI_JOB_STAGE: `${this.stage}`,\n            CI_REGISTRY: gitData.CI_REGISTRY,\n            CI_REGISTRY_IMAGE: gitData.CI_REGISTRY_IMAGE,\n            GITLAB_CI: \"false\",\n        };\n\n        // Create expanded variables\n        const envs = {...predefinedVariables, ...globals.variables || {}, ...jobData.variables || {}, ...variablesFromFiles, ...cliVariables};\n        const expandedGlobalVariables = Utils.expandVariables(globals.variables || {}, envs);\n        const expandedJobVariables = Utils.expandVariables(jobData.variables || {}, envs);\n        this.expandedVariables = {...predefinedVariables, ...expandedGlobalVariables, ...expandedJobVariables, ...variablesFromFiles, ...cliVariables};\n\n        // Set {when, allowFailure} based on rules result\n        if (this.rules) {\n            const ruleResult = Utils.getRulesResult(this.rules, this.expandedVariables);\n            this.when = ruleResult.when;\n            this.allowFailure = ruleResult.allowFailure;\n        }\n\n        if (this.interactive && (this.when !== \"manual\" || this.imageName !== null)) {\n            throw new ExitError(`${this.chalkJobName} @Interactive decorator cannot have image: and must be when:manual`);\n        }\n\n        if (this.injectSSHAgent && this.imageName === null) {\n            throw new ExitError(`${this.chalkJobName} @InjectSSHAgent can only be used with image:`);\n        }\n\n        if (this.imageName && this.mountCache) {\n            for (const c of this.cache) {\n                c.paths.forEach((p) => {\n                    const path = Utils.expandText(p, this.expandedVariables);\n                    if (path.includes(\"*\")) {\n                        throw new ExitError(`${this.name} cannot have * in cache paths, when --mount-cache is enabled`);\n                    }\n                });\n            }\n        }\n    }\n\n    get artifactsToSource() {\n        return this.jobData[\"artifactsToSource\"] == null ? true : this.jobData[\"artifactsToSource\"];\n    }\n\n    get chalkJobName() {\n        return chalk`{blueBright ${this.name.padEnd(this.jobNamePad)}}`;\n    }\n\n    get safeJobName() {\n        return Utils.getSafeJobName(this.name);\n    }\n\n    get needs(): {job: string; artifacts: boolean}[] | null {\n        const needs = this.jobData[\"needs\"];\n        if (!needs) return null;\n        const list: {job: string; artifacts: boolean}[] = [];\n        needs.forEach((need: any) => {\n            list.push({\n                job: typeof need === \"string\" ? need : need.job,\n                artifacts: typeof need === \"string\" ? true : need.artifacts,\n            });\n        });\n        return list;\n    }\n\n    get cache(): CacheEntry[] {\n        let cacheData = this.jobData[\"cache\"];\n        const cacheList: CacheEntry[] = [];\n        if (!cacheData) return [];\n\n        cacheData = Array.isArray(cacheData) ? cacheData : [cacheData];\n        cacheData.forEach((c: any) => {\n            const key = c[\"key\"];\n            const policy = c[\"policy\"] ?? \"pull-push\";\n            if (![\"pull\", \"push\", \"pull-push\"].includes(policy)) {\n                throw new ExitError(\"cache policy is not 'pull', 'push' or 'pull-push'\");\n            }\n            const paths = c[\"paths\"] ?? [];\n            cacheList.push(new CacheEntry(key, paths, policy));\n        });\n        return cacheList;\n    }\n\n    get buildVolumeName(): string {\n        return `gcl-${this.safeJobName}-${this.jobId}-build`;\n    }\n\n    get tmpVolumeName(): string {\n        return `gcl-${this.safeJobName}-${this.jobId}-tmp`;\n    }\n\n    get imageName(): string | null {\n        const image = this.jobData[\"image\"];\n        if (!image) {\n            return null;\n        }\n\n        const imageName = Utils.expandText(image.name, this.expandedVariables);\n        return imageName.includes(\":\") ? imageName : `${imageName}:latest`;\n    }\n\n    get imageEntrypoint(): string[] | null {\n        const image = this.jobData[\"image\"];\n\n        if (!image || !image.entrypoint) {\n            return null;\n        }\n        if (typeof image.entrypoint !== \"object\") {\n            throw new ExitError(\"image:entrypoint must be an array\");\n        }\n        return image.entrypoint;\n    }\n\n    get services(): Service[] {\n        return this.jobData[\"services\"];\n    }\n\n    get producers(): { name: string; dotenv: string | null }[] | null {\n        return this._producers;\n    }\n\n    set producers(producers: { name: string; dotenv: string | null }[] | null) {\n        assert(this._producers == null, \"this._producers can only be set once\");\n        this._producers = producers;\n    }\n\n    get stage(): string {\n        return this.jobData[\"stage\"] || \"test\";\n    }\n\n    get interactive(): boolean {\n        return this.jobData[\"interactive\"] || false;\n    }\n\n    get injectSSHAgent(): boolean {\n        return this.jobData[\"injectSSHAgent\"] || false;\n    }\n\n    get description(): string {\n        return this.jobData[\"description\"] ?? \"\";\n    }\n\n    get artifacts(): { paths?: string[]; exclude?: string[]; reports?: { dotenv?: string } }|null {\n        return this.jobData[\"artifacts\"];\n    }\n\n    get beforeScripts(): string[] {\n        return this.jobData[\"before_script\"] || [];\n    }\n\n    get afterScripts(): string[] {\n        return this.jobData[\"after_script\"] || [];\n    }\n\n    get scripts(): string[] {\n        return this.jobData[\"script\"];\n    }\n\n    get preScriptsExitCode() {\n        return this._prescriptsExitCode;\n    }\n\n    get afterScriptsExitCode() {\n        return this._afterScriptsExitCode;\n    }\n\n    get running() {\n        return this._running;\n    }\n\n    get started() {\n        return this._running || this._prescriptsExitCode !== null;\n    }\n\n    get finished() {\n        return !this._running && this._prescriptsExitCode !== null;\n    }\n\n    get coveragePercent(): string | null {\n        return this._coveragePercent;\n    }\n\n    async start(privileged: boolean): Promise<void> {\n        const startTime = process.hrtime();\n        const writeStreams = this.writeStreams;\n        const safeJobname = this.safeJobName;\n\n        this._running = true;\n\n        await fs.ensureFile(`${this.cwd}/.gitlab-ci-local/output/${safeJobname}.log`);\n        await fs.truncate(`${this.cwd}/.gitlab-ci-local/output/${safeJobname}.log`);\n\n        if (!this.interactive) {\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright starting} ${this.imageName ?? \"shell\"} ({yellow ${this.stage}})\\n`);\n        }\n\n        const prescripts = this.beforeScripts.concat(this.scripts);\n        this._prescriptsExitCode = await this.execScripts(prescripts, privileged);\n        if (this.afterScripts.length === 0 && this._prescriptsExitCode > 0 && !this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, false)}\\n`);\n            await this.cleanupResources();\n            this._running = false;\n            return;\n        }\n\n        if (this.afterScripts.length === 0 && this._prescriptsExitCode > 0 && this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, true)}\\n`);\n            await this.cleanupResources();\n            this._running = false;\n            return;\n        }\n\n        if (this._prescriptsExitCode > 0 && this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, true)}\\n`);\n        }\n\n        if (this._prescriptsExitCode > 0 && !this.allowFailure) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._prescriptsExitCode, false)}\\n`);\n        }\n\n        if (this.afterScripts.length > 0) {\n            this._afterScriptsExitCode = await this.execScripts(this.afterScripts, privileged);\n        }\n\n        if (this._afterScriptsExitCode > 0) {\n            writeStreams.stderr(`${this.getExitedString(startTime, this._afterScriptsExitCode, true, \" after_script\")}\\n`);\n        }\n\n        writeStreams.stdout(`${this.getFinishedString(startTime)}\\n`);\n\n        if (this.jobData.coverage) {\n            this._coveragePercent = await Utils.getCoveragePercent(this.cwd, this.jobData.coverage, safeJobname);\n        }\n\n        await this.cleanupResources();\n        this._running = false;\n    }\n\n    async cleanupResources() {\n        const writeStreams = this.writeStreams;\n        clearTimeout(this._longRunningSilentTimeout);\n\n        for (const id of this._containersToClean) {\n            try {\n                await Utils.spawn(`docker rm -f ${id}`);\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._serviceNetworkId) {\n            try {\n                await Utils.spawn(`docker network rm ${this._serviceNetworkId}`);\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n\n        if (this._containerVolumeNames.length > 0) {\n            try {\n                for (const containerVolume of this._containerVolumeNames) {\n                    await Utils.spawn(`docker volume rm ${containerVolume}`);\n                }\n            } catch (e) {\n                assert(e instanceof Error, \"e is not instanceof Error\");\n                writeStreams.stderr(chalk`{yellow ${e.message}}`);\n            }\n        }\n    }\n\n    private generateInjectSSHAgentOptions() {\n        if (!this.injectSSHAgent) {\n            return \"\";\n        }\n        if (process.platform === \"darwin\" || (process.env.OSTYPE?.match(/^darwin/) ?? null)) {\n            return \"--env SSH_AUTH_SOCK=/run/host-services/ssh-auth.sock -v /run/host-services/ssh-auth.sock:/run/host-services/ssh-auth.sock\";\n        }\n        return `--env SSH_AUTH_SOCK=${process.env.SSH_AUTH_SOCK} -v ${process.env.SSH_AUTH_SOCK}:${process.env.SSH_AUTH_SOCK}`;\n    }\n\n    private generateScriptCommands(scripts: string[]) {\n        let cmd = \"\";\n        scripts.forEach((script) => {\n            // Print command echo'ed in color\n            const split = script.split(/\\r?\\n/);\n            const multilineText = split.length > 1 ? \" # collapsed multi-line command\" : \"\";\n            const text = split[0]?.replace(/[\\\\]/g, \"\\\\\\\\\").replace(/[\"]/g, \"\\\\\\\"\").replace(/[$]/g, \"\\\\$\");\n            cmd += chalk`echo \"{green $ ${text}${multilineText}}\"\\n`;\n\n            // Execute actual script\n            cmd += `${script}\\n`;\n        });\n        return cmd;\n    }\n\n    private async mountCacheCmd(safeJobName: string, writeStreams: WriteStreams) {\n        if (this.imageName && !this.mountCache) return \"\";\n\n        let cmd = \"\";\n        for (const c of this.cache) {\n            const uniqueCacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);\n            c.paths.forEach((p) => {\n                const path = Utils.expandText(p, this.expandedVariables);\n                writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright mounting cache} for path ${path}\\n`);\n                const cacheMount = `gcl-${this.expandedVariables.CI_PROJECT_PATH_SLUG}-${uniqueCacheName}`;\n                cmd += `-v ${cacheMount}:/builds/${safeJobName}/${path} `;\n            });\n        }\n        return cmd;\n    }\n\n    private async execScripts(scripts: string[], privileged: boolean): Promise<number> {\n        const safeJobName = this.safeJobName;\n        const outputFilesPath = `${this.cwd}/.gitlab-ci-local/output/${safeJobName}.log`;\n        const buildVolumeName = this.buildVolumeName;\n        const tmpVolumeName = this.tmpVolumeName;\n        const writeStreams = this.writeStreams;\n        const reportsDotenvVariables = await this.initProducerReportsDotenvVariables(writeStreams);\n        let time;\n        let endTime;\n\n        if (scripts.length === 0 || scripts[0] == null) {\n            return 0;\n        }\n\n        // Copy git tracked files to build folder if shell isolation enabled.\n        if (!this.imageName && this.shellIsolation) {\n            await Utils.rsyncTrackedFiles(this.cwd, `${safeJobName}`);\n        }\n\n        if (this.interactive) {\n            const iCmd = this.generateScriptCommands(scripts);\n            const interactiveCp = childProcess.spawn(iCmd, {\n                shell: \"bash\",\n                stdio: [\"inherit\", \"inherit\", \"inherit\"],\n                cwd: this.cwd,\n                env: {...this.expandedVariables, ...process.env},\n            });\n            return new Promise<number>((resolve, reject) => {\n                interactiveCp.on(\"exit\", (code) => resolve(code ?? 0));\n                interactiveCp.on(\"error\", (err) => reject(err));\n            });\n        }\n\n        this.refreshLongRunningSilentTimeout(writeStreams);\n\n        if (this.imageName) {\n            await this.pullImage(writeStreams, this.imageName);\n\n            let dockerCmd = \"\";\n            if (privileged) {\n                dockerCmd += `docker create --privileged -u 0:0 -i ${this.generateInjectSSHAgentOptions()} `;\n            } else {\n                dockerCmd += `docker create -u 0:0 -i ${this.generateInjectSSHAgentOptions()} `;\n            }\n            if (this.services?.length) {\n                await this.createDockerNetwork(`gitlab-ci-local-${this.jobId}`);\n                dockerCmd += `--network gitlab-ci-local-${this.jobId} `;\n                for (const service of this.services) {\n                    await this.pullImage(writeStreams, service.getName(this.expandedVariables));\n                    await this.startService(writeStreams, service, privileged);\n                    await this.serviceHealthCheck(writeStreams, service);\n                }\n            }\n\n            const volumePromises = [];\n            volumePromises.push(Utils.spawn(`docker volume create ${buildVolumeName}`, this.cwd));\n            volumePromises.push(Utils.spawn(`docker volume create ${tmpVolumeName}`, this.cwd));\n            dockerCmd += `--volume ${buildVolumeName}:/builds/${safeJobName} `;\n            dockerCmd += `--volume ${tmpVolumeName}:/tmp/ `;\n            this._containerVolumeNames.push(buildVolumeName);\n            this._containerVolumeNames.push(tmpVolumeName);\n            await Promise.all(volumePromises);\n\n            dockerCmd += `--workdir /builds/${safeJobName} `;\n\n            for (const volume of this.volumes) {\n                dockerCmd += `--volume ${volume} `;\n            }\n\n            for (const extraHost of this.extraHosts) {\n                dockerCmd += `--add-host=${extraHost} `;\n            }\n\n            if (this.imageEntrypoint) {\n                this.imageEntrypoint.forEach((e) => {\n                    dockerCmd += `--entrypoint \"${e}\" `;\n                });\n            }\n\n            for (const [key, value] of Object.entries(this.expandedVariables)) {\n                dockerCmd += `-e ${key}='${String(value).trim()}' `;\n            }\n            for (const [key, value] of Object.entries(reportsDotenvVariables)) {\n                dockerCmd += `-e ${key}='${String(value).trim()}' `;\n            }\n\n            dockerCmd += await this.mountCacheCmd(safeJobName, writeStreams);\n\n            dockerCmd += `${this.imageName} sh -c \"\\n`;\n            dockerCmd += \"if [ -x /usr/local/bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /usr/local/bin/bash \\n\";\n            dockerCmd += \"elif [ -x /usr/bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /usr/bin/bash \\n\";\n            dockerCmd += \"elif [ -x /bin/bash ]; then\\n\";\n            dockerCmd += \"\\texec /bin/bash \\n\";\n            dockerCmd += \"elif [ -x /usr/local/bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /usr/local/bin/sh \\n\";\n            dockerCmd += \"elif [ -x /usr/bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /usr/bin/sh \\n\";\n            dockerCmd += \"elif [ -x /bin/sh ]; then\\n\";\n            dockerCmd += \"\\texec /bin/sh \\n\";\n            dockerCmd += \"elif [ -x /busybox/sh ]; then\\n\";\n            dockerCmd += \"\\texec /busybox/sh \\n\";\n            dockerCmd += \"else\\n\";\n            dockerCmd += \"\\techo shell not found\\n\";\n            dockerCmd += \"\\texit 1\\n\";\n            dockerCmd += \"fi\\n\\\"\";\n\n            const {stdout: containerId} = await Utils.spawn(dockerCmd, this.cwd);\n            this._containerId = containerId.replace(/\\r?\\n/g, \"\");\n            this._containersToClean.push(this._containerId);\n\n            time = process.hrtime();\n            // Copy source files into container.\n            await Utils.spawn(`docker cp .gitlab-ci-local/builds/.docker/. ${this._containerId}:/builds/${safeJobName}`, this.cwd);\n            this.refreshLongRunningSilentTimeout(writeStreams);\n\n            // Copy file variables into container.\n            const fileVariablesFolder = `/tmp/gitlab-ci-local-file-variables-${this.gitData.CI_PROJECT_PATH_SLUG}/`;\n            if (await fs.pathExists(fileVariablesFolder)) {\n                await Utils.spawn(`docker cp ${fileVariablesFolder} ${this._containerId}:${fileVariablesFolder}/`, this.cwd);\n                this.refreshLongRunningSilentTimeout(writeStreams);\n            }\n\n            endTime = process.hrtime(time);\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright copied to container} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n\n        await this.copyCacheIn(writeStreams);\n        await this.copyArtifactsIn(writeStreams);\n\n        if (this.imageName) {\n            // Files in docker-executor build folder must be root owned.\n            await Utils.spawn(`docker run --rm -v ${tmpVolumeName}:/tmp/ -v ${buildVolumeName}:/app/ firecow/gitlab-ci-local-util bash -c \"chown 0:0 -R /app/ && chmod a+rw -R /app/ && chmod a+rw -R /tmp/\"`);\n        }\n\n        let cmd = \"set -eo pipefail\\n\";\n        cmd += \"exec 0< /dev/null\\n\";\n\n        if (!this.imageName && this.shellIsolation) {\n            cmd += `cd .gitlab-ci-local/builds/${safeJobName}/\\n`;\n        }\n        cmd += this.generateScriptCommands(scripts);\n\n        cmd += \"exit 0\\n\";\n\n        await fs.outputFile(`${this.cwd}/.gitlab-ci-local/scripts/${safeJobName}`, cmd, \"utf-8\");\n        await fs.chmod(`${this.cwd}/.gitlab-ci-local/scripts/${safeJobName}`, \"0755\");\n\n        if (this.imageName) {\n            await Utils.spawn(`docker cp .gitlab-ci-local/scripts/. ${this._containerId}:/gcl-scripts/`, this.cwd);\n        }\n\n        const cp = childProcess.spawn(this._containerId ? `docker start --attach -i ${this._containerId}` : \"bash\", {\n            shell: \"bash\",\n            stdio: [\"pipe\", \"pipe\", \"pipe\"],\n            cwd: this.cwd,\n            env: this.imageName ? {...process.env} : {...this.expandedVariables, ...reportsDotenvVariables, ...process.env},\n        });\n\n        const outFunc = (e: any, stream: (txt: string) => void, colorize: (str: string) => string) => {\n            this.refreshLongRunningSilentTimeout(writeStreams);\n            for (const line of `${e}`.split(/\\r?\\n/)) {\n                if (line.length === 0) {\n                    continue;\n                }\n\n                stream(`${this.chalkJobName} `);\n                if (!line.startsWith(\"\\u001b[32m$\")) {\n                    stream(`${colorize(\">\")} `);\n                }\n                stream(`${line}\\n`);\n                fs.appendFileSync(outputFilesPath, `${line}\\n`);\n            }\n        };\n\n        const exitCode = await new Promise<number>((resolve, reject) => {\n            cp.stdout.on(\"data\", (e) => outFunc(e, writeStreams.stdout.bind(writeStreams), (s) => chalk`{greenBright ${s}}`));\n            cp.stderr.on(\"data\", (e) => outFunc(e, writeStreams.stderr.bind(writeStreams), (s) => chalk`{redBright ${s}}`));\n\n            cp.on(\"exit\", (code) => setTimeout(() => resolve(code ?? 0), 10));\n            cp.on(\"error\", (err) => setTimeout(() => reject(err), 10));\n\n            if (this.imageName) {\n                cp.stdin.end(`/gcl-scripts/${safeJobName}`);\n            } else {\n                cp.stdin.end(`./.gitlab-ci-local/scripts/${safeJobName}`);\n            }\n        });\n\n        if (exitCode == 0) {\n            await this.copyCacheOut(writeStreams);\n            await this.copyArtifactsOut(writeStreams);\n        }\n\n        return exitCode;\n    }\n\n    private async pullImage(writeStreams: WriteStreams, imageToPull: string) {\n        const time = process.hrtime();\n        let pullCmd = \"\";\n        pullCmd += `docker image ls --format '{{.Repository}}:{{.Tag}}' | grep -E '^${imageToPull}$'\\n`;\n        pullCmd += \"if [ \\\"$?\\\" -ne 0 ]; then\\n\";\n        pullCmd += `\\techo \"Pulling ${imageToPull}\"\\n`;\n        pullCmd += `\\tdocker pull ${imageToPull}\\n`;\n        pullCmd += \"fi\\n\";\n        await Utils.spawn(pullCmd, this.cwd);\n        this.refreshLongRunningSilentTimeout(writeStreams);\n        const endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright pulled} ${imageToPull} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n\n    private async initProducerReportsDotenvVariables(writeStreams: WriteStreams) {\n        const producers = this.producers;\n        let producerReportsEnvs = {};\n        for (const producer of producers ?? []) {\n            if (producer.dotenv === null) continue;\n\n            const safeProducerName = Utils.getSafeJobName(producer.name);\n            let dotenvFile;\n            if (!this.shellIsolation && !this.imageName) {\n                dotenvFile = `${this.cwd}/${producer.dotenv}`;\n            } else {\n                dotenvFile = `${this.cwd}/.gitlab-ci-local/artifacts/${safeProducerName}/.gitlab-ci-reports/dotenv/${producer.dotenv}`;\n            }\n            if (await fs.pathExists(dotenvFile)) {\n                const producerReportEnv = dotenv.parse(await fs.readFile(dotenvFile));\n                producerReportsEnvs = {...producerReportsEnvs, ...producerReportEnv};\n            } else {\n                writeStreams.stderr(chalk`${this.chalkJobName} {yellow '${producer.dotenv}' produced by '${producer.name}' could not be found}\\n`);\n            }\n\n        }\n        return producerReportsEnvs;\n    }\n\n    private async copyCacheIn(writeStreams: WriteStreams) {\n        if (this.mountCache && this.imageName) return;\n        if ((!this.imageName && !this.shellIsolation) || this.cache.length === 0) return;\n\n        for (const c of this.cache) {\n            if (![\"pull\", \"pull-push\"].includes(c.policy)) return;\n\n            const time = process.hrtime();\n            const cacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);\n            const cacheFolder = `${this.cwd}/.gitlab-ci-local/cache/${cacheName}`;\n            if (!await fs.pathExists(cacheFolder)) {\n                continue;\n            }\n\n            await Mutex.exclusive(cacheName, async() => {\n                await this.copyIn(cacheFolder);\n            });\n            const endTime = process.hrtime(time);\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright imported cache '${cacheName}'} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n    }\n\n    private async copyArtifactsIn(writeStreams: WriteStreams) {\n        if ((!this.imageName && !this.shellIsolation) || (this.producers ?? []).length === 0) return;\n\n        const time = process.hrtime();\n        const promises = [];\n        for (const producer of this.producers ?? []) {\n            const producerSafeName = Utils.getSafeJobName(producer.name);\n            const artifactFolder = `${this.cwd}/.gitlab-ci-local/artifacts/${producerSafeName}`;\n            if (!await fs.pathExists(artifactFolder)) {\n                throw new ExitError(`${artifactFolder} doesn't exist, did you forget --needs`);\n            }\n            promises.push(this.copyIn(artifactFolder));\n        }\n        await Promise.all(promises);\n        const endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright imported artifacts} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n\n    copyIn(source: string) {\n        const safeJobName = this.safeJobName;\n        if (!this.imageName && this.shellIsolation) {\n            return Utils.spawn(`rsync -a ${source}/. ${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);\n        }\n        return Utils.spawn(`docker cp ${source}/. ${this._containerId}:/builds/${safeJobName}`);\n    }\n\n    private async copyCacheOut(writeStreams: WriteStreams) {\n        if (this.mountCache && this.imageName) return;\n        if ((!this.imageName && !this.shellIsolation) || this.cache.length === 0) return;\n\n        let time, endTime;\n        for (const c of this.cache) {\n            if (![\"push\", \"pull-push\"].includes(c.policy)) return;\n            const cacheName = await c.getUniqueCacheName(this.cwd, this.expandedVariables);\n            for (const path of c.paths) {\n                time = process.hrtime();\n                const expandedPath = Utils.expandText(path, this.expandedVariables);\n                let cmd = \"shopt -s globstar nullglob dotglob\\n\";\n                cmd += `mkdir -p ../../cache/${cacheName}\\n`;\n                cmd += `rsync -Ra ${expandedPath} ../../cache/${cacheName}/. || true\\n`;\n\n                await Mutex.exclusive(cacheName, async() => {\n                    await this.copyOut(cmd, \"cache\", []);\n                });\n                endTime = process.hrtime(time);\n\n                const readdir = await fs.readdir(`${this.cwd}/.gitlab-ci-local/cache/${cacheName}`);\n                if (readdir.length === 0) {\n                    writeStreams.stdout(chalk`${this.chalkJobName} {yellow !! no cache was copied for ${path} !!}\\n`);\n                } else {\n                    writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright exported cache ${expandedPath} '${cacheName}'} in {magenta ${prettyHrtime(endTime)}}\\n`);\n                }\n            }\n        }\n    }\n\n    private async copyArtifactsOut(writeStreams: WriteStreams) {\n        const safeJobName = this.safeJobName;\n\n        if (!this.shellIsolation && !this.imageName || !this.artifacts) return;\n\n        let time, endTime;\n        let cpCmd = \"shopt -s globstar nullglob dotglob\\n\";\n        cpCmd += `mkdir -p ../../artifacts/${safeJobName}\\n`;\n        for (const artifactPath of this.artifacts?.paths ?? []) {\n            const expandedPath = Utils.expandText(artifactPath, this.expandedVariables);\n            cpCmd += `rsync -Ra ${expandedPath} ../../artifacts/${safeJobName}/. || true\\n`;\n        }\n\n        for (const artifactExcludePath of this.artifacts?.exclude ?? []) {\n            const expandedPath = Utils.expandText(artifactExcludePath, this.expandedVariables);\n            cpCmd += `ls -1d '../../artifacts/${safeJobName}/${expandedPath}' | xargs -n1 rm -rf || true\\n`;\n        }\n\n        const reportDotenv = this.artifacts.reports?.dotenv ?? null;\n        if (reportDotenv != null) {\n            cpCmd += `mkdir -p ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv\\n`;\n            cpCmd += `rsync -Ra ${reportDotenv} ../../artifacts/${safeJobName}/.gitlab-ci-reports/dotenv/.\\n`;\n        }\n\n        time = process.hrtime();\n        const dockerCmdExtras = this.mountCache ? [await this.mountCacheCmd(this.safeJobName, writeStreams)] : [];\n        await this.copyOut(cpCmd, \"artifacts\", dockerCmdExtras);\n        endTime = process.hrtime(time);\n\n        const readdir = await fs.readdir(`${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}`);\n        if (readdir.length === 0) {\n            writeStreams.stdout(chalk`${this.chalkJobName} {yellow !! no artifacts was copied !!}\\n`);\n        } else {\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright exported artifacts} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n\n        if (this.artifactsToSource) {\n            time = process.hrtime();\n            await Utils.spawn(`rsync --exclude=/.gitlab-ci-reports/ -a ${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}/. ${this.cwd}`);\n            if (reportDotenv != null) {\n                await Utils.spawn(`rsync -a ${this.cwd}/.gitlab-ci-local/artifacts/${safeJobName}/.gitlab-ci-reports/dotenv/. ${this.cwd}`);\n            }\n            endTime = process.hrtime(time);\n            writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright copied artifacts to cwd} in {magenta ${prettyHrtime(endTime)}}\\n`);\n        }\n    }\n\n    private async copyOut(cmd: string, type: \"artifacts\" | \"cache\", dockerCmdExtras: string[]) {\n        const safeJobName = this.safeJobName;\n        const buildVolumeName = this.buildVolumeName;\n\n        await fs.mkdirp(`${this.cwd}/.gitlab-ci-local/${type}`);\n\n        if (this.imageName) {\n            const {stdout: cid} = await Utils.spawn(`docker create -i ${dockerCmdExtras.join(\" \")} -v ${buildVolumeName}:/builds/${safeJobName}/ -w /builds/${safeJobName}/ firecow/gitlab-ci-local-util bash -c \"${cmd}\"`, this.cwd);\n            const containerId = cid.replace(/\\r?\\n/g, \"\");\n            this._containersToClean.push(containerId);\n            await Utils.spawn(`docker start ${containerId} --attach`);\n            await Utils.spawn(`docker cp ${containerId}:/${type}/. .gitlab-ci-local/${type}/.`, this.cwd);\n        } else if (this.shellIsolation) {\n            await Utils.spawn(`bash -eo pipefail -c \"${cmd}\"`, `${this.cwd}/.gitlab-ci-local/builds/${safeJobName}`);\n        }\n    }\n\n    private refreshLongRunningSilentTimeout(writeStreams: WriteStreams) {\n        clearTimeout(this._longRunningSilentTimeout);\n        this._longRunningSilentTimeout = setTimeout(() => {\n            writeStreams.stdout(chalk`${this.chalkJobName} {grey > still running...}\\n`);\n            this.refreshLongRunningSilentTimeout(writeStreams);\n        }, 10000);\n    }\n\n    private getExitedString(startTime: [number, number], code: number, warning = false, prependString = \"\") {\n        const finishedStr = this.getFinishedString(startTime);\n        if (warning) {\n            return chalk`${finishedStr} {black.bgYellowBright  WARN ${code.toString()} }${prependString}`;\n        }\n\n        return chalk`${finishedStr} {black.bgRed  FAIL ${code.toString()} } ${prependString}`;\n    }\n\n    private getFinishedString(startTime: [number, number]) {\n        const endTime = process.hrtime(startTime);\n        const timeStr = prettyHrtime(endTime);\n        return chalk`${this.chalkJobName} {magentaBright finished} in {magenta ${timeStr}}`;\n    }\n\n    private async createDockerNetwork(networkName: string) {\n        const {stdout: networkId} = await Utils.spawn(`docker network create ${networkName}`);\n        this._serviceNetworkId = networkId.replace(/\\r?\\n/g, \"\");\n    }\n\n    private async startService(writeStreams: WriteStreams, service: Service, privileged: boolean) {\n        let dockerCmd = `docker run -d --network gitlab-ci-local-${this.jobId} `;\n\n        if (privileged) {\n            dockerCmd += \"--privileged \";\n        }\n\n        (service.getEntrypoint() ?? []).forEach((e) => {\n            dockerCmd += `--entrypoint \"${e}\" `;\n        });\n        const serviceAlias = service.getAlias(this.expandedVariables);\n        const serviceName = service.getName(this.expandedVariables);\n        const serviceNameWithoutVersion = serviceName.replace(/(.*)(:.*)/, \"$1\");\n        const aliases = [serviceNameWithoutVersion.replace(\"/\", \"-\"), serviceNameWithoutVersion.replace(\"/\", \"__\")];\n        if (serviceAlias) {\n            aliases.push(serviceAlias);\n        }\n\n        for(const alias of aliases) {\n            dockerCmd += `--network-alias=${alias} `;\n        }\n\n        for (const [key, value] of Object.entries(this.expandedVariables)) {\n            dockerCmd += `-e ${key}=\"${String(value).trim()}\" `;\n        }\n\n        dockerCmd += `${serviceName}`;\n        const command = service.getCommand(this.expandedVariables);\n        if (command) {\n            dockerCmd += ` ${command}`;\n        }\n\n        const time = process.hrtime();\n        const {stdout: containerId} = await Utils.spawn(dockerCmd, this.cwd);\n        this._containersToClean.push(containerId.replace(/\\r?\\n/g, \"\"));\n        this.refreshLongRunningSilentTimeout(writeStreams);\n        const endTime = process.hrtime(time);\n        writeStreams.stdout(chalk`${this.chalkJobName} {magentaBright started service image: ${serviceName} with aliases: ${aliases.join(\", \")}} in {magenta ${prettyHrtime(endTime)}}\\n`);\n    }\n\n    private async serviceHealthCheck(writeStreams: WriteStreams, service: Service) {\n        const dockerInspectCmd = `docker inspect ${service.getName(this.expandedVariables)}`;\n        const {stdout: imageDetails} = await Utils.spawn(dockerInspectCmd, this.cwd);\n        const imageDetailObj = JSON.parse(imageDetails);\n\n        // Copied from the startService block. Important thing is that the aliases match\n        const serviceAlias = service.getAlias(this.expandedVariables);\n        const serviceName = service.getName(this.expandedVariables);\n        const serviceNameWithoutVersion = serviceName.replace(/(.*)(:.*)/, \"$1\");\n        const aliases = [serviceNameWithoutVersion.replace(\"/\", \"-\"), serviceNameWithoutVersion.replace(\"/\", \"__\")];\n        if (serviceAlias) {\n            aliases.push(serviceAlias);\n        }\n\n        // Iterate over each port defined in the image, and try to connect to the alias\n        for(const port of Object.keys(imageDetailObj[0].ContainerConfig.ExposedPorts)) {\n            if(port.endsWith(\"/tcp\")) {\n                const portNum = parseInt(port.replace(\"/tcp\", \"\"));\n\n                let dockerCmd = `docker run -d --network gitlab-ci-local-${this.jobId} `;\n\n                dockerCmd += ` willwill/wait-for-it \"${aliases[0]}:${portNum}\" -t 30`;\n                const time = process.hrtime();\n                const {status: result, stdout} = await Utils.spawn(dockerCmd, this.cwd);\n                this._containersToClean.push(stdout.replace(/\\r?\\n/g, \"\"));\n                const endTime = process.hrtime(time);\n                if(result == 0){\n                    writeStreams.stdout(chalk`${this.chalkJobName} {greenBright service image: ${serviceName} healthcheck passed: ${aliases[0]}:${portNum}} in {green ${prettyHrtime(endTime)}}\\n`);\n                }else{\n                    writeStreams.stdout(chalk`${this.chalkJobName} {redBright service image: ${serviceName} healthcheck failed: ${aliases[0]}:${portNum}} in {red ${prettyHrtime(endTime)}}\\n`);\n                }\n            }\n        }\n    }\n}\n"]}