欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

基于 libevent 开源框架实现的 web 服务器

程序员文章站 2022-06-24 14:32:20
/* 原创文章 转载请附上原链接: https://www.cnblogs.com/jiujue/p/10707153.html */ 自己实现的如有缺漏欢迎提出 直接代码 一切皆在代码中 首先是 主函数文件 和 头文件 头文件: 1 #ifndef _HEAD_H_ 2 #define _HEAD ......

/* 原创文章 转载请附上原链接:    */

自己实现的如有缺漏欢迎提出

直接代码 一切皆在代码中 

首先是 主函数文件 和 头文件

  头文件: 

 1 #ifndef _head_h_
 2 #define _head_h_
 3 
 4 #include<stdio.h>
 5 #include<string.h>
 6 #include<stdlib.h>
 7 #include<time.h>
 8 #include<math.h>
 9 #include <fcntl.h>
10 #include <ctype.h>
11 #include <dirent.h>
12 #include <unistd.h>
13 #include <event2/event.h>
14 #include <event2/listener.h>
15 #include <event2/bufferevent.h>
16 #include <sys/socket.h>
17 #include <sys/types.h>
18 #include <sys/stat.h>
19 #include <arpa/inet.h>
20 
21 int judge_type_dir_or_nondir(const char* name);
22 int send_dir_asheml(struct bufferevent *bufev, char *dirname, void *arg);
23 struct evconnlistener* libev_start(struct event_base*base, const char* ip,int port);
24 int send_html_head(struct bufferevent* bufev, int stat_no, const char* stat_desc, char* type);
25 const char *get_file_type(const char *name);
26 int send_file(struct bufferevent *bufev,char* file);
27 
28 #endif

  主函数文件:

 1 // file name: main.c
 2 // author: jiujue
 3 // created time: 2019年04月05日 星期五 19时54分48秒
 4 
 5 #include "head.h"
 6 
 7 int main(int argc, const char* argv[])
 8 {
 9     if(argc < 4)
10     {
11         printf("argument not enough\neg:./app ip port sourcedir\n");
12         exit(1);
13     }
14 
15 
16     int ret = chdir(argv[3]);
17     if(-1 == ret)
18     {
19         perror("chdir error");
20         exit(1);
21     }
22     else
23     {
24         printf("chdir successful,to -> %s\n",argv[3]);
25     }
26 
27     struct event_base* base = event_base_new();
28 
29     struct evconnlistener* listen = libev_start(base,argv[1],atoi(argv[2]));
30 
31     event_base_dispatch(base);
32 
33     evconnlistener_free(listen);
34     event_base_free(base);
35 
36     return 0 ;
37 }

