2014-03-04 53 views
1

我有簡單Web服務器:C++插座不能接受超過21845個字節

#include <stdio.h> 
#include <stdlib.h> 
#include <iostream> 
#include <fstream> 
#include <unistd.h> 
#include <sys/types.h> 
#include <sys/socket.h> 
#include <netinet/in.h> 
#include <netdb.h> 
#include <arpa/inet.h> 
#include <err.h> 
#include <string.h> 
#include <boost/regex.hpp> 
#include <boost/thread.hpp> 
#include <boost/algorithm/string/replace.hpp> 
#include <boost/algorithm/string.hpp> 
#include "print_r.h" 
#include "Pop.h" 
#include "Headers.h" 
//#include<thread> 


char response[] = "HTTP/1.1 200 OK\r\n" 
"Content-Type: text/html; charset=UTF-8\r\n" 
"Connection: keep-alive\r\n" 
"Server: michal\r\n" 
"Set-Cookie: nazwa=test\r\n" 
"Date: Mon, 24 Feb 2014 11:39:26 GMT\r\n" 
"Vary: Accept-Encoding\r\n\r\n" 
"<html><body><h1>It works!</h1>" 
"<p>This is the default web page for this server.</p>" 
"<p>The web server software is running but no content has been added, yet.</p>" 
"<form method='post' action='/' enctype='multipart/form-data' ><input type='file' name='pliczek1'/><input type='submit' name='sub' value='sender' /><input type='checkbox' name='add[]' value='100001_used' ><input type='hidden' name='hidd' value='testowy hiddenik' /><input type='checkbox' name='add[]' value='100002_used' ><textarea name='txtform'>tekstowe poleąś</textarea></form>" 
"</body></html>\r\n\r\n"; 
void app(int client_fd) 
{ 
    int buffSize = 512; 

    char buff[buffSize]; 
    std::string headers = ""; 
    int i = 0; 
    int npos = 0; 
    int a = 0; 
    while (i = recv(client_fd, buff, buffSize, 0)) 
    { 
     std::cout << i << "\n"; 
     bzero(buff, buffSize); 
     a++; 
     if (i < buffSize) break; 
    } 

    write(client_fd, response, sizeof(response) - 1); /*-1:'\0'*/ 
    close(client_fd); 
} 

int main() 
{ 
    int one = 1, client_fd; 
    struct sockaddr_in svr_addr, cli_addr; 
    socklen_t sin_len = sizeof(cli_addr); 
    std::cout << sizeof(cli_addr); 
    int sock = socket(AF_INET, SOCK_STREAM, 0); 
    if (sock < 0) 
    err(1, "can't open socket"); 

    setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(int)); 

    int port = 8080; 
    svr_addr.sin_family = AF_INET; 
    svr_addr.sin_addr.s_addr = INADDR_ANY; 
    svr_addr.sin_port = htons(port); 

    if (bind(sock, (struct sockaddr *) &svr_addr, sizeof(svr_addr)) == -1) { 
    close(sock); 
    err(1, "Can't bind"); 
    } 

    listen(sock, 5); 

    while (1) { 
    client_fd = accept(sock, (struct sockaddr *) &cli_addr, &sin_len); 
    std::cout << "\n\n+++++++++++++ NEW CLIENT +++++++++++++++\n\n\n"; 
    if (client_fd == -1) { 
     std::cout << ("Can't accept\n"); 
     break; 
    } 
    app(client_fd); 

    } 
} 

我試圖通過發送Web瀏覽器的附件。對於小於21kB的文件,它工作正常,但我無法發送超過21845字節。爲什麼?

+1

因爲你有一個錯誤。你只是很幸運,它爲小信息「工作」。 –

+0

@KarolyHorvath:但是什麼bug? – mitch

回答

0

嘗試閱讀recv調用之間的一個小的延遲。您不能保證一次接收所有數據。你必須等待所有的數據。

+1

謝謝你的幫助,我加了usleep(1);在recv循環 – mitch

+2

不要嘗試解決與睡眠(單獨)的時間問題,這是**從來沒有**解決方案。你認爲你解決了一個問題,但你只是掩蓋了它,並引入了一個更糟糕的錯誤。 –

+0

是的,我同意。現在你知道問題是什麼,你可以專注於正確的解決方案。我會建議保持你的閱讀循環(沒有睡眠),但不斷嘗試閱讀,當沒有數據可能一秒鐘左右。然後你知道你已經獲得了所有的數據。 –

2

你打破了一個非常重要的規則:總是檢查API調用的返回值。

特別是,你不檢查的write的返回值,你只是假設成功爲止。實際上,它通常只發送部分消息,因此您需要進行循環和錯誤檢查。

+0

這裏寫不重要,recv循環執行43次,爲什麼?寫之後 – mitch

+2

你不能保證一次接收所有的TCP數據 –