LibEvent-Demo

2023-05-16

// libevent-test.cpp : 定义控制台应用程序的入口点。
//

//#include "stdafx.h"

#pragma comment(lib,"ws2_32.lib")
#pragma comment(lib,"wsock32.lib")
#pragma comment(lib,"libevent.lib")
#pragma comment(lib,"libevent_core.lib")
#pragma comment(lib,"libevent_extras.lib")


#include <stdio.h>
#include <stdlib.h>
#include <evhttp.h>
#include <event.h>
#include <string.h>
#include "event2/http.h"
#include "event2/event.h"
#include "event2/buffer.h"
#include "event2/bufferevent.h"
#include "event2/bufferevent_compat.h"
#include "event2/http_struct.h"
#include "event2/http_compat.h"
#include "event2/util.h"
#include "event2/listener.h"

#define BUF_MAX 1024*16
#define RESTFUL_URL_KHREALTIMEDATA		"/api/v1/khrealtimedata"
#define RESTFUL_URL_KHQUERYDATA			"/api/v1/khquerydata"
#define RESTFUL_URL_KHINSERT			"/api/v1/khinsert"
#define RESTFUL_URL_KHVARIABLESNUMBER	"/api/v1/khvariablesnumber"
#define RESTFUL_URL_KHVARIABLES			"/api/v1/khvariables"
#define RESTFUL_URL_KHADDTAGS			"/api/v1/khaddtags"
#define RESTFUL_URL_KHDATASAMPLING		"/api/v1/khsamplingdata"
#define RESTFUL_URL_REGISTERCLIENT		"/api/v1/khregisterclient"	//注册客户端
//解析post请求数据
void get_post_message(char *buf, struct evhttp_request *req)
{
	size_t post_size = 0;

	post_size = evbuffer_get_length(req->input_buffer);//获取数据长度
	printf("====line:%d,post len:%d\n",__LINE__,post_size);
	if (post_size <= 0)
	{
		printf("====line:%d,post msg is empty!\n",__LINE__);
		return;
	}
	else
	{
		size_t copy_len = post_size > BUF_MAX ? BUF_MAX : post_size;
		printf("====line:%d,post len:%d, copy_len:%d\n",__LINE__,post_size,copy_len);
		memcpy(buf, evbuffer_pullup(req->input_buffer,-1), copy_len);
		buf[post_size] = '\0';
		printf("====line:%d,post msg:%s\n",__LINE__,buf);
	}
}

//解析http头,主要用于get请求时解析uri和请求参数
char *find_http_header(struct evhttp_request *req,struct evkeyvalq *params,const char *query_char)
{
	if(req == NULL || params == NULL || query_char == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"input params is null.");
		return NULL;
	}

	struct evhttp_uri *decoded = NULL;
	char *query = NULL;	
	char *query_result = NULL;
	const char *path;
	const char *uri = evhttp_request_get_uri(req);//获取请求uri

	if(uri == NULL)
	{
		printf("====line:%d,evhttp_request_get_uri return null\n",__LINE__);
		return NULL;
	}
	else
	{
		printf("====line:%d,Got a GET request for <%s>\n",__LINE__,uri);
	}

	//解码uri
	decoded = evhttp_uri_parse(uri);
	if (!decoded) 
	{
		printf("====line:%d,It's not a good URI. Sending BADREQUEST\n",__LINE__);
		evhttp_send_error(req, HTTP_BADREQUEST, 0);
		return NULL;
	}

	//获取uri中的path部分
	path = evhttp_uri_get_path(decoded);
	if (path == NULL) 
	{
		path = "/";
	}
	else
	{
		printf("====line:%d,path is:%s\n",__LINE__,path);
	}

	//获取uri中的参数部分
	query = (char*)evhttp_uri_get_query(decoded);
	if(query == NULL)
	{
		printf("====line:%d,evhttp_uri_get_query return null\n",__LINE__);
		return NULL;
	}

	//查询指定参数的值
	evhttp_parse_query_str(query, params);			
	query_result = (char*)evhttp_find_header(params, query_char);

	return query_result;
}

