xref: /DragonOS/user/apps/http_server/main.c (revision 8d72b68da9339ec97e1b8929bcf2946f0fd83cd5)
1 #include <arpa/inet.h>
2 #include <fcntl.h>
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <string.h>
6 #include <sys/socket.h>
7 #include <sys/stat.h>
8 #include <unistd.h>
9 
10 #define PORT 12580
11 #define MAX_REQUEST_SIZE 1500
12 #define MAX_RESPONSE_SIZE 1500
13 // 网页根目录
14 #define WEB_ROOT "/wwwroot/First-WebPage-On-DragonOS"
15 #define EXIT_CODE 1
16 #define min(a, b) ((a) < (b) ? (a) : (b))
17 
18 #define DEFAULT_PAGE "/index.html"
19 
20 int security_check(char *path)
21 {
22     // 检查路径是否包含 ..
23     if (strstr(path, ".."))
24     {
25         return 0;
26     }
27     return 1;
28 }
29 
30 ssize_t send_response(int sockfd, char *response)
31 {
32     return write(sockfd, response, strlen(response));
33 }
34 
35 void send_header(int sockfd, int content_length, char *path)
36 {
37     char buffer[MAX_RESPONSE_SIZE];
38     // 获取文件类型
39     char *content_type;
40     if (strstr(path, ".html"))
41     {
42         content_type = "text/html";
43     }
44     else if (strstr(path, ".css"))
45     {
46         content_type = "text/css";
47     }
48     else if (strstr(path, ".js"))
49     {
50         content_type = "application/javascript";
51     }
52     else if (strstr(path, ".png"))
53     {
54         content_type = "image/png";
55     }
56     else if (strstr(path, ".jpg"))
57     {
58         content_type = "image/jpeg";
59     }
60     else if (strstr(path, ".gif"))
61     {
62         content_type = "image/gif";
63     }
64     else
65     {
66         content_type = "text/plain;charset=utf-8";
67     }
68     sprintf(buffer, "HTTP/1.1 200 OK\nContent-Type: %s\nContent-Length: %d\n\n", content_type, content_length);
69     send_response(sockfd, buffer);
70 }
71 
72 void send_file(int sockfd, char *path)
73 {
74     printf("send_file: path: %s\n", path);
75 
76     int fd = open(path, 0);
77     if (fd == -1)
78     {
79         send_response(
80             sockfd,
81             "HTTP/1.1 404 Not Found\nContent-Type: text/html\n\n<html><body><h1>404 Not Found</h1><p>DragonOS Http Server</p></body></html>");
82         return;
83     }
84 
85     int content_length = lseek(fd, 0, SEEK_END);
86     int remaining = content_length;
87     printf("send_file: content_length: %d\n", content_length);
88     lseek(fd, 0, SEEK_SET);
89     send_header(sockfd, content_length, path);
90 
91     char buffer[1048576];
92     int readSize;
93     while (remaining)
94     {
95         // 由于磁盘IO耗时较长,所以每次读取1MB,然后再分批发送
96         int to_read = min(1048576, remaining);
97         readSize = read(fd, &buffer, to_read);
98 
99         remaining -= readSize;
100         void *p = buffer;
101         while (readSize > 0)
102         {
103             int wsize = write(sockfd, p, min(readSize, MAX_RESPONSE_SIZE));
104             if (wsize <= 0)
105             {
106                 printf("send_file failed: wsize: %d\n", wsize);
107                 close(fd);
108                 return;
109             }
110             p += wsize;
111             readSize -= wsize;
112         }
113     }
114 
115     close(fd);
116 }
117 
118 void handle_request(int sockfd, char *request)
119 {
120     char *method, *url, *http_version;
121     char path[MAX_REQUEST_SIZE];
122 
123     method = strtok(request, " ");
124     url = strtok(NULL, " ");
125     http_version = strtok(NULL, "\r\n");
126 
127     printf("handle_request: method: %s, url: %s, http_version: %s\n", method, url, http_version);
128     // 检查空指针等异常情况
129     if (method == NULL || url == NULL || http_version == NULL)
130     {
131         send_response(sockfd, "HTTP/1.1 400 Bad Request\nContent-Type: text/html\n\n<html><body><h1>400 Bad "
132                               "Request</h1><p>DragonOS Http Server</p></body></html>");
133         return;
134     }
135     // 检查url是否为空
136     if (strlen(url) == 0)
137     {
138         send_response(sockfd, "HTTP/1.1 400 Bad Request\nContent-Type: text/html\n\n<html><body><h1>400 Bad "
139                               "Request</h1><p>DragonOS Http Server</p></body></html>");
140         return;
141     }
142     int default_page = 0;
143     if (url[strlen(url) - 1] == '/')
144     {
145         default_page = 1;
146     }
147 
148     if (strcmp(method, "GET") == 0)
149     {
150         if (default_page)
151         {
152             sprintf(path, "%s%s%s", WEB_ROOT, url, DEFAULT_PAGE);
153         }
154         else
155         {
156             sprintf(path, "%s%s", WEB_ROOT, url);
157         }
158         if (!security_check(path))
159         {
160             send_response(
161                 sockfd,
162                 "HTTP/1.1 403 Forbidden\nContent-Type: text/html\n\n<html><body><h1>403 Forbidden</h1><p>DragonOS Http Server</p></body></html>");
163             return;
164         }
165         send_file(sockfd, path);
166     }
167     else
168     {
169         send_response(sockfd, "HTTP/1.1 501 Not Implemented\nContent-Type: text/html\n\n<html><body><h1>501 Not "
170                               "Implemented</h1><p>DragonOS Http Server</p></body></html>");
171     }
172 }
173 
174 int main(int argc, char const *argv[])
175 {
176     int server_fd, new_socket, valread;
177     struct sockaddr_in address;
178     int addrlen = sizeof(address);
179     char buffer[MAX_REQUEST_SIZE] = {0};
180     int opt = 1;
181 
182     // 创建socket
183     if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0)
184     {
185         perror("socket failed");
186         exit(EXIT_CODE);
187     }
188 
189     // 设置socket选项,允许地址重用
190     // if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt)))
191     // {
192     //     perror("setsockopt failed");
193     //     exit(EXIT_CODE);
194     // }
195 
196     // 设置地址和端口
197     address.sin_family = AF_INET;
198     address.sin_addr.s_addr = INADDR_ANY;
199     address.sin_port = htons(PORT);
200 
201     // 把socket绑定到地址和端口上
202     if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0)
203     {
204         perror("bind failed");
205         exit(EXIT_CODE);
206     }
207 
208     // 监听socket
209     if (listen(server_fd, 3) < 0)
210     {
211         perror("listen failed");
212         exit(EXIT_CODE);
213     }
214 
215     while (1)
216     {
217         printf("Waiting for a client...\n");
218 
219         // 等待并接受客户端连接
220         if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t *)&addrlen)) < 0)
221         {
222             perror("accept failed");
223             exit(EXIT_CODE);
224         }
225 
226         // 接收客户端消息
227         valread = read(new_socket, buffer, MAX_REQUEST_SIZE);
228         printf("%s\n", buffer);
229 
230         // 处理请求
231         handle_request(new_socket, buffer);
232 
233         // 关闭客户端连接
234         close(new_socket);
235     }
236 
237     return 0;
238 }