Use a função DnsQuery para resolve nomes de host e endereços de host com o Visual C++ .NET

Este artigo fornece um exemplo de aplicativo de console do Win32 que ilustra como usar a DnsQuery função para resolve nomes de host e IP do host.

Versão original do produto: Winsock
Número de KB original: 831226

Criar um aplicativo de console Win32 de exemplo que ilustra como usar a função DnsQuery

No Winsock, use a getaddrinfo função em vez da getaddrbyname função para hospedar nomes em seu aplicativo. A getaddrbyname função foi substituída pela função para manipular o getaddrinfo endereçamento IPv4 e IPv6.

Winsock nunca foi responsável por caracteres largos até recentemente no Windows Server 2003, onde uma nova versão da getaddrinfo função está incluída. A nova versão se chama GetAddrInfo. Se você precisar de uma solução para todos os sistemas operacionais baseados em NT, use a função DNSQuery do cliente DNS para resolve nomes de host. A DNSQuery função tem uma versão ampla que deve funcionar no Microsoft Windows 2000 e em sistemas operacionais posteriores.

Use as etapas a seguir para criar um aplicativo de console Win32 de exemplo que ilustra como usar a DnsQuery função. A DnsQuery função envia uma consulta a um servidor DNS para resolve o nome do host para um endereço IP e vice-versa.

  1. Inicie o Microsoft Visual Studio .NET.

  2. Em Tipos de Projeto, clique em Projetos do Visual C++ e clique em Projeto Win32 em Modelos.

  3. Digite Q831226 na caixa Nome .

  4. No assistente de aplicativo Do Win32, clique em Aplicativo de Console, clique em Projeto vazio e clique em Concluir.

  5. Em Gerenciador de Soluções, clique com o botão direito do mouse em Arquivos de Origem, clique em Adicionar e clique em Adicionar Novo Item. Adicione um arquivo C++ (.cpp) ao projeto. Nomeie o arquivo Q831226.cpp.

  6. Cole o seguinte código no arquivo 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. No menu Projeto , clique em Propriedades.

  8. Na caixa de diálogo Propriedades do Projeto , expanda o Linker em Propriedades de Configuração, clique em Linha de Comando e adicione as seguintes bibliotecas à caixa Opções Adicionais :

    • Ws2_32.lib
    • Dnsapi.lib
  9. Pressione Ctrl+Shift+B para criar a solução.

Testar o exemplo

  • Localize o endereço IP que corresponde ao nome do host: Q831226.exe -n <hostname> -t A -s <IP address of DNS server>

    Observação

    hostname é o espaço reservado do nome do computador que está sendo consultado.

  • Localize o nome do host que corresponde ao endereço IP: Q831226.exe -n <xxx.xxx.xxx.xxx> -t PTR -s <IP address of DNS server>

    Observação

    xxx.xxx.xxx.xxx é um espaço reservado do endereço IP do computador que está sendo consultado.

Referências

Para obter mais informações sobre a pesquisa DNS, consulte: DnsQuery_A função.