vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

生活百科10个月前发布 aixure
47 0 0

【作者】付广平,任职某银行云技术管理中心,负责云计算相关技术研究。毕业于北京邮电大学,从2013开始从事OpenStack相关工作,参与了OpenStack Nova、Cinder、Oslo等项目社区开发。

目前裸机(物理机)、虚拟机、容器是云计算提供计算服务的三种主流形式。那么如何判断一个虚拟shell环境到底是物理机、虚拟机还是容器呢?

更进一步,如果是物理机,这个物理机厂商是什么,虚拟机到底是KVM还是XEN,容器是Docker还是rkt、lxc等?

更进一步,如果是虚拟机,是否可以判断这个虚拟机是运行在AWS还是阿里或者OpenStack,是否能够获取虚拟机的UUID、instance-type、vpc-id、安全组等信息?

这有点像我们在开发中经常使用的反射(reflection)机制,通过反射可以知道一个类实例(instance)的类(class)是什么,更进一步可以知道这个类的父类是什么、实现了哪些方法、包含哪些属性等。

以下是我用到的一些方法,仅供参考。

一、判断容器

目前还没有什么方法能够100%准确判断虚拟环境是否是容器,至少我没有找到相关文献。

如果环境有systemd-detect-virt命令,则可以直接通过systemd-detect-virt -c命令判断,如果输出为none则不是容器,否则会输出容器类型,比如lxc。目前很少容器里面放systemd的,我见过的就只有LXD的ubuntu镜像,因此这种方法适用性不广。

除此之外,可通过其他tricks判断,最简便的方法判断PID为1的进程,如果该进程就是应用进程则判断是容器,而如果是init进程或者systemd进程,则不一定是容器,当然不能排除是容器的情况,比如LXD/lXC实例的进程就为/sbin/init。

容器和虚拟机不一样的是,容器和宿主机是共享内核的,因此理论上容器内部是没有内核文件的,除非挂载了宿主机的/boot目录:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

另外,我们知道容器是通过cgroup实现资源限制,每个容器都会放到一个cgroup组中,如果是Docker,则cgroup的名称为docker-xxxx,其中xxxx为Docker容器的UUID。

而控制容器的资源,本质就是控制运行在容器内部的进程资源,因此我们可以通过查看容器内部进程为1的cgroup名称获取线索。

如下是我通过Docker跑busybox的cgroup信息:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

我们不仅可以知道这是Docker容器,还获取了Docker容器的UUID为9ba…11。

根据如上的结论,判断一个虚拟环境是否Docker的脚本为:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

当然如果仅仅判断是否Docker容器,还能通过判断是否存在.dockerenv文件区分是否Docker容器:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

rkt容器类似,输出结果如下:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

如上的x2d为-号:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

因此判断一个虚拟环境是否rkt的脚本为:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

好奇AWS lambda的运行环境是什么,于是写了个函数输出/proc/1/cgroup,结果为:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

猜测是一种叫sandbox的运行环境,估计也是一种容器。

判断虚拟环境是否为容器环境相对比较复杂,目前没有完美的方案,总结过程如下:

另外,需要特别注意的是,容器必须最先判断vmware查看虚拟机ip,因为容器本身并没有任何的硬件虚拟化,容器看到的硬件特性信息和宿主机看到的完全一样,因此下面介绍的通过lscpu以及DMI信息判断是否是虚拟机或者物理机,对容器并不适用。换句话说,不能因为lscpu的Hypervisor vendor值为KVM就说明一定是KVM虚拟机,因为它也有可能是容器。下文均假设已经排除为容器的情况。

二、判断物理机

如果使用了systemd,则可以直接通过systemd-detect-virt命令判断是否物理机:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

如果输出为none,则说明是物理机。

当然也可根据lscpu命令输出,看是否有Hypervisor vendor属性,如果没有该属性,则一般为物理机,如果存在该属性则一定是虚拟机:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

获取物理机的信息最直接的方式是查看DMI信息/sys/firmware/dmi/tables/DMI,使用dmidecode命令解码:

如上可以看出这是台物理机,以及厂商、型号、序列号。

通过ipmitool命令可以查看物理服务器的带外IP:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

当然如果是虚拟机,如上命令会执行失败。另外也可以通过其他命令查看物理信息,如lshw命令。

三、判断虚拟机

其实前面已经提到了,如果使用了systemd,则可以直接通过systemd-detect-virt命令判断是否虚拟机:

如果是虚拟机,则会输出虚拟机类型vmware查看虚拟机ip,如kvm、oracle(virtualbox)、xen等。

当然也可根据lscpu命令输出,查看Hypervisor vendor属性值:

通过如上命令,我的一台AWS虚拟机输出为Xen,阿里云虚拟机为KVM,VirtualBox虚拟机也输出为KVM,这是因为我使用了KVM硬件加速虚拟化。

我的搬瓦工虚拟机输出也为KVM,可见搬瓦工主机也是KVM虚拟机。

通过如上方法可以获取虚拟机的虚拟化类型,能否获取更多信息呢?参考物理机的获取方式,我们可以通过dmidecode命令获取更多的虚拟机信息。比如我在一台OpenStack虚拟机运行如下命令:

如上Manufacturer为OpenStack Foundation,说明运行在OpenStack平台,Version为Nova版本,根据OpenStack的releases可知15.0.1对应为OpenStack Ocata版本,而UUID即虚拟机的UUID。

AWS上的一台虚拟机输出为:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

在Version中标明了amazon字样。

阿里云虚拟机如下(感谢L神提供的输出):

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

可见虽然可以从system信息中获取云厂商的线索,但其实虚拟机的system信息并没有统一的标准,有的在version中体现,有的在Product Name中表现,完全取决于云厂商自己的配置。

如上整合如下脚本初略判断:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

如上也可以判断公有云是否基于OpenStack实现。

AWS以及OpenStack系的虚拟机还可以通过metadata或者ConfigDrive获取更多信息,以metadata为例:

获取虚拟机的ID:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

获取instance type(规格):

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

获取虚拟机的公有IP(弹性IP),这个挺有用的,因为在虚拟机没法通过ifconfig查看弹性IP,经常登录虚拟机后,忘记自己的公有IP:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

其他的比如vpc-id、ami id(镜像id)、安全组、公钥名等都可以通过该方式获取。

如果是OpenStack,还可以使用OpenStack的metadata获取更多信息:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

如上可获取虚拟机的租户ID、volume type等信息。

四、总结

如上总结了几种判断虚拟化环境类型的方法,不一定准确,仅供参考,当然也可能还有其他更好的方法。

如下是根据前面的结论写的一个探测虚拟化类型的脚本,不一定健壮完备,仅供参考:

vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶 vmware查看虚拟机ip-如何探测虚拟环境是物理机、虚拟机还是容器? | 运维进阶

限时特惠:本站每日持续更新海量各大内部网赚创业教程,会员可以下载全站资源点击查看详情
站长微信:

© 版权声明

相关文章

暂无评论

暂无评论...