#!/usr/bin/python3 from __future__ import print_function import optparse import datetime import os import re import subprocess import sys import apt from UpdateManager.Core.utils import twrap # set locale early so that the subsequent imports have localized # strings import locale try: locale.setlocale(locale.LC_ALL, "") except: pass from gettext import gettext as _ import gettext gettext.textdomain("update-manager") from HweSupportStatus.consts import ( Messages, LTS_EOL_DATE, HWE_EOL_DATE, NEXT_LTS_DOT1_DATE, ) # HWE stack with a short support period HWE_UNSUPPORTED_BACKPORTS = ( "-lts-utopic", "-lts-vivid", "-lts-wily" ) # from https://wiki.ubuntu.com/Kernel/LTSEnablementStack UNSUPPORTED_KERNEL_IMAGE_REGEX = \ r'linux-image.*-(3\.16|3\.19|4\.2)(\.[0-9]+)?-.*' # HWE stack with a long support period HWE_SUPPORTED_BACKPORT = "-lts-xenial" SUPPORTED_KERNEL_IMAGE_REGEX = r'linux-image.*-4\.4(\.[0-9]+)?-.*' KERNEL_METAPKGS = ( "linux-generic", "linux-image-generic", "linux-signed-generic", "linux-signed-image-generic", ) XORG_METAPKGS = ( "xserver-xorg", "libgl1-mesa-glx", "libwayland-egl1-mesa", ) VBOX_METAPKGS = ( "virtualbox-guest-utils", "virtualbox-guest-source" ) METAPKGS = KERNEL_METAPKGS + XORG_METAPKGS + VBOX_METAPKGS # fglrx drivers were deprecated in 16.04 FGLRX_PKGS = { 'fglrx', 'fglrx-core', 'fglrx-updates', 'fglrx-updates-core' } class Package: """A lightweight apt package """ def __init__(self, name, version, arch, foreign=False): self.name = name self.installed_version = version self.arch = arch self.foreign = foreign def find_hwe_packages(installed_packages): unsupported_hwe_packages = set() supported_hwe_packages = set() for pkg in installed_packages: if pkg.name in FGLRX_PKGS: unsupported_hwe_packages.add(pkg) # metapackages and X are marked with the -lts-$distro string for name in HWE_UNSUPPORTED_BACKPORTS: if pkg.name.endswith(name): unsupported_hwe_packages.add(pkg) # The individual backported kernels have names like # linux-image-3.11.0-17-generic # so we match via a regexp. # # The linux-image-generic-lts-$distro metapkg has additional # dependencies (like linux-firmware) so we can't just walk the # dependency chain. if re.match(UNSUPPORTED_KERNEL_IMAGE_REGEX, pkg.name): unsupported_hwe_packages.add(pkg) # SUPPORTED if pkg.name.endswith(HWE_SUPPORTED_BACKPORT): supported_hwe_packages.add(pkg) if re.match(SUPPORTED_KERNEL_IMAGE_REGEX, pkg.name): supported_hwe_packages.add(pkg) return unsupported_hwe_packages, supported_hwe_packages def is_unsupported_hwe_kernel_running(unsupported_hwe_package): # kernels do not conflict with each other, so we need to check # what version is actually running running_kernel_ver = os.uname()[2] # the running kernel without the abi or buildver running_kernel_ver = running_kernel_ver.split("-")[0] for pkg in unsupported_hwe_package: if not pkg.name.startswith("linux-"): continue # we only care about the version, not abi or build if pkg.installed_version.startswith(running_kernel_ver): return True return False def is_unsupported_xstack_running(unsupported_hwe_packages): # the HWE xstacks conflict with each other, so we can simply test # for existence in the installed unsupported hwe packages for pkg in unsupported_hwe_packages: for xorg_meta in XORG_METAPKGS: if pkg.name.startswith(xorg_meta): return True return False def find_supported_replacement_hwe_packages(unsupported_hwe_packages, installed_packages): unsupported_metapkg_names = set() replacement_names = set() for metapkg in METAPKGS: for unsupported_backport in HWE_UNSUPPORTED_BACKPORTS: metapkg_name = metapkg + unsupported_backport for pkg in unsupported_hwe_packages: if pkg.name == metapkg_name: replacement_name = metapkg + HWE_SUPPORTED_BACKPORT if (replacement_name, pkg.arch) not in \ [(p.name, p.arch) for p in installed_packages]: if pkg.foreign: replacement_name += ':' + pkg.arch replacement_names.add(replacement_name) unsupported_metapkg_names.add(metapkg_name) return unsupported_metapkg_names, replacement_names def is_unsupported_hwe_running(unsupported_hwe_packages): return (is_unsupported_hwe_kernel_running(unsupported_hwe_packages) or is_unsupported_xstack_running(unsupported_hwe_packages)) def advice_about_hwe_status(unsupported_hwe_packages, supported_hwe_packages, installed_packages, has_update_manager, has_fglrx, today, verbose): unsupported_hwe_stack_running = is_unsupported_hwe_running( unsupported_hwe_packages) unsupported_hwe_metapkgs, supported_replacement_hwe = \ find_supported_replacement_hwe_packages(unsupported_hwe_packages, installed_packages) # we need the "-p" option until the next LTS point release is available if today < NEXT_LTS_DOT1_DATE: do_release_upgrade_option = "-p" else: do_release_upgrade_option = "" if unsupported_hwe_stack_running: if today < HWE_EOL_DATE: s = Messages.HWE_SUPPORT_ENDS else: s = Messages.HWE_SUPPORT_HAS_ENDED if has_update_manager: print(s + Messages.UM_UPGRADE) if has_fglrx: print(Messages.FGLRX_DEPRECATION) else: # bug #1341320 - if no metapkg is left we need to show # what is no longer supported if supported_replacement_hwe: print(s + Messages.APT_UPGRADE % ( do_release_upgrade_option, " ".join(supported_replacement_hwe))) else: print(s + Messages.APT_SHOW_UNSUPPORTED % ( " ".join([pkg.name for pkg in unsupported_hwe_packages]))) # some unsupported package installed but not running and not superseded # - this is worth reporting elif (unsupported_hwe_packages and not supported_hwe_packages and not unsupported_hwe_stack_running): s = _(""" You have packages from the Hardware Enablement Stack (HWE) installed that are going out of support on %s. """) % HWE_EOL_DATE if has_update_manager: print(s + Messages.UM_UPGRADE) else: print(s + Messages.APT_UPGRADE % ( do_release_upgrade_option, " ".join(supported_replacement_hwe))) elif supported_hwe_packages: print(Messages.HWE_SUPPORTED) elif verbose: print( _("You are not running a system with a Hardware Enablement Stack. " "Your system is supported until %(month)s %(year)s.") % { 'month': LTS_EOL_DATE.strftime("%B"), 'year': LTS_EOL_DATE.year}) if __name__ == "__main__": parser = optparse.OptionParser(description=_("Check HWE support status")) parser.add_option('--quiet', action='store_true', default=False, help="No output, exit code 10 on unsupported HWE " "packages") parser.add_option('--verbose', action='store_true', default=False, help="more verbose output") parser.add_option('--show-all-unsupported', action='store_true', default=False, help="Show unsupported HWE packages") parser.add_option('--show-replacements', action='store_true', default=False, help="show what packages need installing to be " "supported") # hidden, only useful for testing parser.add_option( '--disable-hwe-check-semaphore-file', default="/var/lib/update-notifier/disable-hwe-eol-messages", help=optparse.SUPPRESS_HELP) options, args = parser.parse_args() if options.quiet: nullfd = os.open(os.devnull, os.O_WRONLY) os.dup2(nullfd, sys.stdout.fileno()) # request from PSE to be able to disable the hwe check via a special # semaphore file HWE_CHECK_DISABLED_FILE = options.disable_hwe_check_semaphore_file if os.path.exists(HWE_CHECK_DISABLED_FILE): if options.verbose: print("Forcefully disabled hwe-support-status via file %s" % HWE_CHECK_DISABLED_FILE, file=sys.stderr) sys.exit(0) foreign_archs = set(subprocess.check_output( ['dpkg', '--print-foreign-architectures'], universal_newlines=True).split()) # do the actual check installed_packages = set() today = datetime.date.today() tagf = apt.apt_pkg.TagFile("/var/lib/dpkg/status") while tagf.step(): if tagf.section.find("Status", "") != "install ok installed": continue pkgname = tagf.section.find("Package") version = tagf.section.find("Version") arch = tagf.section.find("Architecture") foreign = arch in foreign_archs installed_packages.add(Package(pkgname, version, arch, foreign)) installed_pkg_names = [pkg.name for pkg in installed_packages] has_update_manager = "update-manager" in installed_pkg_names has_fglrx = False if FGLRX_PKGS.intersection(installed_pkg_names): has_fglrx = True unsupported_hwe_packages, supported_hwe_packages = find_hwe_packages( installed_packages) if options.show_all_unsupported: if today > HWE_EOL_DATE: print(twrap(" ".join([ pkg.foreign and pkg.name + ':' + pkg.arch or pkg.name for pkg in unsupported_hwe_packages]))) if options.show_replacements: unsupported, replacements = find_supported_replacement_hwe_packages( unsupported_hwe_packages, installed_packages) if replacements: print(" ".join(replacements)) if not options.show_all_unsupported and not options.show_replacements: advice_about_hwe_status( unsupported_hwe_packages, supported_hwe_packages, installed_packages, has_update_manager, has_fglrx, today, options.verbose) if is_unsupported_hwe_running(unsupported_hwe_packages) and \ today > HWE_EOL_DATE: sys.exit(10) sys.exit(0)