接下来是 调用libevent框架了 (重头戏来了 注意回调的设置哦):

  1 // file name: _libev.c
  2 // author: jiujue
  3 // created time: 2019年04月05日 星期五 19时54分08秒
  4 #include "head.h"
  5 #define path_of_404_ "404.html"
  6 
  7 void read_cb(struct bufferevent* bufev, void* arg)
  8 {
  9     printf("<<<<<<<<<<<<<<<<<<<<<<<<<<<,,happen read_cb\n");
 10     char buf[1024] = {0}, method[12] = {0}, path[1024] = {0}, protocol[12] = {0};
 11     bufferevent_read(bufev,buf,sizeof(buf));
 12     //printf("-----------------------recv http request :%s\n",buf);
 13     sscanf(buf,"%s %s %s",method,path,protocol);
 14     char *file = path+1;
 15     if(0 == (strcmp(path,"/") ) )
 16     {
 17         file = (char*)"./";
 18     }
 19     
 20     int isfile = judge_type_dir_or_nondir(file);
 21     printf("fffffffff          is %d \n",isfile);
 22     if(0 == isfile)
 23     {//is palin file
 24         printf("send file <name>>%s\n",file);
 25         send_file(bufev,file);
 26     }
 27     if(isfile == 1){//is dir
 28         printf("send dir <name>>%s\n",file);
 29         send_html_head(bufev,200,"ok",(char*)"text/html");
 30         send_dir_asheml(bufev,file,null);
 31     }
 32     else if(-1 == isfile)
 33     {//is not found file or directory
 34         printf("send 404 <name>>%s\n",file);
 35         send_file(bufev,path_of_404_);
 36     }
 37 
 38 }
 39 
 40 void write_cb(struct bufferevent* bufev, void* arg)
 41 {
 42     struct sockaddr_in *cli = (struct sockaddr_in*)arg;
 43     char buf[1024] = {0};
 44     printf("sent respond to cli,ip ->%s and port ->%d\n",
 45             inet_ntop(af_inet,&(cli->sin_addr.s_addr), buf,sizeof(buf)), 
 46             ntohs(cli->sin_port) );
 47 }
 48 
 49 void event_cb(struct bufferevent* bufev, short ev, void* arg)
 50 {
 51     printf("event_cb successful\n");
 52     if(ev & bev_event_eof)
 53     {
 54         struct sockaddr_in *cli = (struct sockaddr_in*)arg;
 55         char buf[1024] = {0};
 56         printf("have client disconnect, ip ->%s and port ->%d\n",
 57                 inet_ntop(af_inet,&(cli->sin_addr.s_addr), buf,sizeof(buf)), 
 58                 ntohs(cli->sin_port) );
 59 
 60     }
 61     if(ev & bev_event_error )
 62     {
 63         printf("******************************** happy error******************************\n");
 64     }
 65     bufferevent_free(bufev);
 66 }
 67 
 68 void listener_cb(struct evconnlistener *listener, 
 69         evutil_socket_t fd, struct sockaddr* cli, 
 70         int cli_len, void* arg)
 71 {
 72 
 73     printf("<<<<<<<<<<<<<<<<<<<<,,,,,,,,, listener_cb successful\n");
 74     struct event_base* base = (struct event_base*)arg;
 75 
 76     struct bufferevent* bufev = bufferevent_socket_new(base, fd, bev_opt_close_on_free);
 77     bufferevent_setcb(bufev, read_cb, write_cb, event_cb,cli);
 78     bufferevent_enable(bufev,ev_read);
 79 
 80 }
 81 
 82 
 83 struct evconnlistener* libev_start(struct event_base*base, const char* ip,int port)
 84 {
 85 
 86     struct sockaddr_in ser;
 87     ser.sin_family = af_inet;
 88     ser.sin_port = htons(port);
 89     inet_pton(af_inet,ip,&(ser.sin_addr.s_addr));
 90 
 91     struct evconnlistener* ret =  evconnlistener_new_bind(base, listener_cb, base, 
 92             lev_opt_reuseable | lev_opt_close_on_free, 128, 
 93             (struct sockaddr *)&ser, sizeof(ser));
 94 
 95     if(ret == null)
 96     {
 97         return null;
 98     }
 99 
100     return ret;
101 }

然后是 发送文件和目录的回调“

  文件的:

 1 #include "head.h"
 2 
 3 int send_file(struct bufferevent *bufev,char* file)
 4 {
 5 
 6 
 7     int ffd = open(file,o_rdonly);
 8     if(-1 == ffd)
 9     {
10         printf("sourcefilepath : %s\n",file);
11         perror("open sourcefile error");
12 
13     }
14 
15     char file_read_buf[1024];
16     int read_len = 0;
17 
18     char * type = get_file_type(file);
19 
20 
21     send_html_head(bufev,200, "ok", type);
22 
23     while((read_len=read(ffd, file_read_buf,sizeof(file_read_buf))) > 0)
24     {
25         if(0 == read_len)
26         {
27             break;
28         }
29         bufferevent_write(bufev,file_read_buf,read_len);;
30         file_read_buf[strlen(file_read_buf)+1] = '\n';
31         printf("send message :%s\n",file_read_buf);
32         memset(file_read_buf,0,sizeof(file_read_buf));
33     }
34 
35     printf("close ...\n");
36     close(ffd);
37     return 0;
38 }

  目录的(这里拼接网页的时候要细心 非常细心的那种 ):

 1 // file name: _send_dir.c
 2 // author: jiujue
 3 // created time: 2019年04月05日 星期五 19时27分18秒
 4 
 5 #include "head.h"
 6 #define maxforgtml_ 4096
 7 
 8 
 9 int send_dir_asheml(struct bufferevent *bufev, char *dirname, void* arg)
