4

c++/c でソケットを使用して .exe ファイルをダウンロードする方法を知りたいです。私はcygwinとg ++を使用しています

berkeleys ソケットで試してみましたが、ファイルをダウンロードできないようです。以下にコードを埋め込みました。

    #include <fstream>
    #include <stdio.h>
    #include <unistd.h>
    #include <sys/socket.h>
    #include <sys/types.h>
    #include <netinet/in.h>
    #include <stdlib.h>
    #include <string.h>
    #include <netdb.h>
    #include <string>
    #include <sstream>
    #include <iostream>
    #include <map>

    using namespace std;

    int main(){



    int sock_descriptor; // integer number to access socket
struct sockaddr_in serv_addr; // uses predefined sockaddr_in struct
struct hostent *server; // from netdb.h to determine host name out of ip address
char recvBuff[1024];  // Receiving buffer 
char hostname[] = "localhost";

char req[] = "GET /fjernsupport.exe HTTP/1.1"
                "Host: localhost\n"
                "Connection: keep-alive\n"
                "Cache-Control: no-cache\n"
                "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\n"
                "Pragma: no-cache\n"
                "User-Agent: Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.31 (KHTML, like Gecko) Chrome/26.0.1410.64 Safari/537.31\n"
                "Accept-Encoding: gzip,deflate,sdch\n"
                "Accept-Language: en-GB,en;q=0.8,en-US;q=0.6,da;q=0.4\n"
                "Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.3\n\n";

sock_descriptor = socket(AF_INET, SOCK_STREAM, 0); // SOCK_STREAM = TCP, AF_INET = DOMAIN
if(sock_descriptor < 0){
    std::cout << "Failed creating socket\n" << std::endl;
}
bzero((char *)&serv_addr, sizeof(serv_addr));
server = gethostbyname(hostname);
if(server==NULL){
    std::cout << "Failed to find server name" << std::endl;
    return 0;
}
serv_addr.sin_family = AF_INET;
memcpy((char *) &(serv_addr.sin_addr.s_addr), (char *)(server->h_addr), server->h_length);

serv_addr.sin_port = htons(80);  // Ensures integer interpretion is correct
if(connect(sock_descriptor, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0){
    std::cout << "Failed to connect to server" << std::endl;
}else{
    std::cout << "Succesfully connected" << std::endl;
}
cout << "SEND: " << req << endl;
write(sock_descriptor, req, sizeof(req));
bool isFile = false;
ofstream outFile;
outFile.open("outfile.exe", ios::out | ios::binary);
while(true){
    memset(recvBuff, 0, 1024);
    if(read(sock_descriptor, recvBuff, sizeof(recvBuff)-1) > 0){
        //cout << "RECV: " << recvBuff << endl;


                outFile << recvBuff;

                //cout << "WRITING THIS: " << recvBuff << endl;



        //cout << "Newline? " << endl;
    }else{
        outFile.close();
        cout << "Returning here";
        return 0;
    }


}
outFile.close();
cout << "finished";
return 0;

私のコードは完全に接続されており、データを除いて正しい応答ヘッダーを取得しています。データをファイルに移動していません。主な懸念は、受け取ったデータが間違っているのか、それとも outfile.exe の作成方法が間違っているのか、よくわからないことです。

上記のコードは、応答の http ヘッダー ファイルもファイルに書き込んでいることを知っています。しかし、それでもファイルのサイズが十分に大きくなりません。outfile.exe は約 763kbs しかありませんが、fjernsupport.exe は約 3mb です。

4

1 に答える 1

2

ここでは、NUL で終了する文字列であるかのようにバイナリ データを保存しようとしています。

            outFile << recvBuff;

代わりに、read()呼び出しの戻り値を保存し (バッファーに保存されたバイト数を示すため)、そのバイト数をoutFile.

if((bytes = read(sock_descriptor, recvBuff, sizeof(recvBuff))) > 0){
            //...
            outFile.write(recvBuff, bytes);
于 2013-05-17T00:10:28.850 に答える