Uso de la función DnsQuery para resolver nombres de host y direcciones de host con Visual C++ .NET

En este artículo se proporciona un ejemplo de aplicación de consola win32 que muestra cómo usar la DnsQuery función para resolver los nombres de host y la dirección IP del host.

Versión original del producto: Winsock
Número de KB original: 831226

Creación de una aplicación de consola win32 de ejemplo que muestre cómo usar la función DnsQuery

En Winsock, use la getaddrinfo función en lugar de la getaddrbyname función para hospedar nombres en la aplicación. La getaddrbyname función se reemplazó por la función para controlar el getaddrinfo direccionamiento IPv4 e IPv6.

Winsock nunca ha contabilizado caracteres anchos hasta hace poco en Windows Server 2003, donde se incluye una nueva versión de la getaddrinfo función. La nueva versión se denomina GetAddrInfo. Si necesita una solución para todos los sistemas operativos basados en NT, use la función DNSQuery del cliente DNS para resolver los nombres de host. La DNSQuery función tiene una versión amplia que debe funcionar en sistemas operativos Microsoft Windows 2000 y versiones posteriores.

Siga estos pasos para crear una aplicación de consola Win32 de ejemplo que muestre cómo usar la DnsQuery función. La DnsQuery función envía una consulta a un servidor DNS para resolver el nombre de host en una dirección IP y viceversa.

  1. Inicie Microsoft Visual Studio .NET.

  2. En Tipos de proyecto, haga clic en Proyectos de Visual C++ y, a continuación, haga clic en Proyecto Win32 en Plantillas.

  3. Escriba Q831226 en el cuadro Nombre .

  4. En el Asistente para aplicaciones Win32, haga clic en Aplicación de consola, en Proyecto vacío y, a continuación, en Finalizar.

  5. En Explorador de soluciones, haga clic con el botón derecho en Archivos de origen, haga clic en Agregary, a continuación, haga clic en Agregar nuevo elemento. Agregue un archivo de C++ (.cpp) al proyecto. Asigne al archivo el nombre Q831226.cpp.

  6. Pegue el código siguiente en el archivo Q831226.cpp :

    #include <winsock2.h> //winsock
    #include <windns.h> //DNS api's
    #include <stdio.h> //standard i/o
    
    //Usage of the program
    void Usage(char *progname) {
        fprintf(stderr,"Usage\n%s -n [HostName|IP Address] -t [Type] -s [DnsServerIp]\n",progname);
        fprintf(stderr,"Where:\n\t\"HostName|IP Address\" is the name or IP address of the computer ");
        fprintf(stderr,"of the record set being queried\n");
        fprintf(stderr,"\t\"Type\" is the type of record set to be queried A or PTR\n");
        fprintf(stderr,"\t\"DnsServerIp\"is the IP address of DNS server (in dotted decimal notation)");
        fprintf(stderr,"to which the query should be sent\n");
        exit(1);
    }
    
    void ReverseIP(char* pIP)
    {
        char seps[] = ".";
        char *token;
        char pIPSec[4][4];
        int i=0;
        token = strtok( pIP, seps);
        while( token != NULL )
        {
            /* While there are "." characters in "string"*/
            sprintf(pIPSec[i],"%s", token);
            /* Get next "." character: */
            token = strtok( NULL, seps );
            i++;
        }
        sprintf(pIP,"%s.%s.%s.%s.%s", pIPSec[3],pIPSec[2],pIPSec[1],pIPSec[0],"IN-ADDR.ARPA");
    }
    
    // the main function 
    void __cdecl main(int argc, char* argv[])
    {
        DNS_STATUS status; //Return value of DnsQuery_A() function.
        PDNS_RECORD pDnsRecord; //Pointer to DNS_RECORD structure.
        PIP4_ARRAY pSrvList = NULL; //Pointer to IP4_ARRAY structure.
        WORD wType; //Type of the record to be queried.
        char* pOwnerName; //Owner name to be queried.
        char pReversedIP[255];//Reversed IP address.
        char DnsServIp[255]; //DNS server ip address.
        DNS_FREE_TYPE freetype;
        freetype = DnsFreeRecordListDeep;
        IN_ADDR ipaddr;
    
        if (argc > 4)
        {
            for (int i = 1; i < argc; i++)
            {
                if ((argv[i][0] == '-') || (argv[i][0] == '/'))
                {
                    switch (tolower(argv[i][1]))
                    {
                        case 'n':
                            pOwnerName = argv[++i];
                            break;
                        case 't':
                            if (!stricmp(argv[i + 1], "A"))
                                wType = DNS_TYPE_A; //Query host records to resolve a name.
                            else if (!stricmp(argv[i + 1], "PTR"))
                            {
                                //pOwnerName should be in "xxx.xxx.xxx.xxx" format
                                if (strlen(pOwnerName) <= 15)
                                {
                                    //You must reverse the IP address to request a Reverse Lookup 
                                    //of a host name.
                                    sprintf(pReversedIP, "%s", pOwnerName);
                                    ReverseIP(pReversedIP);
                                    pOwnerName = pReversedIP;
                                    wType = DNS_TYPE_PTR; //Query PTR records to resolve an IP address
                                }
                                else
                                {
                                    Usage(argv[0]);
                                }
                            }
                            else
                                Usage(argv[0]);
                            i++;
                            break;
    
                        case 's':
                            // Allocate memory for IP4_ARRAY structure.
                            pSrvList = (PIP4_ARRAY)LocalAlloc(LPTR, sizeof(IP4_ARRAY));
                            if (!pSrvList)
                            {
                                printf("Memory allocation failed \n");
                                exit(1);
                            }
                            if (argv[++i])
                            {
                                strcpy(DnsServIp, argv[i]);
                                pSrvList->AddrCount = 1;
                                pSrvList->AddrArray[0] = inet_addr(DnsServIp); //DNS server IP address
                                break;
                            }
    
                        default:
                            Usage(argv[0]);
                            break;
                    }
                }
                else
                    Usage(argv[0]);
            }
        }
        else
            Usage(argv[0]);
    
        // Calling function DnsQuery to query Host or PTR records 
        status = DnsQuery(pOwnerName, //Pointer to OwnerName. 
        wType, //Type of the record to be queried.
        DNS_QUERY_BYPASS_CACHE, // Bypasses the resolver cache on the lookup. 
        pSrvList, //Contains DNS server IP address.
        &pDnsRecord, //Resource record that contains the response.
        NULL); //Reserved for future use.
    
        if (status)
        {
            if (wType == DNS_TYPE_A)
                printf("Failed to query the host record for %s and the error is %d \n", pOwnerName, status);
            else
                printf("Failed to query the PTR record and the error is %d \n", status);
        }
        else
        {
            if (wType == DNS_TYPE_A)
            {
                //convert the Internet network address into a string
                //in Internet standard dotted format.
                ipaddr.S_un.S_addr = (pDnsRecord->Data.A.IpAddress);
                printf("The IP address of the host %s is %s \n", pOwnerName, inet_ntoa(ipaddr));
    
                // Free memory allocated for DNS records. 
                DnsRecordListFree(pDnsRecord, freetype);
            }
            else
            {
                printf("The host name is %s \n", (pDnsRecord->Data.PTR.pNameHost));
    
                // Free memory allocated for DNS records. 
                DnsRecordListFree(pDnsRecord, freetype);
            }
        }
        LocalFree(pSrvList);
    }
    
  7. En el menú Proyecto , haga clic en Propiedades.

  8. En el cuadro de diálogo Propiedades del proyecto , expanda Enlazador en Propiedades de configuración, haga clic en Línea de comandos y agregue las siguientes bibliotecas al cuadro Opciones adicionales :

    • Ws2_32.lib
    • Dnsapi.lib
  9. Presione Ctrl+Mayús+B para compilar la solución.

Prueba del ejemplo

  • Busque la dirección IP que corresponde al nombre de host: Q831226.exe -n <hostname> -t A -s <IP address of DNS server>

    Nota:

    hostname es el marcador de posición del nombre del equipo que se está consultando.

  • Busque el nombre de host que corresponde a la dirección IP: Q831226.exe -n <xxx.xxx.xxx.xxx> -t PTR -s <IP address of DNS server>

    Nota:

    xxx.xxx.xxx.xxx es un marcador de posición de la dirección IP del equipo que se está consultando.

Referencias

Para obtener más información sobre la búsqueda de DNS, vea: función DnsQuery_A.