Files
csapp2025/weblab/webserver.c
2025-06-08 21:34:30 +08:00

375 lines
9.3 KiB
C

#include <arpa/inet.h>
#include <asm-generic/errno-base.h>
#include <errno.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <time.h>
#include <unistd.h>
#define BUFFER_SIZE 4096
#define CONFIG_LINE_MAX 256
#define LOG_LINE_MAX 512
#define DEFAULT_PORT 8080
#define DEFAULT_ROOT "./webroot"
// 服务器配置结构
typedef struct {
char root[256];
int port;
} ServerConfig;
// 全局变量
ServerConfig config;
FILE *log_file = NULL;
// 函数声明
void load_config(const char *config_file);
void init_log();
void log_request(const char *client_ip, const char *method, const char *path);
void handle_client(int client_socket, struct sockaddr_in client_addr);
void send_response(int client_socket, const char *status,
const char *content_type, const char *body, int body_length);
void send_file(int client_socket, const char *file_path);
void send_404(int client_socket);
char *get_mime_type(const char *file_path);
void signal_handler(int sig);
int main() {
int server_socket, client_socket;
struct sockaddr_in server_addr, client_addr;
socklen_t client_len = sizeof(client_addr);
// 信号处理
signal(SIGINT, signal_handler);
signal(SIGTERM, signal_handler);
// 加载配置文件
load_config("webserver.ini");
// 初始化日志
init_log();
printf("Starting Web Server...\n");
printf("Root directory: %s\n", config.root);
printf("Port: %d\n", config.port);
// 创建socket
server_socket = socket(AF_INET, SOCK_STREAM, 0);
if (server_socket < 0) {
perror("Socket creation failed");
exit(1);
}
// 设置socket选项以重用地址
int opt = 1;
if (setsockopt(server_socket, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) <
0) {
perror("setsockopt failed");
exit(1);
}
// 配置服务器地址
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = INADDR_ANY;
server_addr.sin_port = htons(config.port);
// 绑定socket
if (bind(server_socket, (struct sockaddr *)&server_addr,
sizeof(server_addr)) < 0) {
perror("Bind failed");
close(server_socket);
exit(1);
}
// 监听连接
if (listen(server_socket, 10) < 0) {
perror("Listen failed");
close(server_socket);
exit(1);
}
printf("Server is listening on port %d...\n", config.port);
printf("Press Ctrl+C to stop the server.\n\n");
// 主服务循环
while (1) {
client_socket =
accept(server_socket, (struct sockaddr *)&client_addr, &client_len);
if (client_socket < 0) {
if (errno == EINTR) {
continue;
}
perror("Accept failed");
continue;
}
// 处理客户端请求
handle_client(client_socket, client_addr);
close(client_socket);
}
close(server_socket);
if (log_file) {
fclose(log_file);
}
return EXIT_SUCCESS;
}
// 加载配置文件
void load_config(const char *config_file) {
FILE *file;
char line[CONFIG_LINE_MAX];
char key[128], value[128];
// 设置默认值
strcpy(config.root, DEFAULT_ROOT);
config.port = DEFAULT_PORT;
file = fopen(config_file, "r");
if (!file) {
printf("Warning: Cannot open config file '%s', using defaults.\n",
config_file);
return;
}
while (fgets(line, sizeof(line), file)) {
// 跳过注释和空行
if (line[0] == '#' || line[0] == '\n' || line[0] == '\r') {
continue;
}
// 解析键值对
if (sscanf(line, "%127[^=]=%127s", key, value) == 2) {
// 去除键的空格
char *k = key;
while (*k == ' ' || *k == '\t')
k++;
char *k_end = k + strlen(k) - 1;
while (k_end > k && (*k_end == ' ' || *k_end == '\t' || *k_end == '\n' ||
*k_end == '\r')) {
*k_end = '\0';
k_end--;
}
if (strcmp(k, "root") == 0) {
strcpy(config.root, value);
} else if (strcmp(k, "port") == 0) {
config.port = atoi(value);
if (config.port <= 0 || config.port > 65535) {
config.port = DEFAULT_PORT;
}
}
}
}
fclose(file);
}
// 初始化日志
void init_log() {
log_file = fopen("webserver.log", "a");
if (!log_file) {
printf("Warning: Cannot open log file, logging to stdout only.\n");
}
}
// 记录访问日志
void log_request(const char *client_ip, const char *method, const char *path) {
time_t now;
struct tm *tm_info;
char time_str[64];
char log_line[LOG_LINE_MAX];
time(&now);
tm_info = localtime(&now);
strftime(time_str, sizeof(time_str), "%Y/%m/%d %H:%M:%S", tm_info);
snprintf(log_line, sizeof(log_line), "%s IP:%s %s %s\n", time_str, client_ip,
method, path);
// 输出到控制台
printf("%s", log_line);
// 输出到日志文件
if (log_file) {
fprintf(log_file, "%s", log_line);
fflush(log_file);
}
}
// 处理客户端请求
void handle_client(int client_socket, struct sockaddr_in client_addr) {
char buffer[BUFFER_SIZE];
char method[16], path[256], version[16];
char *client_ip = inet_ntoa(client_addr.sin_addr);
// 接收HTTP请求
int bytes_received = recv(client_socket, buffer, sizeof(buffer) - 1, 0);
if (bytes_received <= 0) {
return;
}
buffer[bytes_received] = '\0';
// 解析HTTP请求行
if (sscanf(buffer, "%15s %255s %15s", method, path, version) != 3) {
send_404(client_socket);
return;
}
// 记录访问日志
log_request(client_ip, method, path);
// 只处理GET请求
if (strcmp(method, "GET") != 0) {
send_response(client_socket, "405 Method Not Allowed", "text/html",
"<h1>405 Method Not Allowed</h1>", 30);
return;
}
// 构建完整文件路径
char file_path[512];
if (strcmp(path, "/") == 0) {
snprintf(file_path, sizeof(file_path), "%s/index.html", config.root);
} else {
snprintf(file_path, sizeof(file_path), "%s%s", config.root, path);
}
// 检查文件是否存在
struct stat file_stat;
if (stat(file_path, &file_stat) == 0 && S_ISREG(file_stat.st_mode)) {
send_file(client_socket, file_path);
} else {
send_404(client_socket);
}
}
// 发送HTTP响应
void send_response(int client_socket, const char *status,
const char *content_type, const char *body,
int body_length) {
char header[1024];
snprintf(header, sizeof(header),
"HTTP/1.1 %s\r\n"
"Content-Type: %s\r\n"
"Content-Length: %d\r\n"
"Connection: close\r\n"
"\r\n",
status, content_type, body_length);
send(client_socket, header, strlen(header), 0);
if (body && body_length > 0) {
send(client_socket, body, body_length, 0);
}
}
// 发送文件
void send_file(int client_socket, const char *file_path) {
FILE *file = fopen(file_path, "rb");
if (!file) {
send_404(client_socket);
return;
}
// 获取文件大小
fseek(file, 0, SEEK_END);
long file_size = ftell(file);
fseek(file, 0, SEEK_SET);
// 获取MIME类型
char *mime_type = get_mime_type(file_path);
// 发送HTTP头
char header[1024];
snprintf(header, sizeof(header),
"HTTP/1.1 200 OK\r\n"
"Content-Type: %s\r\n"
"Content-Length: %ld\r\n"
"Connection: close\r\n"
"\r\n",
mime_type, file_size);
send(client_socket, header, strlen(header), 0);
// 发送文件内容
char file_buffer[4096];
size_t bytes_read;
while ((bytes_read = fread(file_buffer, 1, sizeof(file_buffer), file)) > 0) {
send(client_socket, file_buffer, bytes_read, 0);
}
fclose(file);
}
// 发送404错误
void send_404(int client_socket) {
const char *body = "<html><body><h1>404 Not Found</h1><p>The requested page "
"was not found.</p></body></html>";
send_response(client_socket, "404 Not Found", "text/html", body,
strlen(body));
}
// 获取MIME类型
char *get_mime_type(const char *file_path) {
const char *ext = strrchr(file_path, '.');
if (!ext)
return "application/octet-stream";
if (strcmp(ext, ".html") == 0 || strcmp(ext, ".htm") == 0) {
return "text/html";
} else if (strcmp(ext, ".css") == 0) {
return "text/css";
} else if (strcmp(ext, ".js") == 0) {
return "application/javascript";
} else if (strcmp(ext, ".jpg") == 0 || strcmp(ext, ".jpeg") == 0) {
return "image/jpeg";
} else if (strcmp(ext, ".png") == 0) {
return "image/png";
} else if (strcmp(ext, ".gif") == 0) {
return "image/gif";
} else if (strcmp(ext, ".txt") == 0) {
return "text/plain";
}
return "application/octet-stream";
}
// 信号处理函数
void signal_handler(int sig) {
char *signal_name = (sig == SIGINT) ? "SIGINT" : "SIGTERM";
time_t now;
struct tm *tm_info;
char time_str[64];
// 获取当前时间
time(&now);
tm_info = localtime(&now);
strftime(time_str, sizeof(time_str), "%Y/%m/%d %H:%M:%S", tm_info);
// 记录关闭信息
printf("\n[%s] Received %s signal. Shutting down server...\n", time_str,
signal_name);
// 如果日志文件打开,记录关闭信息并关闭
if (log_file) {
fprintf(log_file, "[%s] Server shutdown initiated by %s\n", time_str,
signal_name);
fflush(log_file);
fclose(log_file);
log_file = NULL;
}
// 等待1秒以确保所有pending的写操作完成
sleep(1);
printf("Server shutdown complete.\n");
exit(0);
}