使用libevent改进的boa服务器

源代码已经迁移到GitHub上。

https://github.com/tricky1997/boa-libevent


boa

boa服务器是一个比较简单的HTTP服务器。不同于Apache、nginx这种大家伙,boa代码量比较少,功能比较简单,常用于嵌入式设备。

对请求的处理方式是select+非阻塞I/O。

主要的数据结构有:

三个请求队列:request_ready,request_block,request_free。

两个fdset: block_read_fdset,block_write_fdset;

主要的设计思想是:在一个while(1)循环里,通过不断的select等待以及对请求队列的轮询,来完成HTTP请求的处理及应答。可以简单概括为“等待+轮询request队列”的方式。

详细些的工作原理可以参看boa源码,以及博客里的《BOA学习笔记》系列。


libevent

提供了跨平台的高效多路I/O函数的封装。支持的后端有/dev/pollkqueue(2)select(2),poll(2)epoll(4) ,Windows select,Solaris's event ports等。

可以监听定时器,信号,以及文件描述符等。

目前流行的轻量级但高效灵活的nginx就是使用的libevent开发。


主要思想:

使用libevent对boa的源代码进行了简单修改:

主体流程不变,没有完全使用异步事件处理机制,依然沿用了无限“等待+轮询request队列”的方式,依然使用了fdset和三个request队列;

使用event_base_loop+event_base_loopexit代替了select调用,使用event_base来托管信号,两个fdset依然保持原来含义,不过不再使用fdset+select来监听事件;


修改后,完全用更高效的libevent替代了select。

改进的boa其实没有太大意义。毕竟成熟的、经得住高流量的服务器软件有nginx、apache、lighttpd等。在嵌入式等要求苛刻的环境下,select一般足以应付连接。


PS:

我只是把它当作libevent的练手小程序了,各位发现bug的话欢迎讨论。

由于没什么测试经验,只在本机上运行,进行了很简单的测试:

打开firefox和chrome试试访问几个页面;

编写了个脚本文件,curl迭代GET一万次,看看是否挂掉或者死循环。

#!/bin/bash
times=0
while [ $times != 10000 ]
do
    curl 127.0.0.1:9001/1/ -silent > /dev/null
    times=`expr $times + 1`
done
没有进行任何高并发情况下的测试,其实我真的很想知道libevent用epoll处理高并发到底多牛逼:(。


主要修改部分:

源代码里,修改的地方一般加上了 /* XXX tricky1997 */的注释,可以全局查找。

主要修改的文件有,boa.c,select.c,signal.c,global.h,boa.h。并新增加了event.c文件。

以下是修改的主要部分:

select.c的修改:

while (1) {
      
        if (sigterm_flag) {
            if (sigterm_flag == 1)
                sigterm_stage1_run(server_s);
            if (sigterm_flag == 2 && !request_ready && !request_block) {
                sigterm_stage2_run();
            }
        }
   
        time(&t_time);

        if (request_block)
            /* move selected req's from request_block to request_ready */
            fdset_update();

        /* any blocked req's move from request_ready to request_block */
        process_requests(server_s);

        if (!sigterm_flag && total_connections < (max_connections - 10)) {
            //BOA_FD_SET(server_s, &block_read_fdset); /* server always set */
            /* XXX tricky1997 */            
            event_add(server_event, NULL);
        }

        req_timeout.tv_sec = (request_ready ? 0 :
                              (ka_timeout ? ka_timeout : REQUEST_TIMEOUT));
        req_timeout.tv_usec = 0l;   /* reset timeout */

		/* XXX trickt1997 */
		FD_ZERO(&block_read_fdset);
		FD_ZERO(&block_write_fdset);
		if( (request_ready || request_block) )
			event_base_loopexit(boa_event_base, &req_timeout);
		event_base_loop(boa_event_base, EVLOOP_ONCE) ;
    }
}


signal.c的修改:

