前面已经实现了服务器的整体框架,现在就来具体实现HTTP服务器处理静态页面的逻辑。
要获取具体的静态文件,就要知道要获取的文件的路径。我们分析url:
协议方案名:使用http:或https:等协议方案名获取访问资源时要指定的协议类型
登录信息:用户名和密码作为从服务器端获取的必要登录信息,是可选项
服务器地址:访问服务器的地址(可以是要进行解析的地址,也可以是IPv4地址)
服务器端口号:指定服务器连接的网络端口号,同样是可选项(默认是80端口)
带层次的文件路径:指定服务器上的文件路径来定位特定的资源
查询字符串:针对已指定文件路径内的资源,可以传入 查询字符串的相关参数
片段标识符:通常可以出已获取资源中的子资源
我们访问服务器的方式是通过IP地址和端口号访问,如果访问某个静态资源,就在IP地址和端口号后面带上文件的路径。我们建立一个目录作为服务器的根目录。
于是我们就可以通过url分析出要访问文件的路径;然后就可以做出响应了。
- 根据url获取到真实的文件路径
先将url路径转化为服务器工作目录下的路径,然后:
如果url没有路径,默认为根目录,尝试访问根目录下的默认文件index.html
如果url是一个目录,尝试访问index.html
如果url不是目录,也就是说是一个文件,什么都不用做
int HandlerFilePath(char url_path[], char* file_path)
{
sprintf(file_path, "./wwwroot%s", url_path);
if(strcmp(url_path, "/") == 0)
{
strcat(file_path, "index.html");
}
else if(IsDir(file_path))
{
strcat(file_path, "index.html");
}
else;
return 200;
}
- 构造响应报文
要构造出浏览器可以正确解析的报文,就要学习HTTP协议中的响应报文格式:
HTTP响应报文格式:
首 行:版本号 状态码 状态码解释
header部分:请求的属性,以冒号为分隔的键值对;每组属性之间用\n分隔
空 行:表示header部分结束
body 部 分:空行后面的内容,如果存在,属性中一定有一个Content-Length字段来标识body部分的长度
根据响应报文格式构造响应报文:
int HandlerStaticFile(int sock, Req* req)
{
//获取到文件路径
char file_path[1024*4] = {0};
int err_code = HandlerFilePath(req->url_path, file_path);
printf("%s\n", file_path);
//打开文件,读取文件内容,写到socket中
int fd = open(file_path, O_RDONLY);
const char* first_line = "HTTP/1.1 200 OK\n";
const char* blank_line = "\n";
struct stat st ;
stat(file_path, &st);
int file_size = st.st_size;
char header[1024*4] = {0};
sprintf(header, "Content-Length: %u\n", file_size);
send(sock, first_line, strlen(first_line), 0);
send(sock, header, strlen(header), 0);
send(sock, blank_line, strlen(blank_line), 0);
sendfile(sock, fd, NULL, file_size);
close(fd);
printf("handlerstatic ok\n");
return err_code;
}