mirror of
https://github.com/virt-manager/virt-manager.git
synced 2025-01-24 02:04:13 +03:00
1f8b712508
Just have one kernel path list, and if type=="xen" and we think xen might have its own kernel path, prepend that to the list of pairs to check. Nowadays xen kernel == hvm kernel for new distros, so we shouldn't force having a specific xen whitelist
916 lines
29 KiB
Python
916 lines
29 KiB
Python
#
|
|
# Copyright 2006-2007, 2013 Red Hat, Inc.
|
|
# Daniel P. Berrange <berrange@redhat.com>
|
|
#
|
|
# This work is licensed under the GNU GPLv2.
|
|
# See the COPYING file in the top-level directory.
|
|
|
|
import configparser
|
|
import logging
|
|
import os
|
|
import re
|
|
|
|
from .osdict import OSDB
|
|
|
|
|
|
###############################################
|
|
# Helpers for detecting distro from given URL #
|
|
###############################################
|
|
|
|
def _grabTreeinfo(fetcher):
|
|
"""
|
|
See if the URL has treeinfo, and if so return it as a ConfigParser
|
|
object.
|
|
"""
|
|
try:
|
|
tmptreeinfo = fetcher.acquireFile(".treeinfo")
|
|
except ValueError:
|
|
return None
|
|
|
|
try:
|
|
treeinfo = configparser.SafeConfigParser()
|
|
treeinfo.read(tmptreeinfo)
|
|
finally:
|
|
os.unlink(tmptreeinfo)
|
|
|
|
try:
|
|
treeinfo.get("general", "family")
|
|
except configparser.NoSectionError:
|
|
logging.debug("Did not find 'family' section in treeinfo")
|
|
return None
|
|
|
|
logging.debug("treeinfo family=%s", treeinfo.get("general", "family"))
|
|
return treeinfo
|
|
|
|
|
|
def _parseSUSEContent(cbuf):
|
|
distribution = None
|
|
distro_version = None
|
|
distro_summary = None
|
|
distro_distro = None
|
|
distro_arch = None
|
|
|
|
# As of 2018 all latest distros match only DISTRO and REPOID below
|
|
for line in cbuf.splitlines()[1:]:
|
|
if line.startswith("LABEL "):
|
|
# opensuse 10.3: LABEL openSUSE 10.3
|
|
# opensuse 11.4: LABEL openSUSE 11.4
|
|
# opensuse 12.3: LABEL openSUSE
|
|
# sles11sp4 DVD: LABEL SUSE Linux Enterprise Server 11 SP4
|
|
distribution = line.split(' ', 1)
|
|
elif line.startswith("DISTRO "):
|
|
# DISTRO cpe:/o:opensuse:opensuse:13.2,openSUSE
|
|
# DISTRO cpe:/o:suse:sled:12:sp3,SUSE Linux Enterprise Desktop 12 SP3
|
|
distro_distro = line.rsplit(',', 1)
|
|
elif line.startswith("VERSION "):
|
|
# opensuse 10.3: VERSION 10.3
|
|
# opensuse 12.3: VERSION 12.3
|
|
distro_version = line.split(' ', 1)
|
|
if len(distro_version) > 1:
|
|
d_version = distro_version[1].split('-', 1)
|
|
if len(d_version) > 1:
|
|
distro_version[1] = d_version[0]
|
|
elif line.startswith("SUMMARY "):
|
|
distro_summary = line.split(' ', 1)
|
|
elif line.startswith("BASEARCHS "):
|
|
# opensuse 11.4: BASEARCHS i586 x86_64
|
|
# opensuse 12.3: BASEARCHS i586 x86_64
|
|
distro_arch = line.split(' ', 1)
|
|
elif line.startswith("DEFAULTBASE "):
|
|
# opensuse 10.3: DEFAULTBASE i586
|
|
distro_arch = line.split(' ', 1)
|
|
elif line.startswith("REPOID "):
|
|
# REPOID obsproduct://build.suse.de/SUSE:SLE-11-SP4:GA/SUSE_SLES/11.4/DVD/x86_64
|
|
# REPOID obsproduct://build.suse.de/SUSE:SLE-12-SP3:GA/SLES/12.3/DVD/aarch64
|
|
distro_arch = line.rsplit('/', 1)
|
|
if distribution and distro_version and distro_arch:
|
|
break
|
|
|
|
if not distribution:
|
|
if distro_summary:
|
|
distribution = distro_summary
|
|
elif distro_distro:
|
|
distribution = distro_distro
|
|
|
|
tree_arch = None
|
|
if distro_arch:
|
|
tree_arch = distro_arch[1].strip()
|
|
# Fix for 13.2 official oss repo
|
|
if tree_arch.find("i586-x86_64") != -1:
|
|
tree_arch = "x86_64"
|
|
else:
|
|
if cbuf.find("x86_64") != -1:
|
|
tree_arch = "x86_64"
|
|
elif cbuf.find("i586") != -1:
|
|
tree_arch = "i586"
|
|
elif cbuf.find("s390x") != -1:
|
|
tree_arch = "s390x"
|
|
|
|
return distribution, distro_version, tree_arch
|
|
|
|
|
|
def _distroFromSUSEContent(fetcher, arch, vmtype):
|
|
try:
|
|
cbuf = fetcher.acquireFileContent("content")
|
|
except ValueError:
|
|
return None
|
|
|
|
distribution, distro_version, tree_arch = _parseSUSEContent(cbuf)
|
|
logging.debug("SUSE content file found distribution=%s distro_version=%s "
|
|
"tree_arch=%s", distribution, distro_version, tree_arch)
|
|
|
|
def _parse_sle_distribution(d):
|
|
sle_version = d[1].strip().rsplit(' ')[4]
|
|
if len(d[1].strip().rsplit(' ')) > 5:
|
|
sle_version = sle_version + '.' + d[1].strip().rsplit(' ')[5][2]
|
|
return ['VERSION', sle_version]
|
|
|
|
dclass = OpensuseDistro
|
|
if distribution:
|
|
if re.match(".*SUSE Linux Enterprise Server*", distribution[1]) or \
|
|
re.match(".*SUSE SLES*", distribution[1]):
|
|
dclass = SLESDistro
|
|
if distro_version is None:
|
|
distro_version = _parse_sle_distribution(distribution)
|
|
elif re.match(".*SUSE Linux Enterprise Desktop*", distribution[1]):
|
|
dclass = SLEDDistro
|
|
if distro_version is None:
|
|
distro_version = _parse_sle_distribution(distribution)
|
|
elif re.match(".*openSUSE.*", distribution[1]):
|
|
dclass = OpensuseDistro
|
|
if distro_version is None:
|
|
distro_version = ['VERSION', distribution[0].strip().rsplit(':')[4]]
|
|
|
|
if distro_version is None:
|
|
logging.debug("No specified SUSE version detected")
|
|
return None
|
|
|
|
ob = dclass(fetcher, tree_arch or arch, vmtype)
|
|
ob.version_from_content = distro_version
|
|
|
|
# Explictly call this, so we populate os_type/variant info
|
|
ob.isValidStore()
|
|
|
|
return ob
|
|
|
|
|
|
def getDistroStore(guest, fetcher):
|
|
stores = []
|
|
logging.debug("Finding distro store for location=%s", fetcher.location)
|
|
|
|
arch = guest.os.arch
|
|
_type = guest.os.os_type
|
|
urldistro = OSDB.lookup_os(guest.os_variant).urldistro
|
|
|
|
treeinfo = _grabTreeinfo(fetcher)
|
|
if not treeinfo:
|
|
dist = _distroFromSUSEContent(fetcher, arch, _type)
|
|
if dist:
|
|
return dist
|
|
|
|
stores = _allstores[:]
|
|
|
|
# If user manually specified an os_distro, bump it's URL class
|
|
# to the top of the list
|
|
if urldistro:
|
|
logging.debug("variant=%s has distro=%s, looking for matching "
|
|
"distro store to prioritize",
|
|
guest.os_variant, urldistro)
|
|
found_store = None
|
|
for store in stores:
|
|
if store.urldistro == urldistro:
|
|
found_store = store
|
|
|
|
if found_store:
|
|
logging.debug("Prioritizing distro store=%s", found_store)
|
|
stores.remove(found_store)
|
|
stores.insert(0, found_store)
|
|
else:
|
|
logging.debug("No matching store found, not prioritizing anything")
|
|
|
|
if treeinfo:
|
|
stores.sort(key=lambda x: not x.uses_treeinfo)
|
|
|
|
for sclass in stores:
|
|
store = sclass(fetcher, arch, _type)
|
|
store.treeinfo = treeinfo
|
|
if store.isValidStore():
|
|
logging.debug("Detected class=%s osvariant=%s",
|
|
store.__class__.__name__, store.os_variant)
|
|
return store
|
|
|
|
# No distro was detected. See if the URL even resolves, and if not
|
|
# give the user a hint that maybe they mistyped. This won't always
|
|
# be true since some webservers don't allow directory listing.
|
|
# http://www.redhat.com/archives/virt-tools-list/2014-December/msg00048.html
|
|
extramsg = ""
|
|
if not fetcher.hasFile(""):
|
|
extramsg = (": " +
|
|
_("The URL could not be accessed, maybe you mistyped?"))
|
|
|
|
raise ValueError(
|
|
_("Could not find an installable distribution at '%s'%s\n\n"
|
|
"The location must be the root directory of an install tree.\n"
|
|
"See virt-install man page for various distro examples." %
|
|
(fetcher.location, extramsg)))
|
|
|
|
|
|
##################
|
|
# Distro classes #
|
|
##################
|
|
|
|
class Distro(object):
|
|
"""
|
|
An image store is a base class for retrieving either a bootable
|
|
ISO image, or a kernel+initrd pair for a particular OS distribution
|
|
"""
|
|
PRETTY_NAME = None
|
|
urldistro = None
|
|
uses_treeinfo = False
|
|
|
|
# osdict variant value
|
|
os_variant = None
|
|
|
|
_boot_iso_paths = None
|
|
_kernel_paths = None
|
|
|
|
def __init__(self, fetcher, arch, vmtype):
|
|
self.fetcher = fetcher
|
|
self.type = vmtype
|
|
self.arch = arch
|
|
|
|
self.uri = fetcher.location
|
|
|
|
# This is set externally
|
|
self.treeinfo = None
|
|
|
|
def isValidStore(self):
|
|
"""Determine if uri points to a tree of the store's distro"""
|
|
raise NotImplementedError
|
|
|
|
def acquireKernel(self, guest):
|
|
kernelpath = None
|
|
initrdpath = None
|
|
if self.treeinfo:
|
|
try:
|
|
kernelpath = self._getTreeinfoMedia("kernel")
|
|
initrdpath = self._getTreeinfoMedia("initrd")
|
|
except configparser.NoSectionError:
|
|
pass
|
|
|
|
if not kernelpath or not initrdpath:
|
|
paths = self._kernel_paths
|
|
|
|
for kpath, ipath in paths:
|
|
if self.fetcher.hasFile(kpath) and self.fetcher.hasFile(ipath):
|
|
kernelpath = kpath
|
|
initrdpath = ipath
|
|
|
|
if not kernelpath or not initrdpath:
|
|
raise RuntimeError(_("Couldn't find kernel for "
|
|
"%(distro)s tree.") %
|
|
{"distro": self.PRETTY_NAME})
|
|
|
|
return self._kernelFetchHelper(guest, kernelpath, initrdpath)
|
|
|
|
def acquireBootDisk(self, guest):
|
|
ignore = guest
|
|
|
|
if self.treeinfo:
|
|
return self.fetcher.acquireFile(self._getTreeinfoMedia("boot.iso"))
|
|
|
|
for path in self._boot_iso_paths:
|
|
if self.fetcher.hasFile(path):
|
|
return self.fetcher.acquireFile(path)
|
|
raise RuntimeError(_("Could not find boot.iso in %s tree." %
|
|
self.PRETTY_NAME))
|
|
|
|
def _check_osvariant_valid(self, os_variant):
|
|
return OSDB.lookup_os(os_variant) is not None
|
|
|
|
def get_osdict_info(self):
|
|
"""
|
|
Return (distro, variant) tuple, checking to make sure they are valid
|
|
osdict entries
|
|
"""
|
|
if not self.os_variant:
|
|
return None
|
|
|
|
if not self._check_osvariant_valid(self.os_variant):
|
|
logging.debug("%s set os_variant to %s, which is not in osdict.",
|
|
self, self.os_variant)
|
|
return None
|
|
|
|
return self.os_variant
|
|
|
|
def _get_method_arg(self):
|
|
return "method"
|
|
|
|
def _getTreeinfoMedia(self, mediaName):
|
|
if self.type == "xen":
|
|
t = "xen"
|
|
else:
|
|
t = self.treeinfo.get("general", "arch")
|
|
|
|
return self.treeinfo.get("images-%s" % t, mediaName)
|
|
|
|
def _fetchAndMatchRegex(self, filename, regex):
|
|
# Fetch 'filename' and return True/False if it matches the regex
|
|
try:
|
|
content = self.fetcher.acquireFileContent(filename)
|
|
except ValueError:
|
|
return False
|
|
|
|
for line in content.splitlines():
|
|
if re.match(regex, line):
|
|
return True
|
|
|
|
logging.debug("%s: found filename=%s but regex didn't match",
|
|
self.__class__.__name__, filename)
|
|
return False
|
|
|
|
def _kernelFetchHelper(self, guest, kernelpath, initrdpath):
|
|
# Simple helper for fetching kernel + initrd and performing
|
|
# cleanup if necessary
|
|
ignore = guest
|
|
kernel = self.fetcher.acquireFile(kernelpath)
|
|
args = ''
|
|
|
|
if not self.fetcher.location.startswith("/"):
|
|
args += "%s=%s" % (self._get_method_arg(), self.fetcher.location)
|
|
|
|
try:
|
|
initrd = self.fetcher.acquireFile(initrdpath)
|
|
return kernel, initrd, args
|
|
except Exception:
|
|
os.unlink(kernel)
|
|
raise
|
|
|
|
|
|
class GenericTreeinfoDistro(Distro):
|
|
PRETTY_NAME = "Generic Treeinfo"
|
|
uses_treeinfo = True
|
|
urldistro = None
|
|
treeinfo_version = None
|
|
|
|
def isValidStore(self):
|
|
return bool(self.treeinfo)
|
|
|
|
def _hasTreeinfoFamily(self, famregex):
|
|
if not self.treeinfo:
|
|
return False
|
|
|
|
treeinfo_family = self.treeinfo.get("general", "family")
|
|
if self.treeinfo.has_option("general", "version"):
|
|
self.treeinfo_version = self.treeinfo.get("general", "version")
|
|
|
|
return bool(re.match(famregex, treeinfo_family))
|
|
|
|
|
|
class RedHatDistro(GenericTreeinfoDistro):
|
|
"""
|
|
Base image store for any Red Hat related distros which have
|
|
a common layout
|
|
"""
|
|
PRETTY_NAME = None
|
|
_version_number = None
|
|
|
|
def isValidStore(self):
|
|
raise NotImplementedError()
|
|
|
|
def _get_method_arg(self):
|
|
if (self._version_number is not None and
|
|
((self.urldistro == "rhel" and self._version_number >= 7) or
|
|
(self.urldistro == "fedora" and self._version_number >= 19))):
|
|
return "inst.repo"
|
|
return "method"
|
|
|
|
|
|
# Fedora distro check
|
|
class FedoraDistro(RedHatDistro):
|
|
PRETTY_NAME = "Fedora"
|
|
urldistro = "fedora"
|
|
|
|
def _parse_fedora_version(self):
|
|
ver = self.treeinfo_version
|
|
latest_variant = OSDB.latest_fedora_version()
|
|
if re.match("fedora[0-9]+", latest_variant):
|
|
latest_vernum = int(latest_variant[6:])
|
|
else:
|
|
logging.debug("Failed to parse version number from latest "
|
|
"fedora variant=%s. Using safe default 22", latest_variant)
|
|
latest_vernum = 22
|
|
|
|
# rawhide trees changed to use version=Rawhide in Apr 2016
|
|
if ver in ["development", "rawhide", "Rawhide"]:
|
|
return latest_vernum, latest_variant
|
|
|
|
# Dev versions can be like '23_Alpha'
|
|
if "_" in ver:
|
|
ver = ver.split("_")[0]
|
|
|
|
# Typical versions are like 'fedora-23'
|
|
vernum = str(ver).split("-")[0]
|
|
if vernum.isdigit():
|
|
vernum = int(vernum)
|
|
else:
|
|
logging.debug("Failed to parse version number from treeinfo "
|
|
"version=%s, using vernum=latest=%s", ver, latest_vernum)
|
|
vernum = latest_vernum
|
|
|
|
if vernum > latest_vernum:
|
|
os_variant = latest_variant
|
|
else:
|
|
os_variant = "fedora" + str(vernum)
|
|
|
|
return vernum, os_variant
|
|
|
|
def isValidStore(self):
|
|
famregex = ".*Fedora.*"
|
|
if not self._hasTreeinfoFamily(famregex):
|
|
return False
|
|
|
|
if not self.treeinfo_version:
|
|
logging.debug("No version found in .treeinfo")
|
|
return False
|
|
logging.debug("Found treeinfo version=%s", self.treeinfo_version)
|
|
|
|
self._version_number, self.os_variant = self._parse_fedora_version()
|
|
return True
|
|
|
|
|
|
# Red Hat Enterprise Linux distro check
|
|
class RHELDistro(RedHatDistro):
|
|
PRETTY_NAME = "Red Hat Enterprise Linux"
|
|
urldistro = "rhel"
|
|
|
|
|
|
def _parseTreeinfoVersion(self, verstr):
|
|
def _safeint(c):
|
|
try:
|
|
val = int(c)
|
|
except Exception:
|
|
val = 0
|
|
return val
|
|
|
|
version = _safeint(verstr[0])
|
|
update = 0
|
|
|
|
# RHEL has version=5.4, scientific linux=54
|
|
updinfo = verstr.split(".")
|
|
if len(updinfo) > 1:
|
|
update = _safeint(updinfo[1])
|
|
elif len(verstr) > 1:
|
|
update = _safeint(verstr[1])
|
|
|
|
return version, update
|
|
|
|
def _setRHELVariant(self, version, update):
|
|
base = "rhel" + str(version)
|
|
if update < 0:
|
|
update = 0
|
|
|
|
ret = None
|
|
while update >= 0:
|
|
tryvar = base + ".%s" % update
|
|
if not self._check_osvariant_valid(tryvar):
|
|
update -= 1
|
|
continue
|
|
|
|
ret = tryvar
|
|
break
|
|
|
|
if not ret:
|
|
# Try plain rhel5, rhel6, whatev
|
|
if self._check_osvariant_valid(base):
|
|
ret = base
|
|
|
|
if ret:
|
|
self.os_variant = ret
|
|
|
|
def _variantFromVersion(self):
|
|
if not self.treeinfo_version:
|
|
return
|
|
|
|
version, update = self._parseTreeinfoVersion(self.treeinfo_version)
|
|
self._version_number = version
|
|
self._setRHELVariant(version, update)
|
|
|
|
|
|
def isValidStore(self):
|
|
# Matches:
|
|
# Red Hat Enterprise Linux
|
|
# RHEL Atomic Host
|
|
famregex = ".*(Red Hat Enterprise Linux|RHEL).*"
|
|
if not self._hasTreeinfoFamily(famregex):
|
|
return False
|
|
|
|
self._variantFromVersion()
|
|
return True
|
|
|
|
|
|
# CentOS distro check
|
|
class CentOSDistro(RHELDistro):
|
|
PRETTY_NAME = "CentOS"
|
|
urldistro = "centos"
|
|
|
|
def isValidStore(self):
|
|
famregex = ".*CentOS.*"
|
|
if not self._hasTreeinfoFamily(famregex):
|
|
return False
|
|
|
|
self._variantFromVersion()
|
|
if self.os_variant:
|
|
new_variant = self.os_variant.replace("rhel", "centos")
|
|
if self._check_osvariant_valid(new_variant):
|
|
self.os_variant = new_variant
|
|
return True
|
|
|
|
|
|
# Scientific Linux distro check
|
|
class SLDistro(RHELDistro):
|
|
PRETTY_NAME = "Scientific Linux"
|
|
urldistro = None
|
|
|
|
def isValidStore(self):
|
|
famregex = ".*Scientific.*"
|
|
if not self._hasTreeinfoFamily(famregex):
|
|
return False
|
|
|
|
self._variantFromVersion()
|
|
return True
|
|
|
|
|
|
class SuseDistro(Distro):
|
|
PRETTY_NAME = "SUSE"
|
|
|
|
_boot_iso_paths = ["boot/boot.iso"]
|
|
version_from_content = []
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
Distro.__init__(self, *args, **kwargs)
|
|
if re.match(r'i[4-9]86', self.arch):
|
|
self.arch = 'i386'
|
|
|
|
oldkern = "linux"
|
|
oldinit = "initrd"
|
|
if self.arch == "x86_64":
|
|
oldkern += "64"
|
|
oldinit += "64"
|
|
|
|
self._kernel_paths = []
|
|
if self.type == "xen":
|
|
# Matches Opensuse > 10.2 and sles 10
|
|
self._kernel_paths.append(
|
|
("boot/%s/vmlinuz-xen" % self.arch,
|
|
"boot/%s/initrd-xen" % self.arch))
|
|
|
|
# Tested with SLES 12 for ppc64le, all s390x
|
|
self._kernel_paths.append(
|
|
("boot/%s/linux" % self.arch,
|
|
"boot/%s/initrd" % self.arch))
|
|
# Tested with Opensuse 10.0
|
|
self._kernel_paths.append(
|
|
("boot/loader/%s" % oldkern,
|
|
"boot/loader/%s" % oldinit))
|
|
# Tested with Opensuse >= 10.2, 11, and sles 10
|
|
self._kernel_paths.append(
|
|
("boot/%s/loader/linux" % self.arch,
|
|
"boot/%s/loader/initrd" % self.arch))
|
|
|
|
|
|
def _variantFromVersion(self):
|
|
distro_version = self.version_from_content[1].strip()
|
|
version = distro_version.split('.', 1)[0].strip()
|
|
self.os_variant = self.urldistro
|
|
if int(version) >= 10:
|
|
if self.os_variant.startswith(("sles", "sled")):
|
|
sp_version = None
|
|
if len(distro_version.split('.', 1)) == 2:
|
|
sp_version = 'sp' + distro_version.split('.', 1)[1].strip()
|
|
self.os_variant += version
|
|
if sp_version:
|
|
self.os_variant += sp_version
|
|
else:
|
|
# Tumbleweed 8 digit date
|
|
if len(version) == 8:
|
|
self.os_variant += "tumbleweed"
|
|
else:
|
|
self.os_variant += distro_version
|
|
else:
|
|
self.os_variant += "9"
|
|
|
|
def isValidStore(self):
|
|
# self.version_from_content is the VERSION line from the contents file
|
|
if (not self.version_from_content or
|
|
self.version_from_content[1] is None):
|
|
return False
|
|
|
|
self._variantFromVersion()
|
|
|
|
self.os_variant = self._detect_osdict_from_url()
|
|
|
|
# Reset kernel name for sle11 source on s390x
|
|
if self.arch == "s390x":
|
|
if self.os_variant == "sles11" or self.os_variant == "sled11":
|
|
self._kernel_paths = [("boot/%s/vmrdr.ikr" % self.arch,
|
|
"boot/%s/initrd" % self.arch)]
|
|
|
|
return True
|
|
|
|
def _get_method_arg(self):
|
|
return "install"
|
|
|
|
|
|
################################
|
|
# osdict autodetection helpers #
|
|
################################
|
|
|
|
def _detect_osdict_from_url(self):
|
|
root = "opensuse"
|
|
oses = [n for n in OSDB.list_os() if n.name.startswith(root)]
|
|
|
|
for osobj in oses:
|
|
codename = osobj.name[len(root):]
|
|
if re.search("/%s/" % codename, self.uri):
|
|
return osobj.name
|
|
return self.os_variant
|
|
|
|
|
|
class SLESDistro(SuseDistro):
|
|
urldistro = "sles"
|
|
|
|
|
|
class SLEDDistro(SuseDistro):
|
|
urldistro = "sled"
|
|
|
|
|
|
# Suse image store is harder - we fetch the kernel RPM and a helper
|
|
# RPM and then munge bits together to generate a initrd
|
|
class OpensuseDistro(SuseDistro):
|
|
urldistro = "opensuse"
|
|
|
|
|
|
class DebianDistro(Distro):
|
|
# ex. http://ftp.egr.msu.edu/debian/dists/sarge/main/installer-i386/
|
|
# daily builds: http://d-i.debian.org/daily-images/amd64/
|
|
PRETTY_NAME = "Debian"
|
|
urldistro = "debian"
|
|
_debname = "debian"
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
Distro.__init__(self, *args, **kwargs)
|
|
|
|
self._kernel_paths = []
|
|
self._url_prefix = ""
|
|
self._treeArch = self._find_treearch()
|
|
self._installer_dirname = self._debname + "-installer"
|
|
|
|
def _find_treearch(self):
|
|
for pattern in ["^.*/installer-(\w+)/?$",
|
|
"^.*/daily-images/(\w+)/?$"]:
|
|
arch = re.findall(pattern, self.uri)
|
|
if not arch:
|
|
continue
|
|
logging.debug("Found pattern=%s treearch=%s in uri",
|
|
pattern, arch[0])
|
|
return arch[0]
|
|
|
|
# Check for standard 'i386' and 'amd64' which will be
|
|
# in the URI name for --location $ISO mounts
|
|
for arch in ["i386", "amd64", "x86_64", "arm64"]:
|
|
if arch in self.uri:
|
|
logging.debug("Found treearch=%s in uri", arch)
|
|
if arch == "x86_64":
|
|
arch = "amd64"
|
|
return arch
|
|
|
|
# Otherwise default to i386
|
|
arch = "i386"
|
|
logging.debug("No treearch found in uri, defaulting to arch=%s", arch)
|
|
return arch
|
|
|
|
def _set_media_paths(self):
|
|
self._boot_iso_paths = ["%s/netboot/mini.iso" % self._url_prefix]
|
|
|
|
hvmroot = "%s/netboot/%s/%s/" % (self._url_prefix,
|
|
self._installer_dirname,
|
|
self._treeArch)
|
|
initrd_basename = "initrd.gz"
|
|
kernel_basename = "linux"
|
|
if self._treeArch in ["ppc64el"]:
|
|
kernel_basename = "vmlinux"
|
|
|
|
if self._treeArch == "s390x":
|
|
hvmroot = "%s/generic/" % self._url_prefix
|
|
kernel_basename = "kernel.%s" % self._debname.lower()
|
|
initrd_basename = "initrd.%s" % self._debname.lower()
|
|
|
|
|
|
if self.type == "xen":
|
|
xenroot = "%s/netboot/xen/" % self._url_prefix
|
|
self._kernel_paths.append(
|
|
(xenroot + "vmlinuz", xenroot + "initrd.gz"))
|
|
self._kernel_paths.append(
|
|
(hvmroot + kernel_basename, hvmroot + initrd_basename))
|
|
|
|
|
|
def _check_manifest(self, filename):
|
|
if not self.fetcher.hasFile(filename):
|
|
return False
|
|
|
|
if self.arch == "s390x":
|
|
regex = ".*generic/kernel\.%s.*" % self._debname.lower()
|
|
else:
|
|
regex = ".*%s.*" % self._installer_dirname
|
|
|
|
return self._fetchAndMatchRegex(filename, regex)
|
|
|
|
def _check_info(self, filename):
|
|
if not self.fetcher.hasFile(filename):
|
|
return False
|
|
|
|
regex = "%s.*" % self._debname.capitalize()
|
|
return self._fetchAndMatchRegex(filename, regex)
|
|
|
|
def _is_regular_tree(self):
|
|
# For regular trees
|
|
if not self._check_manifest("current/images/MANIFEST"):
|
|
return False
|
|
|
|
self._url_prefix = "current/images"
|
|
self._set_media_paths()
|
|
self.os_variant = self._detect_debian_osdict_from_url()
|
|
|
|
return True
|
|
|
|
def _is_daily_tree(self):
|
|
# For daily trees
|
|
if not self._check_manifest("daily/MANIFEST"):
|
|
return False
|
|
|
|
self._url_prefix = "daily"
|
|
self._set_media_paths()
|
|
self.os_variant = self._detect_debian_osdict_from_url()
|
|
|
|
return True
|
|
|
|
def _is_install_cd(self):
|
|
# For install CDs
|
|
if not self._check_info(".disk/info"):
|
|
return False
|
|
|
|
if self.arch == "x86_64":
|
|
kernel_initrd_pair = ("install.amd/vmlinuz",
|
|
"install.amd/initrd.gz")
|
|
elif self.arch == "i686":
|
|
kernel_initrd_pair = ("install.386/vmlinuz",
|
|
"install.386/initrd.gz")
|
|
elif self.arch == "aarch64":
|
|
kernel_initrd_pair = ("install.a64/vmlinuz",
|
|
"install.a64/initrd.gz")
|
|
elif self.arch == "ppc64le":
|
|
kernel_initrd_pair = ("install/vmlinux",
|
|
"install/initrd.gz")
|
|
elif self.arch == "s390x":
|
|
kernel_initrd_pair = ("boot/linux_vm", "boot/root.bin")
|
|
else:
|
|
kernel_initrd_pair = ("install/vmlinuz", "install/initrd.gz")
|
|
self._kernel_paths += [kernel_initrd_pair]
|
|
return True
|
|
|
|
def isValidStore(self):
|
|
return any(check() for check in [
|
|
self._is_regular_tree,
|
|
self._is_daily_tree,
|
|
self._is_install_cd,
|
|
])
|
|
|
|
|
|
################################
|
|
# osdict autodetection helpers #
|
|
################################
|
|
|
|
def _detect_debian_osdict_from_url(self):
|
|
oses = [n for n in OSDB.list_os() if n.name.startswith(self._debname)]
|
|
|
|
if self._url_prefix == "daily":
|
|
logging.debug("Appears to be debian 'daily' URL, using latest "
|
|
"debian OS")
|
|
return oses[0].name
|
|
|
|
for osobj in oses:
|
|
if osobj.codename:
|
|
# Ubuntu codenames look like 'Warty Warthog'
|
|
codename = osobj.codename.split()[0].lower()
|
|
else:
|
|
if " " not in osobj.label:
|
|
continue
|
|
# Debian labels look like 'Debian Sarge'
|
|
codename = osobj.label.split()[1].lower()
|
|
|
|
if ("/%s/" % codename) in self.uri:
|
|
logging.debug("Found codename=%s in the URL string", codename)
|
|
return osobj.name
|
|
|
|
logging.debug("Didn't find any known codename in the URL string")
|
|
return self.os_variant
|
|
|
|
|
|
class UbuntuDistro(DebianDistro):
|
|
# http://archive.ubuntu.com/ubuntu/dists/natty/main/installer-amd64/
|
|
PRETTY_NAME = "Ubuntu"
|
|
urldistro = "ubuntu"
|
|
_debname = "ubuntu"
|
|
|
|
def _is_tree_iso(self):
|
|
# For trees based on ISO's
|
|
if not self._check_info("install/netboot/version.info"):
|
|
return False
|
|
|
|
self._url_prefix = "install"
|
|
self._set_media_paths()
|
|
self.os_variant = self._detect_debian_osdict_from_url()
|
|
|
|
return True
|
|
|
|
def _is_install_cd(self):
|
|
# For install CDs
|
|
if not self._check_info(".disk/info"):
|
|
return False
|
|
|
|
if not self.arch == "s390x":
|
|
kernel_initrd_pair = ("install/vmlinuz", "install/initrd.gz")
|
|
else:
|
|
kernel_initrd_pair = ("boot/kernel.ubuntu", "boot/initrd.ubuntu")
|
|
|
|
self._kernel_paths += [kernel_initrd_pair]
|
|
return True
|
|
|
|
|
|
|
|
class MandrivaDistro(Distro):
|
|
# ftp://ftp.uwsg.indiana.edu/linux/mandrake/official/2007.1/x86_64/
|
|
PRETTY_NAME = "Mandriva/Mageia"
|
|
urldistro = "mandriva"
|
|
|
|
_boot_iso_paths = ["install/images/boot.iso"]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
Distro.__init__(self, *args, **kwargs)
|
|
self._kernel_paths = []
|
|
|
|
# At least Mageia 5 uses arch in the names
|
|
self._kernel_paths += [
|
|
("isolinux/%s/vmlinuz" % self.arch,
|
|
"isolinux/%s/all.rdz" % self.arch)]
|
|
|
|
# Kernels for HVM: valid for releases 2007.1, 2008.*, 2009.0
|
|
self._kernel_paths += [
|
|
("isolinux/alt0/vmlinuz", "isolinux/alt0/all.rdz")]
|
|
|
|
|
|
def isValidStore(self):
|
|
if not self.fetcher.hasFile("VERSION"):
|
|
return False
|
|
return self._fetchAndMatchRegex("VERSION", ".*(Mandriva|Mageia).*")
|
|
|
|
|
|
class ALTLinuxDistro(Distro):
|
|
PRETTY_NAME = "ALT Linux"
|
|
urldistro = "altlinux"
|
|
|
|
_boot_iso_paths = [("altinst", "live")]
|
|
_kernel_paths = [("syslinux/alt0/vmlinuz", "syslinux/alt0/full.cz")]
|
|
|
|
def isValidStore(self):
|
|
# altlinux doesn't have installable URLs, so this is just for ISO
|
|
if not self.fetcher.hasFile(".disk/info"):
|
|
return False
|
|
return self._fetchAndMatchRegex(".disk/info", ".*ALT .*")
|
|
|
|
|
|
# Build list of all *Distro classes
|
|
def _build_distro_list():
|
|
allstores = []
|
|
for obj in list(globals().values()):
|
|
if (isinstance(obj, type) and
|
|
issubclass(obj, Distro) and
|
|
obj.PRETTY_NAME):
|
|
allstores.append(obj)
|
|
|
|
seen_urldistro = []
|
|
for obj in allstores:
|
|
if obj.urldistro and obj.urldistro in seen_urldistro:
|
|
raise RuntimeError("programming error: duplicate urldistro=%s" %
|
|
obj.urldistro)
|
|
seen_urldistro.append(obj.urldistro)
|
|
|
|
# Always stick GenericDistro at the end, since it's a catchall
|
|
allstores.remove(GenericTreeinfoDistro)
|
|
allstores.append(GenericTreeinfoDistro)
|
|
|
|
return allstores
|
|
|
|
_allstores = _build_distro_list()
|