# Copyright 2013-2014 Red Hat, Inc. # # This work is licensed under the GNU GPLv2 or later. # See the COPYING file in the top-level directory. import os import sys import libvirt from . import cli from .cli import fail, fail_conflicting, print_stdout, print_stderr from .guest import Guest from .logger import log from . import xmlutil ################### # Utility helpers # ################### def prompt_yes_or_no(msg): while 1: printmsg = msg + " (y/n): " sys.stdout.write(printmsg) sys.stdout.flush() inp = sys.stdin.readline().lower().strip() if inp in ["y", "yes"]: return True elif inp in ["n", "no"]: return False else: print_stdout(_("Please enter 'yes' or 'no'.")) def get_diff(origxml, newxml): diff = xmlutil.diff(origxml, newxml, "Original XML", "Altered XML") if diff: log.debug("XML diff:\n%s", diff) else: log.debug("No XML diff, didn't generate any change.") return diff def set_os_variant(guest, os_variant): if os_variant is None: return osdata = cli.parse_os_variant(os_variant) if osdata.get_name(): guest.set_os_name(osdata.get_name()) def defined_xml_is_unchanged(conn, domain, original_xml): rawxml = cli.get_xmldesc(domain, inactive=True) new_xml = Guest(conn, parsexml=rawxml).get_xml() return new_xml == original_xml ################## # Action parsing # ################## class Action: """ Helper class tracking one pair of XML ACTION (ex. --edit) and XML OPTION (ex. --disk) """ def __init__(self, action_name, selector, parserclass, parservalue): # one of ["edit", "add-device", "remove-device", "build-xml"] self.action_name = action_name # ex. for `--edit 1` this is selector="1" self.selector = selector # ParserDisk, etc self.parserclass = parserclass # ex for `--disk path=/foo` this is "path=/foo" self.parservalue = parservalue @property def is_edit(self): return self.action_name == "edit" @property def is_add_device(self): return self.action_name == "add-device" @property def is_remove_device(self): return self.action_name == "remove-device" @property def is_build_xml(self): return self.action_name == "build-xml" def validate_action(action, conn, options): if options.os_variant is not None: if action.is_edit: fail(_("--os-variant/--osinfo is not supported with --edit")) if action.is_remove_device: fail(_("--os-variant/--osinfo is not supported with --remove-device")) if action.is_build_xml: fail(_("--os-variant/--osinfo is not supported with --build-xml")) if not action.parserclass.guest_propname and action.is_build_xml: fail(_("--build-xml not supported for {cli_flag}").format( cli_flag=action.parserclass.cli_flag_name())) stub_guest = Guest(conn) if not action.parserclass.prop_is_list(stub_guest): if action.is_remove_device: fail(_("Cannot use --remove-device with {cli_flag}").format( cli_flag=action.parserclass.cli_flag_name())) if action.is_add_device: fail(_("Cannot use --add-device with {cli_flag}").format( cli_flag=action.parserclass.cli_flag_name())) if options.update and not action.parserclass.guest_propname: fail(_("Don't know how to --update for {cli_flag}").format( cli_flag=action.parserclass.cli_flag_name())) def check_action_collision(options): collisions = [] actions = ["edit", "add-device", "remove-device", "build-xml"] for cliname in actions: optname = cliname.replace("-", "_") value = getattr(options, optname) if value not in [False, -1]: collisions.append((cliname, value)) if len(collisions) == 0: fail(_("One of %s must be specified.") % ", ".join(["--" + c for c in actions])) if len(collisions) > 1: fail(_("Conflicting options %s") % ", ".join(["--" + c[0] for c in collisions])) action_name, selector = collisions[0] return action_name, selector def check_xmlopt_collision(options): collisions = [] for parserclass in cli.VIRT_PARSERS: value = getattr(options, parserclass.cli_arg_name) if value: collisions.append((parserclass, value)) if len(collisions) == 0: fail(_("No change specified.")) if len(collisions) != 1: fail(_("Only one change operation may be specified " "(conflicting options %s)") % [c[0].cli_flag_name() for c in collisions]) parserclass, parservalue = collisions[0] return parserclass, parservalue def parse_action(conn, options): # Ensure there wasn't more than one device/xml config option # specified. So reject '--disk X --network X' parserclass, parservalue = check_xmlopt_collision(options) # Ensure only one of these actions was specified # --edit # --remove-device # --add-device # --build-xml action_name, selector = check_action_collision(options) action = Action(action_name, selector, parserclass, parservalue) validate_action(action, conn, options) return action ################ # Change logic # ################ def _find_objects_to_edit(guest, action_name, editval, parserclass): objlist = xmlutil.listify(parserclass.lookup_prop(guest)) idx = None if editval is None: idx = 1 elif (editval.isdigit() or editval.startswith("-") and editval[1:].isdigit()): idx = int(editval) if idx is not None: # Edit device by index if idx == 0: fail(_("Invalid --edit option '%s'") % editval) if not objlist: fail(_("No {cli_flag} objects found in the XML").format( cli_flag=parserclass.cli_flag_name())) if len(objlist) < abs(idx): fail(ngettext("'--edit {number}' requested but there's only " "{maxnum} {cli_flag} object in the XML", "'--edit {number}' requested but there are only " "{maxnum} {cli_flag} objects in the XML", len(objlist)).format( number=idx, maxnum=len(objlist), cli_flag=parserclass.cli_flag_name())) if idx > 0: idx -= 1 inst = objlist[idx] elif editval == "all": # Edit 'all' devices inst = objlist[:] else: # Lookup device by the passed prop string parserobj = parserclass(editval, guest=guest) inst = parserobj.lookup_child_from_option_string() if not inst: fail(_("No matching objects found for %s") % ("--%s %s" % (action_name, editval))) return inst def action_edit(action, guest): parserclass = action.parserclass parservalue = action.parservalue selector = action.selector if parserclass.guest_propname: inst = _find_objects_to_edit(guest, "edit", selector, parserclass) else: inst = guest if (selector and selector != '1' and selector != 'all'): fail(_("'--edit {option}' doesn't make sense with " "{cli_flag}, just use empty '--edit'").format( option=selector, cli_flag=parserclass.cli_flag_name())) devs = [] for editinst in xmlutil.listify(inst): devs += cli.run_parser(guest, parserclass, parservalue, editinst=editinst) return devs def action_add_device(action, guest, os_variant, input_devs): parserclass = action.parserclass parservalue = action.parservalue set_os_variant(guest, os_variant) if input_devs: for dev in input_devs: guest.add_device(dev) devs = input_devs else: devs = cli.run_parser(guest, parserclass, parservalue) for dev in devs: dev.set_defaults(guest) return devs def action_remove_device(action, guest): parserclass = action.parserclass parservalue = action.parservalue[-1] devs = _find_objects_to_edit(guest, "remove-device", parservalue, parserclass) devs = xmlutil.listify(devs) for dev in devs: guest.remove_device(dev) return devs def action_build_xml(action, guest): parserclass = action.parserclass parservalue = action.parservalue devs = cli.run_parser(guest, parserclass, parservalue) for dev in devs: dev.set_defaults(guest) return devs def perform_action(action, guest, options, input_devs): if action.is_add_device: return action_add_device(action, guest, options.os_variant, input_devs) if action.is_remove_device: return action_remove_device(action, guest) if action.is_edit: return action_edit(action, guest) raise xmlutil.DevError( "perform_action() incorrectly called with action_name=%s" % action.action_name) def setup_device(dev): if getattr(dev, "DEVICE_TYPE", None) != "disk": return log.debug("Doing setup for disk=%s", dev) dev.build_storage(cli.get_meter()) def define_changes(conn, inactive_xmlobj, devs, action, confirm): if confirm: if not prompt_yes_or_no( _("Define '%s' with the changed XML?") % inactive_xmlobj.name): return False if action.is_add_device: for dev in devs: setup_device(dev) dom = conn.defineXML(inactive_xmlobj.get_xml()) print_stdout(_("Domain '%s' defined successfully.") % inactive_xmlobj.name) return dom def start_domain_transient(conn, xmlobj, devs, action, confirm): if confirm: if not prompt_yes_or_no( _("Start '%s' with the changed XML?") % xmlobj.name): return False if action.is_add_device: for dev in devs: setup_device(dev) try: dom = conn.createXML(xmlobj.get_xml()) except libvirt.libvirtError as e: fail(_("Failed starting domain '%(domain)s': %(error)s") % { "domain": xmlobj.name, "error": e, }) else: print_stdout(_("Domain '%s' started successfully.") % xmlobj.name) return dom def update_changes(domain, devs, action, confirm): if action.is_add_device: msg_confirm = _("%(xml)s\n\nHotplug this device to the guest " "'%(domain)s'?") msg_success = _("Device hotplug successful.") msg_fail = _("Error attempting device hotplug: %(error)s") elif action.is_remove_device: msg_confirm = _("%(xml)s\n\nHotunplug this device from the guest " "'%(domain)s'?") msg_success = _("Device hotunplug successful.") msg_fail = _("Error attempting device hotunplug: %(error)s") elif action.is_edit: msg_confirm = _("%(xml)s\n\nUpdate this device for the guest " "'%(domain)s'?") msg_success = _("Device update successful.") msg_fail = _("Error attempting device update: %(error)s") else: raise xmlutil.DevError( "update_changes() incorrectly called with action=%s" % action.action_name) for dev in devs: xml = dev.get_xml() if confirm: msg = msg_confirm % { "xml": xml, "domain": domain.name(), } if not prompt_yes_or_no(msg): continue if action.is_add_device: setup_device(dev) try: if action.is_add_device: domain.attachDeviceFlags(xml, libvirt.VIR_DOMAIN_AFFECT_LIVE) elif action.is_remove_device: domain.detachDeviceFlags(xml, libvirt.VIR_DOMAIN_AFFECT_LIVE) elif action.is_edit: domain.updateDeviceFlags(xml, libvirt.VIR_DOMAIN_AFFECT_LIVE) except libvirt.libvirtError as e: if "VIRTXML_TESTSUITE_UPDATE_IGNORE_FAIL" not in os.environ: fail(msg_fail % {"error": e}) print_stdout(msg_success) if confirm: print_stdout("") def prepare_changes(orig_xmlobj, options, action, input_devs=None): """ Perform requested XML edits locally, but don't submit them to libvirt. Optionally perform any XML printing per user request :returns: (list of device objects, altered xmlobj) """ origxml = orig_xmlobj.get_xml() xmlobj = orig_xmlobj.__class__(conn=orig_xmlobj.conn, parsexml=origxml) devs = perform_action(action, xmlobj, options, input_devs) newxml = xmlobj.get_xml() diff = get_diff(origxml, newxml) if not diff: log.warning(_("No XML diff was generated. The requested " "changes will have no effect.")) if options.print_diff: if diff: print_stdout(diff) elif options.print_xml: print_stdout(newxml) return devs, xmlobj ####################### # CLI option handling # ####################### def parse_args(): parser = cli.setupParser( "%(prog)s [options]", _("Edit libvirt XML using command line options."), introspection_epilog=True) cli.add_connect_option(parser, "virt-xml") parser.add_argument("domain", nargs='?', help=_("Domain name, id, or uuid")) actg = parser.add_argument_group(_("XML actions")) actg.add_argument("--edit", nargs='?', default=-1, help=_("Edit VM XML. Examples:\n" "--edit --disk ... (edit first disk device)\n" "--edit 2 --disk ... (edit second disk device)\n" "--edit all --disk ... (edit all disk devices)\n" "--edit target=hda --disk ... (edit disk 'hda')\n")) actg.add_argument("--remove-device", action="store_true", help=_("Remove specified device. Examples:\n" "--remove-device --disk 1 (remove first disk)\n" "--remove-device --disk all (remove all disks)\n" "--remove-device --disk /some/path")) actg.add_argument("--add-device", action="store_true", help=_("Add specified device. Example:\n" "--add-device --disk ...")) actg.add_argument("--build-xml", action="store_true", help=_("Output built device XML. Domain is optional but " "recommended to ensure optimal defaults.")) outg = parser.add_argument_group(_("Output options")) outg.add_argument("--update", action="store_true", help=_("Apply changes to the running VM.\n" "With --add-device, this is a hotplug operation.\n" "With --remove-device, this is a hotunplug operation.\n" "With --edit, this is an update device operation.")) define_g = outg.add_mutually_exclusive_group() define_g.add_argument("--define", action="store_true", help=_("Force defining the domain. Only required if a --print " "option was specified.")) define_g.add_argument("--no-define", dest='define', action="store_false", help=_("Force not defining the domain.")) define_g.set_defaults(define=None) outg.add_argument("--start", action="store_true", help=_("Start the domain.")) outg.add_argument("--print-diff", action="store_true", help=_("Only print the requested change, in diff format")) outg.add_argument("--print-xml", action="store_true", help=_("Only print the requested change, in full XML format")) outg.add_argument("--confirm", action="store_true", help=_("Require confirmation before saving any results.")) cli.add_os_variant_option(parser, virtinstall=False) conv = parser.add_argument_group(_("Conversion options")) cli.ParserConvertToQ35.register() conv.add_argument("--convert-to-q35", nargs="?", const=cli.VirtCLIParser.OPTSTR_EMPTY, help=_("Convert an existing VM from PC/i440FX to Q35.")) cli.ParserConvertToVNC.register() conv.add_argument("--convert-to-vnc", nargs="?", const=cli.VirtCLIParser.OPTSTR_EMPTY, help=_("Convert an existing VM to use VNC graphics. " "This removes any remnants of Spice graphics.")) g = parser.add_argument_group(_("XML options")) cli.add_disk_option(g, editexample=True) cli.add_net_option(g) cli.add_gfx_option(g) cli.add_metadata_option(g) cli.add_memory_option(g) cli.vcpu_cli_options(g, editexample=True) cli.add_xml_option(g) cli.add_guest_xml_options(g) cli.add_boot_options(g) cli.add_device_options(g) misc = parser.add_argument_group(_("Miscellaneous Options")) cli.add_misc_options(misc, prompt=False, printxml=False, dryrun=False) cli.autocomplete(parser) return parser.parse_args() ################### # main() handling # ################### def main(conn=None): cli.earlyLogging() options = parse_args() if (options.confirm or options.print_xml or options.print_diff or options.build_xml): options.quiet = False cli.setupLogging("virt-xml", options.debug, options.quiet) if cli.check_option_introspection(options): return 0 if cli.check_osinfo_list(options): return 0 options.stdinxml = None if not options.domain and not options.build_xml: if not sys.stdin.closed and not sys.stdin.isatty(): if options.confirm: fail(_("Can't use --confirm with stdin input.")) if options.update: fail(_("Can't use --update with stdin input.")) options.stdinxml = sys.stdin.read() else: fail(_("A domain must be specified")) # Default to --define, unless: # --no-define explicitly specified # --print-* option is used # XML input came from stdin if not options.print_xml and not options.print_diff: if options.stdinxml: if not options.define: options.print_xml = True else: if options.define is None: options.define = True if options.confirm and not options.print_xml: options.print_diff = True conn = cli.getConnection(options.connect, conn) action = parse_action(conn, options) domain = None active_xmlobj = None inactive_xmlobj = None if options.domain: domain, inactive_xmlobj, active_xmlobj = cli.get_domain_and_guest( conn, options.domain) else: inactive_xmlobj = Guest(conn, parsexml=options.stdinxml) vm_is_running = bool(active_xmlobj) if action.is_build_xml: built_devs = action_build_xml(action, inactive_xmlobj) for dev in built_devs: # pylint: disable=no-member print_stdout(xmlutil.unindent_device_xml(dev.get_xml())) return 0 input_devs = None performed_update = False if options.update: if options.update and options.start: fail_conflicting("--update", "--start") if vm_is_running: input_devs, dummy = prepare_changes( active_xmlobj, options, action) update_changes(domain, input_devs, action, options.confirm) performed_update = True else: log.warning( _("The VM is not running, --update is inapplicable.")) if not options.define: # --update and --no-define passed, so we are done return 0 original_xml = inactive_xmlobj.get_xml() devs, xmlobj_to_define = prepare_changes( inactive_xmlobj, options, action, input_devs=input_devs) if not options.define: if options.start: start_domain_transient(conn, xmlobj_to_define, devs, action, options.confirm) return 0 dom = define_changes(conn, xmlobj_to_define, devs, action, options.confirm) if not dom: # --confirm user said 'no' return 0 if options.start: try: dom.create() except libvirt.libvirtError as e: # pragma: no cover fail(_("Failed starting domain '%(domain)s': %(error)s") % { "domain": inactive_xmlobj.name, "error": e, }) print_stdout(_("Domain '%s' started successfully.") % inactive_xmlobj.name) elif vm_is_running and not performed_update: print_stdout( _("Changes will take effect after the domain is fully powered off.")) elif defined_xml_is_unchanged(conn, dom, original_xml): log.warning(_("XML did not change after domain define. You may " "have changed a value that libvirt is setting by default.")) return 0 def runcli(): # pragma: no cover try: sys.exit(main()) except SystemExit as sys_e: sys.exit(sys_e.code) except KeyboardInterrupt: log.debug("", exc_info=True) print_stderr(_("Aborted at user request")) except Exception as main_e: fail(main_e)