1

プロジェクトの C でプロキシ アプリケーションに取り組んでいます。getaddrinfo()解析されたホスト名を渡すと失敗するという問題があります。たとえば「www.google.ca」のようにホストをハードコーディングしてもエラーにはなりませんが、(コードが受け取る GET リクエストから) URL を指定するとエラーが発生します (正確なエラーは「不明な名前またはサービス")。私は NetBeans でデバッグを試みましたが、解析された URL がハードコーディングしたものと変わらないことがわかりました。以下は私が使用しているコードです:

リクエストを受信して​​転送を試みるコードのスニペット:

...
//Message is received in the code before this
if (get_host(message, &url) == 0)
{
//Tries to open a socket to the parsed URL. This is where the issue happens
forawrd_fd = create_forward_socket(url, "80");
}
...

ホストの取得機能:

int get_host(char *request, char **host_url)
{
    char url[BUFFER_SIZE];

    if(sscanf(request, "%*s %s HTTP/1.1\r\n", url) != 1)
    {
        return -1;
    }
    else
    {
        int len = strlen(url);

        //If there is a / at the end of the URL remove it
        if(url[len-1] == '/')
        {
            printf("%c%c\n", url[len-2], url[len-1]);
            url[len-1] = '\0';
            printf("%s\n", url);
        }

        *host_url = &url;
        //If the start of the string is http:// remove it
        if(url[0] == 'h' && url[1] == 't' && url[2] == 't'&& url[3] == 'p')
        {
            *host_url += 7;
        }

        return 0;
    }
}

ファイル記述子を取得して getaddrinfo を作成する関数

int create_forward_socket(char* url, const char* port)
{
    //Status variable needed for some calls
    int status, socket_fd, received_data;

    //Setup address info structs
    struct addrinfo hints;
    struct addrinfo *result, *current;

    //Initialize our hints.
    memset(&hints, 0, sizeof hints);
    //IPv4 or IPv6 we don't
    hints.ai_family = AF_UNSPEC;
    //We want a stream socket not datagram
    hints.ai_socktype = SOCK_STREAM;
    //Whatever this means (I forget but again the interwebs says...)
    hints.ai_flags = AI_PASSIVE;

    //Get a linked list of address info that we will choose from
    if ((status = getaddrinfo(url, port, &hints, &result)) != 0) //Status here is -2 when called with the parsed URL
    {
        return -1;
    }

    for (current = result; current != NULL; current = current->ai_next)
    {
        if ((socket_fd = socket(current->ai_family, current->ai_socktype, current->ai_protocol)) != -1)
        {
            if (connect(socket_fd, current->ai_addr, current->ai_addrlen) != -1)
            {
                //We found a usable socket
                break;
            }
            else
            {
                close(socket_fd);
            }
        }
    }

    if (current == NULL)
    {
        return -2;
    }
    else
    {
        return socket_fd;
    }
}

どんな助けでも大歓迎です。私のコードがさらに必要な場合は、お知らせください。記事が長くなりすぎないように、重要だと思うことだけを含めました。

4

1 に答える 1

3

私の推測では、ローカル変数へのポインタを返していると思います。を参照してください。urlはローカル変数であり、*host_url = url;行(これは出力パラメーターであると想定しています)はそれを呼び出し元に返します。しかし、ローカル変数は、関数が戻ったときに破棄され、その後、*host_urlどこの真ん中を指し示します。

コード呼び出しget_host()は次のようになります。

char *host;
get_host(req, &host);
//call getaddrinfo with host

ただし、hostの外部の有効なメモリを指すことはありませんget_host

解決策は、呼び出し元がバッファーを割り当てることです。

int get_host(char *request, char *url)
{ ... }

そして、あなたがそれを呼ぶとき:

char host[BUFFER_SIZE];
get_host(req, host);
//call getaddrinfo with host
于 2013-01-28T00:18:32.213 に答える