void init_signals(void)
{
    struct sigaction sa;
    struct event *sigsegv_event, *sigbus_event, *sigint_event, *sigterm_event, *sighup_event, *sigalrm_event, *sigchld_event;

    sa.sa_flags = 0;

    sigemptyset(&sa.sa_mask);

    sigaddset(&sa.sa_mask, SIGTERM);
    sigaddset(&sa.sa_mask, SIGHUP);
    sigaddset(&sa.sa_mask, SIGPIPE);
    sigaddset(&sa.sa_mask, SIGCHLD);
    sigaddset(&sa.sa_mask, SIGALRM);
    sigaddset(&sa.sa_mask, SIGUSR1);
    sigaddset(&sa.sa_mask, SIGUSR2);
    
    sigsegv_event = event_new(boa_event_base, SIGSEGV, EV_SIGNAL|EV_PERSIST, sigsegv, NULL );
    event_add(sigsegv_event, NULL);

    /* ...其他的event */

    sigalrm_event = event_new(boa_event_base, SIGALRM, EV_SIGNAL|EV_PERSIST, sighup, NULL );
    event_add(sigalrm_event, NULL);

    sa.sa_handler = SIG_IGN;
    sigaction(SIGPIPE, &sa, NULL);
    sigaction(SIGUSR1, &sa, NULL);
    sigaction(SIGUSR2, &sa, NULL);
}

event.c里的函数:

/* used in BOA_FD_SET */
void event_alloc(int fd, fd_set* where){
    int flags = 0;
	struct event_entry* entry;

    if(where == &block_read_fdset)
		flags |= EV_READ;
	else
		flags |= EV_WRITE;
	
	entry = get_event_entry();
	
	event_assign(entry->event, boa_event_base, fd, flags, event_happened, entry);
	event_add(entry->event, NULL);
}

/* normal event handler */
void event_happened(evutil_socket_t fd, short what, void *arg)
{
	struct event_entry* entry = (struct event_entry*) arg;

	if (fd > max_fd) max_fd = fd; 
	if (what & EV_READ)
		FD_SET(fd, &block_read_fdset);
	else
		FD_SET(fd, &block_write_fdset);

	dequeue_event(&event_used, entry);
	enqueue_event(&event_unused, entry);

	event_del(entry->event); 
	//entry->event = NULL;
}



  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
作为一名AI,我并不懂得如何使用libevent,但是我可以为你提供一些libevent的基础知识和参考资料。 libevent是一个开源的事件驱动库,它提供了一种跨平台的方式来处理网络事件。它可以用于编写高性能的网络应用程序,如Web服务器、代理服务器、消息队列等。libevent支持多种网络协议,包括TCP、UDP、HTTP等。 要使用libevent实现WebSocket服务器和客户端,你需要先了解WebSocket协议的基础知识。WebSocket是一种基于TCP的全双工协议,它允许客户端和服务器之间进行双向通信。WebSocket协议的核心是建立一个长时间的TCP连接,然后通过发送HTTP请求和响应来升级到WebSocket连接。一旦WebSocket连接建立,客户端和服务器可以通过发送消息进行通信。 在使用libevent实现WebSocket服务器和客户端时,你需要使用libevent提供的事件循环机制来处理网络事件。你可以创建一个事件循环,并注册事件回调函数来处理不同类型的网络事件,如TCP连接、HTTP请求、WebSocket消息等。在处理WebSocket消息时,你需要按照WebSocket协议的规范解析消息,并根据消息类型进行相应的处理。 下面是一些参考资料,可以帮助你更好地了解libevent和WebSocket协议: 1. libevent官方网站:http://libevent.org/ 2. WebSocket协议规范:https://tools.ietf.org/html/rfc6455 3. libevent实现WebSocket服务器的示例代码:https://github.com/libevent/libevent/blob/master/sample/websocket-server.c 4. libevent实现WebSocket客户端的示例代码:https://github.com/libevent/libevent/blob/master/sample/websocket-client.c 希望这些资料可以帮助你更好地理解使用libevent实现WebSocket服务器和客户端的基本原理。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值