gpt4 book ai didi

c - HTTP 通过 C 中的套接字 : Basic file server fails to properly return file

转载 作者:太空宇宙 更新时间:2023-11-04 06:04:42 25 4
gpt4 key购买 nike

我正在尝试使用 C 中的套接字编写一个基本的 HTTP 文件服务器,但有两个主要问题。

首先,当我尝试通过浏览器访问“ip:port/”时,我得到了 index.html 文件,但它完全是乱码。但是,当我访问“ip:port/not_in_directory”时,我的小自定义 404 消息会正常返回。如果我将默认文件设置为现有的 pdf 文件,也会出现乱码(pdf 乱码的长度)。

其次,当我尝试访问“ip:port/file_that_exists”时,我得到了 404 而不是正确的文件。在调试时,我确实看到 scratch_pad 具有正确的文件名,没有多余或丢失的字符,但是 !access(scratch_pad, R_OK) 仍然是 0。是否有一些必要的转换,即使 access() 需要一个 char*文件名?

编辑:第二个问题已经解决(我使用指向释放缓冲区的指针作为文件名)。但是,浏览器获取的所有内容都是乱码。我正在删除更多不必要的代码。

#define FILE_NOT_FOUND "HTTP/1.0 404 FILE NOT FOUND\r\n"
#define FILE_FOUND "HTTP/1.0 200 OK\r\n"
#define SERVER_NAME "Server: Test Server\r\n"
#define CONTENT_TYPE "Content-Type: text/html\r\n"
#define DATE "Date: "
#define MESSAGE_BREAK "\r\n"
#define NOT_FOUND_HTML "<!DOCTYPE html>\n<html>\n<title>404 Not Found</title>\n<body>\n<p>Nope</p>\n</body>\n</html>\r\n"

if(!access(pulled_name, R_OK | W_OK)) {
printf("file found on server\n");
server_file = fopen(pulled_name, "rb+");
fseek(server_file, 0L, SEEK_END);
file_size = ftell(server_file);
fseek(server_file, 0l, SEEK_SET);
printf("opened successfully and set file_size to %i\n", file_size);
getting_time = time(NULL);
c_time = localtime(&getting_time);
current_time = asctime(c_time);
printf("set time for message\n");

message = malloc((int) strlen(FILE_FOUND)
+ (int) strlen(SERVER_NAME)
+ (int) strlen(CONTENT_TYPE)
+ (int) strlen(DATE)
+ (int) strlen(current_time)
+ (int) strlen(MESSAGE_BREAK));

printf("malloc'ed message\n");

message_size = (int)(strlen(message)*sizeof(char));
memset(message, 0, message_size);
strcpy(message, FILE_FOUND);
strcat(message, SERVER_NAME);
strcat(message, CONTENT_TYPE);
strcat(message, DATE);
strcat(message, current_time);
strcat(message, MESSAGE_BREAK);
printf("built message\n");

printf("set message_size to %i\n", message_size);
while(total_sent < message_size) {
n = send(new_socket_fd, message, message_size, 0);
if (n==-1) {
printf("error sending message\n");
break;
}
total_sent += n;
printf("sent %i\n", total_sent);
}
free(message);

total_sent = 0;
while(total_sent < file_size) {
n = send(new_socket_fd, server_file, file_size, 0);
if (n==-1) {
printf("error sending file\n");
break;
}
total_sent += n;
printf("sent %i\n", total_sent);
}
fclose(server_file);
close(new_socket_fd);
printf("closed client connection!\n");

最佳答案

这个错误有几个实例:

memset(buffer, '\0', sizeof(buffer));

当第一个参数的类型是 char* 时,这意味着缓冲区不会填充空字符,而只会填充前 sizeof(char*) 字节(通常为 4 或 8 个)。在这种情况下,这意味着 buffer 可能不会以空字符终止(因为 recv() 不会附加空字符)。 printf()strtok() 都要求字符串参数以 null 结尾。使用非空终止字符串调用 printf("%s", p); 时,通常会看到在字符串后打印垃圾字符。

通过显式指定第一个参数指向的缓冲区大小来修复所有 memset() 调用:

const size_t BUFFER_SIZE      = 512;
const size_t SCRATCH_PAD_SIZE = 256;
buffer = malloc(BUFFER_SIZE); /*Check return value*/
scratch_pad = malloc(SCRATCH_PAD_SIZE);/*to ensure success.*/
memset(buffer, 0, BUFFER_SIZE);
memset(scratch_pad, 0, SCRATCH_PAD_SIZE);

sizeof(char) 保证为 1,因此它可以从 malloc() 参数中省略。


memset() 的其他误用:

memset(message, '0', sizeof(message)); 
  • '0' 是字符零,不为空
  • sizeof(message)sizeof(char*)

更改为:

memset(message, 0, /* the actual size of message*/ );

关于c - HTTP 通过 C 中的套接字 : Basic file server fails to properly return file,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12640754/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com