/usr/share/pyshared/virtinst/CloneManager.py is in virtinst 0.600.1-3+deb7u2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 | #
# Copyright(c) FUJITSU Limited 2007.
#
# Cloning a virtual machine module.
#
# 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., 51 Franklin Street, Fifth Floor, Boston,
# MA 02110-1301 USA.
"""
Module for cloning an existing virtual machine
General workflow for cloning:
- Instantiate CloneDesign. Requires at least a libvirt connection and
either the name of an existing domain to clone (original_guest), or
a string of libvirt xml representing the guest to clone
(original_guest_xml)
- Run 'setup' from the CloneDesign instance to prep for cloning
- Run 'CloneManager.start_duplicate', passing the CloneDesign instance
"""
import logging
import re
import os
import urlgrabber.progress as progress
import libvirt
import Guest
from VirtualNetworkInterface import VirtualNetworkInterface
from VirtualDisk import VirtualDisk
from virtinst import Storage
from virtinst import _gettext as _
import _util
def _listify(val):
"""
Return (was_val_a_list, listified_val)
"""
if type(val) is list:
return True, val
else:
return False, [val]
def generate_clone_disk_path(origpath, design, newname=None):
origname = design.original_guest
newname = newname or design.clone_name
path = origpath
suffix = ""
# Try to split the suffix off the existing disk name. Ex.
# foobar.img -> foobar-clone.img
#
# If the suffix is greater than 7 characters, assume it isn't
# a file extension and is part of the disk name, at which point
# just stick '-clone' on the end.
if origpath.count(".") and len(origpath.rsplit(".", 1)[1]) <= 7:
path, suffix = origpath.rsplit(".", 1)
suffix = "." + suffix
dirname = os.path.dirname(path)
basename = os.path.basename(path)
clonebase = basename + "-clone"
if origname and basename == origname:
clonebase = newname
clonebase = os.path.join(dirname, clonebase)
return _util.generate_name(
clonebase,
lambda p: VirtualDisk.path_exists(design.original_conn, p),
suffix,
lib_collision=False)
def generate_clone_name(design):
# If the orig name is "foo-clone", we don't want the clone to be
# "foo-clone-clone", we want "foo-clone1"
basename = design.original_guest
match = re.search("-clone[1-9]*$", basename)
start_num = 0
if match:
num_match = re.search("[1-9]+$", match.group())
if num_match:
start_num = int(str(num_match.group()))
basename = basename.replace(match.group(), "")
basename = basename + "-clone"
return _util.generate_name(basename,
design.original_conn.lookupByName,
sep="", start_num=start_num)
#
# This class is the design paper for a clone virtual machine.
#
class CloneDesign(object):
# Reasons why we don't default to cloning.
CLONE_POLICY_NO_READONLY = 1
CLONE_POLICY_NO_SHAREABLE = 2
CLONE_POLICY_NO_EMPTYMEDIA = 3
def __init__(self, connection=None, conn=None):
conn = conn or connection
if not isinstance(conn, libvirt.virConnect):
raise ValueError(_("Connection must be a 'virConnect' instance."))
self._hyper_conn = conn
# original guest name or uuid
self._original_guest = None
self._original_dom = None
self._original_virtual_disks = []
self._original_xml = None
self._guest = None
# clone guest
self._clone_name = None
self._clone_devices = []
self._clone_virtual_disks = []
self._clone_bs = 1024 * 1024 * 10
self._clone_mac = []
self._clone_uuid = None
self._clone_sparse = True
self._clone_xml = None
self._force_target = []
self._skip_target = []
self._preserve = True
self._clone_running = False
# Default clone policy for back compat: don't clone readonly,
# shareable, or empty disks
self._clone_policy = [self.CLONE_POLICY_NO_READONLY,
self.CLONE_POLICY_NO_SHAREABLE,
self.CLONE_POLICY_NO_EMPTYMEDIA]
# Throwaway guest to use for easy validation
self._valid_guest = Guest.Guest(conn=conn)
# Generate a random UUID at the start
while 1:
uuid = _util.uuidToString(_util.randomUUID())
if _util.vm_uuid_collision(self._hyper_conn, uuid):
continue
self.clone_uuid = uuid
break
# Getter/Setter methods
def get_original_guest(self):
return self._original_guest
def set_original_guest(self, original_guest):
if self._lookup_vm(original_guest):
self._original_guest = original_guest
original_guest = property(get_original_guest, set_original_guest,
doc="Original guest name.")
def set_original_xml(self, val):
if type(val) is not str:
raise ValueError(_("Original xml must be a string."))
self._original_xml = val
self._original_guest = _util.get_xml_path(self.original_xml,
"/domain/name")
def get_original_xml(self):
return self._original_xml
original_xml = property(get_original_xml, set_original_xml,
doc="XML of the original guest.")
def get_clone_name(self):
return self._clone_name
def set_clone_name(self, name):
try:
self._valid_guest.set_name(name)
except ValueError, e:
raise ValueError(_("Invalid name for new guest: %s") % e)
self._clone_name = name
clone_name = property(get_clone_name, set_clone_name,
doc="Name to use for the new guest clone.")
def set_clone_uuid(self, uuid):
try:
self._valid_guest.set_uuid(uuid)
except ValueError, e:
raise ValueError(_("Invalid uuid for new guest: %s") % e)
if _util.vm_uuid_collision(self._hyper_conn, uuid):
raise ValueError(_("UUID '%s' is in use by another guest.") %
uuid)
self._clone_uuid = uuid
def get_clone_uuid(self):
return self._clone_uuid
clone_uuid = property(get_clone_uuid, set_clone_uuid,
doc="UUID to use for the new guest clone")
def set_clone_devices(self, devpath):
# Devices here is a string path. Every call to set_clone_devices
# Adds the path (if valid) to the internal _clone_devices list
disklist = []
is_list, pathlist = _listify(devpath)
# Check path is valid
for path in pathlist:
try:
device = VirtualDisk.DEVICE_DISK
if not path:
device = VirtualDisk.DEVICE_CDROM
disk = VirtualDisk(path, size=.0000001,
conn=self._hyper_conn,
device=device)
disklist.append(disk)
except Exception, e:
raise ValueError(_("Could not use path '%s' for cloning: %s") %
(path, str(e)))
if is_list:
self._clone_virtual_disks = []
self._clone_devices = []
self._clone_virtual_disks.extend(disklist)
self._clone_devices.extend(pathlist)
def get_clone_devices(self):
return self._clone_devices
clone_devices = property(get_clone_devices, set_clone_devices,
doc="Paths to use for the new disk locations.")
def get_clone_virtual_disks(self):
return self._clone_virtual_disks
clone_virtual_disks = property(get_clone_virtual_disks,
doc="VirtualDisk instances for the new"
" disk paths")
def set_clone_mac(self, mac):
is_list, maclist = _listify(mac)
for m in maclist:
VirtualNetworkInterface(m, conn=self.original_conn)
if is_list:
self._clone_mac = []
self._clone_mac.extend(maclist)
def get_clone_mac(self):
return self._clone_mac
clone_mac = property(get_clone_mac, set_clone_mac,
doc="MAC address for the new guest clone.")
def get_clone_bs(self):
return self._clone_bs
def set_clone_bs(self, rate):
self._clone_bs = rate
clone_bs = property(get_clone_bs, set_clone_bs,
doc="Block size to use when cloning guest storage.")
def get_original_devices_size(self):
ret = []
for disk in self.original_virtual_disks:
ret.append(disk.size)
return ret
original_devices_size = property(get_original_devices_size,
doc="Size of the original guest's disks."
" DEPRECATED: Get this info from"
" original_virtual_disks")
def get_original_devices(self):
ret = []
for disk in self.original_virtual_disks:
ret.append(disk.path)
return ret
original_devices = property(get_original_devices,
doc="Original disk paths that will be cloned. "
"DEPRECATED: Get this info from "
"original_virtual_disks")
def get_original_virtual_disks(self):
return self._original_virtual_disks
original_virtual_disks = property(get_original_virtual_disks,
doc="VirtualDisk instances of the "
"original disks being cloned.")
def get_hyper_conn(self):
return self._hyper_conn
def set_hyper_conn(self, conn):
self._hyper_conn = conn
original_conn = property(get_hyper_conn, set_hyper_conn,
doc="Libvirt virConnect instance we are cloning "
"on")
def get_original_dom(self):
return self._original_dom
original_dom = property(get_original_dom,
doc="Libvirt virDomain instance of the original "
"guest. May not be available if cloning from "
"XML.")
def get_clone_xml(self):
return self._clone_xml
def set_clone_xml(self, clone_xml):
self._clone_xml = clone_xml
clone_xml = property(get_clone_xml, set_clone_xml,
doc="Generated XML for the guest clone.")
def get_clone_sparse(self):
return self._clone_sparse
def set_clone_sparse(self, flg):
self._clone_sparse = flg
clone_sparse = property(get_clone_sparse, set_clone_sparse,
doc="Whether to attempt sparse allocation during "
"cloning.")
def get_preserve(self):
return self._preserve
def set_preserve(self, flg):
self._preserve = flg
preserve = property(get_preserve, set_preserve,
doc="If true, preserve ALL original disk devices.")
def get_preserve_dest_disks(self):
return not self.preserve
preserve_dest_disks = property(get_preserve_dest_disks,
doc="It true, preserve ALL disk devices for the "
"NEW guest. This means no storage cloning. "
"This is a convenience access for "
"(not CloneManager.preserve)")
def set_force_target(self, dev):
if type(dev) is list:
self._force_target = dev[:]
else:
self._force_target.append(dev)
def get_force_target(self):
return self._force_target
force_target = property(get_force_target, set_force_target,
doc="List of disk targets that we force cloning "
"despite CloneManager's recommendation.")
def set_skip_target(self, dev):
if type(dev) is list:
self._skip_target = dev[:]
else:
self._skip_target.append(dev)
def get_skip_target(self):
return self._skip_target
skip_target = property(get_skip_target, set_skip_target,
doc="List of disk targets that we skip cloning "
"despite CloneManager's recommendation. This "
"takes precedence over force_target.")
def set_clone_policy(self, policy_list):
if type(policy_list) != list:
raise ValueError(_("Cloning policy must be a list of rules."))
self._clone_policy = policy_list
def get_clone_policy(self):
return self._clone_policy
clone_policy = property(get_clone_policy, set_clone_policy,
doc="List of policy rules for determining which "
"vm disks to clone. See CLONE_POLICY_*")
def get_clone_running(self):
return self._clone_running
def set_clone_running(self, val):
self._clone_running = bool(val)
clone_running = property(get_clone_running, set_clone_running,
doc="Allow cloning a running VM. If enabled, "
"domain state is not checked before "
"cloning.")
def _get_replace(self):
return self._valid_guest.replace
def _set_replace(self, val):
self._valid_guest.replace = bool(val)
replace = property(_get_replace, _set_replace,
doc="f enabled, don't check for clone name collision, "
"simply undefine any conflicting guest.")
# Functional methods
def setup_original(self):
"""
Validate and setup all parameters needed for the original (cloned) VM
"""
logging.debug("Validating original guest parameters")
if self.original_guest == None and self.original_xml == None:
raise RuntimeError(_("Original guest name or xml is required."))
if self.original_guest != None and not self.original_xml:
self._original_dom = self._lookup_vm(self.original_guest)
self.original_xml = self._original_dom.XMLDesc(0)
logging.debug("Original XML:\n%s", self.original_xml)
self._guest = Guest.Guest(conn=self._hyper_conn,
parsexml=self.original_xml)
# Pull clonable storage info from the original xml
self._original_virtual_disks = self._get_original_devices_info()
logging.debug("Original paths: %s", self.original_devices)
logging.debug("Original sizes: %s", self.original_devices_size)
# If domain has devices to clone, it must be 'off' or 'paused'
if (not self.clone_running and
(self._original_dom and len(self.original_devices) != 0)):
status = self._original_dom.info()[0]
if status not in [libvirt.VIR_DOMAIN_SHUTOFF,
libvirt.VIR_DOMAIN_PAUSED]:
raise RuntimeError(_("Domain with devices to clone must be "
"paused or shutoff."))
def setup_clone(self):
"""
Validate and set up all parameters needed for the new (clone) VM
"""
logging.debug("Validating clone parameters.")
self._clone_xml = self.original_xml
if len(self.clone_virtual_disks) < len(self.original_virtual_disks):
raise ValueError(_("More disks to clone than new paths specified. "
"(%(passed)d specified, %(need)d needed") %
{"passed" : len(self.clone_virtual_disks),
"need" : len(self.original_virtual_disks) })
logging.debug("Clone paths: %s", self._clone_devices)
self._guest.name = self._clone_name
self._guest.uuid = self._clone_uuid
self._clone_mac.reverse()
for iface in self._guest.get_devices("interface"):
iface.target_dev = None
if self._clone_mac:
mac = self._clone_mac.pop()
else:
while 1:
mac = _util.randomMAC(self.original_conn.getType().lower())
dummy, msg = self._check_mac(mac)
if msg is not None:
continue
else:
break
iface.macaddr = mac
# Changing storage XML
for i in range(len(self._original_virtual_disks)):
orig_disk = self._original_virtual_disks[i]
clone_disk = self._clone_virtual_disks[i]
for disk in self._guest.get_devices("disk"):
if disk.target == orig_disk.target:
xmldisk = disk
if clone_disk.vol_object:
# XXX We could do this with vol upload I think
raise RuntimeError(
_("Clone onto existing storage volume is not "
"supported: '%s'") % clone_disk.path)
# Sync 'size' between the two
if orig_disk.size:
clone_disk.size = orig_disk.size
# Setup proper cloning inputs for the new virtual disks
if orig_disk.vol_object and clone_disk.vol_install:
# Source and dest are managed. If they share the same pool,
# replace vol_install with a CloneVolume instance, otherwise
# simply set input_vol on the dest vol_install
if (clone_disk.vol_install.pool.name() ==
orig_disk.vol_object.storagePoolLookupByVolume().name()):
newname = clone_disk.vol_install.name
clone_disk.vol_install = Storage.CloneVolume(newname,
orig_disk.vol_object)
else:
clone_disk.vol_install.input_vol = orig_disk.vol_object
elif not self.preserve_dest_disks:
clone_disk.clone_path = orig_disk.path
# Change the XML
xmldisk.path = None
xmldisk.type = clone_disk.type
xmldisk.path = clone_disk.path
xmldisk.driver_type = orig_disk.driver_type
# Save altered clone xml
self._clone_xml = self._guest.get_xml_config()
def setup(self):
"""
Helper function that wraps setup_original and setup_clone, with
additional debug logging.
"""
self.setup_original()
logging.debug("Original guest xml is\n%s", self._original_xml)
self.setup_clone()
logging.debug("Clone guest xml is\n%s", self._clone_xml)
def remove_original_vm(self, force=None):
return self._valid_guest.remove_original_vm(force=force)
# Private helper functions
# Check if new mac address is valid
def _check_mac(self, mac):
nic = VirtualNetworkInterface(macaddr=mac, conn=self.original_conn)
return nic.is_conflict_net(self._hyper_conn)
# Parse disk paths that need to be cloned from the original guest's xml
# Return a list of VirtualDisk instances pointing to the original
# storage
def _get_original_devices_info(self):
clonelist = []
retdisks = []
for disk in self._guest.get_devices("disk"):
if self._do_we_clone_device(disk):
clonelist.append(disk)
continue
# Set up virtual disk to encapsulate all relevant path info
for disk in clonelist:
validate = not self.preserve_dest_disks
try:
if (disk.path and validate and
not VirtualDisk.path_exists(self._hyper_conn, disk.path)):
raise ValueError(_("Disk '%s' does not exist.") %
disk.path)
device = VirtualDisk.DEVICE_DISK
if not disk.path:
# Tell VirtualDisk we are a cdrom to allow empty media
device = VirtualDisk.DEVICE_CDROM
d = VirtualDisk(disk.path, conn=self._hyper_conn,
device=device, driverType=disk.driver_type,
validate=validate)
d.target = disk.target
except Exception, e:
logging.debug("", exc_info=True)
raise ValueError(_("Could not determine original disk "
"information: %s" % str(e)))
retdisks.append(d)
return retdisks
# Pull disk #i from the original guest xml, return it's source path
# if it should be cloned
# Cloning policy based on 'clone_policy', 'force_target' and 'skip_target'
def _do_we_clone_device(self, disk):
if not disk.target:
raise ValueError("XML has no 'dev' attribute in disk target")
if disk.target in self.skip_target:
return False
if disk.target in self.force_target:
return True
# No media path
if (not disk.path and
self.CLONE_POLICY_NO_EMPTYMEDIA in self.clone_policy):
return False
# Readonly disks
if (disk.read_only and
self.CLONE_POLICY_NO_READONLY in self.clone_policy):
return False
# Shareable disks
if (disk.shareable and
self.CLONE_POLICY_NO_SHAREABLE in self.clone_policy):
return False
return True
# Simple wrapper for checking a vm exists and returning the domain
def _lookup_vm(self, name):
try:
return self._hyper_conn.lookupByName(name)
except libvirt.libvirtError:
raise ValueError(_("Domain '%s' was not found.") % str(name))
def start_duplicate(design, meter=None):
"""
Actually perform the duplication: cloning disks if needed and defining
the new clone xml.
"""
logging.debug("Starting duplicate.")
if not meter:
meter = progress.BaseMeter()
dom = None
try:
# Replace orig VM if required
design.remove_original_vm()
# Define domain early to catch any xml errors before duping storage
dom = design.original_conn.defineXML(design.clone_xml)
if design.preserve == True:
_do_duplicate(design, meter)
except Exception, e:
logging.debug("Duplicate failed: %s", str(e))
if dom:
dom.undefine()
raise
logging.debug("Duplicating finished.")
# Iterate over the list of disks, and clone them using the appropriate
# clone method
def _do_duplicate(design, meter):
# Now actually do the cloning
for dst_dev in design.clone_virtual_disks:
if dst_dev.clone_path == "/dev/null":
# Not really sure why this check was here, but keeping for compat
logging.debug("Source dev was /dev/null. Skipping")
continue
elif dst_dev.clone_path == dst_dev.path:
logging.debug("Source and destination are the same. Skipping.")
continue
# VirtualDisk.setup handles everything
dst_dev.setup(meter)
|