Простой способ запросить информацию о подключенных USB-устройствах в Python?
Как мы можем запросить информацию о подключенных USB-устройствах в Python? Я хочу получить имя устройства UID (например: SonyEricsson W660), путь к устройству (например: /dev/ttyACM0)
а также каким будет лучший параметр из приведенной выше информации, который будет использоваться в качестве идентификации устройства всякий раз, когда оно снова подключено? (UID?)
Я работаю над Ubuntu 11.04.
ATM у меня есть этот код (используя pyUSB)
busses = usb.busses()
for bus in busses:
devices = bus.devices
for dev in devices:
print repr(dev)
print "Device:", dev.filename
print " idVendor: %d (0x%04x)" % (dev.idVendor, dev.idVendor)
print " idProduct: %d (0x%04x)" % (dev.idProduct, dev.idProduct)
print "Manufacturer:", dev.iManufacturer
print "Serial:", dev.iSerialNumber
print "Product:", dev.iProduct
проблема в том, что я не получаю желаемый результат, вставлю один пример:
<usb.legacy.Device object at 0x1653990>
Device:
idVendor: 4046 (0x0fce)
idProduct: 53411 (0xd0a3)
Manufacturer: 1
Serial: 3
Product: 2
во-первых, я не получаю имя файла, это самое важное для меня. Я предполагаю, что это часть /dev/ttyACM0 и т. д. Во-вторых, я думаю, что был какой-то UID каждого USB-устройства, или я должен использовать как поставщика, так и идентификатор продукта?
EDIT: по-видимому, у меня есть некоторые проблемы с настройкой, я думаю, что использую неправильную библиотеку USB. (используя libusb0.1) банкомат. Вот почему я получаю устройство (dev.filename) строка пуста. Если кто-то может просто сказать, что на какой операционной системе он использует, что USB-библиотека и какая версия PyUSB я думаю, что это решит мои проблемы.
7 ответов
Я могу придумать, как этот код.
поскольку все USB-порты могут быть доступны через /dev/bus/ usb / /
для сгенерированного идентификатора, даже если вы отключите устройство и снова подключите его [ может быть какой-то другой порт ]. Это будет то же самое.
import re
import subprocess
device_re = re.compile("Bus\s+(?P<bus>\d+)\s+Device\s+(?P<device>\d+).+ID\s(?P<id>\w+:\w+)\s(?P<tag>.+)$", re.I)
df = subprocess.check_output("lsusb")
devices = []
for i in df.split('\n'):
if i:
info = device_re.match(i)
if info:
dinfo = info.groupdict()
dinfo['device'] = '/dev/bus/usb/%s/%s' % (dinfo.pop('bus'), dinfo.pop('device'))
devices.append(dinfo)
print devices
образец вывода здесь будет:
[
{'device': '/dev/bus/usb/001/009', 'tag': 'Apple, Inc. Optical USB Mouse [Mitsumi]', 'id': '05ac:0304'},
{'device': '/dev/bus/usb/001/001', 'tag': 'Linux Foundation 2.0 root hub', 'id': '1d6b:0002'},
{'device': '/dev/bus/usb/001/002', 'tag': 'Intel Corp. Integrated Rate Matching Hub', 'id': '8087:0020'},
{'device': '/dev/bus/usb/001/004', 'tag': 'Microdia ', 'id': '0c45:641d'}
]
когда я запускаю ваш код, я получаю следующий вывод, например.
<usb.Device object at 0xef38c0>
Device: 001
idVendor: 7531 (0x1d6b)
idProduct: 1 (0x0001)
Manufacturer: 3
Serial: 1
Product: 2
Примечательно, что а) у меня usb.Device
объекты, тогда как у вас есть usb.legacy.Device
objects и b) у меня есть имена файлов устройств.
каждого usb.Bus
есть
для системы с устаревшим usb и libusb-1.0 этот подход будет работать для извлечения различных фактических строк. Я показываю поставщика и продукт в качестве примеров. Это может вызвать некоторый ввод-вывод, потому что он фактически считывает информацию с устройства (по крайней мере, в первый раз, во всяком случае.) Некоторые устройства не предоставляют эту информацию, поэтому предположение, что они это сделают, вызовет исключение в этом случае; это нормально, поэтому мы проходим.
import usb.core
import usb.backend.libusb1
busses = usb.busses()
for bus in busses:
devices = bus.devices
for dev in devices:
if dev != None:
try:
xdev = usb.core.find(idVendor=dev.idVendor, idProduct=dev.idProduct)
if xdev._manufacturer is None:
xdev._manufacturer = usb.util.get_string(xdev, xdev.iManufacturer)
if xdev._product is None:
xdev._product = usb.util.get_string(xdev, xdev.iProduct)
stx = '%6d %6d: '+str(xdev._manufacturer).strip()+' = '+str(xdev._product).strip()
print stx % (dev.idVendor,dev.idProduct)
except:
pass
для linux я написал скрипт под названием find_port.py который вы можете найти здесь: https://github.com/dhylands/usb-ser-mon/blob/master/find_port.py
Он использует pyudev для перечисления всех устройств tty и может совпадать по различным атрибутам.
используйте опцию --list, чтобы показать все известные последовательные порты USB и их атрибуты. Можно фильтровать по VID, PID, серийному номеру или имени поставщика. Используйте --Help для просмотра параметров фильтрации.
find_port.py печатает имя /dev/ttyXXX, а не /dev/usb/... имя.
Если вам просто нужно имя устройства, вот небольшой хак, который я написал в bash. Чтобы запустить его в python, вам нужен следующий фрагмент. Просто замените $1 и $2 на номер шины и номер устройства, например 001 или 002.
import os
os.system("lsusb | grep \"Bus Device \" | sed 's/\// /' | awk '{for(i=7;i<=NF;++i)print $i}'")
В качестве альтернативы вы можете сохранить его как скрипт bash и запустить его оттуда. Просто сохраните его как скрипт bash, например foo.sh сделайте его исполняемым.
#!/bin/bash
myvar=$(lsusb | grep "Bus Device " | sed 's/\// /' | awk '{for(i=7;i<=NF;++i)print $i}')
echo $myvar
затем вызовите его в скрипте python как
import os
os.system('foo.sh')