haiku/3rdparty/pulkomandy/pkggraph.py

96 lines
3.3 KiB
Python
Raw Normal View History

#!python3
#
# Copyright 2019-2023 Adrien Destugues <pulkomandy@pulkomandy.tk>
#
# Distributed under terms of the MIT license.
from os import listdir
from os.path import isfile, join
import subprocess
import re
import sys
"""
Generate a graph of dependencies for a set of packages packages in an Haiku system.
Usage:
- Without arguments: generate a graph of all packages in /system/packages. This can be quite busy
and hard to understand. It will also take a while to generate the graph, as dot tries to route
thousands of edges.
- With arguments: the arguments are a list of packages to analyze. This allows to print a subset
of the packages for a better view.
Dependencies are resolved: if a package has a specific string in its REQUIRES and another has the
same string in its PROVIDES, a BLUE edge is drawn between the two package.
If a package has a REQUIRES that is not matched by any other package in the set, this REQUIRE entry
is drawn as a node, and the edge pointing to it is RED (so you can easily see missing dependencies
in a package subset). If you use the complete /system/packages hierarchy, there should be no red
edges, all dependencies are satisfied.
The output of the script can be saved to a file for manual analysis (for example, you can search
packages that nothing points to, and see if you want to uninstall them), or piped into dot for
rendering as a PNG, for example:
cd /system/packages
pkggraph.py qt* gst_plugins_ba* | dot -Tpng -o /tmp/packages.png
ShowImage /tmp/packages.png
"""
# Collect the list of packages to be analyzed
path = "/system/packages"
if len(sys.argv) > 1:
packages = sys.argv[1:]
else:
packages = [join(path, f) for f in listdir(path) if(isfile(join(path, f)))]
# List the provides and requires for each package
# pmap maps any provides to the corresponding packagename
# rmap maps a packagename to the corresponding requires
pmap = {}
rmap = {}
for p in packages:
pkgtool = subprocess.Popen(['package', 'list', '-i', p], stdout = subprocess.PIPE)
infos, stderr = pkgtool.communicate()
provides = []
requires = []
for line in infos.split(b'\n'):
if line.startswith(b"\tprovides:"):
provides.append(line.split(b' ')[1])
if line.startswith(b"\trequires:"):
line = line.split(b' ')[1]
if b'>' in line:
line = line.split(b'>')[0]
if b'=' in line:
line = line.split(b'=')[0]
if line != b'haiku' and line != b'haiku_x86':
requires.append(line)
basename = provides[0]
# Merge devel packages with the parent package
basename = basename.decode("utf-8").removesuffix("_devel")
for pro in provides:
pmap[pro] = basename
if len(requires) > 0:
if basename not in rmap:
rmap[basename] = requires
else:
rmap[basename].extend(requires)
# Generate the graph in dot/graphviz format
# For each package, there is an edge to each dependency package
print('strict digraph {\nrankdir="LR"\nsplines=ortho\nnode [ fontname="Noto", fontsize=10];')
for name, dependencies in rmap.items():
for dep in dependencies:
color = "red"
if dep in pmap:
dep = pmap[dep]
color = "blue"
print(f'"{name}" -> "{dep}" [color={color}]')
print("}")