我是网络新手,我尝试在 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);
}
答案 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 但您不会发送额外的字节。