Как правильно получить доступ к varBinds, возвращаемому в ловушках pysnmp

Я использую Python 2.7 и пытаюсь перехватить SNMP-ловушки с помощью pysnmp. Я использую пример из http://pysnmp.sourceforge.net/examples/current/v1arch/manager/ntfrcv/v2c-multiple-transports.html. У меня проблемы с получением ключа / значений из varBinds правильно. Пример кода, кажется, не работает правильно для этого.
Полный код:

from pysnmp.carrier.asynsock.dispatch import AsynsockDispatcher
from pysnmp.carrier.asynsock.dgram import udp, udp6
from pyasn1.codec.ber import decoder
from pysnmp.proto import api

def cbFun(transportDispatcher, transportDomain, transportAddress, wholeMsg):
    while wholeMsg:
        msgVer = int(api.decodeMessageVersion(wholeMsg))
        if msgVer in api.protoModules:
            pMod = api.protoModules[msgVer]
        else:
            print('Unsupported SNMP version %s' % msgVer)
            return
        reqMsg, wholeMsg = decoder.decode(
            wholeMsg, asn1Spec=pMod.Message(),
            )
        print('Notification message from %s:%s: ' % (
            transportDomain, transportAddress
            )
        )
        reqPDU = pMod.apiMessage.getPDU(reqMsg)
        if reqPDU.isSameTypeWith(pMod.TrapPDU()):
            if msgVer == api.protoVersion1:
                j = pMod.apiTrapPDU.getEnterprise(reqPDU)
                print 'Enterprise: ({})'.format(str(j))

                print('Enterprise: %s' % (
                    pMod.apiTrapPDU.getEnterprise(reqPDU).prettyPrint()
                    )
                )
                print('Agent Address: %s' % (
                    pMod.apiTrapPDU.getAgentAddr(reqPDU).prettyPrint()
                    )
                )
                print('Generic Trap: %s' % (
                    pMod.apiTrapPDU.getGenericTrap(reqPDU).prettyPrint()
                    )
                )
                print('Specific Trap: %s' % (
                    pMod.apiTrapPDU.getSpecificTrap(reqPDU).prettyPrint()
                    )
                )
                print('Uptime: %s' % (
                    pMod.apiTrapPDU.getTimeStamp(reqPDU).prettyPrint()
                    )
                )
                varBinds = pMod.apiTrapPDU.getVarBindList(reqPDU)
            else:
                varBinds = pMod.apiPDU.getVarBindList(reqPDU)
            print("Var-binds List: ({})".format(str(varBinds)))

            print('Var-binds:')

            for oid, val in varBinds:
                #print('%s = %s' % (oid.prettyPrint(), val.prettyPrint()))
                print('  %s = %s' % (oid, val))


     return wholeMsg

transportDispatcher = AsynsockDispatcher()

transportDispatcher.registerRecvCbFun(cbFun)

# UDP/IPv4
# ip address changed for public posting
transportDispatcher.registerTransport(
    udp.domainName,     udp.UdpSocketTransport().openServerMode(('255.255.255.255', 162))
)

# UDP/IPv6
transportDispatcher.registerTransport(
udp6.domainName, udp6.Udp6SocketTransport().openServerMode(('::1', 162))
)

transportDispatcher.jobStarted(1)

try:
    # Dispatcher will never finish as job#1 never reaches zero
    transportDispatcher.runDispatcher()
except:
    transportDispatcher.closeDispatcher()
    raise

Я получаю ловушку, однако, когда он запускает код

print('Var-binds:')
for oid, val in varBinds:
    print('%s = %s' % (oid.prettyPrint(), val.prettyPrint()))

Я получаю AttributeError: объект 'str' не имеет атрибута prettyPrint

Печать varBinds я получаю это:
(VarBindList (). SetComponents(VarBind(). SetComponents(ObjectName('1.3.6.1.2.1.1.3.0'), _BindValue(). SetComponents(ObjectSyntax(). SetComponents(None, ApplicationSyntax (). SetComponents(None, Нет, TimeTicks(86300650))))), VarBind(). SetComponents(ObjectName('1.3.6.1.6.3.1.1.4.1.0'), _BindValue(). SetComponents(ObjectSyntax(). SetComponents(SimpleSyntax().setComponents(None, None, ObjectIdentifier('1.3.6.1.4.1.1182.7386.1.1'))))), VarBind().setComponents(ObjectName('1.3.6.1.4.1.1182.7386.1.1.1.0'), _BindValue().setComponents(ObjectSyntax().setComponents(SimpleSyntax().setComponents(Нет, OctetString(hexValue='76616c207573622d636f6e6e2d73746174757320300a')))))))
Если я изменю код на

print('  %s = %s' % (oid, val))

Я получил:

имя =
имя =
имя =

Как правильно получить 3 имени и значения?

1 ответ

Решение

Вы должны использовать .getVarBinds() метод вместо .getVarBindList() как отмечено в последней документации.

Кроме того, вы можете получить пары имя-значение, запросив .items() например, varBinds varBinds.items(),

Также обратите внимание, что существует интерфейс более высокого уровня для приложения Notification Receiver.

Другие вопросы по тегам