net::ERR_INCOMPLETE_CHUNKED_ENCODING 200 (OK)

时间:2021-07-13 09:25:33

标签: c http networking server

我是网络新手,我尝试在 C 上编写我的第一个 http 服务器。但是我遇到了一个问题。当我尝试按块发送图像时,我收到错误 GET http://localhost/img.jpg net::ERR_INCOMPLETE_CHUNKED_ENCODING 200 (OK),html 可以正常工作。 void answerOnGet(int socketCopy, char *fileName) { 结构统计 sbuf;

char filePath[SIZE];
determinateFilePath(fileName, filePath);

stat(filePath, &sbuf);

char responseHeader[2*SIZE];
char fileType[SIZE];

if (strstr(fileName, ".html") || !strcmp(fileName, "/"))
{
    strcpy(fileType, "text/html");
}
else if (strstr(fileName, ".jpg"))
{
    strcpy(fileType, "image/jpg");
}
else if(strstr(fileName, ".ico"))
{
    strcpy(fileType, "image/ico");
}
else
{
    fprintf(stderr, "Type error\n");
    exit(1);
}

FILE *fd = fopen(filePath, "rb");


if(fd == NULL)
{
    perror("Opening error");
    exit(1);
}

snprintf(responseHeader, sizeof(responseHeader),
    "HTTP/1.1 200 OK\r\n"
    "Transfer-Encoding: chunked\r\n"
    "Content-type: %s\r\n\r\n", fileType
);

if(write(socketCopy, responseHeader, strlen(responseHeader)) < 0)
{
    perror("Write error");
    exit(1);
}

long unsigned sizeOfChunk = 0;

if(sbuf.st_size > SIZE)
{
    sizeOfChunk = sbuf.st_size/50;
}
else
{
    sizeOfChunk = sbuf.st_size;
}

char *buf = malloc(sizeof(char) * (sizeOfChunk + 2));

printf("Size of chunk = %ld\n", sizeOfChunk);

char amountOfBytes[SIZE];
while(fread(buf, sizeof(char), sizeOfChunk, fd) == (sizeOfChunk))
{    
    
    
    sprintf(amountOfBytes, "%lX\r\n", sizeOfChunk);

    puts("\nStart to write");
    if(write(socketCopy, amountOfBytes, strlen(amountOfBytes)) < 0)
    {
        perror("Write bytes error");
        exit(1);
    }

    strcat(buf, "\r\n");

    if(write(socketCopy, buf, strlen(buf)) < 0)
    {
        perror("Write error");
        exit(1);
    }
    puts("\nEnd of write");

    bzero(buf, sizeOfChunk);
    bzero(amountOfBytes, strlen(amountOfBytes));
}

puts("Send zero chunk");


if(write(socketCopy, "0\r\n\r\n", strlen("0\r\n\r\n")) < 0)
{
    perror("Write zero chunk error");
    exit(1);
}

fclose(fd);
free(buf);

}

1 个答案:

答案 0 :(得分:2)

你的错误很可能

if(write(socketCopy, buf, strlen(buf)) < 0)

strlen(buf) 有什么作用?它计算到第一个 0 字节的字节数。
这是要写入的正确字节数吗?仅当末尾有 0 字节且之前没有。
有没有?嗯,不。文件中间可以有 0 个字节,fread 不会在最后放一个。

你应该使用 sizeOfChunk 而不是 strlen(buf)。

其他错误:

  • 正如@rveerd 指出的那样,您可能调用 write 并且内核仅写入部分数据。如果发生这种情况,您仍然需要再次调用它来写入其余数据。这可能会发生多次,所以你需要一个循环。我建议创建一个执行循环的函数 actually_write,然后您可以调用 actually_write(socketCopy, buf, sizeOfChunk)

  • 您的程序有时不会在最后发送一些字节。例如,如果文件长度为 1001 字节,那么您将执行 20 字节的块,并且在最后一个完整的块之后还有一个字节,fread 返回 1 但您不会发送额外的字节。