【问题标题】:socket transaction not going through套接字事务未通过
【发布时间】:2013-04-19 16:31:32
【问题描述】:

我正在编写一个在向浏览器发送页面时遇到问题的网络服务器。它正确发送了第一个文件(该文件包括两个 css 文件)。但是,在它发送第二个文件后,浏览器停止请求数据。它总是错过完成页面所需的 css 文件之一。

以下是一些交易:

GET / HTTP/1.1
Host: website.com
Connection: keep-alive
Cache-Control: max-age=0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.31 (KHTML, like Gecko)               Chrome/26.0.1410.43 Safari/537.31
Accept-Encoding: gzip,deflate,sdch
Accept-Language: en-US,en;q=0.8
Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.3

HTTP/1.1 OK 200
Server: thserver
Date: Mon Apr 15 17:29:51 2013
Content-Length: 564

Data received: GET /main.css HTTP/1.1
Host: website.com
Connection: keep-alive
Accept: text/css,*/*;q=0.1
User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.31 (KHTML, like Gecko)     Chrome/26.0.1410.43 Safari/537.31
Referer: http://website.com/
Accept-Encoding: gzip,deflate,sdch
Accept-Language: en-US,en;q=0.8
Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.3

HTTP/1.1 OK 200
Server: thserver
Date: Mon Apr 15 17:29:51 2013
Content-Length: 386

如果有人也知道验证 http 交易的好方法,那也会有所帮助。我确实查看了 chrome 的开发工具并显示它正在等待接收最后一个文件,但我似乎从未在服务器上收到接收请求。

另外,我使用 telnet 来模拟浏览器,它运行良好。这是我输入的内容:

GET / 
GET /index.css
GET /main.css

它每次都发回文件。

代码继续使用 read(2) 轮询套接字,但在第三个文件的请求应该到达时继续返回 EAGAIN 或 EWOULDBLOCK。

这是socket io的代码:

const int port_number   = 80;
const int timeout       = 5000000;
const int wait_time     =   10000;

PORT_CALL {

    /* declarations here */

    read_client:

    result = read( socket[ 0 ], &header_buffer[ bytes_read ], BUFFER_SIZE - bytes_read );

    if( result < 0 ){
        if( errno & ( EWOULDBLOCK | EAGAIN ) ){

            if( bytes_read > 0 ){

                bytes_read = 0;
                read_timeout = 0;
                goto respond_request;
            }
            if( read_timeout < timeout ){
                read_timeout += wait_time;
                usleep( wait_time );
                goto read_client;
            }
        }
        goto exit_thread;

    } else if( result > -1 ){
        bytes_read += result;
        goto read_client;
    }
    respond_request:

    /* respond stuff start */

    /* respond stuff end */

    goto read_client;

    exit_thread:

这是涉及创建套接字的代码

temp->socket_descriptor = socket( AF_INET, SOCK_STREAM, 0 );

temp->server_address.sin_family = AF_INET;
temp->server_address.sin_addr.s_addr = INADDR_ANY;
temp->server_address.sin_port = htons( * temp->port_number );

if( bind( temp->socket_descriptor,
          ( struct sockaddr * ) &temp->server_address,
          sizeof( temp->server_address ) ) < 0 ) ..

listen( temp->socket_descriptor, 10 );

wait_for_client:

new_socket = accept( temp->socket_descriptor,
                     ( struct sockaddr * ) &client_address,
                     &address_length );

我知道我可能要求的太多了。任何人都知道一个很好的调试工具可以让我解决这个问题,我也将不胜感激。如果有人建议使用帮助我解决问题的调试器工具,我也会认为这是一个答案。

编辑:写入和读取功能现在处于循环中,但问题仍然存在。

【问题讨论】:

  • 您是使用wireshark 还是tcpdump 来验证浏览器是否传递了三个请求?如果是这样,您需要弄清楚您的服务器“跳过”了三个请求中的哪一个。要验证交易,您必须将获取的文件与源文件进行比较。此外,您需要验证响应标头是否符合您的预期。
  • 我刚下载了tcpdump,还在学习如何使用。
  • 是的,我确实在 tcp 转储中看到了三个请求。我仍然不确定为什么我没有收到数据,但现在我知道我应该收到请求

标签: c sockets http network-protocols


【解决方案1】:

如果您只是学习如何使用数据包捕获进行调试,也许wireshark 更容易开始。它提供了一个 GUI,可以轻松定义简单的规则并直观地检查捕获。这当然是个人喜好问题,所以只是一个建议。

至于代码,很难说可能是什么问题,因为它不接近SSCCE。不过我有一些建议,

  1. 如果您要检查 if( filename ) 之类的内容,请在释放后检查 filename = NULL;
  2. 如果您想将缓冲区归零,请对其进行 memset 或循环遍历其整个大小。 for( ; header_buffer[ i ]; i++ ) header_buffer[ i ] = 0; 中的测试看起来很危险。
  3. 准备从返回 0 的套接字读取和写入,除了
  4. 循环写入。

你的读取是循环直到它有完整的请求,还是直到它有超过 0 个字节?看起来像后者,这不可能。

编辑,在下面添加文字: 示例中的代码仍然不是很完整(write fex 在哪里?),因此很难猜出确切的问题。然而,有几个相当致命的缺陷。

  1. 不应该像现在这样测试 errno,if( errno &amp; ( EWOULDBLOCK | EAGAIN ) )。在 Linux 上,您可以测试 if (errno == EWOULDBLOCK) 因为 EAGAIN 和 EWOULDBLOCK 是相同的。在它们不同的系统上,使用两个带有 || 的测试。它们之间。如果 errno 为 fex ECONNRESET,则当前代码返回非零值(在 Linux 上)。

  2. 现在代码在技术上涵盖了读取返回 0,但不是作为特殊情况。读取返回 0 表示连接正在关闭,您不应再读取套接字。

关于您的 Web 服务器中可能存在的问题的一个疯狂猜测(与上面的 #2 有关)是客户端在接收第三个文件之前断开连接,并且服务器在关闭的套接字上保持忙碌循环。

我建议浏览Beejs guide 或一些关于系统和套接字编程的书。

【讨论】:

  • 感谢您的回答,我修复了您提到的错误并使我的代码更加整洁。我仍然没有解决问题,但我下载了 Wireshark 并正在使用它。
  • 我从头开始重写代码,没有使程序多线程或动态链接(只是为了避免混乱)。正如你所说,我还考虑了关闭的连接。它现在可以工作了,感谢您的帮助。
猜你喜欢
  • 2020-07-03
  • 1970-01-01
  • 2012-01-05
  • 2021-08-26
  • 1970-01-01
  • 2012-05-14
  • 1970-01-01
  • 2021-06-24
  • 1970-01-01
相关资源
最近更新 更多