摘要:本文首發于泊浮目的專欄背景在上篇文章中源碼剖析之二次開發可擴展框架,我們簡單的了解了一下核心引擎的二次開發技巧。而在沒有足夠人力來維護開發時,我們會將目標定為能夠及時跟上發布版本。
本文首發于泊浮目的專欄:https://segmentfault.com/blog...背景
在上篇文章中(ZStack源碼剖析之二次開發——可擴展框架
),我們簡單的了解了一下ZStack核心引擎的二次開發技巧。在這篇文章中,我們將一起來了解ZStack-Utility(即ZStack的Agent端)的二開姿勢。
我們以ZStack管理節點調用startVm這個api為例子,一起來看一下在agent上的執行邏輯。
def start(self): http_server = kvmagent.get_http_server() http_server.register_async_uri(self.KVM_START_VM_PATH, self.start_vm)
首先,得注冊一個http path用來接受reqeust。
@kvmagent.replyerror def start_vm(self, req): cmd = jsonobject.loads(req[http.REQUEST_BODY]) rsp = StartVmResponse() try: self._record_operation(cmd.vmInstanceUuid, self.VM_OP_START) self._start_vm(cmd) logger.debug("successfully started vm[uuid:%s, name:%s]" % (cmd.vmInstanceUuid, cmd.vmName)) except kvmagent.KvmError as e: e_str = linux.get_exception_stacktrace() logger.warn(e_str) if "burst" in e_str and "Illegal" in e_str and "rate" in e_str: rsp.error = "QoS exceed max limit, please check and reset it in zstack" elif "cannot set up guest memory" in e_str: logger.warn("unable to start vm[uuid:%s], %s" % (cmd.vmInstanceUuid, e_str)) rsp.error = "No enough physical memory for guest" else: rsp.error = e_str err = self.handle_vfio_irq_conflict(cmd.vmInstanceUuid) if err != "": rsp.error = "%s, details: %s" % (err, rsp.error) rsp.success = False return jsonobject.dumps(rsp)
直接進入主干邏輯,self._start_vm(cmd)。
@lock.lock("libvirt-startvm") def _start_vm(self, cmd): try: vm = get_vm_by_uuid_no_retry(cmd.vmInstanceUuid, False) if vm: if vm.state == Vm.VM_STATE_RUNNING: raise kvmagent.KvmError( "vm[uuid:%s, name:%s] is already running" % (cmd.vmInstanceUuid, vm.get_name())) else: vm.destroy() vm = Vm.from_StartVmCmd(cmd) vm.start(cmd.timeout) except libvirt.libvirtError as e: logger.warn(linux.get_exception_stacktrace()) if "Device or resource busy" in str(e.message): raise kvmagent.KvmError( "unable to start vm[uuid:%s, name:%s], libvirt error: %s" % ( cmd.vmInstanceUuid, cmd.vmName, str(e))) try: vm = get_vm_by_uuid(cmd.vmInstanceUuid) if vm and vm.state != Vm.VM_STATE_RUNNING: raise kvmagent.KvmError( "vm[uuid:%s, name:%s, state:%s] is not in running state, libvirt error: %s" % ( cmd.vmInstanceUuid, cmd.vmName, vm.state, str(e))) except kvmagent.KvmError: raise kvmagent.KvmError( "unable to start vm[uuid:%s, name:%s], libvirt error: %s" % (cmd.vmInstanceUuid, cmd.vmName, str(e)))
關鍵邏輯:
vm = Vm.from_StartVmCmd(cmd) vm.start(cmd.timeout)
先看from_StartVmCmd
@staticmethod def from_StartVmCmd(cmd): use_virtio = cmd.useVirtio use_numa = cmd.useNuma elements = {} def make_root(): root = etree.Element("domain") root.set("type", "kvm") # self._root.set("type", "qemu") root.set("xmlns:qemu", "http://libvirt.org/schemas/domain/qemu/1.0") elements["root"] = root def make_cpu(): if use_numa: root = elements["root"] e(root, "vcpu", "128", {"placement": "static", "current": str(cmd.cpuNum)}) # e(root,"vcpu",str(cmd.cpuNum),{"placement":"static"}) tune = e(root, "cputune") e(tune, "shares", str(cmd.cpuSpeed * cmd.cpuNum)) # enable nested virtualization if cmd.nestedVirtualization == "host-model": cpu = e(root, "cpu", attrib={"mode": "host-model"}) e(cpu, "model", attrib={"fallback": "allow"}) elif cmd.nestedVirtualization == "host-passthrough": cpu = e(root, "cpu", attrib={"mode": "host-passthrough"}) e(cpu, "model", attrib={"fallback": "allow"}) elif IS_AARCH64: cpu = e(root, "cpu", attrib={"mode": "host-passthrough"}) e(cpu, "model", attrib={"fallback": "allow"}) else: cpu = e(root, "cpu") # e(cpu, "topology", attrib={"sockets": str(cmd.socketNum), "cores": str(cmd.cpuOnSocket), "threads": "1"}) mem = cmd.memory / 1024 e(cpu, "topology", attrib={"sockets": str(32), "cores": str(4), "threads": "1"}) numa = e(cpu, "numa") e(numa, "cell", attrib={"id": "0", "cpus": "0-127", "memory": str(mem), "unit": "KiB"}) else: root = elements["root"] # e(root, "vcpu", "128", {"placement": "static", "current": str(cmd.cpuNum)}) e(root, "vcpu", str(cmd.cpuNum), {"placement": "static"}) tune = e(root, "cputune") e(tune, "shares", str(cmd.cpuSpeed * cmd.cpuNum)) # enable nested virtualization if cmd.nestedVirtualization == "host-model": cpu = e(root, "cpu", attrib={"mode": "host-model"}) e(cpu, "model", attrib={"fallback": "allow"}) elif cmd.nestedVirtualization == "host-passthrough": cpu = e(root, "cpu", attrib={"mode": "host-passthrough"}) e(cpu, "model", attrib={"fallback": "allow"}) elif IS_AARCH64: cpu = e(root, "cpu", attrib={"mode": "host-passthrough"}) e(cpu, "model", attrib={"fallback": "allow"}) else: cpu = e(root, "cpu") e(cpu, "topology", attrib={"sockets": str(cmd.socketNum), "cores": str(cmd.cpuOnSocket), "threads": "1"}) def make_memory(): root = elements["root"] mem = cmd.memory / 1024 if use_numa: e(root, "maxMemory", str(68719476736), {"slots": str(16), "unit": "KiB"}) # e(root,"memory",str(mem),{"unit":"k"}) e(root, "currentMemory", str(mem), {"unit": "k"}) else: e(root, "memory", str(mem), {"unit": "k"}) e(root, "currentMemory", str(mem), {"unit": "k"}) def make_os(): root = elements["root"] os = e(root, "os") if IS_AARCH64: e(os, "type", "hvm", attrib={"arch": "aarch64"}) e(os, "loader", "/usr/share/edk2.git/aarch64/QEMU_EFI-pflash.raw", attrib={"readonly": "yes", "type": "pflash"}) else: e(os, "type", "hvm", attrib={"machine": "pc"}) # if not booting from cdrom, don"t add any boot element in os section if cmd.bootDev[0] == "cdrom": for boot_dev in cmd.bootDev: e(os, "boot", None, {"dev": boot_dev}) if cmd.useBootMenu: e(os, "bootmenu", attrib={"enable": "yes"}) def make_features(): root = elements["root"] features = e(root, "features") for f in ["acpi", "apic", "pae"]: e(features, f) if cmd.kvmHiddenState == True: kvm = e(features, "kvm") e(kvm, "hidden", None, {"state": "on"}) def make_devices(): root = elements["root"] devices = e(root, "devices") if cmd.addons and cmd.addons["qemuPath"]: e(devices, "emulator", cmd.addons["qemuPath"]) else: e(devices, "emulator", kvmagent.get_qemu_path()) tablet = e(devices, "input", None, {"type": "tablet", "bus": "usb"}) e(tablet, "address", None, {"type":"usb", "bus":"0", "port":"1"}) if IS_AARCH64: keyboard = e(devices, "input", None, {"type": "keyboard", "bus": "usb"}) elements["devices"] = devices def make_cdrom(): devices = elements["devices"] MAX_CDROM_NUM = len(Vm.ISO_DEVICE_LETTERS) EMPTY_CDROM_CONFIGS = None if IS_AARCH64: # AArch64 Does not support the attachment of multiple iso EMPTY_CDROM_CONFIGS = [ EmptyCdromConfig(None, None, None) ] else: # bus 0 unit 0 already use by root volume EMPTY_CDROM_CONFIGS = [ EmptyCdromConfig("hd%s" % Vm.ISO_DEVICE_LETTERS[0], "0", "1"), EmptyCdromConfig("hd%s" % Vm.ISO_DEVICE_LETTERS[1], "1", "0"), EmptyCdromConfig("hd%s" % Vm.ISO_DEVICE_LETTERS[2], "1", "1") ] if len(EMPTY_CDROM_CONFIGS) != MAX_CDROM_NUM: logger.error("ISO_DEVICE_LETTERS or EMPTY_CDROM_CONFIGS config error") def makeEmptyCdrom(targetDev, bus, unit): cdrom = e(devices, "disk", None, {"type": "file", "device": "cdrom"}) e(cdrom, "driver", None, {"name": "qemu", "type": "raw"}) if IS_AARCH64: e(cdrom, "target", None, {"dev": "sdc", "bus": "scsi"}) else: e(cdrom, "target", None, {"dev": targetDev, "bus": "ide"}) e(cdrom, "address", None,{"type" : "drive", "bus" : bus, "unit" : unit}) e(cdrom, "readonly", None) return cdrom if not cmd.bootIso: for config in EMPTY_CDROM_CONFIGS: makeEmptyCdrom(config.targetDev, config.bus, config.unit) return notEmptyCdrom = set([]) for iso in cmd.bootIso: notEmptyCdrom.add(iso.deviceId) cdromConfig = EMPTY_CDROM_CONFIGS[iso.deviceId] if iso.path.startswith("ceph"): ic = IsoCeph() ic.iso = iso devices.append(ic.to_xmlobject(cdromConfig.targetDev, cdromConfig.bus , cdromConfig.unit)) elif iso.path.startswith("fusionstor"): ic = IsoFusionstor() ic.iso = iso devices.append(ic.to_xmlobject(cdromConfig.targetDev, cdromConfig.bus , cdromConfig.unit)) else: cdrom = makeEmptyCdrom(cdromConfig.targetDev, cdromConfig.bus , cdromConfig.unit) e(cdrom, "source", None, {"file": iso.path}) emptyCdrom = set(range(MAX_CDROM_NUM)).difference(notEmptyCdrom) for i in emptyCdrom: cdromConfig = EMPTY_CDROM_CONFIGS[i] makeEmptyCdrom(cdromConfig.targetDev, cdromConfig.bus, cdromConfig.unit) def make_volumes(): devices = elements["devices"] volumes = [cmd.rootVolume] volumes.extend(cmd.dataVolumes) def filebased_volume(_dev_letter, _v): disk = etree.Element("disk", {"type": "file", "device": "disk", "snapshot": "external"}) e(disk, "driver", None, {"name": "qemu", "type": linux.get_img_fmt(_v.installPath), "cache": _v.cacheMode}) e(disk, "source", None, {"file": _v.installPath}) if _v.shareable: e(disk, "shareable") if _v.useVirtioSCSI: e(disk, "target", None, {"dev": "sd%s" % _dev_letter, "bus": "scsi"}) e(disk, "wwn", _v.wwn) e(disk, "address", None, {"type": "drive", "controller": "0", "unit": str(_v.deviceId)}) return disk if _v.useVirtio: e(disk, "target", None, {"dev": "vd%s" % _dev_letter, "bus": "virtio"}) elif IS_AARCH64: e(disk, "target", None, {"dev": "sd%s" % _dev_letter, "bus": "scsi"}) else: e(disk, "target", None, {"dev": "sd%s" % _dev_letter, "bus": "ide"}) return disk def iscsibased_volume(_dev_letter, _v): def blk_iscsi(): bi = BlkIscsi() portal, bi.target, bi.lun = _v.installPath.lstrip("iscsi://").split("/") bi.server_hostname, bi.server_port = portal.split(":") bi.device_letter = _dev_letter bi.volume_uuid = _v.volumeUuid bi.chap_username = _v.chapUsername bi.chap_password = _v.chapPassword return bi.to_xmlobject() def virtio_iscsi(): vi = VirtioIscsi() portal, vi.target, vi.lun = _v.installPath.lstrip("iscsi://").split("/") vi.server_hostname, vi.server_port = portal.split(":") vi.device_letter = _dev_letter vi.volume_uuid = _v.volumeUuid vi.chap_username = _v.chapUsername vi.chap_password = _v.chapPassword return vi.to_xmlobject() if _v.useVirtio: return virtio_iscsi() else: return blk_iscsi() def ceph_volume(_dev_letter, _v): def ceph_virtio(): vc = VirtioCeph() vc.volume = _v vc.dev_letter = _dev_letter return vc.to_xmlobject() def ceph_blk(): if not IS_AARCH64: ic = IdeCeph() else: ic = ScsiCeph() ic.volume = _v ic.dev_letter = _dev_letter return ic.to_xmlobject() def ceph_virtio_scsi(): vsc = VirtioSCSICeph() vsc.volume = _v vsc.dev_letter = _dev_letter return vsc.to_xmlobject() if _v.useVirtioSCSI: disk = ceph_virtio_scsi() if _v.shareable: e(disk, "shareable") return disk if _v.useVirtio: return ceph_virtio() else: return ceph_blk() def fusionstor_volume(_dev_letter, _v): def fusionstor_virtio(): vc = VirtioFusionstor() vc.volume = _v vc.dev_letter = _dev_letter return vc.to_xmlobject() def fusionstor_blk(): ic = IdeFusionstor() ic.volume = _v ic.dev_letter = _dev_letter return ic.to_xmlobject() def fusionstor_virtio_scsi(): vsc = VirtioSCSIFusionstor() vsc.volume = _v vsc.dev_letter = _dev_letter return vsc.to_xmlobject() if _v.useVirtioSCSI: disk = fusionstor_virtio_scsi() if _v.shareable: e(disk, "shareable") return disk if _v.useVirtio: return fusionstor_virtio() else: return fusionstor_blk() def volume_qos(volume_xml_obj): if not cmd.addons: return vol_qos = cmd.addons["VolumeQos"] if not vol_qos: return qos = vol_qos[v.volumeUuid] if not qos: return if not qos.totalBandwidth and not qos.totalIops: return iotune = e(volume_xml_obj, "iotune") if qos.totalBandwidth: e(iotune, "total_bytes_sec", str(qos.totalBandwidth)) if qos.totalIops: # e(iotune, "total_iops_sec", str(qos.totalIops)) e(iotune, "read_iops_sec", str(qos.totalIops)) e(iotune, "write_iops_sec", str(qos.totalIops)) # e(iotune, "read_iops_sec_max", str(qos.totalIops)) # e(iotune, "write_iops_sec_max", str(qos.totalIops)) # e(iotune, "total_iops_sec_max", str(qos.totalIops)) volumes.sort(key=lambda d: d.deviceId) scsi_device_ids = [v.deviceId for v in volumes if v.useVirtioSCSI] for v in volumes: if v.deviceId >= len(Vm.DEVICE_LETTERS): err = "exceeds max disk limit, it"s %s but only 26 allowed" % v.deviceId logger.warn(err) raise kvmagent.KvmError(err) dev_letter = Vm.DEVICE_LETTERS[v.deviceId] if v.useVirtioSCSI: dev_letter = Vm.DEVICE_LETTERS[scsi_device_ids.pop()] if v.deviceType == "file": vol = filebased_volume(dev_letter, v) elif v.deviceType == "iscsi": vol = iscsibased_volume(dev_letter, v) elif v.deviceType == "ceph": vol = ceph_volume(dev_letter, v) elif v.deviceType == "fusionstor": vol = fusionstor_volume(dev_letter, v) else: raise Exception("unknown volume deviceType: %s" % v.deviceType) assert vol is not None, "vol cannot be None" # set boot order for root volume when boot from hd if v.deviceId == 0 and cmd.bootDev[0] == "hd" and cmd.useBootMenu: e(vol, "boot", None, {"order": "1"}) volume_qos(vol) devices.append(vol) def make_nics(): if not cmd.nics: return def nic_qos(nic_xml_object): if not cmd.addons: return nqos = cmd.addons["NicQos"] if not nqos: return qos = nqos[nic.uuid] if not qos: return if not qos.outboundBandwidth and not qos.inboundBandwidth: return bandwidth = e(nic_xml_object, "bandwidth") if qos.outboundBandwidth: e(bandwidth, "outbound", None, {"average": str(qos.outboundBandwidth / 1024 / 8)}) if qos.inboundBandwidth: e(bandwidth, "inbound", None, {"average": str(qos.inboundBandwidth / 1024 / 8)}) devices = elements["devices"] for nic in cmd.nics: interface = e(devices, "interface", None, {"type": "bridge"}) e(interface, "mac", None, {"address": nic.mac}) if nic.ip is not None and nic.ip != "": filterref = e(interface, "filterref", None, {"filter":"clean-traffic"}) e(filterref, "parameter", None, {"name":"IP", "value": nic.ip}) e(interface, "alias", None, {"name": "net%s" % nic.nicInternalName.split(".")[1]}) e(interface, "source", None, {"bridge": nic.bridgeName}) if use_virtio: e(interface, "model", None, {"type": "virtio"}) else: e(interface, "model", None, {"type": "e1000"}) e(interface, "target", None, {"dev": nic.nicInternalName}) nic_qos(interface) def make_meta(): root = elements["root"] e(root, "name", cmd.vmInstanceUuid) e(root, "uuid", uuidhelper.to_full_uuid(cmd.vmInstanceUuid)) e(root, "description", cmd.vmName) e(root, "on_poweroff", "destroy") e(root, "on_crash", "restart") e(root, "on_reboot", "restart") meta = e(root, "metadata") zs = e(meta, "zstack", usenamesapce=True) e(zs, "internalId", str(cmd.vmInternalId)) e(zs, "hostManagementIp", str(cmd.hostManagementIp)) clock = e(root, "clock", None, {"offset": cmd.clock}) if cmd.clock == "localtime": e(clock, "timer", None, {"name": "rtc", "tickpolicy": "catchup"}) e(clock, "timer", None, {"name": "pit", "tickpolicy": "delay"}) e(clock, "timer", None, {"name": "hpet", "present": "no"}) e(clock, "timer", None, {"name": "hypervclock", "present": "yes"}) def make_vnc(): devices = elements["devices"] if cmd.consolePassword == None: vnc = e(devices, "graphics", None, {"type": "vnc", "port": "5900", "autoport": "yes"}) else: vnc = e(devices, "graphics", None, {"type": "vnc", "port": "5900", "autoport": "yes", "passwd": str(cmd.consolePassword)}) e(vnc, "listen", None, {"type": "address", "address": "0.0.0.0"}) def make_spice(): devices = elements["devices"] spice = e(devices, "graphics", None, {"type": "spice", "port": "5900", "autoport": "yes"}) e(spice, "listen", None, {"type": "address", "address": "0.0.0.0"}) e(spice, "image", None, {"compression": "auto_glz"}) e(spice, "jpeg", None, {"compression": "always"}) e(spice, "zlib", None, {"compression": "never"}) e(spice, "playback", None, {"compression": "off"}) e(spice, "streaming", None, {"mode": cmd.spiceStreamingMode}) e(spice, "mouse", None, {"mode": "client"}) e(spice, "filetransfer", None, {"enable": "no"}) e(spice, "clipboard", None, {"copypaste": "no"}) def make_usb_redirect(): if cmd.usbRedirect == "true": devices = elements["devices"] e(devices, "controller", None, {"type": "usb", "model": "ich9-ehci1"}) e(devices, "controller", None, {"type": "usb", "model": "ich9-uhci1", "multifunction": "on"}) e(devices, "controller", None, {"type": "usb", "model": "ich9-uhci2"}) e(devices, "controller", None, {"type": "usb", "model": "ich9-uhci3"}) chan = e(devices, "channel", None, {"type": "spicevmc"}) e(chan, "target", None, {"type": "virtio", "name": "com.redhat.spice.0"}) e(chan, "address", None, {"type": "virtio-serial"}) redirdev2 = e(devices, "redirdev", None, {"type": "spicevmc", "bus": "usb"}) e(redirdev2, "address", None, {"type": "usb", "bus": "0", "port": "2"}) redirdev3 = e(devices, "redirdev", None, {"type": "spicevmc", "bus": "usb"}) e(redirdev3, "address", None, {"type": "usb", "bus": "0", "port": "3"}) redirdev4 = e(devices, "redirdev", None, {"type": "spicevmc", "bus": "usb"}) e(redirdev4, "address", None, {"type": "usb", "bus": "0", "port": "4"}) redirdev5 = e(devices, "redirdev", None, {"type": "spicevmc", "bus": "usb"}) e(redirdev5, "address", None, {"type": "usb", "bus": "0", "port": "6"}) else: # make sure there are three default usb controllers, for usb 1.1/2.0/3.0 devices = elements["devices"] e(devices, "controller", None, {"type": "usb", "index": "0"}) if not IS_AARCH64: e(devices, "controller", None, {"type": "usb", "index": "1", "model": "ehci"}) e(devices, "controller", None, {"type": "usb", "index": "2", "model": "nec-xhci"}) def make_video(): devices = elements["devices"] if IS_AARCH64: video = e(devices, "video") e(video, "model", None, {"type": "virtio"}) elif cmd.videoType != "qxl": video = e(devices, "video") e(video, "model", None, {"type": str(cmd.videoType)}) else: for monitor in range(cmd.VDIMonitorNumber): video = e(devices, "video") e(video, "model", None, {"type": str(cmd.videoType)}) def make_audio_microphone(): if cmd.consoleMode == "spice": devices = elements["devices"] e(devices, "sound",None,{"model":"ich6"}) else: return def make_graphic_console(): if cmd.consoleMode == "spice": make_spice() else: make_vnc() def make_addons(): if not cmd.addons: return devices = elements["devices"] channel = cmd.addons["channel"] if channel: basedir = os.path.dirname(channel.socketPath) linux.mkdir(basedir, 0777) chan = e(devices, "channel", None, {"type": "unix"}) e(chan, "source", None, {"mode": "bind", "path": channel.socketPath}) e(chan, "target", None, {"type": "virtio", "name": channel.targetName}) cephSecretKey = cmd.addons["ceph_secret_key"] cephSecretUuid = cmd.addons["ceph_secret_uuid"] if cephSecretKey and cephSecretUuid: VmPlugin._create_ceph_secret_key(cephSecretKey, cephSecretUuid) pciDevices = cmd.addons["pciDevice"] if pciDevices: make_pci_device(pciDevices) usbDevices = cmd.addons["usbDevice"] if usbDevices: make_usb_device(usbDevices) def make_pci_device(addresses): devices = elements["devices"] for addr in addresses: if match_pci_device(addr): hostdev = e(devices, "hostdev", None, {"mode": "subsystem", "type": "pci", "managed": "yes"}) e(hostdev, "driver", None, {"name": "vfio"}) source = e(hostdev, "source") e(source, "address", None, { "domain": hex(0) if len(addr.split(":")) == 2 else hex(int(addr.split(":")[0], 16)), "bus": hex(int(addr.split(":")[-2], 16)), "slot": hex(int(addr.split(":")[-1].split(".")[0], 16)), "function": hex(int(addr.split(":")[-1].split(".")[1], 16)) }) else: raise kvmagent.KvmError( "can not find pci device for address %s" % addr) def make_usb_device(usbDevices): next_uhci_port = 2 next_ehci_port = 1 next_xhci_port = 1 devices = elements["devices"] for usb in usbDevices: if match_usb_device(usb): hostdev = e(devices, "hostdev", None, {"mode": "subsystem", "type": "usb", "managed": "yes"}) source = e(hostdev, "source") e(source, "address", None, { "bus": str(int(usb.split(":")[0])), "device": str(int(usb.split(":")[1])) }) e(source, "vendor", None, { "id": hex(int(usb.split(":")[2], 16)) }) e(source, "product", None, { "id": hex(int(usb.split(":")[3], 16)) }) # get controller index from usbVersion # eg. 1.1 -> 0 # eg. 2.0.0 -> 1 # eg. 3 -> 2 bus = int(usb.split(":")[4][0]) - 1 if bus == 0: address = e(hostdev, "address", None, {"type": "usb", "bus": str(bus), "port": str(next_uhci_port)}) next_uhci_port += 1 elif bus == 1: address = e(hostdev, "address", None, {"type": "usb", "bus": str(bus), "port": str(next_ehci_port)}) next_ehci_port += 1 elif bus == 2: address = e(hostdev, "address", None, {"type": "usb", "bus": str(bus), "port": str(next_xhci_port)}) next_xhci_port += 1 else: raise kvmagent.KvmError("unknown usb controller %s", bus) else: raise kvmagent.KvmError("cannot find usb device %s", usb) # TODO(WeiW) Validate here def match_pci_device(addr): return True def match_usb_device(addr): if len(addr.split(":")) == 5: return True else: return False def make_balloon_memory(): devices = elements["devices"] b = e(devices, "memballoon", None, {"model": "virtio"}) e(b, "stats", None, {"period": "10"}) def make_console(): devices = elements["devices"] serial = e(devices, "serial", None, {"type": "pty"}) e(serial, "target", None, {"port": "0"}) console = e(devices, "console", None, {"type": "pty"}) e(console, "target", None, {"type": "serial", "port": "0"}) def make_sec_label(): root = elements["root"] e(root, "seclabel", None, {"type": "none"}) def make_controllers(): devices = elements["devices"] e(devices, "controller", None, {"type": "scsi", "model": "virtio-scsi"}) make_root() make_meta() make_cpu() make_memory() make_os() make_features() make_devices() make_video() make_audio_microphone() make_nics() make_volumes() make_cdrom() make_graphic_console() make_usb_redirect() make_addons() make_balloon_memory() make_console() make_sec_label() make_controllers() root = elements["root"] xml = etree.tostring(root) vm = Vm() vm.uuid = cmd.vmInstanceUuid vm.domain_xml = xml vm.domain_xmlobject = xmlobject.loads(xml) return vm
顯然,上述邏輯是在組裝一份xml,便于之后的libvirt使用。
然后是
vm.start(cmd.timeout)
可以看到,這里是直接調用了libvirt的sdk。
這僅僅是一個調用流程。而在很多地方,來自MN的請求會直接調用linux的shell命令,詳情見linux.py。(獲取云盤大小、主存儲容量等)。
問題在基于擴展ZStack的Agent時,如果是一個全新的功能模塊,可能并不會造成和原有代碼的深度耦合。但如果在原有功能上的增強, 對原有代碼進行修改可能會導致我們的業務邏輯和Utility的上游代碼耦合。而在沒有足夠人力來維護、開發ZStack時,我們會將目標定為能夠及時跟上發布版本。 因此,我們要盡量減少沖突。
舉個例子:我們要對啟動vm的邏輯進行增強,添加一個自己的配置寫入xml。這段代碼如果寫進了vm_plugin.py,那么就是一個耦合。耦合多了以后,跟上發布版本就會很困難。
解決方案這是一個參考方案:
如果是引入一個全新的功能模塊,建議重寫一個項目。無論是代碼規范還是自動化測試,都可以有一個很好的實踐。
如果是基于Utility的擴展,比如對于擴展的api——APIStartVmInstanceExMsg。由上游發送http request時,將指定v2版本的agent。比如原有start vm會發送至path:AGENT_IP:7070/vm/start;而如果我們增強了這部分邏輯,將這段代碼copy至vm_plugin_ex.py,并注冊一個path,ex/vm/start。當然port也要重新注冊一個,就像這樣::AGENT_IP:7071/ex/vm/start。
同樣的,對linux.py擴展時,復制一個linux2.py來存放屬于我們自己的擴展邏輯。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/41765.html
摘要:但在實際的二次開發中,這些做法未必能夠完全滿足需求。在源碼剖析之核心庫鑒賞一文中,我們了解到是的基礎設施之一,同時也允許通過顯示聲明的方式來聲明。同理,一些也可以使用繼承進行擴展。 本文首發于泊浮目的專欄:https://segmentfault.com/blog... 前言 在ZStack博文-5.通用插件系統中,官方提出了幾個較為經典的擴展方式。但在實際的二次開發中,這些做法未必...
摘要:本文首發于泊浮目的專欄在前文源碼剖析之二次開發可擴展框架中,我們大概的了解了如何在中進行二次開發。在還有相關的日志,有興趣的讀者可以自行搜索。掛斷點在掛斷點之前,請確定自己的開放了相應的端口。之后記得使用關掉。 本文首發于泊浮目的專欄:https://segmentfault.com/blog... 在前文 ZStack源碼剖析之二次開發——可擴展框架中,我們大概的了解了如何在ZSt...
摘要:本文將對核心引擎的源碼進行剖析。在筆者看來,能夠快速迭代的原因首先是來自于每位工程師的辛勤付出。在中,還有一類很有意思的代碼,一般稱之為。筆者有機會將會在之后的系列文章分析其中的典型案例以及在代碼中使用極其頻繁的核心工具。 本文首發于泊浮目的專欄:https://segmentfault.com/blog... 前言 ZStack是下一代開源的云計算IaaS(基礎架構即服務)軟件。它...
摘要:本文將對核心引擎的源碼進行剖析。在筆者看來,能夠快速迭代的原因首先是來自于每位工程師的辛勤付出。在中,還有一類很有意思的代碼,一般稱之為。筆者有機會將會在之后的系列文章分析其中的典型案例以及在代碼中使用極其頻繁的核心工具。 本文首發于泊浮目的專欄:https://segmentfault.com/blog... 前言 ZStack是下一代開源的云計算IaaS(基礎架構即服務)軟件。它...
摘要:本文將對核心引擎的源碼進行剖析。在筆者看來,能夠快速迭代的原因首先是來自于每位工程師的辛勤付出。在中,還有一類很有意思的代碼,一般稱之為。筆者有機會將會在之后的系列文章分析其中的典型案例以及在代碼中使用極其頻繁的核心工具。 本文首發于泊浮目的專欄:https://segmentfault.com/blog... 前言 ZStack是下一代開源的云計算IaaS(基礎架構即服務)軟件。它...
閱讀 2895·2019-08-30 15:55
閱讀 2000·2019-08-30 14:02
閱讀 1239·2019-08-29 15:23
閱讀 1006·2019-08-29 11:27
閱讀 461·2019-08-26 11:43
閱讀 3191·2019-08-26 10:32
閱讀 1253·2019-08-23 14:41
閱讀 3301·2019-08-23 14:41