列出顶级手动安装的软件包,不包含其依赖项

列出顶级手动安装的软件包,不包含其依赖项

有很多方法可以显示使用手动安装的软件包apt,例如:

apt-mark showmanual

但有时这个输出太多了。例如,如果用户手动安装包foo

apt-get install foo

...并且foo依赖于barbaz,然后apt-mark showmanual将输出:

bar
baz
foo

我们如何只列出顶级手动安装的软件包(IE foo) 没有它们的依赖关系 (IE不是baz,也不是bar)?


下面的代码似乎可以工作,但是GNU parallel调用apt-rdepends几百次太慢了,(4核CPU需要三个小时):

apt-mark showmanual | 
tee /tmp/foo | 
parallel "apt-rdepends -f Depends,PreDepends,Suggests,Recommends {} |
          tail +2" 2> /dev/null | 
tr -s ' ' '\n' | 
grep -v '[():]' | 
sort -Vu | 
grep -wv -f - /tmp/foo

答案1

这可以使用 Python apt API 来完成。您看到的软件包正是其中的真实apt-mark showmanual软件包,并且apt.cache.Cache()is_installedis_auto_installed是假的。但是,处理依赖关系更容易:

#! /usr/bin/env python3

from apt import cache

manual = set(pkg for pkg in cache.Cache() if pkg.is_installed and not pkg.is_auto_installed)
depends = set(dep_pkg.name for pkg in manual for dep in pkg.installed.get_dependencies('PreDepends', 'Depends', 'Recommends') for dep_pkg in dep)

print('\n'.join(pkg.name for pkg in manual if pkg.name not in depends))

即使这列出了一些我不希望在那里看到的软件包(initgrep?!)。

答案2

您可以找到所有手动安装的没有第一级依赖项的软件包,如下所示:

apt-mark showmanual | sort > manually-installed.txt

apt show $(apt-mark showmanual) 2>/dev/null | 
grep -e ^Depends -e ^Pre-Depends > deps1.txt

cat deps1.txt | 
sed 's/^Depends: //; s/^Pre-Depends: //; 
     s/(.*)//g; s/:any//g' > deps2.txt

cat deps2.txt | tr -d ',|' | tr ' ' '\n' | grep -v ^$ |
sort -u > all-dep-packages.txt

grep -v -F -f all-dep-packages.txt manually-installed.txt

您还可以使用以下一行魔术:

apt-mark showmanual | sort | grep -v -F -f <(apt show $(apt-mark showmanual) 2> /dev/null | grep -e ^Depends -e ^Pre-Depends | sed 's/^Depends: //; s/^Pre-Depends: //; s/(.*)//g; s/:any//g' | tr -d ',|' | tr ' ' '\n' | grep -v ^$ | sort -u)

答案3

以下 shell 脚本搜索所有已安装依赖项的父项。

function get_installed_packages() {
    apt list --installed | sed 's#/.*##'
}

function get_installed_packages_with_deps() {
    dpkg-query --show --showformat '${Package} ${Depends} \
        ${Pre-Depends}\n' $(get_installed_packages) | 
    sed 's/ ([^(]*)//g; s/:any\|,//g'
}

function get_package_relations() {
    awk '{print $1 " " $1; for(i = 2; i <= NF; i++) print $1 " " $i;}'
}

function add_marker() {
    echo "~ ~"
}

function resolve_parents() {
    tsort | sed -n '1,/~/ p' | head -n -1
}

(get_installed_packages_with_deps | get_package_relations; add_marker) | 
resolve_parents

tsort在这个脚本中使用了。我假设在没有依赖关系的末尾添加标记时,该标记也将是结果中没有依赖关系的最后一个条目。因此我可以区分最后一个没有依赖项的包和第一个有依赖项的包。

我注意到这个解决方案的一个问题:
依赖图中存在循环。这些条目将被忽略tsort

答案4

这个答案在 Askubuntu 上apt-mark showmanual,它使用与 的内容进行比较/var/log/installer/initial-status.gz

comm -23 <(apt-mark showmanual | sort -u) <(gzip -dc /var/log/installer/initial-status.gz | sed -n 's/^Package: //p' | sort -u)

这给了我自从全新安装 Ubuntu 以来我手动添加的软件包。

相关内容