//处理get请求
void http_handler_testget_msg(struct evhttp_request *req,void *arg)
{
	if(req == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"input param req is null.");
		return;
	}

	char *sign = NULL;
	char *data = NULL;
	struct evkeyvalq sign_params = {0};
	sign = find_http_header(req,&sign_params,"sign");//获取get请求uri中的sign参数
	if(sign == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"request uri no param sign.");
	}
	else
	{
		printf("====line:%d,get request param: sign=[%s]\n",__LINE__,sign);
	}

	data = find_http_header(req,&sign_params,"data");//获取get请求uri中的data参数
	if(data == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"request uri no param data.");
	}
	else
	{
		printf("====line:%d,get request param: data=[%s]\n",__LINE__,data);
	}
	printf("\n");

	//回响应
	struct evbuffer *retbuff = NULL;
	retbuff = evbuffer_new();
	if(retbuff == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"retbuff is null.");
		return;
	}
	evbuffer_add_printf(retbuff,"Receive get request,Thamks for the request!");
	evhttp_send_reply(req,HTTP_OK,"Client",retbuff);
	evbuffer_free(retbuff);
}

//处理post请求
void http_handler_testpost_msg(struct evhttp_request *req,void *arg)
{
	if(req == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"input param req is null.");
		return;
	}

	char buf[BUF_MAX] = {0};
	get_post_message(buf, req);//获取请求数据,一般是json格式的数据
	if(buf == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"get_post_message return null.");
		return;
	}
	else
	{
		//可以使用json库解析需要的数据
		printf("====line:%d,request data:%s",__LINE__,buf);
	}

	//回响应
	struct evbuffer *retbuff = NULL;
	retbuff = evbuffer_new();
	if(retbuff == NULL)
	{
		printf("====line:%d,%s\n",__LINE__,"retbuff is null.");
		return;
	}
	evbuffer_add_printf(retbuff,"Receive post request,Thamks for the request!");
	evhttp_send_reply(req,HTTP_OK,"Client",retbuff);
	evbuffer_free(retbuff);
}
void generic_handler(struct evhttp_request *req, void *arg)
{
	struct evbuffer *buf = evbuffer_new();
	if(!buf)
	{
		puts("failed to create response buffer \n");
		return;
	}

	evbuffer_add_printf(buf, "Server Responsed. Requested: %s\n", evhttp_request_get_uri(req));
	evhttp_send_reply(req, HTTP_OK, "OK", buf);
	evbuffer_free(buf);
}
int init_win_socket()
{
	WSADATA wsaData;
	if(WSAStartup(MAKEWORD(2,2) , &wsaData) != 0) 
	{
		return -1;
	}
	return 0;
}
///
//             http:对每个事件分别处理					//
///
// int main()
// {
// 
// 	WSADATA wsaData;
// 	if(WSAStartup(MAKEWORD(2,2) , &wsaData) != 0) 
// 	{
// 		return -1;
// 	}
// 	short http_port = 4567;
// 	char *http_addr = "127.0.0.1";
// 
// 	//初始化
// 	event_init();
// 	//启动http服务端
// 	struct evhttp *http_server = evhttp_start(http_addr,http_port);
// 	if(http_server == NULL)
// 	{
// 		printf("====line:%d,%s\n",__LINE__,"http server start failed.");
// 		return -1;
// 	}
// 
// 	//设置请求超时时间(s)
// 	evhttp_set_timeout(http_server,5);
// 	//设置事件处理函数,evhttp_set_cb针对每一个事件(请求)注册一个处理函数,
// 	//区别于evhttp_set_gencb函数,是对所有请求设置一个统一的处理函数
// 	evhttp_set_cb(http_server,RESTFUL_URL_KHQUERYDATA,http_handler_testpost_msg,NULL);
// 	evhttp_set_cb(http_server,"/me/testget",http_handler_testget_msg,NULL);
// 
// 
// 
// 	//循环监听
// 	event_dispatch();
// 	//实际上不会释放,代码不会运行到这一步
// 	evhttp_free(http_server);
// 
// 	return 0;
// }
/
//				http:对所有请求统一由一个函数处理													//

