bevBufferevent转发服务器
来源:互联网 发布:c2c 源码 编辑:程序博客网 时间:2024/06/03 20:55
#include <netinet/in.h>
#include <sys/socket.h>
#include <fcntl.h>
#include "event2/event.h"
#include "event2/buffer.h"
#include "event2/bufferevent.h"
#include "event2/listener.h"
#include <assert.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#define MAX_RELAY_LENGTH 65535
#define MAX_MAGIC_ID 65535
#define MAGIC_LEN 16
#define FLAG 1
typedef struct bev_Buffer_ {
struct bufferevent *bev1;
struct bufferevent *bev2;
} bev_Buffer_t;
bev_Buffer_t *bevBuffer = NULL;
static pthread_mutex_t lock;
static short int MagicId[MAX_MAGIC_ID] = {0};
static short int client_id = 0;
static short int device_id = 0;
//struct bufferevent内建了两个event(read/write)和对应的缓冲区(struct evbuffer *input, *output),并提供相应的函数用来操作> 缓冲区(或者直接操作bufferevent)
//接收到数据后,判断是不一样一条消息的结束,结束标志为"over"字符串
static void readcb(struct bufferevent *bev, void *ctx)
{
#if 1
struct evbuffer *input, *output;
char *request_line;
size_t len;
int count = 0;
input = bufferevent_get_input(bev);//其实就是取出bufferevent中的input
output = bufferevent_get_output(bev);//其实就是取出bufferevent中的output
size_t input_len = evbuffer_get_length(input);
size_t output_len = evbuffer_get_length(output);
/************read msg from bufferevent***********/
char msg[MAX_RELAY_LENGTH+1] = {0};
len = bufferevent_read(bev, msg, sizeof(msg));
/*********get client and device MagicId**********/
if (NULL != strstr(msg,"client")) {
printf("msg =:%s len =:%d\n",msg,len);
if (len > MAGIC_LEN) {
client_id = ntohs(((short int*)msg)[0]);
if( (bevBuffer+client_id)->bev2 == 0) {
(bevBuffer+client_id)->bev2 = bev;
//pthread_mutex_unlock(&lock);//unlock mutex
printf("client_id come:%d\n",client_id);
}
if (device_id == client_id) {
for(count = 0; count < MAX_MAGIC_ID; count++) {
if (MagicId[count] == device_id) {
printf("this magic_id already have\n");
return;
} else if (MagicId[count] == 0) {
MagicId[count] = device_id;
printf("magic_id is:%d\n",MagicId[count]);
return;
}
}
}
} else {
client_id = ntohs(((short int*)msg)[0]);
if( (bevBuffer+client_id)->bev2 == 0) {
(bevBuffer+client_id)->bev2 = bev;
//pthread_mutex_unlock(&lock);//unlock mutex
printf("client_id come:%d\n",client_id);
return;
}
if (device_id == client_id) {
for(count = 0; count < MAX_MAGIC_ID; count++) {
if (MagicId[count] == device_id) {
printf("this magic_id already have\n");
return;
} else if (MagicId[count] == 0) {
MagicId[count] = device_id;
printf("magic_id is:%d\n",MagicId[count]);
return;
}
}
}
}
} else if (NULL != strstr(msg,"device")) {
device_id = ntohs(((short int*)msg)[0]);
if( (bevBuffer+device_id)->bev1== 0) {
(bevBuffer+device_id)->bev1 = bev;
//pthread_mutex_unlock(&lock);//unlock mutex
printf("device_id come:%d\n",device_id);
return;
}
}
/*************match bevbuffer and write**********/
for(count = 0; count < MAX_MAGIC_ID; count++) {
if (bev == (bevBuffer + MagicId[count])->bev1) {
if (NULL != (bevBuffer + MagicId[count])->bev2) {
printf("3# current bev is %d ,bev1 %d to bev2 %d\n",bev,(bevBuffer+MagicId[count])->bev1,(bevBuffer+MagicId[count])->bev2);
bufferevent_write((bevBuffer + MagicId[count])->bev2, msg, len);
//pthread_mutex_unlock(&lock);//unlock mutex
return;
}
} else if (bev == (bevBuffer + MagicId[count])->bev2) {
if (NULL != (bevBuffer + MagicId[count])->bev1) {
printf("4# current bev is %d bev2 %d to bev1 %d\n",bev,(bevBuffer+MagicId[count])->bev2,(bevBuffer+MagicId[count])->bev1);
bufferevent_write((bevBuffer + MagicId[count])->bev1, msg, len);
//pthread_mutex_unlock(&lock);//unlock mutex
return;
}
}
}
#endif
}
static void bevBuffer_reset(struct bufferevent *bev)
{
int i = 0;
for(; i < MAX_MAGIC_ID; i++)
{
if((bevBuffer + MagicId[i])->bev1 == bev)
{
printf("#1->i :%d bev1 :%d bev2 :%d magic_id :%d reset\n",i,(bevBuffer + MagicId[i])->bev1,(bevBuffer + MagicId[i])->bev2,MagicId[i]);
/*********reset bevBuffer with magicid****************/
(bevBuffer + MagicId[i])->bev1 = 0;
(bevBuffer + MagicId[i])->bev2 = 0;
MagicId[i] = 0;
return;
}
if((bevBuffer + MagicId[i])->bev2 == bev)
{
printf("#2->i :%d bev2 :%d bev1 :%d magic_id :%d reset\n",i,(bevBuffer + MagicId[i])->bev2,(bevBuffer + MagicId[i])->bev1,MagicId[i]);
/*********reset bevBuffer with magicid****************/
(bevBuffer + MagicId[i])->bev1 = 0;
(bevBuffer + MagicId[i])->bev2 = 0;
MagicId[i] = 0;
return;
}
}
}
static void errorcb(struct bufferevent *bev, short error, void *ctx)
{
if (error & BEV_EVENT_EOF)
{
/* connection has been closed, do any clean up here */
printf("connection closed\n");
}
else if (error & BEV_EVENT_ERROR)
{
/* check errno to see what error occurred */
printf("some other error\n");
}
else if (error & BEV_EVENT_TIMEOUT)
{
/* must be a timeout event handle, handle it */
printf("Timed out\n");
}
bevBuffer_reset(bev);
bufferevent_free(bev);
printf("errorcb done\n");
}
static void do_accept(evutil_socket_t listener, short event, void *arg)
{
struct event_base *base = (struct event_base *)arg;
//struct sockaddr_storage ss;
struct sockaddr_in ss;
socklen_t slen = sizeof(ss);
evutil_socket_t fd;
//int fd = accept(listener, (struct sockaddr*)&ss, &slen);
fd = accept(listener, (struct sockaddr*)&ss, &slen);
if (fd < 0)
{
perror("accept");
}
else if (fd > FD_SETSIZE)
{
close(fd);
}
else
{
printf("\n\naccept:<IP>:%s<Port>:%d\n",inet_ntoa(ss.sin_addr),ntohs(ss.sin_port));
struct bufferevent *bev;
evutil_make_socket_nonblocking(fd);
//使用bufferevent_socket_new创建一个struct bufferevent *bev,关联该sockfd,托管给event_base
//BEV_OPT_CLOSE_ON_FREE表示释放bufferevent时关闭底层传输端口。这将关闭底层套接字,释放底层bufferevent等。
bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
bevBuffer_reset(bev);
//设置读写对应的回调函数
printf("readcb start\n");
//mutex lock
//pthread_mutex_lock(&lock);
bufferevent_setcb(bev, readcb, NULL, errorcb, NULL);
printf("readcb done\n");
//bufferevent_setwatermark(bev, EV_READ, 0, MAX_LINE);
//启用读写事件,其实是调用了event_add将相应读写事件加入事件监听队列poll。正如文档所说,如果相应事件不置为true,bufferevent是不会读写数据的
bufferevent_enable(bev, EV_READ|EV_WRITE);
//printf("event enable\n");
}
}
void run(void)
{
evutil_socket_t listener;
struct sockaddr_in sin;
struct event_base *base;
struct event *listener_event;
base = event_base_new();
if (!base)
return; /*XXXerr*/
sin.sin_family = AF_INET;
sin.sin_addr.s_addr = 0;
//sin.sin_addr.s_addr = htonl(INADDR_ANY);
//sin.sin_port = htons(8800);
sin.sin_port = htons(11440);
listener = socket(AF_INET, SOCK_STREAM, 0);
evutil_make_socket_nonblocking(listener);
#ifndef WIN32
{
int one = 1;
setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));
}
#endif
if (bind(listener, (struct sockaddr*)&sin, sizeof(sin)) < 0)
{
perror("bind");
return;
}
if (listen(listener, 16)<0)
{
perror("listen");
return;
}
bevBuffer = malloc(MAX_MAGIC_ID*sizeof(bev_Buffer_t));
memset(bevBuffer, 0, MAX_MAGIC_ID*sizeof(bev_Buffer_t));
//mutex lock by ycs
//pthread_mutex_init(&lock,NULL);
printf("start listen \n");
listener_event = event_new(base, listener, EV_READ|EV_PERSIST, do_accept, (void*)base);
/*XXX check it */
event_add(listener_event, NULL);
event_base_dispatch(base);
printf("event dispatch\n");
}
int main(int argc, char **argv)
{
setvbuf(stdout, NULL, _IONBF, 0);
run();
return 0;
}
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
- #include <netinet/in.h>
- #include <sys/socket.h>
- #include <fcntl.h>
- #include <event2/event.h>
- #include <event2/buffer.h>
- #include <event2/bufferevent.h>
- #include <assert.h>
- #include <unistd.h>
- #include <string.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <errno.h>
- void do_read(evutil_socket_t fd, short events, void *arg);
- //struct bufferevent内建了两个event(read/write)和对应的缓冲区(struct evbuffer *input, *output),并提供相应的函数用来操作>
- 缓冲区(或者直接操作bufferevent)
- //接收到数据后,判断是不一样一条消息的结束,结束标志为"over"字符串
- void readcb(struct bufferevent *bev, void *ctx)
- {
- printf("called readcb!\n");
- struct evbuffer *input, *output;
- char *request_line;
- size_t len;
- input = bufferevent_get_input(bev);//其实就是取出bufferevent中的input
- output = bufferevent_get_output(bev);//其实就是取出bufferevent中的output
- size_t input_len = evbuffer_get_length(input);
- printf("input_len: %d\n", input_len);
- size_t output_len = evbuffer_get_length(output);
- printf("output_len: %d\n", output_len);
- while(1)
- {
- request_line = evbuffer_readln(input, &len, EVBUFFER_EOL_CRLF);//从evbuffer前面取出一行,用一个新分配的空字符结束
- 的字符串返回这一行,EVBUFFER_EOL_CRLF表示行尾是一个可选的回车,后随一个换行符
- if(NULL == request_line)
- {
- printf("The first line has not arrived yet.\n");
- free(request_line);//之所以要进行free是因为 line = mm_malloc(n_to_copy+1)),在这里进行了malloc
- break;
- }
- else
- <span style="white-space: pre;"> </span>{
- printf("Get one line date: %s\n", request_line);
- if(strstr(request_line, "over") != NULL)//用于判断是不是一条消息的结束
- {
- char *response = "Response ok! Hello Client!\r\n";
- evbuffer_add(output, response, strlen(response));//Adds data to an event buffer
- printf("服务端接收一条数据完成,回复客户端一条消息: %s\n", response);
- free(request_line);
- break;
- }
- }
- free(request_line);
- }
- size_t input_len1 = evbuffer_get_length(input);
- printf("input_len1: %d\n", input_len1);
- size_t output_len1 = evbuffer_get_length(output);
- printf("output_len1: %d\n\n", output_len1);
- }
- void errorcb(struct bufferevent *bev, short error, void *ctx)
- {
- if (error & BEV_EVENT_EOF)
- {
- /* connection has been closed, do any clean up here */
- printf("connection closed\n");
- }
- else if (error & BEV_EVENT_ERROR)
- {
- /* check errno to see what error occurred */
- printf("some other error\n");
- }
- else if (error & BEV_EVENT_TIMEOUT)
- {
- /* must be a timeout event handle, handle it */
- printf("Timed out\n");
- }
- bufferevent_free(bev);
- }
- void do_accept(evutil_socket_t listener, short event, void *arg)
- {
- struct event_base *base = arg;
- struct sockaddr_storage ss;
- socklen_t slen = sizeof(ss);
- int fd = accept(listener, (struct sockaddr*)&ss, &slen);
- if (fd < 0)
- {
- perror("accept");
- }
- else if (fd > FD_SETSIZE)
- {
- close(fd);
- }
- else
- {
- struct bufferevent *bev;
- evutil_make_socket_nonblocking(fd);
- //使用bufferevent_socket_new创建一个struct bufferevent *bev,关联该sockfd,托管给event_base
- ////BEV_OPT_CLOSE_ON_FREE表示释放bufferevent时关闭底层传输端口。这将关闭底层套接字,释放底层bufferevent等。
- bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
- //设置读写对应的回调函数
- bufferevent_setcb(bev, readcb, NULL, errorcb, NULL);
- // bufferevent_setwatermark(bev, EV_READ, 0, MAX_LINE);
- //启用读写事件,其实是调用了event_add将相应读写事件加入事件监听队列poll。正如文档所说,如果相应事件不置为true,buf
- ferevent是不会读写数据的
- bufferevent_enable(bev, EV_READ|EV_WRITE);
- }
- }
- void run(void)
- {
- evutil_socket_t listener;
- struct sockaddr_in sin;
- struct event_base *base;
- struct event *listener_event;
- base = event_base_new();
- if (!base)
- return; /*XXXerr*/
- sin.sin_family = AF_INET;
- sin.sin_addr.s_addr = 0;
- sin.sin_port = htons(8000);
- listener = socket(AF_INET, SOCK_STREAM, 0);
- evutil_make_socket_nonblocking(listener);
- #ifndef WIN32
- {
- int one = 1;
- setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));
- }
- #endif
- if (bind(listener, (struct sockaddr*)&sin, sizeof(sin)) < 0)
- {
- perror("bind");
- return;
- }
- if (listen(listener, 16)<0)
- {
- perror("listen");
- return;
- }
- listener_event = event_new(base, listener, EV_READ|EV_PERSIST, do_accept, (void*)base);
- /*XXX check it */
- event_add(listener_event, NULL);
- event_base_dispatch(base);
- }
- int main(int argc, char **argv)
- {
- setvbuf(stdout, NULL, _IONBF, 0);
- run();
- return 0;
- }
- bevBufferevent转发服务器
- Postfix转发服务器配置
- 多线程通信,服务器转发
- 服务器实现iptables转发
- 视频转发服务器构建
- http 请求转发服务器
- 服务器请求端口转发
- TCP转发服务器的例子
- RTSP流媒体转发服务器源码
- 服务器转发软件设计100元
- DNS服务器--转发解析请求
- SecureCRT 端口转发连接服务器
- vlc搭建流媒体转发服务器
- Exchange 邮件服务器无法转发邮件
- 应用SOCKS服务器实现P2P转发
- VC编写服务器转发式的聊天室
- 关于 darwin 转发服务器收集的资料
- 重定向跟服务器内部转发总结
- Android Studio如何集成Genymotion
- CPU卡发卡步骤
- open流程分析
- 原来 Java8 Stream 中的排序是插入排序
- C#[Win32&WinCE&WM]应用程序只能运行一个实例:MutexHelper
- bevBufferevent转发服务器
- Android apk 反编译
- 重载与构造函数的解析
- HTML5表格元素(20160805-0005)
- mongodb查看锁信息
- 写给一年后的自己
- Scheme版本修订进展
- Range Sum Query - Immutable
- 网络加载框架Retrofit2