mirror of https://gitee.com/openkylin/apport.git
Add general hooks
Forwarded: not-needed Gbp-Pq: Name Add-general-hooks.patch
This commit is contained in:
parent
2fd154b495
commit
4a9a585cd9
|
@ -19,6 +19,7 @@
|
|||
import base64
|
||||
import datetime
|
||||
import glob
|
||||
import json
|
||||
import os
|
||||
import re
|
||||
import select
|
||||
|
@ -1085,6 +1086,22 @@ def attach_default_grub(report, key=None):
|
|||
report[key] = "".join(filtered)
|
||||
|
||||
|
||||
def attach_casper_md5check(report, location):
|
||||
"""Attach the results of the casper md5check of install media."""
|
||||
result = "unknown"
|
||||
mismatches = []
|
||||
if os.path.exists(location):
|
||||
attach_root_command_outputs(report, {"CasperMD5json": f"cat '{location}'"})
|
||||
if "CasperMD5json" in report:
|
||||
check = json.loads(report["CasperMD5json"])
|
||||
result = check["result"]
|
||||
mismatches = check["checksum_missmatch"]
|
||||
report["CasperMD5CheckResult"] = result
|
||||
if mismatches:
|
||||
report["CasperMD5CheckMismatches"] = " ".join(mismatches)
|
||||
report.pop("CasperMD5json", None)
|
||||
|
||||
|
||||
# backwards compatible API
|
||||
shared_libraries = apport.fileutils.shared_libraries
|
||||
links_with_shared_library = apport.fileutils.links_with_shared_library
|
||||
|
|
|
@ -0,0 +1,42 @@
|
|||
"""Redirect reports on packages from the Ubuntu Cloud Archive to the
|
||||
launchpad cloud-archive project.
|
||||
|
||||
Copyright (C) 2013 Canonical Ltd.
|
||||
Author: James Page <james.page@ubuntu.com>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU General Public License as published by the
|
||||
Free Software Foundation; either version 2 of the License, or (at your
|
||||
option) any later version. See http://www.gnu.org/copyleft/gpl.html for
|
||||
the full text of the license.
|
||||
"""
|
||||
|
||||
from apport import packaging
|
||||
|
||||
|
||||
def add_info(report, unused_ui):
|
||||
"""Redirect reports on packages from the Ubuntu Cloud Archive to the
|
||||
launchpad cloud-archive project."""
|
||||
package = report.get("Package")
|
||||
if not package:
|
||||
return
|
||||
package = package.split()[0]
|
||||
try:
|
||||
if (
|
||||
"~cloud" in packaging.get_version(package)
|
||||
and packaging.get_package_origin(package) == "Canonical"
|
||||
):
|
||||
report[
|
||||
"CrashDB"
|
||||
] = """\
|
||||
{
|
||||
"impl": "launchpad",
|
||||
"project": "cloud-archive",
|
||||
"bug_pattern_url": "http://people.canonical.com/"
|
||||
"~ubuntu-archive/bugpatterns/bugpatterns.xml",
|
||||
}
|
||||
"""
|
||||
except ValueError as error:
|
||||
if "does not exist" in str(error):
|
||||
return
|
||||
raise error
|
|
@ -0,0 +1,122 @@
|
|||
# This hook collects logs for Power systems and more specific logs for Pseries,
|
||||
# PowerNV platforms.
|
||||
#
|
||||
# Author: Thierry FAUCK <thierry@linux.vnet.ibm.com>
|
||||
#
|
||||
# This program is free software; you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation; either version 2 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program; if not, write to the Free Software
|
||||
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
|
||||
"""IBM Power System related information"""
|
||||
|
||||
import os
|
||||
import pathlib
|
||||
import platform
|
||||
import subprocess
|
||||
import tempfile
|
||||
|
||||
from apport.hookutils import (
|
||||
attach_file,
|
||||
attach_file_if_exists,
|
||||
attach_root_command_outputs,
|
||||
command_available,
|
||||
command_output,
|
||||
)
|
||||
|
||||
|
||||
def _add_tar(report, path, key):
|
||||
(tar_fd, tar_file) = tempfile.mkstemp(prefix="apport.", suffix=".tar")
|
||||
os.close(tar_fd)
|
||||
subprocess.call(["tar", "chf", tar_file, path])
|
||||
if os.path.getsize(tar_file) > 0:
|
||||
report[key] = (tar_file,)
|
||||
# NB, don't cleanup the temp file, it'll get read later by the apport main
|
||||
# code
|
||||
|
||||
|
||||
# TODO: Split into smaller functions/methods
|
||||
# pylint: disable-next=too-many-branches,too-many-statements
|
||||
def add_info(report, unused_ui):
|
||||
"""Add IBM Power System related information to report."""
|
||||
arch = platform.machine()
|
||||
if arch not in ["ppc64", "ppc64le"]:
|
||||
return
|
||||
|
||||
is_kernel = report["ProblemType"].startswith("Kernel") or "linux" in report.get(
|
||||
"Package", ""
|
||||
)
|
||||
|
||||
try:
|
||||
contents = pathlib.Path("/proc/cpuinfo").read_text("utf-8")
|
||||
is_pseries = "pSeries" in contents
|
||||
is_power_nv = "PowerNV" in contents
|
||||
is_power_kvm = "emulated by qemu" in contents
|
||||
except IOError:
|
||||
is_pseries = False
|
||||
is_power_nv = False
|
||||
is_power_kvm = False
|
||||
|
||||
if is_pseries or is_power_nv:
|
||||
if is_kernel:
|
||||
_add_tar(report, "/proc/device-tree/", "DeviceTree.tar")
|
||||
attach_file(report, "/proc/misc", "ProcMisc")
|
||||
attach_file(report, "/proc/locks", "ProcLocks")
|
||||
attach_file(report, "/proc/loadavg", "ProcLoadAvg")
|
||||
attach_file(report, "/proc/swaps", "ProcSwaps")
|
||||
attach_file(report, "/proc/version", "ProcVersion")
|
||||
report["cpu_smt"] = command_output(["ppc64_cpu", "--smt"])
|
||||
report["cpu_cores"] = command_output(["ppc64_cpu", "--cores-present"])
|
||||
report["cpu_coreson"] = command_output(["ppc64_cpu", "--cores-on"])
|
||||
# To be executed as root
|
||||
if is_kernel:
|
||||
attach_root_command_outputs(
|
||||
report,
|
||||
{
|
||||
"cpu_runmode": "ppc64_cpu --run-mode",
|
||||
"cpu_freq": "ppc64_cpu --frequency",
|
||||
"cpu_dscr": "ppc64_cpu --dscr",
|
||||
"nvram": "cat /dev/nvram",
|
||||
},
|
||||
)
|
||||
attach_file_if_exists(report, "/var/log/platform")
|
||||
|
||||
if is_pseries and not is_power_kvm:
|
||||
attach_file(report, "/proc/ppc64/lparcfg", "ProcLparCfg")
|
||||
attach_file(report, "/proc/ppc64/eeh", "ProcEeh")
|
||||
attach_file(report, "/proc/ppc64/systemcfg", "ProcSystemCfg")
|
||||
report["lscfg_vp"] = command_output(["lscfg", "-vp"])
|
||||
report["lsmcode"] = command_output(["lsmcode", "-A"])
|
||||
report["bootlist"] = command_output(["bootlist", "-m", "both", "-r"])
|
||||
report["lparstat"] = command_output(["lparstat", "-i"])
|
||||
if command_available("lsvpd"):
|
||||
report["lsvpd"] = command_output(["lsvpd", "--debug"])
|
||||
if command_available("lsvio"):
|
||||
report["lsvio"] = command_output(["lsvio", "-des"])
|
||||
if command_available("servicelog"):
|
||||
report["servicelog_dump"] = command_output(["servicelog", "--dump"])
|
||||
if command_available("servicelog_notify"):
|
||||
report["servicelog_list"] = command_output(["servicelog_notify", "--list"])
|
||||
if command_available("usysattn"):
|
||||
report["usysattn"] = command_output(["usysattn"])
|
||||
if command_available("usysident"):
|
||||
report["usysident"] = command_output(["usysident"])
|
||||
if command_available("serv_config"):
|
||||
report["serv_config"] = command_output(["serv_config", "-l"])
|
||||
|
||||
if is_power_nv:
|
||||
_add_tar(report, "/proc/ppc64/", "ProcPpc64.tar")
|
||||
attach_file_if_exists(report, "/sys/firmware/opal/msglog")
|
||||
if os.path.exists("/var/log/dump"):
|
||||
report["VarLogDump_list"] = command_output(["ls", "-l", "/var/log/dump"])
|
||||
if is_kernel:
|
||||
_add_tar(report, "/var/log/opal-elog", "OpalElog.tar")
|
|
@ -0,0 +1,69 @@
|
|||
"""Bugs and crashes for the Ubuntu GNOME flavour.
|
||||
|
||||
Copyright (C) 2013 Canonical Ltd.
|
||||
Author: Martin Pitt <martin.pitt@ubuntu.com>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU General Public License as published by the
|
||||
Free Software Foundation; either version 2 of the License, or (at your
|
||||
option) any later version. See http://www.gnu.org/copyleft/gpl.html for
|
||||
the full text of the license.
|
||||
"""
|
||||
|
||||
# pylint: disable=invalid-name
|
||||
# pylint: enable=invalid-name
|
||||
|
||||
|
||||
def add_info(report, unused_ui):
|
||||
"""Handle bugs and crashes for the Ubuntu GNOME flavour."""
|
||||
release = report.get("DistroRelease", "")
|
||||
|
||||
msg = (
|
||||
"The GNOME3 PPA you are using is no longer supported"
|
||||
" for this Ubuntu release. Please "
|
||||
)
|
||||
# redirect reports against PPA packages to ubuntu-gnome project
|
||||
if "[origin: LP-PPA-gnome3-team-gnome3" in report.get("Package", ""):
|
||||
report[
|
||||
"CrashDB"
|
||||
] = """\
|
||||
{
|
||||
"impl": "launchpad",
|
||||
"project": "ubuntu-gnome",
|
||||
"bug_pattern_url": "http://people.canonical.com/"
|
||||
"~ubuntu-archive/bugpatterns/bugpatterns.xml",
|
||||
"dupdb_url": "http://phillw.net/ubuntu-gnome/apport_duplicates/",
|
||||
}
|
||||
"""
|
||||
|
||||
# using the staging PPA?
|
||||
if "LP-PPA-gnome3-team-gnome3-staging" in report.get("Package", ""):
|
||||
report.setdefault("Tags", "")
|
||||
report["Tags"] += " gnome3-staging"
|
||||
if release in {"Ubuntu 14.04", "Ubuntu 16.04"}:
|
||||
report["UnreportableReason"] = (
|
||||
f'{msg} run "ppa-purge ppa:gnome3-team/gnome3-staging".'
|
||||
)
|
||||
|
||||
# using the next PPA?
|
||||
elif "LP-PPA-gnome3-team-gnome3-next" in report.get("Package", ""):
|
||||
report.setdefault("Tags", "")
|
||||
report["Tags"] += " gnome3-next"
|
||||
if release in {"Ubuntu 14.04", "Ubuntu 16.04"}:
|
||||
report["UnreportableReason"] = (
|
||||
f'{msg} run "ppa-purge ppa:gnome3-team/gnome3-next".'
|
||||
)
|
||||
|
||||
elif release in {"Ubuntu 14.04", "Ubuntu 16.04"}:
|
||||
report["UnreportableReason"] = (
|
||||
f'{msg} run "ppa-purge ppa:gnome3-team/gnome3".'
|
||||
)
|
||||
|
||||
if "[origin: LP-PPA-gnome3-team-gnome3" in report.get("Dependencies", ""):
|
||||
report.setdefault("Tags", "")
|
||||
report["Tags"] += " gnome3-ppa"
|
||||
if (
|
||||
release in {"Ubuntu 14.04", "Ubuntu 16.04"}
|
||||
and "UnreportableReason" not in report
|
||||
):
|
||||
report["UnreportableReason"] = f"{msg} use ppa-purge to remove the PPA."
|
|
@ -0,0 +1,635 @@
|
|||
"""Attach generally useful information, not specific to any package.
|
||||
|
||||
Copyright (C) 2009 Canonical Ltd.
|
||||
Authors: Matt Zimmerman <mdz@canonical.com>,
|
||||
Brian Murray <brian@ubuntu.com>
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU General Public License as published by the
|
||||
Free Software Foundation; either version 2 of the License, or (at your
|
||||
option) any later version. See http://www.gnu.org/copyleft/gpl.html for
|
||||
the full text of the license.
|
||||
"""
|
||||
|
||||
import os
|
||||
import pathlib
|
||||
import platform
|
||||
import re
|
||||
import subprocess
|
||||
import sys
|
||||
import time
|
||||
from gettext import gettext as _
|
||||
from glob import glob
|
||||
|
||||
import apport.hookutils
|
||||
import apport.packaging
|
||||
import problem_report
|
||||
|
||||
|
||||
def add_info(report, ui):
|
||||
# TODO: Split into smaller functions/methods
|
||||
# pylint: disable=invalid-name,too-many-branches,too-many-locals,too-many-statements
|
||||
# pylint: disable=too-many-nested-blocks
|
||||
"""Attach generally useful information, not specific to any package."""
|
||||
_add_release_info(report)
|
||||
_add_kernel_info(report)
|
||||
add_proposed_info(report)
|
||||
|
||||
# collect a condensed version of /proc/cpuinfo
|
||||
apport.hookutils.attach_file(report, "/proc/cpuinfo", "ProcCpuinfo")
|
||||
short_cpuinfo = []
|
||||
for item in reversed(report.get("ProcCpuinfo", "").split("\n")):
|
||||
short_cpuinfo.append(item)
|
||||
if item.startswith("processor\t:"):
|
||||
break
|
||||
short_cpuinfo = reversed(short_cpuinfo)
|
||||
report["ProcCpuinfoMinimal"] = "\n".join(short_cpuinfo)
|
||||
report.pop("ProcCpuinfo")
|
||||
|
||||
hook_errors = [k for k in report.keys() if k.startswith("HookError_")]
|
||||
if hook_errors:
|
||||
report.add_tags(["apport-hook-error"])
|
||||
|
||||
# locally installed python versions can cause a multitude of errors
|
||||
if (
|
||||
report.get("ProblemType") == "Package"
|
||||
or "python" in report.get("InterpreterPath", "")
|
||||
or "python" in report.get("ExecutablePath", "")
|
||||
):
|
||||
for python in ("python", "python3"):
|
||||
add_python_details(f"{python.title()}Details", python, report)
|
||||
|
||||
try:
|
||||
report["ApportVersion"] = apport.packaging.get_version("apport")
|
||||
except ValueError:
|
||||
# might happen on local installs
|
||||
pass
|
||||
|
||||
# We want to know if people have modified apport's crashdb.conf in case
|
||||
# crashes are reported to Launchpad when they shouldn't be e.g. for a
|
||||
# non-development release.
|
||||
apport.hookutils.attach_conffiles(report, "apport", ui=ui)
|
||||
|
||||
# Should the system have been rebooted?
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/var/run/reboot-required.pkgs", "RebootRequiredPkgs"
|
||||
)
|
||||
|
||||
casper_md5check = "casper-md5check.json"
|
||||
if "LiveMediaBuild" in report:
|
||||
apport.hookutils.attach_casper_md5check(report, f"/run/{casper_md5check}")
|
||||
else:
|
||||
apport.hookutils.attach_casper_md5check(
|
||||
report, f"/var/log/installer/{casper_md5check}"
|
||||
)
|
||||
|
||||
if report.get("ProblemType") == "Package":
|
||||
# every error report regarding a package should have package manager
|
||||
# version information
|
||||
apport.hookutils.attach_related_packages(report, ["dpkg", "apt"])
|
||||
_check_for_disk_error(report)
|
||||
# check to see if the real root on a persistent media is full
|
||||
if "LiveMediaBuild" in report:
|
||||
st = os.statvfs("/cdrom")
|
||||
free_mb = st.f_bavail * st.f_frsize / 1000000
|
||||
if free_mb < 10:
|
||||
report["UnreportableReason"] = (
|
||||
f"Your system partition has less than {free_mb} MB"
|
||||
f" of free space available, which leads to problems"
|
||||
f" using applications and installing updates."
|
||||
f" Please free some space."
|
||||
)
|
||||
|
||||
_match_error_messages(report)
|
||||
|
||||
# these attachments will not exist if ProblemType is Bug as the package
|
||||
# hook runs after the general hook
|
||||
for attachment in ("DpkgTerminalLog", "VarLogDistupgradeApttermlog"):
|
||||
if attachment in report:
|
||||
log_file = _get_attachment_contents(report, attachment)
|
||||
untrimmed_dpkg_log = log_file
|
||||
_check_attachment_for_errors(report, attachment)
|
||||
trimmed_log = _get_attachment_contents(report, attachment)
|
||||
trimmed_log = trimmed_log.split("\n")
|
||||
lines = []
|
||||
for line in untrimmed_dpkg_log.splitlines():
|
||||
if line not in trimmed_log:
|
||||
lines.append(str(line))
|
||||
elif line in trimmed_log:
|
||||
trimmed_log.remove(line)
|
||||
dpkg_log_without_error = "\n".join(lines)
|
||||
|
||||
# crash reports from live system installer often expose target mount
|
||||
for f in ("ExecutablePath", "InterpreterPath"):
|
||||
if f in report and report[f].startswith("/target/"):
|
||||
report[f] = report[f][7:]
|
||||
|
||||
# Allow filing update-manager bugs with obsolete packages
|
||||
if report.get("Package", "").startswith("update-manager"):
|
||||
os.environ["APPORT_IGNORE_OBSOLETE_PACKAGES"] = "1"
|
||||
|
||||
# file bugs against OEM project for modified packages
|
||||
if "Package" in report:
|
||||
v = report["Package"].split()[1]
|
||||
oem_project = get_oem_project(report)
|
||||
if oem_project and ("common" in v or oem_project in v):
|
||||
report["CrashDB"] = "canonical-oem"
|
||||
|
||||
if "Package" in report:
|
||||
package = report["Package"].split()[0]
|
||||
if package:
|
||||
apport.hookutils.attach_conffiles(report, package, ui=ui)
|
||||
|
||||
# do not file bugs against "upgrade-system" if it is not installed
|
||||
# (LP#404727)
|
||||
if package == "upgrade-system" and "not installed" in report["Package"]:
|
||||
report["UnreportableReason"] = (
|
||||
"You do not have the upgrade-system package installed."
|
||||
" Please report package upgrade failures against the package"
|
||||
" that failed to install, or against upgrade-manager."
|
||||
)
|
||||
|
||||
# build a duplicate signature tag for package reports
|
||||
if report.get("ProblemType") == "Package":
|
||||
if "DpkgTerminalLog" in report:
|
||||
# this was previously trimmed in check_attachment_for_errors
|
||||
termlog = report["DpkgTerminalLog"]
|
||||
elif "VarLogDistupgradeApttermlog" in report:
|
||||
termlog = _get_attachment_contents(report, "VarLogDistupgradeApttermlog")
|
||||
else:
|
||||
termlog = None
|
||||
if termlog:
|
||||
(package, version) = report["Package"].split(None, 1)
|
||||
# for packages that run update-grub include /etc/default/grub
|
||||
UPDATE_BOOT = [
|
||||
"friendly-recovery",
|
||||
"linux",
|
||||
"memtest86+",
|
||||
"plymouth",
|
||||
"ubuntu-meta",
|
||||
"virtualbox-ose",
|
||||
]
|
||||
ug_failure = (
|
||||
r"/etc/kernel/post(inst|rm)\.d/"
|
||||
r"zz-update-grub exited with return code [1-9]+"
|
||||
)
|
||||
mkconfig_failure = (
|
||||
r"/usr/sbin/grub-mkconfig.*/etc/default/grub: Syntax error"
|
||||
)
|
||||
if re.search(ug_failure, termlog) or re.search(mkconfig_failure, termlog):
|
||||
if report["SourcePackage"] in UPDATE_BOOT:
|
||||
apport.hookutils.attach_default_grub(report, "EtcDefaultGrub")
|
||||
dupe_sig = ""
|
||||
dupe_sig_created = False
|
||||
# messages we expect to see from a package manager (LP: #1692127)
|
||||
pkg_mngr_msgs = re.compile(
|
||||
r"^(Authenticating|De-configuring|Examining|Installing"
|
||||
r"|Preparing|Processing triggers|Purging|Removing|Replaced"
|
||||
r"|Replacing|Setting up|Unpacking|Would remove).*\.\.\.\s*$"
|
||||
)
|
||||
for line in termlog.split("\n"):
|
||||
if pkg_mngr_msgs.search(line):
|
||||
dupe_sig = f"{line}\n"
|
||||
dupe_sig_created = True
|
||||
continue
|
||||
dupe_sig += f"{line}\n"
|
||||
# this doesn't catch 'dpkg-divert: error' LP: #1581399
|
||||
if "dpkg: error" in dupe_sig and line.startswith(" "):
|
||||
if "trying to overwrite" in line:
|
||||
conflict_pkg = re.search("in package (.*) ", line)
|
||||
if conflict_pkg and not apport.packaging.is_distro_package(
|
||||
conflict_pkg.group(1)
|
||||
):
|
||||
report["UnreportableReason"] = _(
|
||||
"An Ubuntu package has a file conflict with a "
|
||||
"package that is not a genuine Ubuntu package."
|
||||
)
|
||||
report.add_tags(["package-conflict"])
|
||||
if dupe_sig_created:
|
||||
# the duplicate signature should be the first failure
|
||||
report["DuplicateSignature"] = (
|
||||
f"package:{package}:{version}\n{dupe_sig}"
|
||||
)
|
||||
break
|
||||
if dupe_sig:
|
||||
if dpkg_log_without_error.find(dupe_sig) != -1:
|
||||
report["UnreportableReason"] = _(
|
||||
"You have already encountered this package"
|
||||
" installation failure."
|
||||
)
|
||||
|
||||
|
||||
def _match_error_messages(report):
|
||||
# There are enough of these now that it is probably worth refactoring...
|
||||
# -mdz
|
||||
if report.get("ProblemType") == "Package":
|
||||
if "failed to install/upgrade: corrupted filesystem tarfile" in report.get(
|
||||
"Title", ""
|
||||
):
|
||||
report["UnreportableReason"] = (
|
||||
"This failure was caused by a corrupted package download"
|
||||
" or file system corruption."
|
||||
)
|
||||
|
||||
if "is already installed and configured" in report.get("ErrorMessage", ""):
|
||||
report["SourcePackage"] = "dpkg"
|
||||
|
||||
|
||||
def _check_attachment_for_errors(report, attachment):
|
||||
# TODO: Split into smaller functions/methods
|
||||
# pylint: disable=too-many-branches,too-many-statements,too-many-nested-blocks
|
||||
if report.get("ProblemType") == "Package":
|
||||
wrong_grub_msg = _(
|
||||
"""\
|
||||
Your system was initially configured with grub version 2, but you have\
|
||||
removed it from your system in favor of grub 1 without configuring it.\
|
||||
To ensure your bootloader configuration is updated whenever a new kernel\
|
||||
is available, open a terminal and run:
|
||||
|
||||
sudo apt-get install grub-pc
|
||||
"""
|
||||
)
|
||||
|
||||
trim_dpkg_log(report)
|
||||
log_file = _get_attachment_contents(report, attachment)
|
||||
|
||||
grub_hook_failure = "DpkgTerminalLog" in report and bool(
|
||||
re.search(
|
||||
r"^Not creating /boot/grub/menu.lst as you wish",
|
||||
report["DpkgTerminalLog"],
|
||||
re.MULTILINE,
|
||||
)
|
||||
)
|
||||
|
||||
if report["Package"] not in ["grub", "grub2"]:
|
||||
# linux-image postinst emits this when update-grub fails
|
||||
# https://wiki.ubuntu.com/KernelTeam/DebuggingUpdateErrors
|
||||
grub_errors = [
|
||||
r"^User postinst hook script \[.*update-grub\] exited with value",
|
||||
r"^run-parts: /etc/kernel/post(inst|rm).d"
|
||||
r"/zz-update-grub exited with return code [1-9]+",
|
||||
r"^/usr/sbin/grub-probe: error",
|
||||
]
|
||||
|
||||
for grub_error in grub_errors:
|
||||
if attachment in report and re.search(
|
||||
grub_error, log_file, re.MULTILINE
|
||||
):
|
||||
# File these reports on the grub package instead
|
||||
grub_package = apport.packaging.get_file_package(
|
||||
"/usr/sbin/update-grub"
|
||||
)
|
||||
if (
|
||||
grub_package is None
|
||||
or grub_package == "grub"
|
||||
and "grub-probe" not in log_file
|
||||
):
|
||||
report["SourcePackage"] = "grub"
|
||||
if os.path.exists("/boot/grub/grub.cfg") and grub_hook_failure:
|
||||
report["UnreportableReason"] = wrong_grub_msg
|
||||
else:
|
||||
report["SourcePackage"] = "grub2"
|
||||
|
||||
if report["Package"] != "initramfs-tools":
|
||||
# update-initramfs emits this when it fails, usually invoked
|
||||
# from the linux-image postinst
|
||||
# https://wiki.ubuntu.com/KernelTeam/DebuggingUpdateErrors
|
||||
if attachment in report and re.search(
|
||||
r"^update-initramfs: failed for ", log_file, re.MULTILINE
|
||||
):
|
||||
# File these reports on the initramfs-tools package instead
|
||||
report["SourcePackage"] = "initramfs-tools"
|
||||
|
||||
if report["Package"].startswith("linux-image-") and attachment in report:
|
||||
# /etc/kernel/*.d failures from kernel package postinst
|
||||
match = re.search(
|
||||
r"^run-parts: (/etc/kernel/\S+\.d/\S+) exited with return code \d+",
|
||||
log_file,
|
||||
re.MULTILINE,
|
||||
)
|
||||
if match:
|
||||
path = match.group(1)
|
||||
package = apport.packaging.get_file_package(path)
|
||||
if package:
|
||||
report["SourcePackage"] = package
|
||||
report["ErrorMessage"] = match.group(0)
|
||||
if package == "grub-pc" and grub_hook_failure:
|
||||
report["UnreportableReason"] = wrong_grub_msg
|
||||
else:
|
||||
report["UnreportableReason"] = (
|
||||
"This failure was caused by a program"
|
||||
" which did not originate from Ubuntu"
|
||||
)
|
||||
|
||||
error_message = report.get("ErrorMessage")
|
||||
corrupt_package = (
|
||||
"This failure was caused by a corrupted package download"
|
||||
" or file system corruption."
|
||||
)
|
||||
out_of_memory = "This failure was caused by the system running out of memory."
|
||||
|
||||
if "failed to install/upgrade: corrupted filesystem tarfile" in report.get(
|
||||
"Title", ""
|
||||
):
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if "dependency problems - leaving unconfigured" in error_message:
|
||||
report["UnreportableReason"] = (
|
||||
"This failure is a followup error from a previous"
|
||||
" package install failure."
|
||||
)
|
||||
|
||||
if "failed to allocate memory" in error_message:
|
||||
report["UnreportableReason"] = out_of_memory
|
||||
|
||||
if "cannot access archive" in error_message:
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if re.search(
|
||||
r"(failed to read|failed in write|short read) on buffer copy", error_message
|
||||
):
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if re.search(
|
||||
r"(failed to read|failed to write|failed to seek"
|
||||
r"|unexpected end of file or stream)",
|
||||
error_message,
|
||||
):
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if re.search(
|
||||
r"(--fsys-tarfile|dpkg-deb --control) returned error exit status 2",
|
||||
error_message,
|
||||
):
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if attachment in report and re.search(
|
||||
r"dpkg-deb: error.*is not a debian format archive", log_file, re.MULTILINE
|
||||
):
|
||||
report["UnreportableReason"] = corrupt_package
|
||||
|
||||
if "is already installed and configured" in report.get("ErrorMessage", ""):
|
||||
# there is insufficient information in the data currently gathered
|
||||
# so gather more data
|
||||
report["SourcePackage"] = "dpkg"
|
||||
report["AptdaemonVersion"] = apport.packaging.get_version("aptdaemon")
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/var/log/dpkg.log", "DpkgLog"
|
||||
)
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/var/log/apt/term.log", "AptTermLog"
|
||||
)
|
||||
# gather filenames in /var/crash to see if there is one for dpkg
|
||||
reports = glob("/var/crash/*")
|
||||
if reports:
|
||||
report["CrashReports"] = apport.hookutils.command_output(
|
||||
["stat", "-c", "%a:%u:%g:%s:%y:%x:%n"] + reports
|
||||
)
|
||||
report.add_tags(["already-installed"])
|
||||
|
||||
|
||||
def _check_for_disk_error(report):
|
||||
devs_to_check = []
|
||||
if "Dmesg.txt" not in report and "CurrentDmesg.txt" not in report:
|
||||
return
|
||||
if "Df.txt" not in report:
|
||||
return
|
||||
df_output = report["Df.txt"]
|
||||
device_error = False
|
||||
for line in df_output:
|
||||
line = line.strip("\n")
|
||||
if line.endswith("/") or line.endswith("/usr") or line.endswith("/var"):
|
||||
# without manipulation it'd look like /dev/sda1
|
||||
device = line.split(" ")[0].strip("0123456789")
|
||||
device = device.replace("/dev/", "")
|
||||
devs_to_check.append(device)
|
||||
dmesg = report.get("CurrentDmesg.txt", report["Dmesg.txt"])
|
||||
for line in dmesg:
|
||||
line = line.strip("\n")
|
||||
if "I/O error" in line:
|
||||
# no device in this line
|
||||
if "journal commit I/O error" in line:
|
||||
continue
|
||||
for dev in devs_to_check:
|
||||
if re.search(dev, line):
|
||||
error_device = dev
|
||||
device_error = True
|
||||
break
|
||||
if device_error:
|
||||
report["UnreportableReason"] = (
|
||||
f"This failure was caused by a hardware error on /dev/{error_device}"
|
||||
)
|
||||
|
||||
|
||||
def _add_kernel_info(report):
|
||||
# This includes the Ubuntu packaged kernel version
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/proc/version_signature", "ProcVersionSignature"
|
||||
)
|
||||
|
||||
|
||||
def _add_release_info(report):
|
||||
# https://bugs.launchpad.net/bugs/364649
|
||||
media = "/var/log/installer/media-info"
|
||||
apport.hookutils.attach_file_if_exists(report, media, "InstallationMedia")
|
||||
# Preinstalled Raspberry Pi images include a build date breadcrumb
|
||||
apport.hookutils.attach_file_if_exists(report, "/.disk/info", "ImageMediaBuild")
|
||||
if "ImageMediaBuild" in report:
|
||||
report.add_tags([f"{report['Architecture']}-image"])
|
||||
try:
|
||||
compatible = pathlib.Path("/proc/device-tree/compatible").read_bytes()
|
||||
is_a_pi = any(
|
||||
vendor == "raspberrypi"
|
||||
for s in compatible.split(b"\0")
|
||||
if s
|
||||
for vendor, model in (s.decode("ascii").split(",", 1),)
|
||||
)
|
||||
except FileNotFoundError:
|
||||
is_a_pi = False
|
||||
if is_a_pi:
|
||||
report.add_tags(["raspi-image"])
|
||||
|
||||
# if we are running from a live system, add the build timestamp
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/cdrom/.disk/info", "LiveMediaBuild"
|
||||
)
|
||||
if os.path.exists("/cdrom/.disk/info"):
|
||||
report["CasperVersion"] = apport.packaging.get_version("casper")
|
||||
|
||||
# https://wiki.ubuntu.com/FoundationsTeam/Specs/OemTrackingId
|
||||
apport.hookutils.attach_file_if_exists(
|
||||
report, "/var/lib/ubuntu_dist_channel", "DistributionChannelDescriptor"
|
||||
)
|
||||
|
||||
os_release = platform.freedesktop_os_release()
|
||||
release_codename = os_release.get("VERSION_CODENAME")
|
||||
if release_codename:
|
||||
report.add_tags([release_codename])
|
||||
|
||||
if os.path.exists(media):
|
||||
mtime = os.stat(media).st_mtime
|
||||
human_mtime = time.strftime("%Y-%m-%d", time.gmtime(mtime))
|
||||
delta = time.time() - mtime
|
||||
report["InstallationDate"] = (
|
||||
f"Installed on {human_mtime} ({round(delta / 86400)} days ago)"
|
||||
)
|
||||
|
||||
log = "/var/log/dist-upgrade/main.log"
|
||||
if os.path.exists(log):
|
||||
mtime = os.stat(log).st_mtime
|
||||
human_mtime = time.strftime("%Y-%m-%d", time.gmtime(mtime))
|
||||
delta = time.time() - mtime
|
||||
|
||||
# Would be nice if this also showed which release was originally
|
||||
# installed
|
||||
report["UpgradeStatus"] = (
|
||||
f"Upgraded to {release_codename} on {human_mtime}"
|
||||
f" ({round(delta / 86400)} days ago)"
|
||||
)
|
||||
else:
|
||||
report["UpgradeStatus"] = "No upgrade log present (probably fresh install)"
|
||||
|
||||
|
||||
def add_proposed_info(report):
|
||||
"""Tag if package comes from -proposed."""
|
||||
if "Package" not in report:
|
||||
return
|
||||
try:
|
||||
(package, version) = report["Package"].split()[:2]
|
||||
except ValueError:
|
||||
print("WARNING: malformed Package field: " + report["Package"])
|
||||
return
|
||||
|
||||
apt_cache = subprocess.run(
|
||||
["apt-cache", "showpkg", package],
|
||||
check=False,
|
||||
stdout=subprocess.PIPE,
|
||||
text=True,
|
||||
)
|
||||
if apt_cache.returncode != 0:
|
||||
print(f"WARNING: apt-cache showpkg {package} failed")
|
||||
return
|
||||
|
||||
found_proposed = False
|
||||
found_updates = False
|
||||
found_security = False
|
||||
for line in apt_cache.stdout.splitlines():
|
||||
if line.startswith(version + " ("):
|
||||
if "-proposed_" in line:
|
||||
found_proposed = True
|
||||
if "-updates_" in line:
|
||||
found_updates = True
|
||||
if "-security" in line:
|
||||
found_security = True
|
||||
|
||||
if found_proposed and not found_updates and not found_security:
|
||||
report.add_tags(["package-from-proposed"])
|
||||
|
||||
|
||||
def get_oem_project(report):
|
||||
"""Determine OEM project name from Distribution Channel Descriptor.
|
||||
|
||||
Return None if it cannot be determined or does not exist.
|
||||
"""
|
||||
dcd = report.get("DistributionChannelDescriptor", None)
|
||||
if dcd and dcd.startswith("canonical-oem-"):
|
||||
return dcd.split("-")[2]
|
||||
return None
|
||||
|
||||
|
||||
def trim_dpkg_log(report):
|
||||
"""Trim DpkgTerminalLog to the most recent installation session."""
|
||||
if "DpkgTerminalLog" not in report:
|
||||
return
|
||||
if not report["DpkgTerminalLog"].strip():
|
||||
report["UnreportableReason"] = "/var/log/apt/term.log does not contain any data"
|
||||
return
|
||||
lines = []
|
||||
dpkg_log = report["DpkgTerminalLog"]
|
||||
if isinstance(dpkg_log, bytes):
|
||||
trim_re = re.compile(b"^\\(.* ... \\d+ .*\\)$")
|
||||
start_re = re.compile(b"^Log started:")
|
||||
else:
|
||||
trim_re = re.compile("^\\(.* ... \\d+ .*\\)$")
|
||||
start_re = re.compile("^Log started:")
|
||||
for line in dpkg_log.splitlines():
|
||||
if start_re.match(line) or trim_re.match(line):
|
||||
lines = []
|
||||
continue
|
||||
lines.append(line)
|
||||
# If trimming the log file fails, return the whole log file.
|
||||
if not lines:
|
||||
return
|
||||
if isinstance(lines[0], str):
|
||||
report["DpkgTerminalLog"] = "\n".join(lines)
|
||||
else:
|
||||
report["DpkgTerminalLog"] = "\n".join(
|
||||
[str(line.decode("UTF-8", "replace")) for line in lines]
|
||||
)
|
||||
|
||||
|
||||
def _get_attachment_contents(report, attachment):
|
||||
if isinstance(report[attachment], problem_report.CompressedValue):
|
||||
contents = report[attachment].get_value().decode("UTF-8")
|
||||
else:
|
||||
contents = report[attachment]
|
||||
return contents
|
||||
|
||||
|
||||
def add_python_details(key, python, report):
|
||||
"""Add comma separated details about which python is being used"""
|
||||
python_path = apport.hookutils.command_output(["which", python])
|
||||
if python_path.startswith("Error: "):
|
||||
report[key] = "N/A"
|
||||
return
|
||||
python_link = apport.hookutils.command_output(["readlink", "-f", python_path])
|
||||
python_pkg = apport.fileutils.find_file_package(python_path)
|
||||
if python_pkg:
|
||||
python_pkg_version = apport.packaging.get_version(python_pkg)
|
||||
python_version = apport.hookutils.command_output([python_link, "--version"])
|
||||
data = f"{python_link}, {python_version}"
|
||||
if python_pkg:
|
||||
data += f", {python_pkg}, {python_pkg_version}"
|
||||
else:
|
||||
data += ", unpackaged"
|
||||
report[key] = data
|
||||
|
||||
|
||||
def main(): # pylint: disable=missing-function-docstring
|
||||
# for testing: update report file given on command line
|
||||
if len(sys.argv) != 2:
|
||||
sys.stderr.write(f"Usage for testing this hook: {sys.argv[0]} <report file>\n")
|
||||
sys.exit(1)
|
||||
|
||||
report_file = sys.argv[1]
|
||||
|
||||
report = apport.Report()
|
||||
with open(report_file, "rb") as report_fd:
|
||||
report.load(report_fd)
|
||||
report_keys = set(report.keys())
|
||||
|
||||
new_report = report.copy()
|
||||
add_info(new_report, None)
|
||||
|
||||
new_report_keys = set(new_report.keys())
|
||||
|
||||
# Show differences
|
||||
# N.B. Some differences will exist if the report file is not from your
|
||||
# system because the hook runs against your local system.
|
||||
changed = 0
|
||||
for key in sorted(report_keys | new_report_keys):
|
||||
if key in new_report_keys and key not in report_keys:
|
||||
print(f"+{key}: {new_report[key]}")
|
||||
changed += 1
|
||||
elif key in report_keys and key not in new_report_keys:
|
||||
print(f"-{key}: (deleted)")
|
||||
changed += 1
|
||||
elif key in report_keys and key in new_report_keys:
|
||||
if report[key] != new_report[key]:
|
||||
print(f"~{key}: (changed)")
|
||||
changed += 1
|
||||
print(f"{changed} items changed")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
Loading…
Reference in New Issue