int main(int argc, char* argv[])
{
#ifdef WIN32
	init_win_socket();
#endif

	short          http_port = 4567;
	char          *http_addr = "127.0.0.1";

	struct event_base * base = event_base_new();

	struct evhttp * http_server = evhttp_new(base);
	if(!http_server)
	{
		return -1;
	}

	int ret = evhttp_bind_socket(http_server,http_addr,http_port);
	if(ret!=0)
	{
		return -1;
	}

	evhttp_set_gencb(http_server, generic_handler, NULL);

	printf("http server start OK! \n");

	event_base_dispatch(base);

	evhttp_free(http_server);

	WSACleanup();
	return 0;
}
//
//					服务端程序demo
//
// #include <stdio.h>
// #include <time.h>
// 
// #include <event2/bufferevent.h>
// #include <event2/buffer.h>
// #include <event2/listener.h>
// #include <event2/util.h>
// #include <event2/event.h>
// 
// const int PORT = 2500;
// const int BUFFER_SIZE = 1024;
// 
// void listener_cb(struct evconnlistener *, evutil_socket_t,
// struct sockaddr *, int socklen, void *);
// void conn_writecb(struct bufferevent *, void *);
// void conn_readcb(struct bufferevent *, void *);
// void conn_eventcb(struct bufferevent *, short, void *);
// void delay(int ms);
// 
// int main(int argc, char **argv)
// {
// 	printf("I am server\n");
// #ifdef WIN32
// 	WSAData wsaData;
// 	WSAStartup(MAKEWORD(2, 0), &wsaData);
// #endif
// 	struct sockaddr_in sin;
// 	memset(&sin, 0, sizeof(sin));
// 	sin.sin_family = AF_INET;
// 	sin.sin_port = htons(PORT);
// 
// 	struct evconnlistener *listener;
// 	struct event_base *base = event_base_new();
// 	if (!base)
// 	{
// 		printf("Could not initialize libevent\n");
// 		return 1;
// 	}
// 	listener = evconnlistener_new_bind(base, listener_cb, (void *)base,
// 		LEV_OPT_REUSEABLE|LEV_OPT_CLOSE_ON_FREE, -1,
// 		(struct sockaddr*)&sin,
// 		sizeof(sin));
// 
// 	if (!listener)
// 	{
// 		printf("Could not create a listener\n");
// 		return 1;
// 	}
// 
// 	event_base_dispatch(base);
// 	evconnlistener_free(listener);
// 	event_base_free(base);
// 
// 	printf("done\n");
// 	return 0;
// }
// 
// void listener_cb(struct evconnlistener *listener, evutil_socket_t fd,struct sockaddr *sa, int socklen, void *user_data)
// {
// 	printf("Detect an connection\n");
// 	struct event_base *base = (struct event_base *)user_data;
// 	struct bufferevent *bev;
// 
// 	bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
// 	if (!bev)
// 	{
// 		printf("Could not create a bufferevent\n");
// 		event_base_loopbreak(base);
// 		return;
// 	}
// 	bufferevent_setcb(bev, conn_readcb, conn_writecb, conn_eventcb, NULL);
// 	bufferevent_enable(bev, EV_READ|EV_WRITE);
// 	//服务器监听到连接时,给客户端发送第一条消息
// 	char *reply_msg = "I receive a message from server";
// 	bufferevent_write(bev, reply_msg, strlen(reply_msg));
// }
// 
// //conn_writecwritecb函数将在bufferevent中的output evbuffer缓冲区发送完成后被调用。
// //此时evbuffer_get_length(output) = 0,说明output evbuffer缓冲区被清空。
// //假设发现有10000条记录要发送出去,1次发送10000条将占用大量内存,所以,我们要分批发送
// //先发送100条数据,假设每条数据为1024字节bufferevent_write(bev,buf,1024 *100);
// //系统在这100条记录发送完成后将调用conn_writecbb回调函数,然后在该函数中循环发送剩下的
// //数据
// void conn_writecb(struct bufferevent *bev, void *user_data)
// {
// 	//    struct evbuffer *output = bufferevent_get_output(bev);
// 	//    if (evbuffer_get_length(output) == 0)
// 	//    {
// 	//        printf("Output evbuffer is flushed\n");
// 	//        bufferevent_free(bev);
// 	//    }
// 	//delay 1 second
// 	delay(1000);
// 	static int msg_num = 1;
// 	char reply_msg[1000] = {'\0'};
// 	char *str = "I receive a message from server ";
// 	memcpy(reply_msg,str,strlen(str));
// 	sprintf(reply_msg+strlen(str),"%d",msg_num);
// 	bufferevent_write(bev, reply_msg, strlen(reply_msg));
// 	msg_num++;
// }
// 
// void conn_readcb(struct bufferevent *bev, void *user_data)
// {
// 	struct evbuffer *input =bufferevent_get_input(bev);
// 	size_t sz=evbuffer_get_length(input);
// 	if (sz > 0)
// 	{
// 		char msg[BUFFER_SIZE] = {'\0'};
// 		bufferevent_read(bev, msg, sz);
// 		printf("%s\n", msg);
// 	}
// }
// 
// void conn_eventcb(struct bufferevent *bev, short events, void *user_data)
// {
// 	if (events & BEV_EVENT_EOF)
// 	{
// 		printf("Connection closed\n");
// 	}
// 	else if (events & BEV_EVENT_ERROR)
// 	{
// 		printf("Got an error on the connection: %s\n",strerror(errno));
// 	}
// 
// 	bufferevent_free(bev);
// }
// 
// void delay(int ms)
// {
// 	clock_t start = clock();
// 	while(clock() - start < ms);
// }

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

