作者微信:15013593099 欢迎交流
libvirt可以屏蔽不同虚拟化技术的差异,对不同类型的虚拟机进行管理。本文拟用libvirt技术实现一个虚拟机监控中间键,对虚拟机的监控指标主要分为计算资源,内存资源,i\o资源,网络资源。
(1)获得虚拟机hypervisor连接
在对虚拟机进行操作之前,首先需要得到于虚拟机hypervisor的连接,其实libvirt编程借口就是hypervisor向外提供的接口。不同的虚拟华技术向外提供的接口名称,接口形式可能不同,但是向外提供的功能和信息都大同小异。libvirt只不过把这些接口封装起来,在向外提供一个统一的API,这样就实现了对hypervisor差异的屏蔽。
所以在开始使用libvirt API提供的方法之前,先要跟虚拟机的hypercisor获得连接,说白了就是告诉libvirt我们要操作的虚拟机用的是什么虚拟化技术。
- virConnectPtr conn;
- conn = virConnectOpen("xen:///"); //这里是Xen这种虚拟机的动,如果要用别的虚机类型,就把这里换掉
- if (conn == NULL) {
- fprintf(stderr, "Failed to open connection to qemu:///system\n");
- return 1;
- }
至此,我们就得到了虚拟机的连接conn.
(2) 服务器内虚拟机的数量
在服务器内部,虚拟机分为正在运行的和没有运行的两类,分别叫ActiveDom和InactiveDom;一个虚拟机正常创建之后,就处在active状态,当对他执行shut down之后,虚拟机就处在Inactive状态;对处于Inactive状态的虚拟机执行Run之后,他就处于active状态了,需要注意的是:对activeDom执行pause之后,这个虚拟机仍然处于active状态。
1.查看activeDom的数量使用如下方法:
- numOfActiveDom = virConnectNumOfDomains(conn); //numOfActiveDom就是活动虚拟机的数量
2.获得active虚拟机的编号:
- int activeDom[numOfActiveDom];
- numOfActiveDom = virConnectListDomains(conn , activeDom , numOfActiveDom);
- //数组activeDom里存放的就是所有活动虚拟机的id。
3.通过这些id就可以拿到相关虚拟机的实体
- virDomainPtr *allDomains; //allDomain是用来装所有活动虚拟机实体的数组
- allDomains = malloc(sizeof(virDomainPtr)* numOfactiveDom);
- //困惑1:由于virDomain是未公开的结构体,所以 sizeof(virDomain) 不 可使用,因此试了sizeof(virDomainPtr),尽然也可以。
- printf("活动的Dom:\n");
- for(i=0 ; i<numOfActiveDom ; i++){
- printf("id:%d\n",activeDom[i]);
- allDomains[numOfDom] = virDomainLookupByID(conn ,activeDom[i]); //通过id取Domain
- } //至此,所有的活动虚拟机实力已经得到。
4.获得inactiveDom实体的方法类似。只不过由于inactiveDom没有id,所以需要通过名字得到实体。
- char **inactiveDom;
- virDomainPtr *allDomains;
- numOfInactiveDom = virConnectNumOfDefinedDomains(conn);
- allDomains = malloc(sizeof(virDomainPtr)* numOfactiveDom);
- if (numOfInactiveDom != 0){
- printf("Inactive的数量是:%d\n",numOfInactiveDom);
- inactiveDom = malloc(sizeof(char*) * (numOfInactiveDom));
- numOfInactiveDom = virConnectListDefinedDomains(conn , inactiveDom , numOfInactiveDom);
- printf("vir之后的numOfInactiveDom*************:%d\n", numOfInactiveDom);
- printf("\n不活动的Dom:\n");
- for(i=0 ; i<numOfInactiveDom ; i++){
- printf("name:%s\n",inactiveDom[i]);
- allDomains[numOfDom] = virDomainLookupByName(conn , inactiveDom[i]);//通过name取domain
- numOfDom++;
- }
关于XML:
在libvirt中,虚拟机的信息是以XML的形式存储的,我们可以用方法来获取一个虚拟机(也就是一个Domain)的XML文件
- char *caps;
- caps = malloc(sizeof(char)*1024);
- caps = virDomainGetXMLDesc(allDomains[numOfDom],4);
- free(caps);
使用
fprintf(stdout, "Capabilities:\n%s\n",caps);
把caps输出如下:
<domain type='xen' id='1'> //虚拟机类型于虚拟机id
<name>test1</name>
<uuid>3eae3e71-a7c2-2bb0-035b-a6e1011ddc24</uuid>
<memory>556032</memory>
<currentMemory>524288</currentMemory>
<vcpu>1</vcpu>
<os>
<type>hvm</type>
<loader>/usr/lib/xen/boot/hvmloader</loader>
<boot dev='hd'/>
</os>
<features>
<acpi/>
<apic/>
<pae/>
</features>
<clock offset='utc'/>
<on_poweroff>destroy</on_poweroff>
<on_reboot>restart</on_reboot>
<on_crash>restart</on_crash>
<devices>
<emulator>/usr/lib/xen/bin/qemu-dm</emulator>
<disk type='file' device='disk'>
<driver name='file'/>
<source file='/var/lib/xen/images/test1.img'/>
<target dev='hda' bus='ide'/>
</disk>
<interface type='bridge'>
<mac address='00:16:3e:74:03:53'/>
<source bridge='xenbr0'/>
<script path='vif-bridge'/>
<target dev='vif1.0'/>
</interface>
<serial type='pty'>
<source path='/dev/pts/2'/>
<target port='0'/>
</serial>
<console type='pty' tty='/dev/pts/2'>
<source path='/dev/pts/2'/>
<target port='0'/>
</console>
<input type='mouse' bus='ps2'/>
<graphics type='vnc' port='5901' autoport='yes' keymap='en-us'/>
</devices>
</domain>
device件值对里装的是I/O信息,Interface里装的是网络信息
(一)计算资源与内存资源的监控libvirt中提供virDomainGetInfo方法可以将一个domain的计算资源和内存资源的使用情况封装在一个结构体中,如下:
- struct virDomainInfo{
- unsigned char state : //the running state, one of virDomainState
- unsigned long maxMem :// the maximum memory in KBytes allowed
- unsigned long memory :// the memory in KBytes used by the domain
- unsigned short nrVirtCpu :// the number of virtual CPUs for the domain
- unsigned long long cpuTime : //the CPU time used in nanoseconds
- }
获取内存资源和计算资源监控信息并打印的方法如下:
- virDomainInfoPtr domaininfo;
- domaininfo = malloc(sizeof(virDomainInfo)* 1);
- m = virDomainGetInfo(allDomains[numOfDom] , domaininfo);
- printf("状态:%d ",(*domaininfo).state);
- printf("最大内存:%ld ",((*domaininfo).maxMem)/1024);
- printf("内存:%ld ",((*domaininfo).memory)/1024);
- printf("虚拟cpu个数:%d ",(*domaininfo).nrVirtCpu);
- printf("cpu时间:%lld ",(*domaininfo).cpuTime);
(二)获得网络流量信息
可以使用 virDomainInterfaceStats方法,将一个domain的网络流量信息封装在一个virDomainInterfaceStatsStruct结构体中,该方法需要传递一个虚拟网卡接口的参数,这个参数可以通过XML文件取得:
<interface type='bridge'>
<mac address='00:16:3e:74:03:53'/>
<source bridge='xenbr0'/>
<script path='vif-bridge'/>
<target dev='vif1.0 '/> /*这个就是需要的参数*/
</interface>
具体代码如下:
- virDomainInterfaceStatsPtr interfacestats;
- interfacestats = malloc(sizeof(virDomainInterfaceStatsStruct)* 10);
- m = virDomainInterfaceStats(allDomains[numOfDom], i_path, interfacestats, 10);
- printf("rx字节数:%ld ",(*interfacestats).rx_bytes);
- printf("tx字节数:%ld ",(*interfacestats).tx_bytes);
- printf("rx丢包:%ld ",(*interfacestats).rx_drop);
- printf("tx丢包:%ld\n",(*interfacestats).tx_drop);
整体运行结果如下图,具体代码见附件: