0

Indy 10.5.7 で c++builderXE を使用しており、別のエージェント snmp からトラップを受信しようとしています。

トラップを受信するプログラムの実行方法を説明する情報がありません。

以下に、私が今使用しようとしているコードのスニペットを示します。

ReceiveTrap() メソッドは常に 0 を返します。これは、データが受信されなかったことを意味します。

スペア API を使用して数年前に作成した別のプログラムで PC 構成をテストしたところ、トラップが受信されたので、これは構成の問題ではないはずです。

以下のルーチンで私が間違っている帽子の提案はありますか?

void __fastcall TForm1::LabelReceiveTrapClick(TObject * Sender)
{
    static bool status = false;
    int ists;
    String Fun           = "[SimpleReceiveTrap] ";
    TSNMPInfo * infoSnmp = 0;

    try
    {
        status = !status;

        if (status)
        {
            std::auto_ptr< TIdSNMP >clientSnmp(new TIdSNMP(NULL));
            clientSnmp->Community      = "public";
            clientSnmp->ReceiveTimeout = 1000;
            clientSnmp->Binding->Port  = 162;
            while (status)
            {
                Application->ProcessMessages();
                ists = clientSnmp->ReceiveTrap();
                Mylog(L"%s ReceiveTrap status = [%d]", Fun.c_str(), ists);
                if (ists > 0)
                {
                    infoSnmp = clientSnmp->Trap;
                }
            }
        }
    }
    catch (Exception & ex)
    {
        Mylog(L"%s ERROR", Fun.c_str(), ex.Message.c_str());
    }
}
4

2 に答える 2

1

これは、トラップを受信するためのリッスン ポートを設定する正しい方法ではありません。プロパティを読み取ると、およびプロパティBindingを使用してソケットがローカル IP/ポートに割り当てられ、バインドされます。すでにバインドされているソケットのローカルを変更することはできないため、プロパティの割り当ては事実上ノーオペレーションです。TIdSNMP::BoundIPTIdSNMP::BoundPortPortBinding->Port

さらに言えば、とにかく間違ったソケットを操作しようとしています。ソケットは、Bindingリモート SNMP システムにクエリを送信するために使用されます。 TIdSNMPトラップを受信するために別のソケットを使用します。 そのソケットのリッスン ポートを指定するためTIdSNMPの別のプロパティがあります。TrapPortBindingアクセスすると、トラップ ソケットが割り当てられ、 および にバインドされBinding->IPますTIdSNMP::TrapPort。プロパティのTrapPortデフォルトは 162 です。

std::auto_ptr< TIdSNMP >clientSnmp(new TIdSNMP(NULL));
clientSnmp->Community = "public";
clientSnmp->ReceiveTimeout = 1000;
clientSnmp->TrapPort = 162; // <--
...
ists = clientSnmp->ReceiveTrap();

Indy の変更ログを見ると、10.5.7 がリリースされて以来、リッスン ソケットにトラップ関連の変更がいくつか加えられているため、バグ修正を取得するには、新しい Indy バージョンにアップグレードする必要がある場合があります。または、最新バージョンをダウンロードしIdSNMP.pasて、プロジェクトに直接追加することもできます。

于 2012-08-10T04:39:55.110 に答える
0

Indi コンポーネントのみを使用すると、rev 2c のトラップを読み取ることができません。

私が使用したコードの下:

データを取得するには、FPiette コンポーネント スイートの TWSocket を使用します。

void __fastcall TForm1::LabelStartServerTracSnmpClick(TObject * Sender)
{
    String Fun = "[LabelStartServerTracSnmp] ";
    try
    {
        if (WSocket1->State == wsClosed)
        {
            WSocket1->Proto = "udp";
            WSocket1->Addr  = "0.0.0.0";
            WSocket1->Port  = 162;
            WSocket1->Listen();
        }
        else
        {
            WSocket1->Close();
        }
    }
    catch (Exception & ex)
    {
        Mylog(L"%s ERROR: [%s]", Fun.c_str(), ex.Message.c_str());
    }
}

受信したデータを分析するには、Indy を使用します

void __fastcall TForm1::WSocket1DataAvailable(TObject * Sender, WORD ErrCode)
{
    char buffer[1024];
    int len, cnt, srcLen;
    TSockAddrIn srcSocket;
    String rcvmsg, remHost, s1, s2, Fun = "[WSocket1DataAvailable] ";
    TIdSNMP * clientSnmp = NULL;
    TSNMPInfo * infoSnmp = NULL;
    try
    {
        srcLen = sizeof(srcSocket);
        len    = WSocket1->ReceiveFrom(buffer, sizeof(buffer), srcSocket, srcLen);
        if (len >= 0)
        {
            buffer[len] = 0;
            rcvmsg      = String(buffer, len);

            __try
            {
                clientSnmp = new TIdSNMP(NULL);
                infoSnmp   = new TSNMPInfo(clientSnmp);
                infoSnmp->DecodeBuf(rcvmsg);
                cnt = infoSnmp->ValueCount;
                if (cnt > 0)
                {
                    // ---------------------------------------------------
                    for (int idx = 0; idx < cnt; ++idx)
                    {
                        s1 = infoSnmp->ValueOID[idx];
                        s2 = infoSnmp->Value[idx];
                        Mylog(L"[%s] Trap : [%s] => [%s]", s1.c_str(), s2.c_str());
                    }
                }
            }
            __finally
            {
                if (infoSnmp)
                {
                    delete infoSnmp;
                    infoSnmp = 0;
                }
                if (clientSnmp)
                {
                    delete clientSnmp;
                    clientSnmp = 0;
                }
            }
        }
    }
    catch (Exception & ex)
    {
        Mylog(L"%s ERROR", Fun.c_str(), ex.Message.c_str());
    }

}
于 2012-09-03T07:48:08.233 に答える