前面解析了request请求报文的行。现在介绍对请求头的解析,解析就是将头部字段名和对应的值存储起来,存在下面的结构体里面,再把结构体放到链表里。
typedef struct tk_http_header{
void* key_start;
void* key_end;
void* value_start;
void* value_end;
struct list_head list;
}tk_http_header_t;
链表结构体如下:
typedef struct list_head {
struct list_head *prev, *next;
}list_head;
处理请求头的函数
tk_http_parse_request_body
int tk_http_parse_request_body(tk_http_request_t *request){
// 状态列表
enum{
sw_start = 0,
sw_key,
sw_spaces_before_colon,
sw_spaces_after_colon,
sw_value,
sw_cr,
sw_crlf,
sw_crlfcr
}state;
state = request->state;
size_t pi;
unsigned char ch, *p;
tk_http_header_t *hd;
for (pi = request->pos; pi < request->last; pi++) {
p = (unsigned char*)&request->buff[pi % MAX_BUF];
ch = *p;
switch(state){
case sw_start:
if(ch == CR || ch == LF)
break;
request->cur_header_key_start = p;
state = sw_key;
break;
case sw_key:
if(ch == ' '){
request->cur_header_key_end = p;
state = sw_spaces_before_colon;
break;
}
if(ch == ':'){
request->cur_header_key_end = p;
state = sw_spaces_after_colon;
break;
}
break;
case sw_spaces_before_colon:
if(ch == ' ')
break;
else if(ch == ':'){
state = sw_spaces_after_colon;
break;
}
else
return TK_HTTP_PARSE_INVALID_HEADER;
case sw_spaces_after_colon:
if (ch == ' ')
break;
state = sw_value;
request->cur_header_value_start = p;
break;
case sw_value:
if(ch == CR){
request->cur_header_value_end = p;
state = sw_cr;
}
if(ch == LF){
request->cur_header_value_end = p;
state = sw_crlf;
}
break;
case sw_cr:
if(ch == LF){
state = sw_crlf;
hd = (tk_http_header_t *) malloc(sizeof(tk_http_header_t));
hd->key_start = request->cur_header_key_start;
hd->key_end = request->cur_header_key_end;
hd->value_start = request->cur_header_value_start;
hd->value_end = request->cur_header_value_end;
list_add(&(hd->list), &(request->list));
break;
}
else
return TK_HTTP_PARSE_INVALID_HEADER;
case sw_crlf:
if(ch == CR)
state = sw_crlfcr;
else{
request->cur_header_key_start = p;
state = sw_key;
}
break;
case sw_crlfcr:
switch(ch){
case LF:
goto done;
default:
return TK_HTTP_PARSE_INVALID_HEADER;
}
}
}
request->pos = pi;
request->state = state;
return TK_AGAIN;
done:
request->pos = pi + 1;
request->state = sw_start;
return 0;
}
链表里的常用操作(头插入,尾插入,删除,判断空)函数如下
// 插入新节点
static inline void __list_add(struct list_head *_new, struct list_head *prev, struct list_head *next) {
_new->next = next;
next->prev = _new;
prev->next = _new;
_new->prev = prev;
}
// 头部新增
static inline void list_add(struct list_head *_new, struct list_head *head) {
__list_add(_new, head, head->next);
}
// 尾部新增
static inline void list_add_tail(struct list_head *_new, struct list_head *head) {
__list_add(_new, head->prev, head);
}
// 删除节点
static inline void __list_del(struct list_head *prev, struct list_head *next) {
prev->next = next;
next->prev = prev;
}
// 删除entry节点
static inline void list_del(struct list_head *entry) {
__list_del(entry->prev, entry->next);
}
// 链表判空
static inline int list_empty(struct list_head *head) {
return (head->next == head) && (head->prev == head);
}