From 3371b9540356b5875c12e4578a9af7b1d8adfbc6 Mon Sep 17 00:00:00 2001
From: Rhys Arkins <rhys@arkins.net>
Date: Mon, 9 Mar 2020 15:56:50 +0100
Subject: [PATCH] feat: clean up dangling docker containers at startup (#5678)

---
 .../__snapshots__/artifacts.spec.ts.snap      | 18 +++++++++++++++
 lib/util/exec/__snapshots__/exec.spec.ts.snap |  2 ++
 lib/util/exec/docker/index.ts                 | 22 +++++++++++++++++++
 lib/util/exec/index.ts                        | 13 +++++++++--
 lib/util/index.ts                             |  4 ++--
 lib/workers/global/index.ts                   |  2 +-
 6 files changed, 56 insertions(+), 5 deletions(-)

diff --git a/lib/manager/cocoapods/__snapshots__/artifacts.spec.ts.snap b/lib/manager/cocoapods/__snapshots__/artifacts.spec.ts.snap
index 62b4123ef4..1d56f775ee 100644
--- a/lib/manager/cocoapods/__snapshots__/artifacts.spec.ts.snap
+++ b/lib/manager/cocoapods/__snapshots__/artifacts.spec.ts.snap
@@ -15,6 +15,12 @@ exports[`.updateArtifacts() catches write error 2`] = `Array []`;
 
 exports[`.updateArtifacts() dynamically selects Docker image tag 1`] = `
 Array [
+  Object {
+    "cmd": "docker ps --filter label=renovate_child -aq | xargs docker rm -f",
+    "options": Object {
+      "encoding": "utf-8",
+    },
+  },
   Object {
     "cmd": "docker pull renovate/cocoapods:1.2.4",
     "options": Object {
@@ -49,6 +55,12 @@ Array [
 
 exports[`.updateArtifacts() falls back to the \`latest\` Docker image tag 1`] = `
 Array [
+  Object {
+    "cmd": "docker ps --filter label=renovate_child -aq | xargs docker rm -f",
+    "options": Object {
+      "encoding": "utf-8",
+    },
+  },
   Object {
     "cmd": "docker pull renovate/cocoapods:latest",
     "options": Object {
@@ -163,6 +175,12 @@ Array [
 
 exports[`.updateArtifacts() returns updated Podfile 2`] = `
 Array [
+  Object {
+    "cmd": "docker ps --filter label=renovate_child -aq | xargs docker rm -f",
+    "options": Object {
+      "encoding": "utf-8",
+    },
+  },
   Object {
     "cmd": "docker pull renovate/cocoapods",
     "options": Object {
diff --git a/lib/util/exec/__snapshots__/exec.spec.ts.snap b/lib/util/exec/__snapshots__/exec.spec.ts.snap
index 0c830a3709..052196846b 100644
--- a/lib/util/exec/__snapshots__/exec.spec.ts.snap
+++ b/lib/util/exec/__snapshots__/exec.spec.ts.snap
@@ -4,6 +4,7 @@ exports[`Child process execution wrapper Supports image prefetch 1`] = `
 Array [
   "echo hello",
   "echo hello",
+  "docker ps --filter label=renovate_child -aq | xargs docker rm -f",
   "docker pull example/image",
   "docker ps --filter name=example_image -aq | xargs docker rm -f",
   "docker run --rm --name=example_image --label=renovate_child example/image bash -l -c \\"echo hello\\"",
@@ -11,6 +12,7 @@ Array [
   "docker run --rm --name=example_image --label=renovate_child example/image bash -l -c \\"echo hello\\"",
   "echo hello",
   "echo hello",
+  "docker ps --filter label=renovate_child -aq | xargs docker rm -f",
   "docker ps --filter name=example_image -aq | xargs docker rm -f",
   "docker run --rm --name=example_image --label=renovate_child example/image bash -l -c \\"echo hello\\"",
   "docker ps --filter name=example_image -aq | xargs docker rm -f",
diff --git a/lib/util/exec/docker/index.ts b/lib/util/exec/docker/index.ts
index 031989ada9..f1043fa1bc 100644
--- a/lib/util/exec/docker/index.ts
+++ b/lib/util/exec/docker/index.ts
@@ -134,6 +134,28 @@ export async function removeDockerContainer(image): Promise<void> {
   }
 }
 
+// istanbul ignore next
+export async function removeDanglingContainers(): Promise<void> {
+  try {
+    const res = await rawExec(
+      `docker ps --filter label=renovate_child -aq | xargs docker rm -f`,
+      { encoding: 'utf-8' }
+    );
+    if (res?.stdout?.trim().length) {
+      const containerIds = res.stdout
+        .trim()
+        .split('\n')
+        .map(container => container.trim())
+        .filter(Boolean);
+      logger.debug({ containerIds }, 'Removed dangling child containers');
+    } else {
+      logger.trace('No dangling containers to remove');
+    }
+  } catch (err) {
+    logger.warn({ err }, 'Error removing dangling containers');
+  }
+}
+
 export async function generateDockerCommand(
   commands: string[],
   options: DockerOptions,
diff --git a/lib/util/exec/index.ts b/lib/util/exec/index.ts
index 2593b29552..0ed33a1df9 100644
--- a/lib/util/exec/index.ts
+++ b/lib/util/exec/index.ts
@@ -1,7 +1,11 @@
 import { dirname, join } from 'path';
 import { hrtime } from 'process';
 import { ExecOptions as ChildProcessExecOptions } from 'child_process';
-import { generateDockerCommand, removeDockerContainer } from './docker';
+import {
+  generateDockerCommand,
+  removeDockerContainer,
+  removeDanglingContainers,
+} from './docker';
 import { getChildProcessEnv } from './env';
 import { logger } from '../../logger';
 import {
@@ -22,11 +26,16 @@ const execConfig: ExecConfig = {
   cacheDir: null,
 };
 
-export function setExecConfig(config: Partial<RenovateConfig>): void {
+export async function setExecConfig(
+  config: Partial<RenovateConfig>
+): Promise<void> {
   for (const key of Object.keys(execConfig)) {
     const value = config[key];
     execConfig[key] = value || null;
   }
+  if (execConfig.binarySource === 'docker') {
+    await removeDanglingContainers();
+  }
 }
 
 type ExtraEnv<T = unknown> = Record<string, T>;
diff --git a/lib/util/index.ts b/lib/util/index.ts
index 181fa5ee30..ca1081ab0e 100644
--- a/lib/util/index.ts
+++ b/lib/util/index.ts
@@ -1,7 +1,7 @@
 import { setExecConfig } from './exec';
 import { setFsConfig } from './fs';
 
-export function setUtilConfig(config: any): void {
-  setExecConfig(config);
+export async function setUtilConfig(config: any): Promise<void> {
+  await setExecConfig(config);
   setFsConfig(config);
 }
diff --git a/lib/workers/global/index.ts b/lib/workers/global/index.ts
index 9131e7e461..d633666b3f 100644
--- a/lib/workers/global/index.ts
+++ b/lib/workers/global/index.ts
@@ -77,7 +77,7 @@ export async function start(): Promise<0 | 1> {
         break;
       }
       const repoConfig = await getRepositoryConfig(config, repository);
-      setUtilConfig(repoConfig);
+      await setUtilConfig(repoConfig);
       if (repoConfig.hostRules) {
         hostRules.clear();
         repoConfig.hostRules.forEach(rule => hostRules.add(rule));
-- 
GitLab