LibEvent-Demo 的相关文章

随机推荐

  • ROI管理与客户需求管理

    转自 xff1a http semwatch org 2010 02 roi management and client demand 作者 xff1a Gaoge 日期 xff1a 二月 26 2010 分类 xff1a 付费搜索SEM
  • 人脸识别过程

    做人脸识别的朋友很多不知道人脸识别过程 xff0c 人脸识别厂家人脸识别过程 人脸识别第一步先进行图像采集 xff0c 人脸定位 xff0c 特征提取 xff0c 特征对比 xff0c 人脸资料存储 xff0c 识别成功就可以开门 xff0
  • 个人博客的三种方式

    创建个人博客的三种方式 1 博客平台 简书 CSDN 知乎专栏等 特点 xff1a 简单 xff0c 可控性低 申请个账号就可以写 xff0c 其他的都不用管 限制多 xff0c 界面 排版 有没有广告自己说了不算 xff0c 什么能发什么
  • GitHub 简介

    用详细的图文对GitHub进行简单的介绍 git是一个版本控制工具 xff0c github是一个用git做版本控制的项目托管平台 主页介绍 xff1a overview xff1a 总览 相当于个人主页 repositories xff1
  • Markdown 编辑器推荐和常用语法介绍

    Markdown 是非常好用的文档编写方式 下面是我的 Markdown 学习总结 不求详尽 xff0c 只求简明 编辑器 xff1a 我目前使用的 Markdown 编辑器是 Joplin xff0c 感觉挺好用 我对编辑器的需求是 xf
  • 使用 Hexo 在 Github 上建博客

    先确认 git 与 npm 已经安装 xff0c 在终端输入以下命令 git version npm version 安装 hexo xff0c 在终端输入 npm install hexo cli g 安装过程中如果报错 解决方法 xff
  • ping命令的过程

    ping命令的过程 1 ping是什么 xff1f PING Packet Internet Groper xff0c 因特网包探索器 xff0c 用于测试网络连通性的程序 Ping发送一个ICMP Internet Control Mes
  • Hexo 更换主题

    更换 Hexo 主题非常容易 xff0c 只要在 themes 文件夹内 xff0c 新增一个任意名称的文件夹 xff0c 并修改 config yml 内的 theme 设定 xff0c 即可切换主题 具体步骤 xff1a 1 安装主题
  • Python 爬虫零基础教程(0):简介及准备

    其他的教程往往从语法开始 xff0c 而我们直接开始爬虫 xff0c 语法等知识边做边学 这第0篇我们简单介绍下爬虫和编程工具 爬虫是什么 爬虫是自动浏览 保存网页内容的程序或脚本 爬虫不同于黑客 xff0c 爬虫爬取的是允许访问的内容 工
  • 如何解决Unable to parse template "Interface"Error Message;

    Unable to parse template Interface Error Message This Template did not Produce a Java Class or an interface关于这个错误 xff0c
  • inflate函数使用总结

    inflate 两个参数和三个参数的区别 以前使用没有关注过 xff0c 因为觉得没报bug就行了 xff0c 两个三个参数无所谓 xff0c 经过导师提醒 xff0c 决定好好看看源码和相关知识 xff0c 总觉一下区别 xff0c 以免
  • 除了csdn网站打不开,一直刷新等待没反应,其他网站都能正常访问

    DNS错误 自动获取DNS服务器地址 清除DNS缓存信息 ipconfig span class token operator span flushdns 重置winsock 目录设置 netsh winsock reset
  • org.slf4j用法

    org slf4j用法 Scala 1 创建Logger对象 private val logger Logger 61 LoggerFactory getLogger classOf HttpBmlClient 2 打印错误信息 同时抛出异
  • 安装图形界面、VNCserver

    centos7 安装图形界面 xff1a 第一步 xff1a 安装Gnome包 在命令行下 输入下面的命令来安装Gnome包 yumgroupinstall 34 GNOMEDesktop 34 34 GraphicalAdministra
  • MySQL全量、增量备份与恢复的简单方法

    本文主要给大家介绍MySQL全量 增量备份与恢复的简单方法 xff0c 文章内容都是笔者用心摘选和编辑的 xff0c 具有一定的针对性 xff0c 对大家的参考意义还是比较大的 xff0c 下面跟笔者一起了解下MySQL全量 增量备份与恢复
  • windows安装zabbix代理

    一 关闭windows防火墙或者开通10050和10051端口 直接windows关闭防火墙或者在防火墙中放行10050和10051 二 xff0e 下载 安装并修改windows代理 1 下载zabbix agentd包 官网下载地址 x
  • android8.0 Fingerprint 指纹输错5次后亮屏显示错误信息

    当有指纹解锁时 xff0c 会执行AuthenticationClient java gt onAuthenticated 一直在监听解锁行为 64 Override public boolean onAuthenticated int f
  • secureCRT 抓取串口数据

    language 61 34 VBScript 34 interface 61 34 1 0 34 Dim outputFile fout Dim outputPath Dim user outputPath 61 34 D out txt
  • 2022年编程语言热度排行榜来啦,快来看看你学习的语言排第几

    提示 xff1a 文章写完后 xff0c 目录可以自动生成 xff0c 如何生成可参考右边的帮助文档 前言 一直以来 xff0c 编程语言都是程序员非常关注的话题 年末将至 xff0c 是否会有程序员发出疑问 2022 年行业需求最大的编程
  • LibEvent-Demo

    libevent test cpp 定义控制台应用程序的入口点 include 34 stdafx h 34 pragma comment lib 34 ws2 32 lib 34 pragma comment lib 34 wsock32