Libvirt (正體中文)
Libvirt 是一組軟體的匯集,提供了管理虛擬機器和其它虛擬化功能(如:儲存和虛擬網路等)的便利途徑。這些軟體主要包括:一個長期稳定的 API、一個守護程序(libvirtd)、一些命令列工具(如virsh、virt-install等)和一些GUI工具(如virt-manager等)。Libvirt 的主要目標是提供一個單一途徑以管理多種不同虛擬化方案以及虛擬化主機,包括:KVM/QEMU,Xen,LXC,OpenVZ 或 VirtualBox hypervisors (詳見這裡)。
Libvirt 的一些主要功能如下:
- VM management(虛擬機器管理):如:開機、停止、暫停、保存、恢復和迁移等;多種不同類型裝置的熱插拔操作,包括硬碟、網路卡、記憶體、CPU等。
- Remote machine support(支援遠端連線):Libvirt 的所有功能都可以在執行著libvirt daemon的機器上執行,包括遠端機器。通過最簡便且無需額外配置的 SSH 協定,遠端連線可支援多種網路連線方式。
- Storage management(存儲管理):任何執行 libvirt 守護程序的主機都可以用于管理多種類型的的存儲:創建多種類型的鏡像檔(qcow2,qed,vmdk,raw,...),掛載 NFS 共用,枚舉現有 LVM 捲軸,新增新的 LVM 磁區,對裸磁碟裝置分割,掛載 iSCSI 共用,以及更多......
- Network interface management(網路介面管理):任何執行 libvirt 守護程序的主機都可以用于管理物理的和邏輯的網路介面,枚舉現有介面,配置(和創建)介面、橋接、VLAN、連接埠綁定。
- Virtual NAT and Route based networking(虛擬NAT 和基於路由的網路):任何執行libvirt 守護程序的主機都可以管理和創建虛擬網路。Libvirt 虛擬網路使用iptables防火牆規則實現一个路由器,為虛擬機器提供到主機網路的透明存取。
安裝
基於守護程序/用戶端的架構的 libvirt 只需要安裝在需要實現虛擬化的機器上。當然,libvirt伺服器和libvirt用戶端可以是相同的物理機器。
伺服器端
安裝 libvirt 以及至少一個管理程式(hypervisor):
- 截至 2015-02-01,啟動 libvirtd必須在系統上安裝 qemu(参參見FS#41888)。好在, libvirt 的 KVM/QEMU 驅動 是 libvirt 的,如果 KVM 功能已啟用,首選驅動則支援全虛擬化和硬體加速加速的用戶機。詳見 QEMU。
- 其他虛擬機器後端,包括 LXC, Xen 和 VirtualBox。請參見他們各自的安裝說明。
其它虛擬機器管理式的列表见这里。
對於網路連線,安裝這些包:
- ebtables 和 dnsmasq 用於 預設的 NAT/DHCP網路
- bridge-utils 用於橋接網路
- openbsd-netcat 經由 SSH 遠端管理
用戶端
用戶端是用於管理和存取虛擬機器的使用者界面。
- virsh 用於管理和配置(域)的命令列程式;它包含在 libvirt 中。
虛擬機* virt-manager 用於管理虛擬機的GUI。
- virtviewer[broken link: replaced by virt-viewer] 用於顯示並與虛擬化用戶機作業系統進行交互
- gnome-boxes 簡單的 GNOME 3 程式,存取遠端虛擬系統
- libvirt-sandboxAUR 應用程式沙箱工具包
相容 libvirt 的軟體列表見[1].
配置
對於系統 級別的管理工作(如:全局配置和鏡像位置),libvirt 要求至少要設定授權和啟動守護程序。
設定授權
來自 libvirt:連線授權:
- Libvirt daemon允許管理員分別為用戶端連線的每個網路 socket 選擇不同授權機制。這主要是通過 libvirt 守護程序的主設定檔 /etc/libvirt/libvirtd.conf來實現的。每個 libvirt socket 可以有獨立的授權機制配置。目前的可選項有none、polkit和sasl。
由于  libvirt 在安裝时將把 polkit 作為依赖一並安裝,所以 polkit 通常是 unix_sock_auth 參數的預設值(來源)。但基於檔案的權限仍然可用。
使用 polkit
libvirt  daemon在 polkit 策略設定檔(/usr/share/polkit-1/actions/org.libvirt.unix.policy)中提供了2種  行為[broken link: invalid section] 策略:
-  org.libvirt.unix.manage面向完全的管理存取(讀寫模式的後台 socket),以及
-  org.libvirt.unix.monitor面向僅监視查看存取(唯讀 socket)。
預設的面向讀寫模式後台 socket 的策略將請求認證為管理者。這類似於 sudo 認證,但它並不要求用戶應用最終以 root 身分執行。預設策略下也仍然允許任何應用連線到唯讀 socket。
Arch Linux 預設 wheel 組的所有實驗者都是管理員身分:定義於 /etc/polkit-1/rules.d/50-default.rules(參閱管理员标识[broken link: invalid section])。這樣就不必新增組和規則檔案。 如果用户是 wheel 群組的成員:只要連線到了讀寫模式 socket(例如通过 virt-manager)就會被提示輸入該使用者的口令。
從 libvirt 1.2.16 開始(提案見:[2]),libvirt 組的成員使用者預設可以無密碼存取讀寫模式 socket。最簡單的判斷方法就是看 libvirt 組是否存在並且使用者是否為該組成員。如果要把 kvm 組存取讀寫模式後台 socket 的認證策略改為免認證模式,可創建下面的檔案:
/etc/polkit-1/rules.d/50-libvirt.rules
/* Allow users in kvm group to manage the libvirt daemon without authentication(允許 kvm 組的使用者管理 libvirt 而無需認證)
*/
polkit.addRule(function(action, subject) {
    if (action.id == "org.libvirt.unix.manage" &&
        subject.isInGroup("kvm")) {
            return polkit.Result.YES;
    }
});
然後 新增使用者 到 kvm 組并重新登入。kvm 也可以是任何其它存在的組並且用戶是該組的成員(詳情請閱讀 使用者和使用者群組)。
修改組之後別忘了重新登入才能生效。
基於檔案的權限授權
為了給 libvirt 組使用者定義基於檔案的權限以管理虛擬機器,取消下列行的註釋:
/etc/libvirt/libvirtd.conf
#unix_sock_group = "libvirt" #unix_sock_ro_perms = "0777" # set to 0770 to deny non-group libvirt users #unix_sock_rw_perms = "0770" #auth_unix_ro = "none" #auth_unix_rw = "none"
有些資料提到可以通過改變某些特定 libvirt 目錄的權限以簡化管理。需要記住的是:包更新時,這些變更會遺失。如要修改這些系統目錄的權限,需要 root 使用者權限。
守護程序
libvirtd.service 和 virtlogd.service2個服務都要 啟動。可以把 libvirtd.service 設定為 啟用,這時系統將同時啟用 virtlogd.service 和 virtlockd.socket 両個服務 單元,因此後二者不必再設置為啟用。
非加密的 TCP/IP sockets
編輯 /etc/libvirt/libvirtd.conf:
/etc/libvirt/libvirtd.conf
listen_tls = 0 listen_tcp = 1 auth_tcp=none
It is also necessary to start the server in listening mode by editing /etc/conf.d/libvirtd:
/etc/conf.d/libvirtd
LIBVIRTD_ARGS="--listen"
测试
To test if libvirt is working properly on a system level:
$ virsh -c qemu:///system
To test if libvirt is working properly for a user-session:
$ virsh -c qemu:///session
管理
Libvirt management is done mostly with three tools: virt-manager (GUI), virsh, and guestfish (which is part of libguestfsAUR).
virsh
The virsh program is for managing guest domains (virtual machines) and works well for scripting, virtualization administration. Though most virsh commands require root privileges to run due to the communication channels used to talk to the hypervisor, typical management, creation, and running of domains (like that done with VirtualBox) can be done as a regular user.
Virsh includes an interactive terminal that can be entered if no commands are passed (options are allowed though): virsh.  The interactive terminal has support for tab completion.
From the command line:
$ virsh [option] <command> [argument]...
From the interactive terminal:
virsh # <command> [argument]...
Help is available:
$ virsh help [option*] or [group-keyword*]
儲存pool
A pool is a location where storage volumes can be kept. What libvirt defines as volumes others may define as "virtual disks" or "virtual machine images". Pool locations may be a directory, a network filesystem, or partition (this includes a LVM). Pools can be toggled active or inactive and allocated for space.
On the system-level, /var/lib/libvirt/images/ will be activated by default; on a user-session, virt-manager creates $HOME/VirtualMachines.
Print active and inactive storage pools:
$ virsh pool-list --all
用 virsh 新增存儲Pool
If wanted to add a storage pool, here are examples of the command form, adding a directory, and adding a LVM volume:
$ virsh pool-define-as name type [source-host] [source-path] [source-dev] [source-name] [<target>] [--source-format format] $ virsh pool-define-as poolname dir - - - - /home/username/.local/libvirt/images $ virsh pool-define-as poolname fs - - /dev/vg0/images - mntpoint
The above command defines the information for the pool, to build it:
$ virsh pool-build poolname $ virsh pool-start poolname $ virsh pool-autostart poolname
To remove it:
$ virsh pool-undefine poolname
- It is a good practice to dedicate a volume group to the storage pool only.
- Choose a LVM volume group that differs from the pool name, otherwise when the storage pool is deleted the LVM group will be too.
用 virt-manager 新增儲存Pool
First, connect to a hypervisor (e.g. QEMU/KVM system, or user-session). Then, right-click on a connection and select Details; select the Storage tab, push the + button on the lower-left, and follow the wizard.
存儲磁區
Once the pool has been created, volumes can be created inside the pool. If building a new domain (virtual machine), this step can be skipped as a volume can be created in the domain creation process.
用 virsh 新增磁區
Create volume, list volumes, resize, and delete:
$ virsh vol-create-as poolname volumename 10GiB $ virsh vol-list poolname $ virsh vol-resize --pool poolname volumename 12GiB $ virsh vol-delete --pool poolname volumename $ virsh vol-dumpxml --pool poolname volumename # for details.
virt-manager 後備儲存類型的 bug
On newer versions of virt-manager you can now specify a backing store to use when creating a new disk. This is very useful, in that you can have new domains be based on base images saving you both time and disk space when provisioning new virtual systems. There is a bug (https://bugzilla.redhat.com/show_bug.cgi?id=1235406) in the current version of virt-manager which causes virt-manager to choose the wrong type of the backing image in the case where the backing image is a qcow2 type. In this case, it will errantly pick the backing type as raw. This will cause the new image to be unable to read from the backing store, and effectively remove the utility of having a backing store at all.
There is a workaround for this issue. qemu-img has long been able to do this operation directly. If you wish to have a backing store for your new domain before this bug is fixed, you may use the following command.
$ qemu-img create -f qcow2 -o backing_file=<path to backing image>,backing_fmt=qcow2 <disk name> <disk size>
Then you can use this image as the base for your new domain and it will use the backing store as a COW volume saving you time and disk space.
虛擬機
Virtual machines are called domains.  If working from the command line, use virsh to list, create, pause, shutdown domains, etc.  virt-viewer can be used to view domains started with virsh.  Creation of domains is typically done either graphically with virt-manager or with virt-install (a command line program that is part of the virt-manager package).
Creating a new domain typically involves using some installation media, such as an .iso from the storage pool or an optical drive.
Print active and inactive domains:
# virsh list --all
/var/lib/libvirt/images/ to be accessed.  If using SELinux and there are issues with the volumes, ensure that volumes are in that directory, or ensure that other storage pools are correctly labeled.用 virt-install 新增虛擬機器
For an extremely detailed domain (virtual machine) setup, it is easier to #用 virt-manager 新增虛擬機器.  However, basics can easily be done with virt-install and still run quite well.  Minimum specifications are --name, --memory, guest storage (--disk, --filesystem, or --nodisks), and an install method (generally an .iso or CD).
Arch Linux install (two GiB, raw format volume create; user-networking):
$ virt-install \ --name arch-linux_testing \ --memory 1024 \ --vcpus=2,maxvcpus=4 \ --cpu host \ --cdrom $HOME/Downloads/arch-linux_install.iso \ --disk size=2,format=raw \ --network user \ --virt-type kvm
Fedora testing (Xen hypervisor, non-default pool, do not originally view):
$ virt-install \ --connect xen:/// \ --name fedora-testing \ --memory 2048 \ --vcpus=2 \ --cpu=host \ --cdrom /tmp/fedora20_x84-64.iso \ --os-type=linux --os-variant=fedora20 \ --disk pool=testing,size=4 \ --network bridge=br0 \ --graphics=vnc \ --noautoconsole $ virt-viewer --connect xen:/// fedora-testing
Windows:
$ virt-install \ --name=windows7 \ --memory 2048 \ --cdrom /dev/sr0 \ --os-variant=win7 \ --disk /mnt/storage/domains/windows7.qcow2,size=20GiB \ --network network=vm-net \ --graphics spice
osinfo-query --fields=name,version os to get argument for --os-variant; this will help define some specifications for the domain.  However, --memory and --disk will need to be entered; one can look within the appropriate /usr/share/libosinfo/db/oses/os.xml if needing these specifications.  After installing, it will likely be preferable to install the Spice Guest Tools that include the VirtIO drivers. For a Windows VirtIO network driver there is also virtio-winAUR. These drivers are referenced by a <model type='virtio' /> in the guest's .xml configuration section for the device. A bit more information can also be found on the QEMU article.Import existing volume:
$ virt-install \ --name demo \ --memory 512 \ --disk /home/user/VMs/mydisk.img \ --import
用 virt-manager 新增虛擬機器
首先,連線到虛擬機的Hypervisor(例如 QEMU/KVM system 或使用這 session,在“連線”上右鍵按一下並選擇 新增,然後跟隨 新增虛擬機器精靈 完成。
- 在第五步中打開進階選項並確認虛擬化類型设为 KVM(這通常是首選模式)。如果要求附加的硬體配置,選中安裝前定制選項。
管理虛擬機
啟動虛擬機器:
$ virsh start domain $ virt-viewer --connect qemu:///session domain
Gracefully attempt to shutdown a domain; force off a domain:
$ virsh shutdown domain $ virsh destroy domain
Autostart domain on libvirtd start:
$ virsh autostart domain $ virsh autostart domain --disable
Shutdown domain on host shutdown:
-  Running domains can be automatically suspended/shutdown at host shutdown using the libvirt-guests.servicesystemd service. This same service will resume/startup the suspended/shutdown domain automatically at host startup. Read/etc/conf.d/libvirt-guestsfor service options.
Edit a domain's XML configuration:
$ virsh edit domain
網路
此處是有有關libvirt 網路的一个正宗的概述。
預設情況下,當 libvird 服務啟動後,即創建了一个名為 default 的 NAT 網橋与外部網路聯通(警告:參閱 #"default" 網路的 bug[broken link: invalid section])。對於其他的網路連線需求,可創建下列4種類型的網路讓虛擬機連線到網路:
-  bridge — 這是一個虛擬裝置,它通過一個物理介面直接共享網路資料。使用場合為:宿主機有 靜態 網路、虛擬機不需與其它虛擬機連接、虛擬機要佔用全部進出流量,並且虛擬機執行於系統 層級。有關如何在現有預設網橋時增加另一個網橋的方法,參閱 網橋。網橋創建後,需要將它指定到相應客戶機的 .xml設定檔中。
- network — 這是一個虛擬網路,它可以與其它虛擬機共用。使用場景为:宿主機有 動態 網路(例如:NetworkManager)或使用無線網路。
- macvtap — 直接連線到宿主機的一個物理網路介面。
- user — 本地網路,僅用於使用者 工作階段。
絕大多數用戶都可以通過 virsh 的各種可選項創建具有各種功能的網路,一般來說比通過 GUI 程式(像 virt-manager 等)更容易做到。也可以按 #用 virt-install 新增虛擬機[broken link: invalid section] 實現。
快照
Snapshots take the disk, memory, and device state of a domain at a point-of-time, and save it for future use. They have many uses, from saving a "clean" copy of an OS image to saving a domain's state before a potentially destructive operation. Snapshots are identified with a unique name.
Snapshots are saved within the volume itself and the volume must be the format: qcow2 or raw. Snapshots use deltas so they have the potentiality to not take much space.
新增快照
Once a snapshot is taken it is saved as a new block device and the original snapshot is taken offline. Snapshots can be chosen from and also merged into another (even without shutting down the domain).
Print a running domain's volumes (running domains can be printed with virsh list):
# virsh domblklist domain
Target Source ------------------------------------------------ vda /vms/domain.img
To see a volume's physical properties:
# qemu-img info /vms/domain.img
image: /vms/domain.img file format: qcow2 virtual size: 50G (53687091200 bytes) disk size: 2.1G cluster_size: 65536
Create a disk-only snapshot (the option --atomic will prevent the volume from being modified if snapshot creation fails):
# virsh snapshot-create-as domain snapshot1 --disk-only --atomic
List snapshots:
# virsh snapshot-list domain
Name Creation Time State ------------------------------------------------------------ snapshot1 2012-10-21 17:12:57 -0700 disk-snapshot
One can they copy the original image with cp --sparse=true or rsync -S and then merge the the original back into snapshot:
# virsh blockpull --domain domain --path /vms/domain.snapshot1
domain.snapshot1 becomes a new volume.  After this is done the original volume (domain.img and snapshot metadata can be deleted.   The virsh blockcommit would work opposite to blockpull but it seems to be currently under development (including snapshot-revert feature, scheduled to be released sometime next year.
其他管理操作
Connect to non-default hypervisor:
$ virsh --connect xen:/// virsh # uri xen:///
Connect to the QEMU hypervisor over SSH; and the same with logging:
$ virsh --connect qemu+ssh://username@host/system $ LIBVIRT_DEBUG=1 virsh --connect qemu+ssh://username@host/system
Connect a graphic console over SSH:
$ virt-viewer --connect qemu+ssh://username@host/system domain $ virt-manager --connect qemu+ssh://username@host/system domain
Connect to the VirtualBox hypervisor (VirtualBox support in libvirt is not stable yet and may cause libvirtd to crash):
$ virsh --connect vbox:///system
Network configurations:
$ virsh -c qemu:///system net-list --all $ virsh -c qemu:///system net-dumpxml default
Python 連接代碼
The libvirt-python package provides a python2 API in /usr/lib/python2.7/site-packages/libvirt.py.
General examples are given in /usr/share/doc/libvirt-python-your_libvirt_version/examples/
Unofficial example using qemu and openssh:
#! /usr/bin/env python2
# -*- coding: utf-8 -*-
import socket
import sys
import libvirt
if (__name__ == "__main__"):
   conn = libvirt.open("qemu+ssh://xxx/system")
   print "Trying to find node on xxx"
   domains = conn.listDomainsID()
   for domainID in domains:
       domConnect = conn.lookupByID(domainID)
       if domConnect.name() == 'xxx-node':
           print "Found shared node on xxx with ID " + str(domainID)
           domServ = domConnect
           break
