使用Snmp++获取MIB表


        Snmp++
是一套强大的网络管理应用开发包。它提供了Snmp网管协议所描述的所有命令,并且提供SMI数据类型的解析。MIB数据包含普通数据和表数据。在提取表数据时,由于表项的数量和Oid都不确定,所以不能通过某个特定的Oid直接获得取值。通常,关于Snmp的书上都会介绍使用GetNext命令来实现表的遍历,这种方法比较简单,这里主要讨论该算法的原理和如何用Snmp++实现。

MIB表是通过行和列来描述的。其中列表头是各个表项的原始Oid,而行表头则是index。这样以来一个Oid和一个index就唯一地确定了表中的一项。比如在接口表中,ifDescrOid 1.3.6 .1.2.1.2.2.1.2)为一列,而具体对于某一个接口则为一行。这样,某一具体表项的Oid就表示为:列Oid+index的形式。下图形象地描述了一张表的格式。

 

ifIndex

1.3.6 .1.2.1.2.2.1.1

ifDescr

1.3.6 .1.2.1.2.2.1.2

ifType

1.3.6 .1.2.1.2.2.1.3

ifMtu

1.3.6 .1.2.1.2.2.1.4

Ifspeed

1.3.6 .1.2.1.2.2.1.5

Index0

xxx

xxx

xxx

xxx

xxx

Index1

xxx

xxx

xxx

xxx

xxx

Index2

xxx

xxx

xxx

xxx

xxx

按照协议描述,最基本的方法是通过index来获取某一表项。但事实上,index本身也是一个表项,再加之有些表需要多个index,并且各种index的数据类型不同,比如要手工处理ip地址类型的index就比较困难,所以这种方法具有很难的操作性。因此,在实际编程时,可以采取一些比较技巧化的方法。

Snmp中对于GetNext命令的描述可知,如果GetNext的参数为一个表中某一列的表头Oid,比如前面的ifDescr 1.3.6 .1.2.1.2.2.1.2),则得到的值为该列第一行元素值,并可得到该值的Oid。再对取得的Oid使用GetNext就可获得该列第二行的值。如此下去,如果到了该列的最后一行,那么用GetNext将得到下一列的第一行。如果到了该表的最后一个元素,那么用GetNext将得到按MIB树所得的下一个元素值。显然,在越界的情况下,其Oid的前部分已不同于本列表头的Oid,所以,可以通过得到的Oid值来判断是否越界。

以下为其主要的代码:

 

/*从代理提取某一特定表项,即表中的一列*/

void get_Table(Oid *item_oid, CTarget *target)

{

   GenAddress address;

   target->get_address( address);
            
Oid full_oids[MAX_INDEX];       //用来保存得到的所有oid

   int index_count=0;                //该列的行数

   bool tag=true;                    //标志循环是否结束

   for(;tag==true;)

   {

      Pdu pdu;

      Vb vb;

      vb.set_oid(item_oid);

      pdu += vb;

      int status;

      if ((status=snmp->get_next( pdu, *target))== SNMP_CLASS_SUCCESS)

      {

         pdu.get_vb( vb,0);

Oid full_oid;       //该表项的Oid

            vb.get_oid(full_oid);    

            //判断是否已越界,如果是则结束循环

            if(item_oid ->nCompare(item_oid ->len(), full_oid)==0)

            {

               vb.get_oid(full_oids[index_count]);

               index_count++;

               /*

               在这里进行数据处理

               */

            }

            else

            {

               tag=false;

            }

      }

      else

      {

         tag=false;

      }

   }

}

 

以上的算法是从表中提取一列,那么如何提取一行呢?对于上述算法可以加以改进以适应我们的需要。但是,GetNext命令是按列遍历的,当我们要用它获得一行的时候还是必须先至少获得一列的信息,也就是说,在行遍历算法中还是要包含上述代码。这种方法经过实践是成功的,在这里仅对算法进行一下描述。

首先还是要执行上述代码,但是,在数据处理的时候必须保存所得到的完整Oid。我们知道,得到的Oid实际上是由列Oid+index构成,而列Oid是已知的,那么如果我们将得到的Oid前面的列Oid部分替换为另外的列Oid就可以获得该行另一列的完整Oid。在多数情况下,同一表中不同的列Oid仅相差一个数字,所以,替换方法也比较简单。这里假设我们只需替换一位(其它情况下只需做修改即可),算法如下:

 

//按行提取表数据

Oid row_oid[MAX_OID_NUM];

/*

首先在此处包含前面按列提取的代码,

并在数据处理处将full_oid保存在数组row_oid

*/

   //假设前面已经将第一列的所有表项Oid保存在数组row_oid中了

   //按行循环

for(int i=0; i<index_count; i++)

   {

      Oid oid;     

      //获取该行中每一列的数据

      for(oid=第一列oid; oid<最后一列oid; oid=下一列的oid)

      {

         Pdu pdu;

         Vb vb;

         //替换,如有需要,可以采用其它方法,这里选取最简单的情况只替换一位

            row_oid[i][oid.len()-1]=oid[oid.len()-1];

            vb.set_oid(row_oid[i]);

         pdu += vb;

            int status;

         if ((status=snmp->get( pdu, *target))== SNMP_CLASS_SUCCESS)

            {

            pdu.get_vb( vb,0);

            /*

            在这里进行数据处理

            */

            }

            else

            {

               cout << snmp->error_msg( status) << "/n";

            }

    }

实际上,Snmp提取表数据的方法有很多,这里介绍的是最简单最基础的一种方法。而且对于不同的开发包,还有更好的对表支持的方法,比如AdventSnmp开发包就有直接的表操作函数。无论采用何种方法,了解最基本的工作原理都非常有帮助。

你可能感兴趣的:(编程,算法,table,Class,vb,图形)