这个问题与我是否应该关注过量的、未运行的Docker容器有关。

我想知道如何移除旧容器。docker rm 3e552code34a允许你删除一个,但我已经有很多了。Docker rm——help没有提供选择选项(像all一样,或者通过映像名称)。

也许有一个存放这些容器的目录,我可以很容易地手动删除它们?


当前回答

一个伟大的自动替代- containrrr/watchtower

它包含图像自动清理,更新和监控选项

version: "3.9"
services:
  service:
    container_name: watchtower
    image: containrrr/watchtower:latest
    hostname: watchtower
    environment:
      - TZ=Europe/Bucharest
      - WATCHTOWER_CLEANUP=true
      - WATCHTOWER_INCLUDE_RESTARTING=true
      - WATCHTOWER_POLL_INTERVAL=86400
      #- WATCHTOWER_SCHEDULE=0 0 4 * * *
      - WATCHTOWER_LABEL_ENABLE=false
      - WATCHTOWER_MONITOR_ONLY=false
      - WATCHTOWER_NO_PULL=false
      - WATCHTOWER_NO_RESTART=false
    volumes:
      - /var/run/docker.sock:/var/run/docker.sock
    restart: unless-stopped

其他回答

我建议你先停止图像,然后删除。

你可以这样说:

$ docker stop $(docker ps -a)
$ docker rm $(docker ps -a)

您可以使用一些Docker UI应用程序来删除容器。

抱歉做广告,但我总是用我自己的应用程序做同样的事情。如果您正在寻找一个简单的应用程序来管理Docker映像或容器,您可以尝试一下:https://github.com/alex-agency/AMHub。

这是一个运行在Docker容器内的Docker UI web应用程序。 为了安装它,你只需要调用这个命令:

docker run -d -p 80:80 -p 8000:8000 -e DOCKER=$(which docker) -v /var/run/docker.sock:/docker.sock alexagency/amhub

更新后的答案 现在使用docker系统修剪或docker容器修剪。请看VonC的最新答案。

以前的回答 综合上面几个不同的提示,删除所有非运行容器的最优雅的方法似乎是:

Docker rm $(Docker ps -q -f status=exit)

-q只打印容器id(没有列标题) -f允许你过滤打印的容器列表(在这种情况下,我们过滤只显示退出的容器)

下面是我的docker-cleanup脚本,它删除未标记的容器和图像。请检查来源的任何更新。

#!/bin/sh
# Cleanup docker files: untagged containers and images.
#
# Use `docker-cleanup -n` for a dry run to see what would be deleted.

untagged_containers() {
  # Print containers using untagged images: $1 is used with awk's print: 0=line, 1=column 1.
  docker ps -a | awk '$2 ~ "[0-9a-f]{12}" {print $'$1'}'
}

untagged_images() {
  # Print untagged images: $1 is used with awk's print: 0=line, 3=column 3.
  # NOTE: intermediate images (via -a) seem to only cause
  # "Error: Conflict, foobarid wasn't deleted" messages.
  # Might be useful sometimes when Docker messed things up?!
  # docker images -a | awk '$1 == "<none>" {print $'$1'}'
  docker images | tail -n +2 | awk '$1 == "<none>" {print $'$1'}'
}

# Dry-run.
if [ "$1" = "-n" ]; then
  echo "=== Containers with uncommitted images: ==="
  untagged_containers 0
  echo

  echo "=== Uncommitted images: ==="
  untagged_images 0

  exit
fi

# Remove containers with untagged images.
echo "Removing containers:" >&2
untagged_containers 1 | xargs --no-run-if-empty docker rm --volumes=true

# Remove untagged images
echo "Removing images:" >&2
untagged_images 3 | xargs --no-run-if-empty docker rmi

来源:https://github.com/blueyed/dotfiles/blob/master/usr/bin/docker-cleanup

因此,我个人建议将此作为映像和容器的部署脚本的一部分,只保留最近的n个容器和映像。我用与git标签相同的版本控制模式标记Docker映像,并且总是用“latest”标记最新的Docker映像。这意味着在没有清理任何东西的情况下,我的Docker映像最终看起来像这样:

REPOSITORY              TAG       IMAGE ID        CREATED         VIRTUAL SIZE
some_repo/some_image    0.0.5     8f1a7c7ba93c    23 hours ago    925.4 MB
some_repo/some_image    latest    8f1a7c7ba93c    23 hours ago    925.4 MB
some_repo/some_image    0.0.4     0beabfa514ea    45 hours ago    925.4 MB
some_repo/some_image    0.0.3     54302cd10bf2    6 days ago      978.5 MB
some_repo/some_image    0.0.2     0078b30f3d9a    7 days ago      978.5 MB
some_repo/some_image    0.0.1     sdfgdf0f3d9a    8 days ago      938.5 MB

现在,当然我不想让我所有的图像(或容器)在我所有的生产盒上永久保存。我只需要最后的3或4个回滚来去掉其他的东西。尤尼克斯的尾巴是你最好的朋友。由于docker images和docker ps都是按日期排序的,我们可以使用tail选择除前三个之外的所有图像并删除它们:

docker rmi $(docker images -q | tail -n +4)

将其与部署脚本一起运行(或在本地运行),以始终保留足够的映像,以便能够轻松地回滚,而不会占用太多空间或将旧映像弄得乱七八糟。

就我个人而言,我只在我的生产盒上保留一个容器,但如果你想要更多的容器,你可以对容器做同样的事情:

docker rm $(docker ps -aq | tail -n +4)

最后,在我的简化示例中,我们一次只处理一个存储库,但是如果您有更多的存储库,您可以使用相同的思想来变得更复杂一些。假设我只想保留some_repo/some_image中的最后三张图像。我只需要混合grep和awk就可以了:

docker rmi $(docker images -a | grep 'some_repo/some_image' | awk '{print $3}' | tail -n +4)

同样,同样的想法也适用于容器,但到这里你已经明白了,所以我就不举例子了。