我通过c#中的snmpsharpnet查询Synlogy NAS,并为UPS电池获取以下值:
OID:1.3.6.1.4.1.6574.4.3.1.1.0 类型:不透明 数据:9F 78 04 42 C8 00 00
但是,它应该是一个浮点值=> 100.00
电压相同:
OID:1.3.6.1.4.1.6574.4.4.1.1.0 类型:不透明 数据:9F 78 04 43 65 00 00
浮点值=> 230.00
我如何获得价值?
我的代码:
// SNMP community name
OctetString communityo = new OctetString(community);
// Define agent parameters class
AgentParameters param = new AgentParameters(communityo);
// Set SNMP version to 1 (or 2)
param.Version = SnmpVersion.Ver1;
// Construct the agent address object
// IpAddress class is easy to use here because
// it will try to resolve constructor parameter if it doesn't
// parse to an IP address
IpAddress agent = new IpAddress(host);
// Construct target
UdpTarget target = new UdpTarget((IPAddress)agent, 161, 2000, 1);
// Pdu class used for all requests
Pdu pdu = new Pdu(PduType.Get);
pdu.VbList.Add(batteryoid); //
pdu.VbList.Add(voltageoid); //
pdu.VbList.Add(statusoid); //
// Make SNMP request
SnmpV1Packet result = (SnmpV1Packet)target.Request(pdu, param);
// If result is null then agent didn't reply or we couldn't parse the reply.
if (result != null)
{
// ErrorStatus other then 0 is an error returned by
// the Agent - see SnmpConstants for error definitions
if (result.Pdu.ErrorStatus != 0)
{
// agent reported an error with the request
Console.WriteLine("Error in SNMP reply. Error {0} index {1}",
result.Pdu.ErrorStatus,
result.Pdu.ErrorIndex);
}
else
{
// Reply variables are returned in the same order as they were added
// to the VbList
Console.WriteLine("battery ({0}) ({1}): {2}",
result.Pdu.VbList[0].Oid.ToString(),
SnmpConstants.GetTypeName(result.Pdu.VbList[0].Value.Type),
result.Pdu.VbList[0].Value.ToString());
Console.WriteLine("voltage ({0}) ({1}): {2}",
result.Pdu.VbList[1].Oid.ToString(),
SnmpConstants.GetTypeName(result.Pdu.VbList[1].Value.Type),
result.Pdu.VbList[1].Value.ToString());
Console.WriteLine("status ({0}) ({1}): {2}",
result.Pdu.VbList[2].Oid.ToString(),
SnmpConstants.GetTypeName(result.Pdu.VbList[2].Value.Type),
result.Pdu.VbList[2].Value.ToString());
}
}
else
{
Console.WriteLine("No response received from SNMP agent.");
}
target.Close();
答案 0 :(得分:1)
来自SNMP的不透明数据意味着某些字段的值本身就是BER编码的ASN.1数据 - 它使用与SNMP查询本身编码相同的语言进行编码。
对于某些快速背景,ASN.1是数据建模的标准 - 它定义了如何以标准方式构造数据,以便其他程序可以理解该结构。 ASN.1 不是用于以位和字节编码该结构的系统 - 这是BER或DER编码标准的工作。
您拥有的数据 - 9F 78 04 42 C8 00 00
- 是BER编码数据。 BER数据分为三个部分:type
,length
和value
。
如果我们使用该模板反汇编您的字节,我们会提出9f78
是类型; 04
是长度,42c80000
是值。
根据NetSnmp's documentation的一些内容,BER中的{type}代码9f78
表示“FLOATTYPE”。看起来很有希望。
下一个字节是长度字节,在这种情况下为04
。这意味着我们在值中有一个4字节的浮点数。
最后,我们有值代码42c80000
。根据{{3}},此值将被解释为标准RFC 6340 32位浮点数。 IEEE 754是用于将浮点值编码为位的最广泛使用的标准 - 这是大多数计算机和编程语言使用的标准。 C#的float
类型定义为IEEE 754 32位单精度浮点值,double
是IEEE 754 64位双精度浮点值,例如
如果我们使用IEEE 754将原始字节转换为浮点数,我们发现42c80000
被视为IEEE 754 32位浮点数的值是'100.0'。瞧,我们有你的价值。
现在我们已经了解了您的数据,让我们弄清楚如何让代码处理它。
从这些设备获得的值应始终采用相同的格式,因此我们可以假设它们始终以9f78 04
开头。所以你应该检查你得到的值是否包含那些字节,然后将它们切掉,留下剩下的四个存储浮点值的字节。您只需将这些字节转换为浮点数。
值得庆幸的是,.Net有一个课程可以帮到你 - handy-dandy online converter:
private static void BytesToSingle()
{
// Remember that 0x42c80000 has the 0x42 at the highest byte, 00 at the lowest byte.
byte[] snmpData = new byte[] { 0x00, 0x00, 0xC8, 0x42 };
single floatVal = BitConverter.ToSingle( snmpData, 0 );
// Prints "100.0".
Console.WriteLine( floatVal );
}