我正在尝试比较两个 debian 11 系统(服务器 1 和服务器 2)、一个 debian 9 和一个 11 系统(服务器 3 和服务器 2),并确保它们具有相同的软件包。我打算使用 ansible 安装缺少的软件包。
我查看了Debian 在哪里存储已安装软件包的列表?但它不显示 apt 包版本,也显示依赖项。我只需要安装在 system1 上的带有版本号的 debian 软件包,这样我就可以在 system2 上进行比较并安装缺少的软件包。
我正在尝试比较两个 debian 11 系统(服务器 1 和服务器 2)、一个 debian 9 和一个 11 系统(服务器 3 和服务器 2),并确保它们具有相同的软件包。我打算使用 ansible 安装缺少的软件包。
我查看了Debian 在哪里存储已安装软件包的列表?但它不显示 apt 包版本,也显示依赖项。我只需要安装在 system1 上的带有版本号的 debian 软件包,这样我就可以在 system2 上进行比较并安装缺少的软件包。
将列出所有手动安装的包,即显式安装而不是作为依赖项拉入(并保持安装)的包。
要获取相应的版本,请使用
但是,您不能使用它来可靠地将包选择从一个系统复制到另一个系统,因为依赖性计算会在不同的时间点产生不同的结果(随着包依赖性的发展)。这意味着
apt
与几个月或几年前相比,现在要求安装一组软件包不会在系统上产生完全相同的结果——它可能会产生相同的结果,特别是如果两个系统都是使用Debian 的相同基本版本,但您不能依赖它来执行此操作。还要注意,依赖特定版本并不是那么有用:
apt
只能安装存储库中当前可用的任何版本,因此除非您在“源”系统上保留了包含版本的存储库,并且知道那里安装了哪些版本不会提供有关目标的可操作信息。另请参阅为什么以前版本的 Debian 软件包在软件包存储库中消失了?(与版本控制的系统配置高度相关)您真的应该使用如何将已安装的软件包选择从一个 Debian 系统复制到另一个系统中描述的方法之一?(Debian Wheezy) — 本质上,用于
dpkg --get-selections
列出所有软件包“选择”,dpkg --set-selections
然后apt-get dselect-upgrade
将它们应用到目标系统上。基本方法是:
它返回所有已安装软件包的列表(这就是 ^ii 的用途)。
这不包括版本号,但无论如何您都无法根据版本号进行安装,因为这些版本号可能在正在使用的存储库的包池中发生了变化。
我不清楚你为什么认为你需要版本号,因为你只能在运行 apt-get update 后实际安装 apt 中的版本,并且由于两个系统都是 Debian 11,它们将具有相同的版本他们池中的软件包数量,除非他们使用 3rd 方 repos 或其他东西。
您可以为每个系统生成一个列表,然后仅安装未出现在目标计算机列表中的软件包。
当我用脚本做这样的事情时,我让它变得更加健壮,所以我会根据源系统列表测试包是否安装在目标系统上,如果不存在则只安装,但你必须在循环,因为如果包由于某种原因在目标系统的存储库中不可用,那么整个事情都会失败,除非你一个一个地做。
明确的方法是生成两个列表,创建一个小的差异列表,不在目标系统上的包,然后循环遍历那个小列表并一个一个地安装它们。循环处理孤立的问题,因为该安装将失败,它将继续执行列表中的下一个。
Bash比较两个列表找到丢失的项目
那有一个如何仅与 Bash 比较列表的示例,但它比您想要的要复杂一些。
像这样的东西我认为你最好使用完整的脚本来完成这项工作,我个人会在 Perl 中完成,但它会是你喜欢使用的任何东西。
创建孤儿
另一个问题是,这将一个一个地安装丢失的包,其中可能包含各种依赖项,但它也可能包含您不想要的孤立包。
例如,如果系统经历了多次升级到下一个 Debian 稳定版,可能会出现孤立包,这可能会留下无法删除的孤立包。
这种方法的主要缺点是您实际上不想安装软件包的依赖项,因为它们可能会更改或获得新版本,然后这些将成为孤立的但不会被 apt autoremove 删除,因为您已明确安装它们所以它们是标记为显式安装,而不是引入的依赖项。这会更频繁地命中 lib 包,但它也可能会在安装时遇到其他引入各种包的东西。
你可以用更多的脚本来解决这个问题,但这会变得很复杂,所以如果目标只是匹配包,只需一个一个地安装它们直到完成。就添加一些混乱而言,它确实会稍微弄乱安装,但这是一种简单的方法。
小心推荐!
请注意,在执行此操作时使用 --no-install-recommends 非常重要,否则最终会导致一团糟。由于这个原因,我总是将我的系统配置为从不安装推荐。推荐可以很容易地以菊花链方式连接到真正失控的东西,我在网上看到了一些示例,其中在不使用 --no-install-recommends 时安装单个小型 CLI 程序,由于菊花链,它试图拉入数百兆字节的包问题。
例如,在 /etc/apt/apt.conf.d/ 中有一些文件,比如 80basics,您可以在其中添加以下行:
我从未见过任何我想自动安装推荐或建议的情况。
可能是更好的方法来做到这一点
我猜有更好的方法来做到这一点,因为这是同步系统的标准问题,可能是一些本地 Debian 工具或一些 3rd 方工具,如果这样的事情没有,我会感到惊讶不存在,虽然我不知道有一个。
另一个
dpkg
答案的变体是:它输出包名称和包版本的排序列表。每行都有包名称、制表符和包版本字符串。您可以添加一个 glob 表达式(在单引号中)作为
-W
选项的参数,以将列表缩减为匹配的包名称(例如dpkg-query -W '*apt*'
,对于名称中包含的包apt
),或者您可以将输出通过管道传输到grep
或awk
或sed
过滤行,或将输出重定向到文件并cat
使用文本编辑器或文本编辑器读取。