Простой способ запросить информацию о подключенных 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'}
]

Если вы работаете в Windows, вы можете использовать pywin32.

Я нашел пример здесь:

import win32com.client

wmi = win32com.client.GetObject ("winmgmts:")
for usb in wmi.InstancesOf ("Win32_USBHub"):
    print usb.DeviceID

вы можете попробовать привязки libusb


когда я запускаю ваш код, я получаю следующий вывод, например.

<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')