是否可以使用 python 获取知道其序列号的 pendrive 的安装点?
Is it possible to get the mountpoint of a pendrive knowing its serialnumber using python?
我有 2 台 PC(一台 Linux 和一台 Windows)连接到位于不同楼层的本地网络。那个楼层的人将他们的 USB 笔式驱动器连接到其中一台 PC,我想将不同的特定文件集复制给不同的人。
之前,
- 我做的太难了(到地板上手动做)
- 后来我写了一个 python 程序,将特定的文件集复制到
具体的人对我的决定通过
ssh
。 (即,我登录到
具体机器通过 ssh
,要求用户 (通过 phone 电话) 一一插入他们的
笔式驱动器,然后我执行接受一个 python 程序
争论。这个论点不过是我想复制的名字,
并通过接收参数,程序决定哪些文件是
复制到笔式驱动器)。
过程还是有点乏味,..
因为只能连接一个笔式驱动器,所以我必须为每个用户重复执行此操作。
因此,为了减少消耗的总时间,我在两个系统上都连接了 USB 集线器,以便在给定时间内多次插入笔式驱动器。问题来了,决定哪个设备属于谁。
问题 : 是否可以使用 python 从 SerialNumber
找到笔式驱动器的挂载点? (要是python就好了,因为主程序是用python写的)
我考虑 SerialNumber
结束的原因,
UUID
- 它在设备格式化时改变
Vendor
、ProdID
和 Manufacturer
- 不确定它们是否会
不同的。 (即,如果它来自同一个制造商和同一个
型号)
我尝试 wmi
windows.. 并从 SO 那里得到了这段代码,(抱歉,我没有 link。很久以前就用过了)
import win32com.client
wmi = win32com.client.GetObject ("winmgmts:")
for usb in wmi.InstancesOf ("Win32_USBHub"):
print usb.DeviceID
我得到的输出是
USB\VID_5986&PID_0292&4817B6D&0&6
USB\VID_8087&PID_0024&55D1EEC&0&1
USB\VID_8087&PID_0024&88B8ABA&0&1
USB\ROOT_HUB20&11F77F7&0
USB\ROOT_HUB20&62BF53D&0
USB\VID_03F0&PID_3307\JN0W5LAB0ZHQ5VK8
它与linux中的情况类似,我能得到的只是序列号,使用usb-devices
。但是无法获取对应的挂载点
任何想法请...
要在 Linux 上执行此操作,您需要解析 /proc/mounts
以确定设备名称到挂载点的映射,即 /dev/sdc2
- > /var/run/media/myaut/hyperx
。
诀窍是找出需要序列号的设备名称。最简单的方法是使用 udev
- 它在 /dev/disk/by-id
:
中生成符号链接时使用串行
/dev/disk/by-id/usb-Generic_Flash_Disk_12345678-0:0 -> ../../sdd
但我们并没有寻求最简单的解决方案,对吗?诀窍是 udev
规则可能会改变,而 sysfs
(来自内核)更可靠。我实现了一个执行此操作的脚本:
import os
import sys
import glob
SYS_USB_DEVICES = '/sys/bus/usb/devices'
SYS_BLOCK_DEVICES = '/sys/class/block'
try:
serial = sys.argv[1]
except IndexError:
print >> sys.stderr, "Usage: findflash.py SERIAL"
sys.exit(1)
# PASS 1 Find USB node with corresponding to serial
for usbid in os.listdir(SYS_USB_DEVICES):
usbserpath = os.path.join(SYS_USB_DEVICES, usbid, 'serial')
if not os.path.exists(usbserpath):
continue
with open(usbserpath) as f:
usb_serial = f.read().strip()
if serial == usb_serial:
# Found it!
break
else:
print >> sys.stderr, "Cannot find usb device with serial {0}".format(serial)
sys.exit(1)
# Find SCSI ids corresponding to this device
# I didn't check SYSFS documentation, but tested it on openSUSE 13.1
# The form of path is:
# <SUBDEVICE>/host<SCSI_HOST_ID>/target<SCSI_TARGET_ID>/<CTRL><CHANNEL>:<TGT>:<LUN>
# We need only basename
devs = glob.glob(os.path.join(SYS_USB_DEVICES, usbid,
'*/host*/target*/*:*:*:*'))
devs = map(os.path.basename, devs)
# PASS 2 - find mountpoints for devices with SCSI ids we discover earlier
# Parse mountpoint formatted as "/dev/... /path/to/mntpt ..."
def parse_mntpt(line):
dev, mntpt, _ = line.split(None, 2)
dev = os.path.basename(dev)
return dev, mntpt
mntpts = {}
with open('/proc/mounts') as f:
mntpts = dict(map(parse_mntpt, f.readlines()))
# List of ('scsi id', 'dev name', 'mnt pt (if exists)')
devlist = []
def create_dev(scsiid, devname):
global mntpts
devlist.append((scsiid, devname, mntpts.get(devname)))
for devname in os.listdir(SYS_BLOCK_DEVICES):
devpath = os.path.join(SYS_BLOCK_DEVICES, devname)
devlink = os.path.join(devpath, 'device')
# Node is "virtual", i.e. partition, ignore it
if not os.path.islink(devlink):
continue
scsiid = os.path.basename(os.readlink(devlink))
if scsiid not in devs:
continue
create_dev(scsiid, devname)
# Find partition names
parts = glob.glob(os.path.join(devpath, '*/partition'))
for partpath in parts:
partname = os.path.basename(os.path.dirname(partpath))
create_dev(scsiid, partname)
# END - print results
fmtstr = '{0:8} {1:5} {2}'
print fmtstr.format('SCSI ID', 'DEV', 'MOUNT POINT')
for scsiid, devname, mntpt in devlist:
print fmtstr.format(scsiid, devname, mntpt)
这是示例输出:
$ python findflash.py 12345678
SCSI ID DEV MOUNT POINT
8:0:0:0 sdd None
8:0:0:0 sdd1 /var/run/media/myaut/Debian0wheezy020140723-17:30
8:0:0:0 sdd2 None
8:0:0:0 sdd5 None
8:0:0:1 sr0 None
我不能说 Windows 这很容易。我有一个代码(在 C/WinAPI 中)能够从系统中收集所有磁盘设备,但它的逻辑与文件系统表示相去甚远,所以我仍然没有找到解决方案。
Windows 的复杂性来自于:
- 有一组函数,如 SetupDi*,可让您枚举磁盘设备。他们的名字以 PnP 风格出现,与卷名无关。
- 有 DOS 风格 API(即在 A: 和 C: 级别工作)
- WinNT 风格 API 知道卷,可能还有挂载点。
当然这三层之间的联系并不明显(有方法可以通过 size/offset 来匹配分区,但这太疯狂了)。我仍然害怕在我的库中实现它:(
我有 2 台 PC(一台 Linux 和一台 Windows)连接到位于不同楼层的本地网络。那个楼层的人将他们的 USB 笔式驱动器连接到其中一台 PC,我想将不同的特定文件集复制给不同的人。
之前,
- 我做的太难了(到地板上手动做)
- 后来我写了一个 python 程序,将特定的文件集复制到
具体的人对我的决定通过
ssh
。 (即,我登录到 具体机器通过ssh
,要求用户 (通过 phone 电话) 一一插入他们的 笔式驱动器,然后我执行接受一个 python 程序 争论。这个论点不过是我想复制的名字, 并通过接收参数,程序决定哪些文件是 复制到笔式驱动器)。
过程还是有点乏味,.. 因为只能连接一个笔式驱动器,所以我必须为每个用户重复执行此操作。
因此,为了减少消耗的总时间,我在两个系统上都连接了 USB 集线器,以便在给定时间内多次插入笔式驱动器。问题来了,决定哪个设备属于谁。
问题 : 是否可以使用 python 从 SerialNumber
找到笔式驱动器的挂载点? (要是python就好了,因为主程序是用python写的)
我考虑 SerialNumber
结束的原因,
UUID
- 它在设备格式化时改变Vendor
、ProdID
和Manufacturer
- 不确定它们是否会 不同的。 (即,如果它来自同一个制造商和同一个 型号)
我尝试 wmi
windows.. 并从 SO 那里得到了这段代码,(抱歉,我没有 link。很久以前就用过了)
import win32com.client
wmi = win32com.client.GetObject ("winmgmts:")
for usb in wmi.InstancesOf ("Win32_USBHub"):
print usb.DeviceID
我得到的输出是
USB\VID_5986&PID_0292&4817B6D&0&6
USB\VID_8087&PID_0024&55D1EEC&0&1
USB\VID_8087&PID_0024&88B8ABA&0&1
USB\ROOT_HUB20&11F77F7&0
USB\ROOT_HUB20&62BF53D&0
USB\VID_03F0&PID_3307\JN0W5LAB0ZHQ5VK8
它与linux中的情况类似,我能得到的只是序列号,使用usb-devices
。但是无法获取对应的挂载点
任何想法请...
要在 Linux 上执行此操作,您需要解析 /proc/mounts
以确定设备名称到挂载点的映射,即 /dev/sdc2
- > /var/run/media/myaut/hyperx
。
诀窍是找出需要序列号的设备名称。最简单的方法是使用 udev
- 它在 /dev/disk/by-id
:
/dev/disk/by-id/usb-Generic_Flash_Disk_12345678-0:0 -> ../../sdd
但我们并没有寻求最简单的解决方案,对吗?诀窍是 udev
规则可能会改变,而 sysfs
(来自内核)更可靠。我实现了一个执行此操作的脚本:
import os
import sys
import glob
SYS_USB_DEVICES = '/sys/bus/usb/devices'
SYS_BLOCK_DEVICES = '/sys/class/block'
try:
serial = sys.argv[1]
except IndexError:
print >> sys.stderr, "Usage: findflash.py SERIAL"
sys.exit(1)
# PASS 1 Find USB node with corresponding to serial
for usbid in os.listdir(SYS_USB_DEVICES):
usbserpath = os.path.join(SYS_USB_DEVICES, usbid, 'serial')
if not os.path.exists(usbserpath):
continue
with open(usbserpath) as f:
usb_serial = f.read().strip()
if serial == usb_serial:
# Found it!
break
else:
print >> sys.stderr, "Cannot find usb device with serial {0}".format(serial)
sys.exit(1)
# Find SCSI ids corresponding to this device
# I didn't check SYSFS documentation, but tested it on openSUSE 13.1
# The form of path is:
# <SUBDEVICE>/host<SCSI_HOST_ID>/target<SCSI_TARGET_ID>/<CTRL><CHANNEL>:<TGT>:<LUN>
# We need only basename
devs = glob.glob(os.path.join(SYS_USB_DEVICES, usbid,
'*/host*/target*/*:*:*:*'))
devs = map(os.path.basename, devs)
# PASS 2 - find mountpoints for devices with SCSI ids we discover earlier
# Parse mountpoint formatted as "/dev/... /path/to/mntpt ..."
def parse_mntpt(line):
dev, mntpt, _ = line.split(None, 2)
dev = os.path.basename(dev)
return dev, mntpt
mntpts = {}
with open('/proc/mounts') as f:
mntpts = dict(map(parse_mntpt, f.readlines()))
# List of ('scsi id', 'dev name', 'mnt pt (if exists)')
devlist = []
def create_dev(scsiid, devname):
global mntpts
devlist.append((scsiid, devname, mntpts.get(devname)))
for devname in os.listdir(SYS_BLOCK_DEVICES):
devpath = os.path.join(SYS_BLOCK_DEVICES, devname)
devlink = os.path.join(devpath, 'device')
# Node is "virtual", i.e. partition, ignore it
if not os.path.islink(devlink):
continue
scsiid = os.path.basename(os.readlink(devlink))
if scsiid not in devs:
continue
create_dev(scsiid, devname)
# Find partition names
parts = glob.glob(os.path.join(devpath, '*/partition'))
for partpath in parts:
partname = os.path.basename(os.path.dirname(partpath))
create_dev(scsiid, partname)
# END - print results
fmtstr = '{0:8} {1:5} {2}'
print fmtstr.format('SCSI ID', 'DEV', 'MOUNT POINT')
for scsiid, devname, mntpt in devlist:
print fmtstr.format(scsiid, devname, mntpt)
这是示例输出:
$ python findflash.py 12345678
SCSI ID DEV MOUNT POINT
8:0:0:0 sdd None
8:0:0:0 sdd1 /var/run/media/myaut/Debian0wheezy020140723-17:30
8:0:0:0 sdd2 None
8:0:0:0 sdd5 None
8:0:0:1 sr0 None
我不能说 Windows 这很容易。我有一个代码(在 C/WinAPI 中)能够从系统中收集所有磁盘设备,但它的逻辑与文件系统表示相去甚远,所以我仍然没有找到解决方案。
Windows 的复杂性来自于:
- 有一组函数,如 SetupDi*,可让您枚举磁盘设备。他们的名字以 PnP 风格出现,与卷名无关。
- 有 DOS 风格 API(即在 A: 和 C: 级别工作)
- WinNT 风格 API 知道卷,可能还有挂载点。
当然这三层之间的联系并不明显(有方法可以通过 size/offset 来匹配分区,但这太疯狂了)。我仍然害怕在我的库中实现它:(