10 {
11     printf("******************send_dir name is %s\n",dirname);
12 
13     char* buf_dir_html = (char *)malloc(maxforgtml_); 
14     struct dirent **ptr;
15     int dir_total_num = scandir(dirname,&ptr,null,alphasort);
16     
17     //html head
18     sprintf(buf_dir_html,"<!doctype html>\
19                                 <html>\
20                                     <head>\
21                                         <title>curent dir:%s</title>\
22                                     </head>\
23                                     <body>\
24                                         <h1>curretn dir content:%s </h1>\
25                                         <table>\
26                                             <tr>\
27                                                 <td>name</td><td>size</td><td>type</td>\
28                                             </tr>",dirname,dirname);
29     bufferevent_write(bufev,buf_dir_html,strlen(buf_dir_html));
30 
31     for(int i=0;i<dir_total_num;++i)
32     {
33         char buf_current_name[1024] = {0};
34         if( 0 == strcmp(dirname,"./"))
35         {
36             sprintf(buf_current_name,"%s%s",dirname,ptr[i]->d_name);
37         }
38         else
39         {
40             sprintf(buf_current_name,"%s/%s",dirname,ptr[i]->d_name);
41         }
42         printf("++++++++++++++++++++send cur dir <name>>%s\n",buf_current_name);
43         struct stat st;
44         memset(&st,0,sizeof(st));
45         stat(buf_current_name,&st);
46     
47         sprintf(buf_dir_html,
48                                         "<tr>\
49                                             <td><a href=\"%s\">%s</a></td>\
50                                             <td>%ld</td>\
51                                             <td>%s</td>\
52                                         </tr>",
53                 buf_current_name,
54                 ptr[i]->d_name,st.st_size,
55                 judge_type_dir_or_nondir(buf_current_name)!= 0?"dir":"plain file");
56         bufferevent_write(bufev,buf_dir_html,strlen(buf_dir_html));
57         memset((char*)buf_dir_html,0,sizeof(buf_dir_html));
58     }
59 
60     //html end
61     sprintf(buf_dir_html,
62                                         "</table>\
63                                     </body>\
64                                 </html>");
65     bufferevent_write(bufev,buf_dir_html,strlen(buf_dir_html));
66     bufferevent_write(bufev,"\r\n",2);
67 
68     free(buf_dir_html);
69     return 0;
70 }

 

最后就是一些小函数了: 判断文件类型 和是否为目录:

  判断文件类型(这里如果出问题 打开图片等时会出现问题):

 1 // file name: _get_file_type.c
 2 // author: jiujue
 3 // created time: 2019年04月06日 星期六 19时14分07秒
 4 
 5 
 6 #include "head.h"
 7 
 8 const char *get_file_type(const char *name)
 9 {
10     char* dot;
11 
12     
13     dot = strrchr(name, '.');   
14     if (dot == null)
15         return "text/plain; charset=utf-8";
16     if (strcmp(dot, ".html") == 0 || strcmp(dot, ".htm") == 0)
17         return "text/html; charset=utf-8";
18     if (strcmp(dot, ".jpg") == 0 || strcmp(dot, ".jpeg") == 0)
19         return "image/jpeg";
20     if (strcmp(dot, ".gif") == 0)
21         return "image/gif";
22     if (strcmp(dot, ".png") == 0)
23         return "image/png";
24     if (strcmp(dot, ".css") == 0)
25         return "text/css";
26     if (strcmp(dot, ".au") == 0)
27         return "audio/basic";
28     if (strcmp( dot, ".wav" ) == 0)
29         return "audio/wav";
30     if (strcmp(dot, ".avi") == 0)
31         return "video/x-msvideo";
32     if (strcmp(dot, ".mov") == 0 || strcmp(dot, ".qt") == 0)
33         return "video/quicktime";
34     if (strcmp(dot, ".mpeg") == 0 || strcmp(dot, ".mpe") == 0)
35         return "video/mpeg";
36     if (strcmp(dot, ".vrml") == 0 || strcmp(dot, ".wrl") == 0)
37         return "model/vrml";
38     if (strcmp(dot, ".midi") == 0 || strcmp(dot, ".mid") == 0)
39         return "audio/midi";
40     if (strcmp(dot, ".mp3") == 0)
41         return "audio/mpeg";
42     if (strcmp(dot, ".ogg") == 0)
43         return "application/ogg";
44     if (strcmp(dot, ".pac") == 0)
45         return "application/x-ns-proxy-autoconfig";
46 
47     return "text/plain; charset=utf-8";
48 }

  判断是否为目录:

 1 // file name: _judge_type.c
 2 // author: jiujue
 3 // created time: 2019年04月05日 星期五 20时54分34秒
 4 
 5 #include "head.h"
 6 
 7 int judge_type_dir_or_nondir(const char* name)
 8 {
 9     struct stat st;
10     int ret = stat(name,&st);
11     if(-1 == ret)
12     {
13         return -1;
14     }
15     if(s_isreg(st.st_mode))
16     {
17         return 0;
18     }
19     if(s_isdir(st.st_mode))
20     {
21         return 1;
22     }
23     else
24     {
25         return 2;
26     }
27 
28 }
29 
30 
31 #if 0
32 int main(int argc,char* argv[])
33 {
34     int ret =  judge_type_dir_or_nondir(argv[1]);
35     if(ret == 1)
36     {
37         printf("is dir ");
38     }
39     if(ret == 0)
40     {
41         printf("is  file");
42     }
43     return 0;
44 }
45 #endif

注:以上代码已测验,基本没有问题(bug 肯定有 欢迎提出)

结语:有问题欢迎提在下方 ,本人在校学生,时间较为充裕, 有时间会回复的。

/* 原创文章 转载请附上原链接:    */