1622 lines
57 KiB
Python
Executable File
1622 lines
57 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
# Copyright (c) 2017-2025 Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
|
|
#
|
|
# pylint: disable=C0103,C0114,C0115,C0116,C0301,C0302
|
|
# pylint: disable=R0902,R0904,R0911,R0912,R0914,R0915,R1705,R1710,E1121
|
|
|
|
# Note: this script requires at least Python 3.6 to run.
|
|
# Don't add changes not compatible with it, it is meant to report
|
|
# incompatible python versions.
|
|
|
|
"""
|
|
Dependency checker for Sphinx documentation Kernel build.
|
|
|
|
This module provides tools to check for all required dependencies needed to
|
|
build documentation using Sphinx, including system packages, Python modules
|
|
and LaTeX packages for PDF generation.
|
|
|
|
It detect packages for a subset of Linux distributions used by Kernel
|
|
maintainers, showing hints and missing dependencies.
|
|
|
|
The main class SphinxDependencyChecker handles the dependency checking logic
|
|
and provides recommendations for installing missing packages. It supports both
|
|
system package installations and Python virtual environments. By default,
|
|
system pacage install is recommended.
|
|
"""
|
|
|
|
import argparse
|
|
import os
|
|
import re
|
|
import subprocess
|
|
import sys
|
|
from glob import glob
|
|
|
|
|
|
def parse_version(version):
|
|
"""Convert a major.minor.patch version into a tuple"""
|
|
return tuple(int(x) for x in version.split("."))
|
|
|
|
|
|
def ver_str(version):
|
|
"""Returns a version tuple as major.minor.patch"""
|
|
|
|
return ".".join([str(x) for x in version])
|
|
|
|
|
|
RECOMMENDED_VERSION = parse_version("3.4.3")
|
|
MIN_PYTHON_VERSION = parse_version("3.7")
|
|
|
|
|
|
class DepManager:
|
|
"""
|
|
Manage package dependencies. There are three types of dependencies:
|
|
|
|
- System: dependencies required for docs build;
|
|
- Python: python dependencies for a native distro Sphinx install;
|
|
- PDF: dependencies needed by PDF builds.
|
|
|
|
Each dependency can be mandatory or optional. Not installing an optional
|
|
dependency won't break the build, but will cause degradation at the
|
|
docs output.
|
|
"""
|
|
|
|
# Internal types of dependencies. Don't use them outside DepManager class.
|
|
_SYS_TYPE = 0
|
|
_PHY_TYPE = 1
|
|
_PDF_TYPE = 2
|
|
|
|
# Dependencies visible outside the class.
|
|
# The keys are tuple with: (type, is_mandatory flag).
|
|
#
|
|
# Currently we're not using all optional dep types. Yet, we'll keep all
|
|
# possible combinations here. They're not many, and that makes easier
|
|
# if later needed and for the name() method below
|
|
|
|
SYSTEM_MANDATORY = (_SYS_TYPE, True)
|
|
PYTHON_MANDATORY = (_PHY_TYPE, True)
|
|
PDF_MANDATORY = (_PDF_TYPE, True)
|
|
|
|
SYSTEM_OPTIONAL = (_SYS_TYPE, False)
|
|
PYTHON_OPTIONAL = (_PHY_TYPE, False)
|
|
PDF_OPTIONAL = (_PDF_TYPE, True)
|
|
|
|
def __init__(self, pdf):
|
|
"""
|
|
Initialize internal vars:
|
|
|
|
- missing: missing dependencies list, containing a distro-independent
|
|
name for a missing dependency and its type.
|
|
- missing_pkg: ancillary dict containing missing dependencies in
|
|
distro namespace, organized by type.
|
|
- need: total number of needed dependencies. Never cleaned.
|
|
- optional: total number of optional dependencies. Never cleaned.
|
|
- pdf: Is PDF support enabled?
|
|
"""
|
|
self.missing = {}
|
|
self.missing_pkg = {}
|
|
self.need = 0
|
|
self.optional = 0
|
|
self.pdf = pdf
|
|
|
|
@staticmethod
|
|
def name(dtype):
|
|
"""
|
|
Ancillary routine to output a warn/error message reporting
|
|
missing dependencies.
|
|
"""
|
|
if dtype[0] == DepManager._SYS_TYPE:
|
|
msg = "build"
|
|
elif dtype[0] == DepManager._PHY_TYPE:
|
|
msg = "Python"
|
|
else:
|
|
msg = "PDF"
|
|
|
|
if dtype[1]:
|
|
return f"ERROR: {msg} mandatory deps missing"
|
|
else:
|
|
return f"Warning: {msg} optional deps missing"
|
|
|
|
@staticmethod
|
|
def is_optional(dtype):
|
|
"""Ancillary routine to report if a dependency is optional"""
|
|
return not dtype[1]
|
|
|
|
@staticmethod
|
|
def is_pdf(dtype):
|
|
"""Ancillary routine to report if a dependency is for PDF generation"""
|
|
if dtype[0] == DepManager._PDF_TYPE:
|
|
return True
|
|
|
|
return False
|
|
|
|
def add_package(self, package, dtype):
|
|
"""
|
|
Add a package at the self.missing() dictionary.
|
|
Doesn't update missing_pkg.
|
|
"""
|
|
is_optional = DepManager.is_optional(dtype)
|
|
self.missing[package] = dtype
|
|
if is_optional:
|
|
self.optional += 1
|
|
else:
|
|
self.need += 1
|
|
|
|
def del_package(self, package):
|
|
"""
|
|
Remove a package at the self.missing() dictionary.
|
|
Doesn't update missing_pkg.
|
|
"""
|
|
if package in self.missing:
|
|
del self.missing[package]
|
|
|
|
def clear_deps(self):
|
|
"""
|
|
Clear dependencies without changing needed/optional.
|
|
|
|
This is an ackward way to have a separate section to recommend
|
|
a package after system main dependencies.
|
|
|
|
TODO: rework the logic to prevent needing it.
|
|
"""
|
|
|
|
self.missing = {}
|
|
self.missing_pkg = {}
|
|
|
|
def check_missing(self, progs):
|
|
"""
|
|
Update self.missing_pkg, using progs dict to convert from the
|
|
agnostic package name to distro-specific one.
|
|
|
|
Returns an string with the packages to be installed, sorted and
|
|
with eventual duplicates removed.
|
|
"""
|
|
|
|
self.missing_pkg = {}
|
|
|
|
for prog, dtype in sorted(self.missing.items()):
|
|
# At least on some LTS distros like CentOS 7, texlive doesn't
|
|
# provide all packages we need. When such distros are
|
|
# detected, we have to disable PDF output.
|
|
#
|
|
# So, we need to ignore the packages that distros would
|
|
# need for LaTeX to work
|
|
if DepManager.is_pdf(dtype) and not self.pdf:
|
|
self.optional -= 1
|
|
continue
|
|
|
|
if not dtype in self.missing_pkg:
|
|
self.missing_pkg[dtype] = []
|
|
|
|
self.missing_pkg[dtype].append(progs.get(prog, prog))
|
|
|
|
install = []
|
|
for dtype, pkgs in self.missing_pkg.items():
|
|
install += pkgs
|
|
|
|
return " ".join(sorted(set(install)))
|
|
|
|
def warn_install(self):
|
|
"""
|
|
Emit warnings/errors related to missing packages.
|
|
"""
|
|
|
|
output_msg = ""
|
|
|
|
for dtype in sorted(self.missing_pkg.keys()):
|
|
progs = " ".join(sorted(set(self.missing_pkg[dtype])))
|
|
|
|
try:
|
|
name = DepManager.name(dtype)
|
|
output_msg += f'{name}:\t{progs}\n'
|
|
except KeyError:
|
|
raise KeyError(f"ERROR!!!: invalid dtype for {progs}: {dtype}")
|
|
|
|
if output_msg:
|
|
print(f"\n{output_msg}")
|
|
|
|
class AncillaryMethods:
|
|
"""
|
|
Ancillary methods that checks for missing dependencies for different
|
|
types of types, like binaries, python modules, rpm deps, etc.
|
|
"""
|
|
|
|
@staticmethod
|
|
def which(prog):
|
|
"""
|
|
Our own implementation of which(). We could instead use
|
|
shutil.which(), but this function is simple enough.
|
|
Probably faster to use this implementation than to import shutil.
|
|
"""
|
|
for path in os.environ.get("PATH", "").split(":"):
|
|
full_path = os.path.join(path, prog)
|
|
if os.access(full_path, os.X_OK):
|
|
return full_path
|
|
|
|
return None
|
|
|
|
@staticmethod
|
|
def get_python_version(cmd):
|
|
"""
|
|
Get python version from a Python binary. As we need to detect if
|
|
are out there newer python binaries, we can't rely on sys.release here.
|
|
"""
|
|
|
|
result = SphinxDependencyChecker.run([cmd, "--version"],
|
|
capture_output=True, text=True)
|
|
version = result.stdout.strip()
|
|
|
|
match = re.search(r"(\d+\.\d+\.\d+)", version)
|
|
if match:
|
|
return parse_version(match.group(1))
|
|
|
|
print(f"Can't parse version {version}")
|
|
return (0, 0, 0)
|
|
|
|
@staticmethod
|
|
def find_python():
|
|
"""
|
|
Detect if are out there any python 3.xy version newer than the
|
|
current one.
|
|
|
|
Note: this routine is limited to up to 2 digits for python3. We
|
|
may need to update it one day, hopefully on a distant future.
|
|
"""
|
|
patterns = [
|
|
"python3.[0-9]",
|
|
"python3.[0-9][0-9]",
|
|
]
|
|
|
|
# Seek for a python binary newer than MIN_PYTHON_VERSION
|
|
for path in os.getenv("PATH", "").split(":"):
|
|
for pattern in patterns:
|
|
for cmd in glob(os.path.join(path, pattern)):
|
|
if os.path.isfile(cmd) and os.access(cmd, os.X_OK):
|
|
version = SphinxDependencyChecker.get_python_version(cmd)
|
|
if version >= MIN_PYTHON_VERSION:
|
|
return cmd
|
|
|
|
@staticmethod
|
|
def check_python():
|
|
"""
|
|
Check if the current python binary satisfies our minimal requirement
|
|
for Sphinx build. If not, re-run with a newer version if found.
|
|
"""
|
|
cur_ver = sys.version_info[:3]
|
|
if cur_ver >= MIN_PYTHON_VERSION:
|
|
ver = ver_str(cur_ver)
|
|
print(f"Python version: {ver}")
|
|
|
|
# This could be useful for debugging purposes
|
|
if SphinxDependencyChecker.which("docutils"):
|
|
result = SphinxDependencyChecker.run(["docutils", "--version"],
|
|
capture_output=True, text=True)
|
|
ver = result.stdout.strip()
|
|
match = re.search(r"(\d+\.\d+\.\d+)", ver)
|
|
if match:
|
|
ver = match.group(1)
|
|
|
|
print(f"Docutils version: {ver}")
|
|
|
|
return
|
|
|
|
python_ver = ver_str(cur_ver)
|
|
|
|
new_python_cmd = SphinxDependencyChecker.find_python()
|
|
if not new_python_cmd:
|
|
print(f"ERROR: Python version {python_ver} is not spported anymore\n")
|
|
print(" Can't find a new version. This script may fail")
|
|
return
|
|
|
|
# Restart script using the newer version
|
|
script_path = os.path.abspath(sys.argv[0])
|
|
args = [new_python_cmd, script_path] + sys.argv[1:]
|
|
|
|
print(f"Python {python_ver} not supported. Changing to {new_python_cmd}")
|
|
|
|
try:
|
|
os.execv(new_python_cmd, args)
|
|
except OSError as e:
|
|
sys.exit(f"Failed to restart with {new_python_cmd}: {e}")
|
|
|
|
@staticmethod
|
|
def run(*args, **kwargs):
|
|
"""
|
|
Excecute a command, hiding its output by default.
|
|
Preserve comatibility with older Python versions.
|
|
"""
|
|
|
|
capture_output = kwargs.pop('capture_output', False)
|
|
|
|
if capture_output:
|
|
if 'stdout' not in kwargs:
|
|
kwargs['stdout'] = subprocess.PIPE
|
|
if 'stderr' not in kwargs:
|
|
kwargs['stderr'] = subprocess.PIPE
|
|
else:
|
|
if 'stdout' not in kwargs:
|
|
kwargs['stdout'] = subprocess.DEVNULL
|
|
if 'stderr' not in kwargs:
|
|
kwargs['stderr'] = subprocess.DEVNULL
|
|
|
|
# Don't break with older Python versions
|
|
if 'text' in kwargs and sys.version_info < (3, 7):
|
|
kwargs['universal_newlines'] = kwargs.pop('text')
|
|
|
|
return subprocess.run(*args, **kwargs)
|
|
|
|
class MissingCheckers(AncillaryMethods):
|
|
"""
|
|
Contains some ancillary checkers for different types of binaries and
|
|
package managers.
|
|
"""
|
|
|
|
def __init__(self, args, texlive):
|
|
"""
|
|
Initialize its internal variables
|
|
"""
|
|
self.pdf = args.pdf
|
|
self.virtualenv = args.virtualenv
|
|
self.version_check = args.version_check
|
|
self.texlive = texlive
|
|
|
|
self.min_version = (0, 0, 0)
|
|
self.cur_version = (0, 0, 0)
|
|
|
|
self.deps = DepManager(self.pdf)
|
|
|
|
self.need_symlink = 0
|
|
self.need_sphinx = 0
|
|
|
|
self.verbose_warn_install = 1
|
|
|
|
self.virtenv_dir = ""
|
|
self.install = ""
|
|
self.python_cmd = ""
|
|
|
|
self.virtenv_prefix = ["sphinx_", "Sphinx_" ]
|
|
|
|
def check_missing_file(self, files, package, dtype):
|
|
"""
|
|
Does the file exists? If not, add it to missing dependencies.
|
|
"""
|
|
for f in files:
|
|
if os.path.exists(f):
|
|
return
|
|
self.deps.add_package(package, dtype)
|
|
|
|
def check_program(self, prog, dtype):
|
|
"""
|
|
Does the program exists and it is at the PATH?
|
|
If not, add it to missing dependencies.
|
|
"""
|
|
found = self.which(prog)
|
|
if found:
|
|
return found
|
|
|
|
self.deps.add_package(prog, dtype)
|
|
|
|
return None
|
|
|
|
def check_perl_module(self, prog, dtype):
|
|
"""
|
|
Does perl have a dependency? Is it available?
|
|
If not, add it to missing dependencies.
|
|
|
|
Right now, we still need Perl for doc build, as it is required
|
|
by some tools called at docs or kernel build time, like:
|
|
|
|
scripts/documentation-file-ref-check
|
|
|
|
Also, checkpatch is on Perl.
|
|
"""
|
|
|
|
# While testing with lxc download template, one of the
|
|
# distros (Oracle) didn't have perl - nor even an option to install
|
|
# before installing oraclelinux-release-el9 package.
|
|
#
|
|
# Check it before running an error. If perl is not there,
|
|
# add it as a mandatory package, as some parts of the doc builder
|
|
# needs it.
|
|
if not self.which("perl"):
|
|
self.deps.add_package("perl", DepManager.SYSTEM_MANDATORY)
|
|
self.deps.add_package(prog, dtype)
|
|
return
|
|
|
|
try:
|
|
self.run(["perl", f"-M{prog}", "-e", "1"], check=True)
|
|
except subprocess.CalledProcessError:
|
|
self.deps.add_package(prog, dtype)
|
|
|
|
def check_python_module(self, module, is_optional=False):
|
|
"""
|
|
Does a python module exists outside venv? If not, add it to missing
|
|
dependencies.
|
|
"""
|
|
if is_optional:
|
|
dtype = DepManager.PYTHON_OPTIONAL
|
|
else:
|
|
dtype = DepManager.PYTHON_MANDATORY
|
|
|
|
try:
|
|
self.run([self.python_cmd, "-c", f"import {module}"], check=True)
|
|
except subprocess.CalledProcessError:
|
|
self.deps.add_package(module, dtype)
|
|
|
|
def check_rpm_missing(self, pkgs, dtype):
|
|
"""
|
|
Does a rpm package exists? If not, add it to missing dependencies.
|
|
"""
|
|
for prog in pkgs:
|
|
try:
|
|
self.run(["rpm", "-q", prog], check=True)
|
|
except subprocess.CalledProcessError:
|
|
self.deps.add_package(prog, dtype)
|
|
|
|
def check_pacman_missing(self, pkgs, dtype):
|
|
"""
|
|
Does a pacman package exists? If not, add it to missing dependencies.
|
|
"""
|
|
for prog in pkgs:
|
|
try:
|
|
self.run(["pacman", "-Q", prog], check=True)
|
|
except subprocess.CalledProcessError:
|
|
self.deps.add_package(prog, dtype)
|
|
|
|
def check_missing_tex(self, is_optional=False):
|
|
"""
|
|
Does a LaTeX package exists? If not, add it to missing dependencies.
|
|
"""
|
|
if is_optional:
|
|
dtype = DepManager.PDF_OPTIONAL
|
|
else:
|
|
dtype = DepManager.PDF_MANDATORY
|
|
|
|
kpsewhich = self.which("kpsewhich")
|
|
for prog, package in self.texlive.items():
|
|
|
|
# If kpsewhich is not there, just add it to deps
|
|
if not kpsewhich:
|
|
self.deps.add_package(package, dtype)
|
|
continue
|
|
|
|
# Check if the package is needed
|
|
try:
|
|
result = self.run(
|
|
[kpsewhich, prog], stdout=subprocess.PIPE, text=True, check=True
|
|
)
|
|
|
|
# Didn't find. Add it
|
|
if not result.stdout.strip():
|
|
self.deps.add_package(package, dtype)
|
|
|
|
except subprocess.CalledProcessError:
|
|
# kpsewhich returned an error. Add it, just in case
|
|
self.deps.add_package(package, dtype)
|
|
|
|
def get_sphinx_fname(self):
|
|
"""
|
|
Gets the binary filename for sphinx-build.
|
|
"""
|
|
if "SPHINXBUILD" in os.environ:
|
|
return os.environ["SPHINXBUILD"]
|
|
|
|
fname = "sphinx-build"
|
|
if self.which(fname):
|
|
return fname
|
|
|
|
fname = "sphinx-build-3"
|
|
if self.which(fname):
|
|
self.need_symlink = 1
|
|
return fname
|
|
|
|
return ""
|
|
|
|
def get_sphinx_version(self, cmd):
|
|
"""
|
|
Gets sphinx-build version.
|
|
"""
|
|
try:
|
|
result = self.run([cmd, "--version"],
|
|
stdout=subprocess.PIPE,
|
|
stderr=subprocess.STDOUT,
|
|
text=True, check=True)
|
|
except (subprocess.CalledProcessError, FileNotFoundError):
|
|
return None
|
|
|
|
for line in result.stdout.split("\n"):
|
|
match = re.match(r"^sphinx-build\s+([\d\.]+)(?:\+(?:/[\da-f]+)|b\d+)?\s*$", line)
|
|
if match:
|
|
return parse_version(match.group(1))
|
|
|
|
match = re.match(r"^Sphinx.*\s+([\d\.]+)\s*$", line)
|
|
if match:
|
|
return parse_version(match.group(1))
|
|
|
|
def check_sphinx(self, conf):
|
|
"""
|
|
Checks Sphinx minimal requirements
|
|
"""
|
|
try:
|
|
with open(conf, "r", encoding="utf-8") as f:
|
|
for line in f:
|
|
match = re.match(r"^\s*needs_sphinx\s*=\s*[\'\"]([\d\.]+)[\'\"]", line)
|
|
if match:
|
|
self.min_version = parse_version(match.group(1))
|
|
break
|
|
except IOError:
|
|
sys.exit(f"Can't open {conf}")
|
|
|
|
if not self.min_version:
|
|
sys.exit(f"Can't get needs_sphinx version from {conf}")
|
|
|
|
self.virtenv_dir = self.virtenv_prefix[0] + "latest"
|
|
|
|
sphinx = self.get_sphinx_fname()
|
|
if not sphinx:
|
|
self.need_sphinx = 1
|
|
return
|
|
|
|
self.cur_version = self.get_sphinx_version(sphinx)
|
|
if not self.cur_version:
|
|
sys.exit(f"{sphinx} didn't return its version")
|
|
|
|
if self.cur_version < self.min_version:
|
|
curver = ver_str(self.cur_version)
|
|
minver = ver_str(self.min_version)
|
|
|
|
print(f"ERROR: Sphinx version is {curver}. It should be >= {minver}")
|
|
self.need_sphinx = 1
|
|
return
|
|
|
|
# On version check mode, just assume Sphinx has all mandatory deps
|
|
if self.version_check and self.cur_version >= RECOMMENDED_VERSION:
|
|
sys.exit(0)
|
|
|
|
def catcheck(self, filename):
|
|
"""
|
|
Reads a file if it exists, returning as string.
|
|
If not found, returns an empty string.
|
|
"""
|
|
if os.path.exists(filename):
|
|
with open(filename, "r", encoding="utf-8") as f:
|
|
return f.read().strip()
|
|
return ""
|
|
|
|
def get_system_release(self):
|
|
"""
|
|
Determine the system type. There's no unique way that would work
|
|
with all distros with a minimal package install. So, several
|
|
methods are used here.
|
|
|
|
By default, it will use lsb_release function. If not available, it will
|
|
fail back to reading the known different places where the distro name
|
|
is stored.
|
|
|
|
Several modern distros now have /etc/os-release, which usually have
|
|
a decent coverage.
|
|
"""
|
|
|
|
system_release = ""
|
|
|
|
if self.which("lsb_release"):
|
|
result = self.run(["lsb_release", "-d"], capture_output=True, text=True)
|
|
system_release = result.stdout.replace("Description:", "").strip()
|
|
|
|
release_files = [
|
|
"/etc/system-release",
|
|
"/etc/redhat-release",
|
|
"/etc/lsb-release",
|
|
"/etc/gentoo-release",
|
|
]
|
|
|
|
if not system_release:
|
|
for f in release_files:
|
|
system_release = self.catcheck(f)
|
|
if system_release:
|
|
break
|
|
|
|
# This seems more common than LSB these days
|
|
if not system_release:
|
|
os_var = {}
|
|
try:
|
|
with open("/etc/os-release", "r", encoding="utf-8") as f:
|
|
for line in f:
|
|
match = re.match(r"^([\w\d\_]+)=\"?([^\"]*)\"?\n", line)
|
|
if match:
|
|
os_var[match.group(1)] = match.group(2)
|
|
|
|
system_release = os_var.get("NAME", "")
|
|
if "VERSION_ID" in os_var:
|
|
system_release += " " + os_var["VERSION_ID"]
|
|
elif "VERSION" in os_var:
|
|
system_release += " " + os_var["VERSION"]
|
|
except IOError:
|
|
pass
|
|
|
|
if not system_release:
|
|
system_release = self.catcheck("/etc/issue")
|
|
|
|
system_release = system_release.strip()
|
|
|
|
return system_release
|
|
|
|
class SphinxDependencyChecker(MissingCheckers):
|
|
"""
|
|
Main class for checking Sphinx documentation build dependencies.
|
|
|
|
- Check for missing system packages;
|
|
- Check for missing Python modules;
|
|
- Check for missing LaTeX packages needed by PDF generation;
|
|
- Propose Sphinx install via Python Virtual environment;
|
|
- Propose Sphinx install via distro-specific package install.
|
|
"""
|
|
def __init__(self, args):
|
|
"""Initialize checker variables"""
|
|
|
|
# List of required texlive packages on Fedora and OpenSuse
|
|
texlive = {
|
|
"amsfonts.sty": "texlive-amsfonts",
|
|
"amsmath.sty": "texlive-amsmath",
|
|
"amssymb.sty": "texlive-amsfonts",
|
|
"amsthm.sty": "texlive-amscls",
|
|
"anyfontsize.sty": "texlive-anyfontsize",
|
|
"atbegshi.sty": "texlive-oberdiek",
|
|
"bm.sty": "texlive-tools",
|
|
"capt-of.sty": "texlive-capt-of",
|
|
"cmap.sty": "texlive-cmap",
|
|
"ctexhook.sty": "texlive-ctex",
|
|
"ecrm1000.tfm": "texlive-ec",
|
|
"eqparbox.sty": "texlive-eqparbox",
|
|
"eu1enc.def": "texlive-euenc",
|
|
"fancybox.sty": "texlive-fancybox",
|
|
"fancyvrb.sty": "texlive-fancyvrb",
|
|
"float.sty": "texlive-float",
|
|
"fncychap.sty": "texlive-fncychap",
|
|
"footnote.sty": "texlive-mdwtools",
|
|
"framed.sty": "texlive-framed",
|
|
"luatex85.sty": "texlive-luatex85",
|
|
"multirow.sty": "texlive-multirow",
|
|
"needspace.sty": "texlive-needspace",
|
|
"palatino.sty": "texlive-psnfss",
|
|
"parskip.sty": "texlive-parskip",
|
|
"polyglossia.sty": "texlive-polyglossia",
|
|
"tabulary.sty": "texlive-tabulary",
|
|
"threeparttable.sty": "texlive-threeparttable",
|
|
"titlesec.sty": "texlive-titlesec",
|
|
"ucs.sty": "texlive-ucs",
|
|
"upquote.sty": "texlive-upquote",
|
|
"wrapfig.sty": "texlive-wrapfig",
|
|
}
|
|
|
|
super().__init__(args, texlive)
|
|
|
|
self.need_pip = False
|
|
self.rec_sphinx_upgrade = 0
|
|
|
|
self.system_release = self.get_system_release()
|
|
self.activate_cmd = ""
|
|
|
|
# Some distros may not have a Sphinx shipped package compatible with
|
|
# our minimal requirements
|
|
self.package_supported = True
|
|
|
|
# Recommend a new python version
|
|
self.recommend_python = None
|
|
|
|
# Certain hints are meant to be shown only once
|
|
self.distro_msg = None
|
|
|
|
self.latest_avail_ver = (0, 0, 0)
|
|
self.venv_ver = (0, 0, 0)
|
|
|
|
prefix = os.environ.get("srctree", ".") + "/"
|
|
|
|
self.conf = prefix + "Documentation/conf.py"
|
|
self.requirement_file = prefix + "Documentation/sphinx/requirements.txt"
|
|
|
|
def get_install_progs(self, progs, cmd, extra=None):
|
|
"""
|
|
Check for missing dependencies using the provided program mapping.
|
|
|
|
The actual distro-specific programs are mapped via progs argument.
|
|
"""
|
|
install = self.deps.check_missing(progs)
|
|
|
|
if self.verbose_warn_install:
|
|
self.deps.warn_install()
|
|
|
|
if not install:
|
|
return
|
|
|
|
if cmd:
|
|
if self.verbose_warn_install:
|
|
msg = "You should run:"
|
|
else:
|
|
msg = ""
|
|
|
|
if extra:
|
|
msg += "\n\t" + extra.replace("\n", "\n\t")
|
|
|
|
return(msg + "\n\tsudo " + cmd + " " + install)
|
|
|
|
return None
|
|
|
|
#
|
|
# Distro-specific hints methods
|
|
#
|
|
|
|
def give_debian_hints(self):
|
|
"""
|
|
Provide package installation hints for Debian-based distros.
|
|
"""
|
|
progs = {
|
|
"Pod::Usage": "perl-modules",
|
|
"convert": "imagemagick",
|
|
"dot": "graphviz",
|
|
"ensurepip": "python3-venv",
|
|
"python-sphinx": "python3-sphinx",
|
|
"rsvg-convert": "librsvg2-bin",
|
|
"virtualenv": "virtualenv",
|
|
"xelatex": "texlive-xetex",
|
|
"yaml": "python3-yaml",
|
|
}
|
|
|
|
if self.pdf:
|
|
pdf_pkgs = {
|
|
"fonts-dejavu": [
|
|
"/usr/share/fonts/truetype/dejavu/DejaVuSans.ttf",
|
|
],
|
|
"fonts-noto-cjk": [
|
|
"/usr/share/fonts/noto-cjk/NotoSansCJK-Regular.ttc",
|
|
"/usr/share/fonts/opentype/noto/NotoSansCJK-Regular.ttc",
|
|
"/usr/share/fonts/opentype/noto/NotoSerifCJK-Regular.ttc",
|
|
],
|
|
"tex-gyre": [
|
|
"/usr/share/texmf/tex/latex/tex-gyre/tgtermes.sty"
|
|
],
|
|
"texlive-fonts-recommended": [
|
|
"/usr/share/texlive/texmf-dist/fonts/tfm/adobe/zapfding/pzdr.tfm",
|
|
],
|
|
"texlive-lang-chinese": [
|
|
"/usr/share/texlive/texmf-dist/tex/latex/ctex/ctexhook.sty",
|
|
],
|
|
}
|
|
|
|
for package, files in pdf_pkgs.items():
|
|
self.check_missing_file(files, package, DepManager.PDF_MANDATORY)
|
|
|
|
self.check_program("dvipng", DepManager.PDF_MANDATORY)
|
|
|
|
if not self.distro_msg:
|
|
self.distro_msg = \
|
|
"Note: ImageMagick is broken on some distros, affecting PDF output. For more details:\n" \
|
|
"\thttps://askubuntu.com/questions/1158894/imagemagick-still-broken-using-with-usr-bin-convert"
|
|
|
|
return self.get_install_progs(progs, "apt-get install")
|
|
|
|
def give_redhat_hints(self):
|
|
"""
|
|
Provide package installation hints for RedHat-based distros
|
|
(Fedora, RHEL and RHEL-based variants).
|
|
"""
|
|
progs = {
|
|
"Pod::Usage": "perl-Pod-Usage",
|
|
"convert": "ImageMagick",
|
|
"dot": "graphviz",
|
|
"python-sphinx": "python3-sphinx",
|
|
"rsvg-convert": "librsvg2-tools",
|
|
"virtualenv": "python3-virtualenv",
|
|
"xelatex": "texlive-xetex-bin",
|
|
"yaml": "python3-pyyaml",
|
|
}
|
|
|
|
fedora_tex_pkgs = [
|
|
"dejavu-sans-fonts",
|
|
"dejavu-sans-mono-fonts",
|
|
"dejavu-serif-fonts",
|
|
"texlive-collection-fontsrecommended",
|
|
"texlive-collection-latex",
|
|
"texlive-xecjk",
|
|
]
|
|
|
|
fedora = False
|
|
rel = None
|
|
|
|
match = re.search(r"(release|Linux)\s+(\d+)", self.system_release)
|
|
if match:
|
|
rel = int(match.group(2))
|
|
|
|
if not rel:
|
|
print("Couldn't identify release number")
|
|
noto_sans_redhat = None
|
|
self.pdf = False
|
|
elif re.search("Fedora", self.system_release):
|
|
# Fedora 38 and upper use this CJK font
|
|
|
|
noto_sans_redhat = "google-noto-sans-cjk-fonts"
|
|
fedora = True
|
|
else:
|
|
# Almalinux, CentOS, RHEL, ...
|
|
|
|
# at least up to version 9 (and Fedora < 38), that's the CJK font
|
|
noto_sans_redhat = "google-noto-sans-cjk-ttc-fonts"
|
|
|
|
progs["virtualenv"] = "python-virtualenv"
|
|
|
|
if not rel or rel < 8:
|
|
print("ERROR: Distro not supported. Too old?")
|
|
return
|
|
|
|
# RHEL 8 uses Python 3.6, which is not compatible with
|
|
# the build system anymore. Suggest Python 3.11
|
|
if rel == 8:
|
|
self.check_program("python3.9", DepManager.SYSTEM_MANDATORY)
|
|
progs["python3.9"] = "python39"
|
|
progs["yaml"] = "python39-pyyaml"
|
|
|
|
self.recommend_python = True
|
|
|
|
# There's no python39-sphinx package. Only pip is supported
|
|
self.package_supported = False
|
|
|
|
if not self.distro_msg:
|
|
self.distro_msg = \
|
|
"Note: RHEL-based distros typically require extra repositories.\n" \
|
|
"For most, enabling epel and crb are enough:\n" \
|
|
"\tsudo dnf install -y epel-release\n" \
|
|
"\tsudo dnf config-manager --set-enabled crb\n" \
|
|
"Yet, some may have other required repositories. Those commands could be useful:\n" \
|
|
"\tsudo dnf repolist all\n" \
|
|
"\tsudo dnf repoquery --available --info <pkgs>\n" \
|
|
"\tsudo dnf config-manager --set-enabled '*' # enable all - probably not what you want"
|
|
|
|
if self.pdf:
|
|
pdf_pkgs = [
|
|
"/usr/share/fonts/google-noto-cjk/NotoSansCJK-Regular.ttc",
|
|
"/usr/share/fonts/google-noto-sans-cjk-fonts/NotoSansCJK-Regular.ttc",
|
|
]
|
|
|
|
self.check_missing_file(pdf_pkgs, noto_sans_redhat, DepManager.PDF_MANDATORY)
|
|
|
|
self.check_rpm_missing(fedora_tex_pkgs, DepManager.PDF_MANDATORY)
|
|
|
|
self.check_missing_tex(DepManager.PDF_MANDATORY)
|
|
|
|
# There's no texlive-ctex on RHEL 8 repositories. This will
|
|
# likely affect CJK pdf build only.
|
|
if not fedora and rel == 8:
|
|
self.deps.del_package("texlive-ctex")
|
|
|
|
return self.get_install_progs(progs, "dnf install")
|
|
|
|
def give_opensuse_hints(self):
|
|
"""
|
|
Provide package installation hints for openSUSE-based distros
|
|
(Leap and Tumbleweed).
|
|
"""
|
|
progs = {
|
|
"Pod::Usage": "perl-Pod-Usage",
|
|
"convert": "ImageMagick",
|
|
"dot": "graphviz",
|
|
"python-sphinx": "python3-sphinx",
|
|
"virtualenv": "python3-virtualenv",
|
|
"xelatex": "texlive-xetex-bin texlive-dejavu",
|
|
"yaml": "python3-pyyaml",
|
|
}
|
|
|
|
suse_tex_pkgs = [
|
|
"texlive-babel-english",
|
|
"texlive-caption",
|
|
"texlive-colortbl",
|
|
"texlive-courier",
|
|
"texlive-dvips",
|
|
"texlive-helvetic",
|
|
"texlive-makeindex",
|
|
"texlive-metafont",
|
|
"texlive-metapost",
|
|
"texlive-palatino",
|
|
"texlive-preview",
|
|
"texlive-times",
|
|
"texlive-zapfchan",
|
|
"texlive-zapfding",
|
|
]
|
|
|
|
progs["latexmk"] = "texlive-latexmk-bin"
|
|
|
|
match = re.search(r"(Leap)\s+(\d+).(\d)", self.system_release)
|
|
if match:
|
|
rel = int(match.group(2))
|
|
|
|
# Leap 15.x uses Python 3.6, which is not compatible with
|
|
# the build system anymore. Suggest Python 3.11
|
|
if rel == 15:
|
|
if not self.which(self.python_cmd):
|
|
self.check_program("python3.11", DepManager.SYSTEM_MANDATORY)
|
|
progs["python3.11"] = "python311"
|
|
self.recommend_python = True
|
|
|
|
progs.update({
|
|
"python-sphinx": "python311-Sphinx python311-Sphinx-latex",
|
|
"virtualenv": "python311-virtualenv",
|
|
"yaml": "python311-PyYAML",
|
|
})
|
|
else:
|
|
# Tumbleweed defaults to Python 3.11
|
|
|
|
progs.update({
|
|
"python-sphinx": "python313-Sphinx python313-Sphinx-latex",
|
|
"virtualenv": "python313-virtualenv",
|
|
"yaml": "python313-PyYAML",
|
|
})
|
|
|
|
# FIXME: add support for installing CJK fonts
|
|
#
|
|
# I tried hard, but was unable to find a way to install
|
|
# "Noto Sans CJK SC" on openSUSE
|
|
|
|
if self.pdf:
|
|
self.check_rpm_missing(suse_tex_pkgs, DepManager.PDF_MANDATORY)
|
|
if self.pdf:
|
|
self.check_missing_tex()
|
|
|
|
return self.get_install_progs(progs, "zypper install --no-recommends")
|
|
|
|
def give_mageia_hints(self):
|
|
"""
|
|
Provide package installation hints for Mageia and OpenMandriva.
|
|
"""
|
|
progs = {
|
|
"Pod::Usage": "perl-Pod-Usage",
|
|
"convert": "ImageMagick",
|
|
"dot": "graphviz",
|
|
"python-sphinx": "python3-sphinx",
|
|
"rsvg-convert": "librsvg2",
|
|
"virtualenv": "python3-virtualenv",
|
|
"xelatex": "texlive",
|
|
"yaml": "python3-yaml",
|
|
}
|
|
|
|
tex_pkgs = [
|
|
"texlive-fontsextra",
|
|
"texlive-fonts-asian",
|
|
"fonts-ttf-dejavu",
|
|
]
|
|
|
|
if re.search(r"OpenMandriva", self.system_release):
|
|
packager_cmd = "dnf install"
|
|
noto_sans = "noto-sans-cjk-fonts"
|
|
tex_pkgs = [
|
|
"texlive-collection-basic",
|
|
"texlive-collection-langcjk",
|
|
"texlive-collection-fontsextra",
|
|
"texlive-collection-fontsrecommended"
|
|
]
|
|
|
|
# Tested on OpenMandriva Lx 4.3
|
|
progs["convert"] = "imagemagick"
|
|
progs["yaml"] = "python-pyyaml"
|
|
progs["python-virtualenv"] = "python-virtualenv"
|
|
progs["python-sphinx"] = "python-sphinx"
|
|
progs["xelatex"] = "texlive"
|
|
|
|
self.check_program("python-virtualenv", DepManager.PYTHON_MANDATORY)
|
|
|
|
# On my tests with openMandriva LX 4.0 docker image, upgraded
|
|
# to 4.3, python-virtualenv package is broken: it is missing
|
|
# ensurepip. Without it, the alternative would be to run:
|
|
# python3 -m venv --without-pip ~/sphinx_latest, but running
|
|
# pip there won't install sphinx at venv.
|
|
#
|
|
# Add a note about that.
|
|
|
|
if not self.distro_msg:
|
|
self.distro_msg = \
|
|
"Notes:\n"\
|
|
"1. for venv, ensurepip could be broken, preventing its install method.\n" \
|
|
"2. at least on OpenMandriva LX 4.3, texlive packages seem broken"
|
|
|
|
else:
|
|
packager_cmd = "urpmi"
|
|
noto_sans = "google-noto-sans-cjk-ttc-fonts"
|
|
|
|
progs["latexmk"] = "texlive-collection-basic"
|
|
|
|
if self.pdf:
|
|
pdf_pkgs = [
|
|
"/usr/share/fonts/google-noto-cjk/NotoSansCJK-Regular.ttc",
|
|
"/usr/share/fonts/TTF/NotoSans-Regular.ttf",
|
|
]
|
|
|
|
self.check_missing_file(pdf_pkgs, noto_sans, DepManager.PDF_MANDATORY)
|
|
self.check_rpm_missing(tex_pkgs, DepManager.PDF_MANDATORY)
|
|
|
|
return self.get_install_progs(progs, packager_cmd)
|
|
|
|
def give_arch_linux_hints(self):
|
|
"""
|
|
Provide package installation hints for ArchLinux.
|
|
"""
|
|
progs = {
|
|
"convert": "imagemagick",
|
|
"dot": "graphviz",
|
|
"latexmk": "texlive-core",
|
|
"rsvg-convert": "extra/librsvg",
|
|
"virtualenv": "python-virtualenv",
|
|
"xelatex": "texlive-xetex",
|
|
"yaml": "python-yaml",
|
|
}
|
|
|
|
archlinux_tex_pkgs = [
|
|
"texlive-basic",
|
|
"texlive-binextra",
|
|
"texlive-core",
|
|
"texlive-fontsrecommended",
|
|
"texlive-langchinese",
|
|
"texlive-langcjk",
|
|
"texlive-latexextra",
|
|
"ttf-dejavu",
|
|
]
|
|
|
|
if self.pdf:
|
|
self.check_pacman_missing(archlinux_tex_pkgs,
|
|
DepManager.PDF_MANDATORY)
|
|
|
|
self.check_missing_file(["/usr/share/fonts/noto-cjk/NotoSansCJK-Regular.ttc"],
|
|
"noto-fonts-cjk",
|
|
DepManager.PDF_MANDATORY)
|
|
|
|
|
|
return self.get_install_progs(progs, "pacman -S")
|
|
|
|
def give_gentoo_hints(self):
|
|
"""
|
|
Provide package installation hints for Gentoo.
|
|
"""
|
|
texlive_deps = [
|
|
"dev-texlive/texlive-fontsrecommended",
|
|
"dev-texlive/texlive-latexextra",
|
|
"dev-texlive/texlive-xetex",
|
|
"media-fonts/dejavu",
|
|
]
|
|
|
|
progs = {
|
|
"convert": "media-gfx/imagemagick",
|
|
"dot": "media-gfx/graphviz",
|
|
"rsvg-convert": "gnome-base/librsvg",
|
|
"virtualenv": "dev-python/virtualenv",
|
|
"xelatex": " ".join(texlive_deps),
|
|
"yaml": "dev-python/pyyaml",
|
|
"python-sphinx": "dev-python/sphinx",
|
|
}
|
|
|
|
if self.pdf:
|
|
pdf_pkgs = {
|
|
"media-fonts/dejavu": [
|
|
"/usr/share/fonts/dejavu/DejaVuSans.ttf",
|
|
],
|
|
"media-fonts/noto-cjk": [
|
|
"/usr/share/fonts/noto-cjk/NotoSansCJKsc-Regular.otf",
|
|
"/usr/share/fonts/noto-cjk/NotoSerifCJK-Regular.ttc",
|
|
],
|
|
}
|
|
for package, files in pdf_pkgs.items():
|
|
self.check_missing_file(files, package, DepManager.PDF_MANDATORY)
|
|
|
|
# Handling dependencies is a nightmare, as Gentoo refuses to emerge
|
|
# some packages if there's no package.use file describing them.
|
|
# To make it worse, compilation flags shall also be present there
|
|
# for some packages. If USE is not perfect, error/warning messages
|
|
# like those are shown:
|
|
#
|
|
# !!! The following binary packages have been ignored due to non matching USE:
|
|
#
|
|
# =media-gfx/graphviz-12.2.1-r1 X pdf -python_single_target_python3_13 qt6 svg
|
|
# =media-gfx/graphviz-12.2.1-r1 X pdf python_single_target_python3_12 -python_single_target_python3_13 qt6 svg
|
|
# =media-gfx/graphviz-12.2.1-r1 X pdf qt6 svg
|
|
# =media-gfx/graphviz-12.2.1-r1 X pdf -python_single_target_python3_10 qt6 svg
|
|
# =media-gfx/graphviz-12.2.1-r1 X pdf -python_single_target_python3_10 python_single_target_python3_12 -python_single_target_python3_13 qt6 svg
|
|
# =media-fonts/noto-cjk-20190416 X
|
|
# =app-text/texlive-core-2024-r1 X cjk -xetex
|
|
# =app-text/texlive-core-2024-r1 X -xetex
|
|
# =app-text/texlive-core-2024-r1 -xetex
|
|
# =dev-libs/zziplib-0.13.79-r1 sdl
|
|
#
|
|
# And will ignore such packages, installing the remaining ones. That
|
|
# affects mostly the image extension and PDF generation.
|
|
|
|
# Package dependencies and the minimal needed args:
|
|
portages = {
|
|
"graphviz": "media-gfx/graphviz",
|
|
"imagemagick": "media-gfx/imagemagick",
|
|
"media-libs": "media-libs/harfbuzz icu",
|
|
"media-fonts": "media-fonts/noto-cjk",
|
|
"texlive": "app-text/texlive-core xetex",
|
|
"zziblib": "dev-libs/zziplib sdl",
|
|
}
|
|
|
|
extra_cmds = ""
|
|
if not self.distro_msg:
|
|
self.distro_msg = "Note: Gentoo requires package.use to be adjusted before emerging packages"
|
|
|
|
use_base = "/etc/portage/package.use"
|
|
files = glob(f"{use_base}/*")
|
|
|
|
for fname, portage in portages.items():
|
|
install = False
|
|
|
|
while install is False:
|
|
if not files:
|
|
# No files under package.usage. Install all
|
|
install = True
|
|
break
|
|
|
|
args = portage.split(" ")
|
|
|
|
name = args.pop(0)
|
|
|
|
cmd = ["grep", "-l", "-E", rf"^{name}\b" ] + files
|
|
result = self.run(cmd, stdout=subprocess.PIPE, text=True)
|
|
if result.returncode or not result.stdout.strip():
|
|
# File containing portage name not found
|
|
install = True
|
|
break
|
|
|
|
# Ensure that needed USE flags are present
|
|
if args:
|
|
match_fname = result.stdout.strip()
|
|
with open(match_fname, 'r', encoding='utf8',
|
|
errors='backslashreplace') as fp:
|
|
for line in fp:
|
|
for arg in args:
|
|
if arg.startswith("-"):
|
|
continue
|
|
|
|
if not re.search(rf"\s*{arg}\b", line):
|
|
# Needed file argument not found
|
|
install = True
|
|
break
|
|
|
|
# Everything looks ok, don't install
|
|
break
|
|
|
|
# emit a code to setup missing USE
|
|
if install:
|
|
extra_cmds += (f"sudo su -c 'echo \"{portage}\" > {use_base}/{fname}'\n")
|
|
|
|
# Now, we can use emerge and let it respect USE
|
|
return self.get_install_progs(progs,
|
|
"emerge --ask --changed-use --binpkg-respect-use=y",
|
|
extra_cmds)
|
|
|
|
def get_install(self):
|
|
"""
|
|
OS-specific hints logic. Seeks for a hinter. If found, use it to
|
|
provide package-manager specific install commands.
|
|
|
|
Otherwise, outputs install instructions for the meta-packages.
|
|
|
|
Returns a string with the command to be executed to install the
|
|
the needed packages, if distro found. Otherwise, return just a
|
|
list of packages that require installation.
|
|
"""
|
|
os_hints = {
|
|
re.compile("Red Hat Enterprise Linux"): self.give_redhat_hints,
|
|
re.compile("Fedora"): self.give_redhat_hints,
|
|
re.compile("AlmaLinux"): self.give_redhat_hints,
|
|
re.compile("Amazon Linux"): self.give_redhat_hints,
|
|
re.compile("CentOS"): self.give_redhat_hints,
|
|
re.compile("openEuler"): self.give_redhat_hints,
|
|
re.compile("Oracle Linux Server"): self.give_redhat_hints,
|
|
re.compile("Rocky Linux"): self.give_redhat_hints,
|
|
re.compile("Springdale Open Enterprise"): self.give_redhat_hints,
|
|
|
|
re.compile("Ubuntu"): self.give_debian_hints,
|
|
re.compile("Debian"): self.give_debian_hints,
|
|
re.compile("Devuan"): self.give_debian_hints,
|
|
re.compile("Kali"): self.give_debian_hints,
|
|
re.compile("Mint"): self.give_debian_hints,
|
|
|
|
re.compile("openSUSE"): self.give_opensuse_hints,
|
|
|
|
re.compile("Mageia"): self.give_mageia_hints,
|
|
re.compile("OpenMandriva"): self.give_mageia_hints,
|
|
|
|
re.compile("Arch Linux"): self.give_arch_linux_hints,
|
|
re.compile("Gentoo"): self.give_gentoo_hints,
|
|
}
|
|
|
|
# If the OS is detected, use per-OS hint logic
|
|
for regex, os_hint in os_hints.items():
|
|
if regex.search(self.system_release):
|
|
return os_hint()
|
|
|
|
#
|
|
# Fall-back to generic hint code for other distros
|
|
# That's far from ideal, specially for LaTeX dependencies.
|
|
#
|
|
progs = {"sphinx-build": "sphinx"}
|
|
if self.pdf:
|
|
self.check_missing_tex()
|
|
|
|
self.distro_msg = \
|
|
f"I don't know distro {self.system_release}.\n" \
|
|
"So, I can't provide you a hint with the install procedure.\n" \
|
|
"There are likely missing dependencies."
|
|
|
|
return self.get_install_progs(progs, None)
|
|
|
|
#
|
|
# Common dependencies
|
|
#
|
|
def deactivate_help(self):
|
|
"""
|
|
Print a helper message to disable a virtual environment.
|
|
"""
|
|
|
|
print("\n If you want to exit the virtualenv, you can use:")
|
|
print("\tdeactivate")
|
|
|
|
def get_virtenv(self):
|
|
"""
|
|
Give a hint about how to activate an already-existing virtual
|
|
environment containing sphinx-build.
|
|
|
|
Returns a tuble with (activate_cmd_path, sphinx_version) with
|
|
the newest available virtual env.
|
|
"""
|
|
|
|
cwd = os.getcwd()
|
|
|
|
activates = []
|
|
|
|
# Add all sphinx prefixes with possible version numbers
|
|
for p in self.virtenv_prefix:
|
|
activates += glob(f"{cwd}/{p}[0-9]*/bin/activate")
|
|
|
|
activates.sort(reverse=True, key=str.lower)
|
|
|
|
# Place sphinx_latest first, if it exists
|
|
for p in self.virtenv_prefix:
|
|
activates = glob(f"{cwd}/{p}*latest/bin/activate") + activates
|
|
|
|
ver = (0, 0, 0)
|
|
for f in activates:
|
|
# Discard too old Sphinx virtual environments
|
|
match = re.search(r"(\d+)\.(\d+)\.(\d+)", f)
|
|
if match:
|
|
ver = (int(match.group(1)), int(match.group(2)), int(match.group(3)))
|
|
|
|
if ver < self.min_version:
|
|
continue
|
|
|
|
sphinx_cmd = f.replace("activate", "sphinx-build")
|
|
if not os.path.isfile(sphinx_cmd):
|
|
continue
|
|
|
|
ver = self.get_sphinx_version(sphinx_cmd)
|
|
|
|
if not ver:
|
|
venv_dir = f.replace("/bin/activate", "")
|
|
print(f"Warning: virtual environment {venv_dir} is not working.\n" \
|
|
"Python version upgrade? Remove it with:\n\n" \
|
|
"\trm -rf {venv_dir}\n\n")
|
|
else:
|
|
if self.need_sphinx and ver >= self.min_version:
|
|
return (f, ver)
|
|
elif parse_version(ver) > self.cur_version:
|
|
return (f, ver)
|
|
|
|
return ("", ver)
|
|
|
|
def recommend_sphinx_upgrade(self):
|
|
"""
|
|
Check if Sphinx needs to be upgraded.
|
|
|
|
Returns a tuple with the higest available Sphinx version if found.
|
|
Otherwise, returns None to indicate either that no upgrade is needed
|
|
or no venv was found.
|
|
"""
|
|
|
|
# Avoid running sphinx-builds from venv if cur_version is good
|
|
if self.cur_version and self.cur_version >= RECOMMENDED_VERSION:
|
|
self.latest_avail_ver = self.cur_version
|
|
return None
|
|
|
|
# Get the highest version from sphinx_*/bin/sphinx-build and the
|
|
# corresponding command to activate the venv/virtenv
|
|
self.activate_cmd, self.venv_ver = self.get_virtenv()
|
|
|
|
# Store the highest version from Sphinx existing virtualenvs
|
|
if self.activate_cmd and self.venv_ver > self.cur_version:
|
|
self.latest_avail_ver = self.venv_ver
|
|
else:
|
|
if self.cur_version:
|
|
self.latest_avail_ver = self.cur_version
|
|
else:
|
|
self.latest_avail_ver = (0, 0, 0)
|
|
|
|
# As we don't know package version of Sphinx, and there's no
|
|
# virtual environments, don't check if upgrades are needed
|
|
if not self.virtualenv:
|
|
if not self.latest_avail_ver:
|
|
return None
|
|
|
|
return self.latest_avail_ver
|
|
|
|
# Either there are already a virtual env or a new one should be created
|
|
self.need_pip = True
|
|
|
|
if not self.latest_avail_ver:
|
|
return None
|
|
|
|
# Return if the reason is due to an upgrade or not
|
|
if self.latest_avail_ver != (0, 0, 0):
|
|
if self.latest_avail_ver < RECOMMENDED_VERSION:
|
|
self.rec_sphinx_upgrade = 1
|
|
|
|
return self.latest_avail_ver
|
|
|
|
def recommend_package(self):
|
|
"""
|
|
Recommend installing Sphinx as a distro-specific package.
|
|
"""
|
|
|
|
print("\n2) As a package with:")
|
|
|
|
old_need = self.deps.need
|
|
old_optional = self.deps.optional
|
|
|
|
self.pdf = False
|
|
self.deps.optional = 0
|
|
old_verbose = self.verbose_warn_install
|
|
self.verbose_warn_install = 0
|
|
|
|
self.deps.clear_deps()
|
|
|
|
self.deps.add_package("python-sphinx", DepManager.PYTHON_MANDATORY)
|
|
|
|
cmd = self.get_install()
|
|
if cmd:
|
|
print(cmd)
|
|
|
|
self.deps.need = old_need
|
|
self.deps.optional = old_optional
|
|
self.verbose_warn_install = old_verbose
|
|
|
|
def recommend_sphinx_version(self, virtualenv_cmd):
|
|
"""
|
|
Provide recommendations for installing or upgrading Sphinx based
|
|
on current version.
|
|
|
|
The logic here is complex, as it have to deal with different versions:
|
|
|
|
- minimal supported version;
|
|
- minimal PDF version;
|
|
- recommended version.
|
|
|
|
It also needs to work fine with both distro's package and
|
|
venv/virtualenv
|
|
"""
|
|
|
|
if self.recommend_python:
|
|
cur_ver = sys.version_info[:3]
|
|
if cur_ver < MIN_PYTHON_VERSION:
|
|
print(f"\nPython version {cur_ver} is incompatible with doc build.\n" \
|
|
"Please upgrade it and re-run.\n")
|
|
return
|
|
|
|
# Version is OK. Nothing to do.
|
|
if self.cur_version != (0, 0, 0) and self.cur_version >= RECOMMENDED_VERSION:
|
|
return
|
|
|
|
if self.latest_avail_ver:
|
|
latest_avail_ver = ver_str(self.latest_avail_ver)
|
|
|
|
if not self.need_sphinx:
|
|
# sphinx-build is present and its version is >= $min_version
|
|
|
|
# only recommend enabling a newer virtenv version if makes sense.
|
|
if self.latest_avail_ver and self.latest_avail_ver > self.cur_version:
|
|
print(f"\nYou may also use the newer Sphinx version {latest_avail_ver} with:")
|
|
if f"{self.virtenv_prefix}" in os.getcwd():
|
|
print("\tdeactivate")
|
|
print(f"\t. {self.activate_cmd}")
|
|
self.deactivate_help()
|
|
return
|
|
|
|
if self.latest_avail_ver and self.latest_avail_ver >= RECOMMENDED_VERSION:
|
|
return
|
|
|
|
if not self.virtualenv:
|
|
# No sphinx either via package or via virtenv. As we can't
|
|
# Compare the versions here, just return, recommending the
|
|
# user to install it from the package distro.
|
|
if not self.latest_avail_ver or self.latest_avail_ver == (0, 0, 0):
|
|
return
|
|
|
|
# User doesn't want a virtenv recommendation, but he already
|
|
# installed one via virtenv with a newer version.
|
|
# So, print commands to enable it
|
|
if self.latest_avail_ver > self.cur_version:
|
|
print(f"\nYou may also use the Sphinx virtualenv version {latest_avail_ver} with:")
|
|
if f"{self.virtenv_prefix}" in os.getcwd():
|
|
print("\tdeactivate")
|
|
print(f"\t. {self.activate_cmd}")
|
|
self.deactivate_help()
|
|
return
|
|
print("\n")
|
|
else:
|
|
if self.need_sphinx:
|
|
self.deps.need += 1
|
|
|
|
# Suggest newer versions if current ones are too old
|
|
if self.latest_avail_ver and self.latest_avail_ver >= self.min_version:
|
|
if self.latest_avail_ver >= RECOMMENDED_VERSION:
|
|
print(f"\nNeed to activate Sphinx (version {latest_avail_ver}) on virtualenv with:")
|
|
print(f"\t. {self.activate_cmd}")
|
|
self.deactivate_help()
|
|
return
|
|
|
|
# Version is above the minimal required one, but may be
|
|
# below the recommended one. So, print warnings/notes
|
|
if self.latest_avail_ver < RECOMMENDED_VERSION:
|
|
print(f"Warning: It is recommended at least Sphinx version {RECOMMENDED_VERSION}.")
|
|
|
|
# At this point, either it needs Sphinx or upgrade is recommended,
|
|
# both via pip
|
|
|
|
if self.rec_sphinx_upgrade:
|
|
if not self.virtualenv:
|
|
print("Instead of install/upgrade Python Sphinx pkg, you could use pip/pypi with:\n\n")
|
|
else:
|
|
print("To upgrade Sphinx, use:\n\n")
|
|
else:
|
|
print("\nSphinx needs to be installed either:\n1) via pip/pypi with:\n")
|
|
|
|
if not virtualenv_cmd:
|
|
print(" Currently not possible.\n")
|
|
print(" Please upgrade Python to a newer version and run this script again")
|
|
else:
|
|
print(f"\t{virtualenv_cmd} {self.virtenv_dir}")
|
|
print(f"\t. {self.virtenv_dir}/bin/activate")
|
|
print(f"\tpip install -r {self.requirement_file}")
|
|
self.deactivate_help()
|
|
|
|
if self.package_supported:
|
|
self.recommend_package()
|
|
|
|
print("\n" \
|
|
" Please note that Sphinx currentlys produce false-positive\n" \
|
|
" warnings when the same name is used for more than one type (functions,\n" \
|
|
" structs, enums,...). This is known Sphinx bug. For more details, see:\n" \
|
|
"\thttps://github.com/sphinx-doc/sphinx/pull/8313")
|
|
|
|
def check_needs(self):
|
|
"""
|
|
Main method that checks needed dependencies and provides
|
|
recommendations.
|
|
"""
|
|
self.python_cmd = sys.executable
|
|
|
|
# Check if Sphinx is already accessible from current environment
|
|
self.check_sphinx(self.conf)
|
|
|
|
if self.system_release:
|
|
print(f"Detected OS: {self.system_release}.")
|
|
else:
|
|
print("Unknown OS")
|
|
if self.cur_version != (0, 0, 0):
|
|
ver = ver_str(self.cur_version)
|
|
print(f"Sphinx version: {ver}\n")
|
|
|
|
# Check the type of virtual env, depending on Python version
|
|
virtualenv_cmd = None
|
|
|
|
if sys.version_info < MIN_PYTHON_VERSION:
|
|
min_ver = ver_str(MIN_PYTHON_VERSION)
|
|
print(f"ERROR: at least python {min_ver} is required to build the kernel docs")
|
|
self.need_sphinx = 1
|
|
|
|
self.venv_ver = self.recommend_sphinx_upgrade()
|
|
|
|
if self.need_pip:
|
|
if sys.version_info < MIN_PYTHON_VERSION:
|
|
self.need_pip = False
|
|
print("Warning: python version is not supported.")
|
|
else:
|
|
virtualenv_cmd = f"{self.python_cmd} -m venv"
|
|
self.check_python_module("ensurepip")
|
|
|
|
# Check for needed programs/tools
|
|
self.check_perl_module("Pod::Usage", DepManager.SYSTEM_MANDATORY)
|
|
|
|
self.check_program("make", DepManager.SYSTEM_MANDATORY)
|
|
self.check_program("which", DepManager.SYSTEM_MANDATORY)
|
|
|
|
self.check_program("dot", DepManager.SYSTEM_OPTIONAL)
|
|
self.check_program("convert", DepManager.SYSTEM_OPTIONAL)
|
|
|
|
self.check_python_module("yaml")
|
|
|
|
if self.pdf:
|
|
self.check_program("xelatex", DepManager.PDF_MANDATORY)
|
|
self.check_program("rsvg-convert", DepManager.PDF_MANDATORY)
|
|
self.check_program("latexmk", DepManager.PDF_MANDATORY)
|
|
|
|
# Do distro-specific checks and output distro-install commands
|
|
cmd = self.get_install()
|
|
if cmd:
|
|
print(cmd)
|
|
|
|
# If distro requires some special instructions, print here.
|
|
# Please notice that get_install() needs to be called first.
|
|
if self.distro_msg:
|
|
print("\n" + self.distro_msg)
|
|
|
|
if not self.python_cmd:
|
|
if self.need == 1:
|
|
sys.exit("Can't build as 1 mandatory dependency is missing")
|
|
elif self.need:
|
|
sys.exit(f"Can't build as {self.need} mandatory dependencies are missing")
|
|
|
|
# Check if sphinx-build is called sphinx-build-3
|
|
if self.need_symlink:
|
|
sphinx_path = self.which("sphinx-build-3")
|
|
if sphinx_path:
|
|
print(f"\tsudo ln -sf {sphinx_path} /usr/bin/sphinx-build\n")
|
|
|
|
self.recommend_sphinx_version(virtualenv_cmd)
|
|
print("")
|
|
|
|
if not self.deps.optional:
|
|
print("All optional dependencies are met.")
|
|
|
|
if self.deps.need == 1:
|
|
sys.exit("Can't build as 1 mandatory dependency is missing")
|
|
elif self.deps.need:
|
|
sys.exit(f"Can't build as {self.deps.need} mandatory dependencies are missing")
|
|
|
|
print("Needed package dependencies are met.")
|
|
|
|
DESCRIPTION = """
|
|
Process some flags related to Sphinx installation and documentation build.
|
|
"""
|
|
|
|
|
|
def main():
|
|
"""Main function"""
|
|
parser = argparse.ArgumentParser(description=DESCRIPTION)
|
|
|
|
parser.add_argument(
|
|
"--no-virtualenv",
|
|
action="store_false",
|
|
dest="virtualenv",
|
|
help="Recommend installing Sphinx instead of using a virtualenv",
|
|
)
|
|
|
|
parser.add_argument(
|
|
"--no-pdf",
|
|
action="store_false",
|
|
dest="pdf",
|
|
help="Don't check for dependencies required to build PDF docs",
|
|
)
|
|
|
|
parser.add_argument(
|
|
"--version-check",
|
|
action="store_true",
|
|
dest="version_check",
|
|
help="If version is compatible, don't check for missing dependencies",
|
|
)
|
|
|
|
args = parser.parse_args()
|
|
|
|
checker = SphinxDependencyChecker(args)
|
|
|
|
checker.check_python()
|
|
checker.check_needs()
|
|
|
|
# Call main if not used as module
|
|
if __name__ == "__main__":
|
|
main()
|