代码拉取完成,页面将自动刷新
同步操作将从 linze/TinyWebServer 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <cassert>
#include <sys/epoll.h>
#include "./lock/locker.h"
#include "./threadpool/threadpool.h"
#include "./timer/lst_timer.h"
#include "./http/http_conn.h"
#include "./log/log.h"
#include "./CGImysql/sql_connection_pool.h"
#define MAX_FD 65536 //最大文件描述符
#define MAX_EVENT_NUMBER 10000 //最大事件数
#define TIMESLOT 5 //最小超时单位
//这三个函数在http_conn.cpp中定义,改变链接属性
extern int addfd(int epollfd,int fd,bool one_shot);
extern int remove(int epollfd,int fd);
extern int setnonblocking( int fd );
//设置定时器相关参数
static int pipefd[2];
static sort_timer_lst timer_lst;
static int epollfd = 0;
//信号处理函数
void sig_handler( int sig )
{
//为保证函数的可重入性,保留原来的errno
int save_errno = errno;
int msg = sig;
send( pipefd[1], ( char* )&msg, 1, 0 );
errno = save_errno;
}
//设置信号函数
void addsig(int sig,void(handler)(int),bool restart=true)
{
struct sigaction sa;
memset(&sa,'\0',sizeof(sa));
sa.sa_handler=handler;
if(restart)
sa.sa_flags |= SA_RESTART;
sigfillset(&sa.sa_mask);
assert(sigaction(sig,&sa,NULL)!=-1);
}
//定时处理任务,重新定时以不断触发SIGALRM信号
void timer_handler()
{
timer_lst.tick();
alarm( TIMESLOT );
}
//定时器回调函数,删除非活动连接在socket上的注册事件,并关闭
void cb_func( client_data* user_data )
{
epoll_ctl( epollfd, EPOLL_CTL_DEL, user_data->sockfd, 0 );
assert( user_data );
close( user_data->sockfd );
LOG_INFO("close fd %d", user_data->sockfd);
Log::get_instance()->flush();
//printf( "close fd %d\n", user_data->sockfd );
}
void show_error(int connfd,const char* info)
{
printf("%s",info);
send(connfd,info,strlen(info),0);
close(connfd);
}
//设置信号为LT阻塞模式
void addfd_(int epollfd,int fd,bool one_shot)
{
epoll_event event;
event.data.fd=fd;
event.events=EPOLLIN|EPOLLRDHUP;
if(one_shot)
event.events|=EPOLLONESHOT;
epoll_ctl(epollfd,EPOLL_CTL_ADD,fd,&event);
}
int main(int argc,char *argv[])
{
//Log::get_instance()->init("./mylog.log",8192,2000000,10);//异步日志模型
Log::get_instance()->init("./mylog.log",8192,2000000,0);//同步日志模型
if(argc<=1)
{
printf("usage: %s ip_address port_number\n",basename(argv[0]));
return 1;
}
//const char* ip=argv[1];
int port=atoi(argv[1]);
//忽略SIGPIPE信号
addsig(SIGPIPE,SIG_IGN);
//创建线程池
threadpool<http_conn>* pool=NULL;
try
{
pool=new threadpool<http_conn>;
}
catch(...){
return 1;
}
//单例模式创建数据库连接池
connection_pool *connPool=connection_pool::GetInstance("localhost","root","root","qgydb",3306,5);
http_conn* users=new http_conn[MAX_FD];
assert(users);
int user_count=0;
//初始化数据库读取表
users->initmysql_result();
//创建套接字,返回listenfd
int listenfd=socket(PF_INET,SOCK_STREAM,0);
assert(listenfd>=0);
//struct linger tmp={1,0};
//SO_LINGER若有数据待发送,延迟关闭
//setsockopt(listenfd,SOL_SOCKET,SO_LINGER,&tmp,sizeof(tmp));
int ret=0;
struct sockaddr_in address;
bzero(&address,sizeof(address));
address.sin_family=AF_INET;
//inet_pton(AF_INET,ip,&address.sin_addr);
address.sin_addr.s_addr=htonl(INADDR_ANY);
address.sin_port=htons(port);
// 设置端口复用,绑定端口
int flag = 1;
setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &flag, sizeof(flag));
ret=bind(listenfd,(struct sockaddr*)&address,sizeof(address));
//printf("bind ret = %d\n", ret);
assert(ret>=0);
ret=listen(listenfd,5);
assert(ret>=0);
//创建内核事件表
epoll_event events[MAX_EVENT_NUMBER];
int epollfd=epoll_create(5);
assert(epollfd!=-1);
addfd_(epollfd,listenfd,false);
http_conn::m_epollfd=epollfd;
//创建管道
ret = socketpair( PF_UNIX, SOCK_STREAM, 0, pipefd );
assert( ret != -1 );
setnonblocking( pipefd[1] );
addfd( epollfd, pipefd[0], false);
// add all the interesting signals here
addsig(SIGALRM, sig_handler, false);
addsig(SIGTERM, sig_handler, false);
bool stop_server = false;
client_data* users_timer = new client_data[MAX_FD];
bool timeout = false;
alarm( TIMESLOT );
//printf("监听......\n");
while(!stop_server)
{
int number=epoll_wait(epollfd,events,MAX_EVENT_NUMBER,-1);
if(number<0&&errno!=EINTR)
{
//printf("epoll failure\n");
LOG_ERROR("%s","epoll failure");
break;
}
for(int i=0;i<number;i++)
{
int sockfd=events[i].data.fd;
//处理新到的客户连接
if(sockfd==listenfd)
{
struct sockaddr_in client_address;
socklen_t client_addrlength=sizeof(client_address);
int connfd=accept(listenfd,(struct sockaddr*)&client_address,&client_addrlength);
if(connfd<0)
{
//printf("errno is:%d\n",errno);
LOG_ERROR("%s:errno is:%d","accept error",errno);
continue;
}
if(http_conn::m_user_count>=MAX_FD)
{
show_error(connfd,"Internal server busy");
LOG_ERROR("%s","Internal server busy");
continue;
}
users[connfd].init(connfd,client_address);
//初始化client_data数据
//创建定时器,设置回调函数和超时时间,绑定用户数据,将定时器添加到链表中
users_timer[connfd].address = client_address;
users_timer[connfd].sockfd = connfd;
util_timer* timer = new util_timer;
timer->user_data = &users_timer[connfd];
timer->cb_func = cb_func;
time_t cur = time( NULL );
timer->expire = cur + 3 * TIMESLOT;
users_timer[connfd].timer = timer;
timer_lst.add_timer( timer );
}
else if(events[i].events & (EPOLLRDHUP | EPOLLHUP|EPOLLERR))
{
users[sockfd].close_conn();
//服务器端关闭连接,移除对应的定时器
cb_func( &users_timer[sockfd] );
util_timer *timer=users_timer[sockfd].timer;
if( timer )
{
timer_lst.del_timer( timer );
}
}
//处理信号
else if( ( sockfd == pipefd[0] ) && ( events[i].events & EPOLLIN ) )
{
int sig;
char signals[1024];
ret = recv( pipefd[0], signals, sizeof( signals ), 0 );
if( ret == -1 )
{
// handle the error
continue;
}
else if( ret == 0 )
{
continue;
}
else
{
for( int i = 0; i < ret; ++i )
{
switch( signals[i] )
{
case SIGALRM:
{
timeout = true;
break;
}
case SIGTERM:
{
stop_server = true;
}
}
}
}
}
//处理客户连接上接收到的数据
else if(events[i].events&EPOLLIN)
{
util_timer* timer = users_timer[sockfd].timer;
if(users[sockfd].read_once()){
LOG_INFO("deal with the client(%s)",inet_ntoa(users[sockfd].get_address()->sin_addr));
Log::get_instance()->flush();
//若监测到读事件,将该事件放入请求队列
pool->append(users+sockfd);
//若有数据传输,则将定时器往后延迟3个单位
//并对新的定时器在链表上的位置进行调整
if( timer )
{
time_t cur = time( NULL );
timer->expire = cur + 3 * TIMESLOT;
//printf( "adjust timer once\n" );
LOG_INFO("%s","adjust timer once");
Log::get_instance()->flush();
timer_lst.adjust_timer( timer );
}
}
else
{
users[sockfd].close_conn();
cb_func( &users_timer[sockfd] );
if( timer )
{
timer_lst.del_timer( timer );
}
}
}
else if(events[i].events&EPOLLOUT)
{
if(!users[sockfd].write())
users[sockfd].close_conn();
}
//else
//{
//}
}
if(timeout){
timer_handler();
timeout=false;
}
}
close(epollfd);
close(listenfd);
close(pipefd[1]);
close(pipefd[0]);
delete [] users;
delete [] users_timer;
delete pool;
//销毁数据库连接池
connPool->DestroyPool();
return 0;
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。