文章目录
- 简介
- guestfish
- libguestfs项目
- 例子
- 原理
- 代码
- libguestfs架构
- 参考
简介
guestfish
Guestfish 是libguestfs项目中的一个工具软件,提供修改虚机镜像内部配置的功能。它不需要把虚机镜像挂接到本地,而是为你提供一个shell接口,你可以查看、编辑和删除镜像内的文件。
Guestfish提供了结构化的libguestfs API访问,可以通过shell脚本、命令行或交互方式访问。它使用libguestfs并公开了guestfs API的所有功能。Libguestfs是一个用于访问和修改磁盘映像和虚拟机的库。
libguestfs项目
从官网的描述看,libguestfs项目是一些用来操作虚拟机镜像的工具和库的集合,从描述看在镜像管理方面很强大。
例子
一个简单查看镜像文件系统的例子:
Welcome to guestfish, the guest filesystem shell for
editing virtual machine filesystems and disk images.
Type: ‘help’ for help on commands
‘man’ to read the manual
‘quit’ to quit the shell
><fs> help
Add disk images to examine using the ‘-a’ or ‘-d’ options, or the ‘add’
command.
Or create a new disk image using ‘-N’, or the ‘alloc’ or ‘sparse’ commands.
Once you have done this, use the ‘run’ command.
For more information about a command, use ‘help cmd’.
To read the manual, type ‘man’.
><fs> add linux.qcow2
><fs> run
100% ⟦▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒⟧ --:--
><fs> list-filesystems
/dev/sda1: vfat
/dev/sda2: ext4
/dev/sda3: swap
><fs>
可以看出,通过guestfish可以列出镜像里面的文件系统。
原理
在运行guestfish查看镜像的过程中,后台ps查看下qemu进程,可以找到这样一个进程:
/usr/bin/qemu-system-x86_64 -global virtio-blk-pci.scsi=off -no-user-config -nodefaults -display none -machine accel=kvm:tcg -cpu max,la57=off -m 1280 -no-reboot -rtc driftfix=slew -no-hpet -global kvm-pit.lost_tick_policy=discard -kernel /var/tmp/.guestfs-1000/appliance.d/kernel -initrd /var/tmp/.guestfs-1000/appliance.d/initrd -object rng-random,filename=/dev/urandom,id=rng0 -device virtio-rng-pci,rng=rng0 -device virtio-scsi-pci,id=scsi -drive file=/home/czw/vm/img/linux.qcow2,cache=writeback,id=hd0,if=none -device scsi-hd,drive=hd0 -drive file=/var/tmp/.guestfs-1000/appliance.d/root,snapshot=on,id=appliance,cache=unsafe,if=none,format=raw -device scsi-hd,drive=appliance -device virtio-serial-pci -serial stdio -chardev socket,path=/run/user/1000/libguestfsdU5W8O/guestfsd.sock,id=channel0 -device virtserialport,chardev=channel0,name=org.libguestfs.channel.0 -append panic=1 console=ttyS0 edd=off udevtimeout=6000 udev.event-timeout=6000 no_timer_check printk.time=1 cgroup_disable=memory usbcore.nousb cryptomgr.notests tsc=reliable 8250.nr_uarts=1 root=UUID=3c1ee3f6-e2da-41b9-ac29-ca925bafb519 selinux=0 quiet TERM=xterm-256color
这个进程让guestfish有了能够操作镜像的能力,分析这个qemu进程的参数:
- 指定虚机kernel为guestfish自己的
- 指定initrd为guestfish自己的
- 指定镜像文件为所操作的镜像文件
- 关联一个本地socket文件:/run/user/1000/libguestfsdU5W8O/guestfsd.sock
进一步查看这个本地socket文件谁在使用:
lsof /run/user/1000/libguestfsdU5W8O/guestfsd.sock
COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME
guestfish 18535 czw 6u unix 0x00000000649fd22e 0t0 192221 /run/user/1000/libguestfsdU5W8O/guestfsd.sock type=STREAM (CONNECTED)
至此,可以大致猜测出guestfish的实现原理:
- 使用自己定制的内核启动虚拟机并将待操作的磁盘文件分配给虚拟机,这里称该虚拟机为guest
- 通过本地socket文件,guestfish进程可以与guest中的某个程序通信,从而通过guest中的程序对磁盘文件进行操作
- 当guestfish退出的时候,自动销毁guest
代码
为了验证上面实验的猜想,可以直接下载源代码,guestfish是libguestfs项目的一部分,直接clone整个项目:
git clone https://github.com/libguestfs/libguestfs
cd libguestfs
git submodule update --init
autoreconf -i
./configure CFLAGS=-fPIC
make
如果是debian/ubutnu的环境,参考https://manpages.ubuntu.com/manpages/noble/ja/man1/guestfs-building.1.html,安装编译依赖。
编译完,在fish/.lib
文件夹下就是gestfish可执行程序,他的代码路径为:fish/fish.c
简单看下代码,核心是调用libguestfs库提供的guestfs_
开头的函数,详细的流程直接看代码就好了。
libguestfs架构
直接展示官方的架构图:
大致思路就是:libguestfs通过rpc与qemu虚机内部的guestfsd通信,将对镜像的操作转换为通过guestfsd间接操作镜像。
参考
guestfish - the guest filesystem shell
libguestfs
guestfs-internals - architecture and internals of libguestfs