2016-12-29 1 views
1

Ich versuche, den Hostnamen des MX-Datensatzes mit niedriger Priorität mit dem folgenden Code abrufen, aber ich kann die erforderlichen Details (Hostname, Ttl, Priorität) nicht analysieren von der angegebenen Nameserver-Antwort.Extrahieren Sie DNS-Eintrag (MX) Details von Nameserver Antwort C++

u_char nsbuf[4096], dispbuf[4096]; 
ns_msg msg; 
ns_rr rr; 
int i, j, l; 
std::string domain("gmail.com"); 
l = res_query(domain.c_str(), ns_c_any, ns_t_mx, nsbuf, sizeof (nsbuf)); 

    ns_initparse(nsbuf, l, &msg); 
    printf("%s :\n", domain.c_str()); 
    l = ns_msg_count(msg, ns_s_an); 
    for (j = 0; j < l; j++) 
    { 
     int prr = ns_parserr(&msg, ns_s_an, j, &rr); 


     ns_sprintrr(&msg, &rr, NULL, NULL, reinterpret_cast<char*> (dispbuf), sizeof (dispbuf)); 

     printf("%s\n", dispbuf); 
    } 

Der obige Code gibt das Ergebnis als

gmail.com. 15M IN MX 30 alt3.gmail-smtp-in.l.google.com.

Gibt es eine verfügbare Funktion, um Hostname, Priorität, TTL usw. in separaten Puffern wie unten zu erhalten?

Host -> alt3.gmail-smtp-in.l.google.com

Priorität -> 30

ttl -> 15M

Und sollten wir manuell überprüfen für den Datensatz mit höherer Priorität, oder gibt es eine Dienstprogrammfunktion oder einen Code, der die Anforderung erfüllen könnte?

Edit:

Ich habe versucht, den folgenden Code Daten zu extrahieren

#include <cstdlib> 
#include <stdio.h> 
#include <iostream> 
#include <stdlib.h> 
#include <unistd.h> 
#include <netinet/in.h> 
#include <resolv.h> 
#include <cstring> 
#include <string> 
#include <string.h> 

using namespace std; 
int main(int argc, char** argv) { 

    u_char nsbuf[4096]; 
    u_char dispbuf[4096]; 
    ns_msg msg; 
    ns_rr rr; 
    int i, j, l; 
    std::string domain("gmail.com"); 
    l = res_query(domain.c_str(), ns_c_any, ns_t_mx, nsbuf, sizeof (nsbuf)); 
    if (l < 0) { 
     perror(domain.c_str()); 
    } else { 
#ifdef USE_PQUERY 
     res_pquery(&_res, nsbuf, l, stdout); 
#else 
     ns_initparse(nsbuf, l, &msg); 
     l = ns_msg_count(msg, ns_s_an); 
     for (j = 0; j < l; j++) { 
      int prr = ns_parserr(&msg, ns_s_an, j, &rr); 

      //BLOCK 1 
      char *cp; 
      cp = (char *) ns_rr_name(rr); 
      printf("CP->%s\n", (char *) cp); 
      int i1 = ns_rr_type(rr); 
      printf("Type->%d\n", i1); 
      int i2 = ns_rr_class(rr); 
      printf("Class->%d\n", i2); 
      int i3 = ns_rr_ttl(rr); 
      printf("TTL->%d\n", i3); 
      int i4 = ns_rr_rdlen(rr); 
      printf("DataLength->%d\n", i4); 

      //BLOCK 2 
      const u_char *rdata = ns_rr_rdata(rr) +1; 
      printf("DataU_char-> %s\n", reinterpret_cast<const char*> (rdata)); 

      int len = strlen(reinterpret_cast<const char*> (rdata)); 
      printf("len->%d\n", len); 

      char rdatatemp[1024]; 
      strncpy(rdatatemp, reinterpret_cast<const char*> (rdata), sizeof (rdatatemp)); 
      printf("DataChar->%s\n", rdatatemp); 

      ns_sprintrr(&msg, &rr, NULL, NULL, reinterpret_cast<char*> (dispbuf), sizeof (dispbuf)); 
      printf("FullRecord->%s\n", dispbuf); 
      printf("\n"); 
     } 
#endif 
    } 
    return 0; 
} 

Der obige Code funktioniert gut für TXT-Eintrag, aber für MX-Eintrag, nicht korrekt analysiert und die folgende ist das Ergebnis

Ausgang:

C P-> gmail.com
Type-> 15
Klasse-> 1
TTL-> 130
DataLength-> 32
DataU_char-> gmail-SMTP-inlgoogle
len-> 33
DataChar-> gmail-smtp-inlgoogle
FullRecord-> gmail.com. 2m10s IN MX 30 alt3.gmail-smtp-in.l.google.com.

CP-> gmail.com
Type-> 15
Klasse-> 1
TTL-> 130
DataLength-> 9
DataU_char-> ALT2.
len-> 10
DataChar-> alt2 .
FullRecord-> gmail.com. 2m10s IN MX 20 alt2.gmail-smtp-in.l.google.com.

So in DataChar & DataU_char spezielle Symbole gedruckt werden.
"alt2 . " wird anstelle von "alt2.gmail-smtp-in.l.google.com" gedruckt.
Auch der DataLength-Wert ist falsch.
Auch ich bin nicht in der Lage, die Priorität des Datensatzes zu bekommen.
Fehle ich etwas hier, oder ist es der Fehler mit der C++ Bibliothek selbst?

Antwort

1

libresolv hat keine öffentlichen Funktionen zum Entpacken bestimmter Ressourcen-Record-Typen, aber sind Funktionen darin, um Ihnen zu helfen, es selbst zu tun.

Insbesondere sehen dn_expand, die ein (komprimiert) Domain-Namen lesen kann und ns_get16, die ein Big-Endian-Zwei-Byte-Feld aus einem Datensatz gelesen wird, so in Ihrem Fall:

char exchange[NS_MAXDNAME]; 

const u_char *rdata = ns_rr_rdata(rr); 

const uint16_t pri = ns_get16(rdata); 
int len = dn_expand(nsbuf, nsbuf + msg_len, rdata + 2, exchange, sizeof(exchange)); 

printf("Pri->%d\n", pri); 
printf("Exchange->%s\n", exchange); 

wo msg_len ersetzt Ihre überschriebene l Variable, die die Länge des empfangenen Pakets enthält.

Die rdata + 2 im Anruf an dn_expand() überspringt das 16-Bit-Prioritätsfeld.

+0

Ich habe gerade versucht, was war in der Antwort und bekam Priorität richtig, aber der Austausch war leer. Die Variable len war -1 und ich konnte nicht herausfinden, was msg_len ist. –

+1

@ fury.slay in Ihrem ursprünglichen Code, den Sie 'l = res_query (...)' zuweisen, aber dann überschreiben Sie diese Variable mit dem Ergebnis von 'ns_msg_count()'. Sie müssen diesen ursprünglichen Wert von "l" in "msg_len" speichern – Alnitak

+0

sollten wir manuell überprüfen, welche die geringste Priorität hat und diesen Datensatz verwenden, oder gibt es eine API dafür? –

Verwandte Themen