Linux使用Libevent库实现一个网页服务器---C语言程序

Web服务器

这一个库的实现
其他的知识都是这一个专栏里面的文章

实际使用

编译的时候需要有一个libevent库

gcc httpserv.c -o httpserv -levent

实际使用的时候需要指定端口以及共享的目录

./httpserv 80 .

这一个函数会吧这一个文件夹下面的所有文件共享出去

在这里插入图片描述

实际的效果, 这里我是把我的笔记共享了一下

实现目标

  1. 使用libevent库进行处理客户端连接(listener_cb)
  2. 时候http协议和浏览器进行连接
  3. 获取连接以后把服务启的某个文件夹下面的文件目录返回
  4. 可以根据返回的目录获取文件信息(bufferevent的读回调函数)
  5. 日志会在http.log文件里面
/*************************************************************************
  > File Name: http0.c
  > Author: XvSenfeng
  > Mail: 1458612070@qq.com 
  > Created Time: Thu 18 Apr 2024 11:11:53 AM CST
 ************************************************************************/

#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <event2/event.h>
#include <event2/listener.h>
#include <event2/bufferevent.h>
#include <signal.h>
#include <arpa/inet.h>
#include <ctype.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <dirent.h>

//读取一行
int get_line(struct bufferevent *bev, char *buf, int size){
	int i = 0;
	char c = '\0';
	int n;
	static char temp, todeal = 0;
	if(todeal == 1){
		buf[0] = temp;
		i++;
		todeal = 0;
	}
	while((i < size - 1) && (c != '\n')){
		n = bufferevent_read(bev, &c, 1);
		if(n > 0){
			if(c == '\r'){
				n = bufferevent_read(bev, &temp, 1);
				if((n > 0) && (temp == '\n')){
					buf[i++] = temp;
					break;
				}else{
					buf[i++] = '\n';
					todeal = 1;
					break;
				}
			}
			buf[i] = c;
			i++;
		}else{
			c = '\n';
		}
	}
	buf[i] = '\0';
	if(n == -1){
		i = n;
	}
	return i;
}
//根据文件的后缀, 获取文件的类型(用于HTTP协议通讯)
//name:文件名
//type:传出参数
void get_file_type(const char *name, char *type){
	if(strstr(name, ".html")){
		strcpy(type, "text/html; charset=utf-8");
	}else if(strstr(name, ".jpg")){
		strcpy(type, "image/jpeg");
	}else if(strstr(name, ".png")){
		strcpy(type, "image/png");
	}else if(strstr(name, ".gif")){
		strcpy(type, "image/gif");
	}else if(strstr(name, ".wav")){
		strcpy(type, "audio/wav");
	}else if(strstr(name, ".mp3")){
		strcpy(type, "audio/mp3");
	}else if(strstr(name, ".mp4")){
		strcpy(type, "video/mp4");
	}else{
		strcpy(type, "text/plain; charset=utf-8");
	}
}
//发送一个文件
//filename:文件名
//bev:使用的事件缓冲区
void send_file(const char *filename, struct bufferevent *bev){
	char buf[1024];
    //打开文件
	int fd = open(filename, O_RDONLY);
	if(fd == -1){
		perror("open error");
		exit(1);
	}
	//发送文件内容
	int len = 0;
	while((len = read(fd, buf, sizeof(buf))) > 0){
		bufferevent_write(bev, buf, len);
	}
	close(fd);
}
//发送HTTP协议的头部
//错误号,错误描述,文件类型,文件长度,bufferevent,文件名
void send_respond(int no, char *disp, char *type, long size, struct bufferevent *bev){
	//发送http响应
	char buf[1024] = {0};
	sprintf(buf, "HTTP/1.1 %d %s\r\n", no, disp);
	bufferevent_write(bev, buf, strlen(buf));
	sprintf(buf, "Content-Type: %s\r\n", type);
	bufferevent_write(bev, buf, strlen(buf));
	sprintf(buf, "Content-Length: %ld\r\n", size);
	bufferevent_write(bev, buf, strlen(buf));
	sprintf(buf, "Connection: close\r\n");
	bufferevent_write(bev, buf, strlen(buf));
	sprintf(buf, "\r\n");
	bufferevent_write(bev, buf, strlen(buf));
}
//把文本转化为URL格式, 可用于网址
void strencode(char* to, size_t tosize, const char* from)
{
	int tolen;

	for (tolen = 0; *from != '\0' && tolen + 4 < tosize; ++from)
	{
		if (isalnum(*from) || strchr("/_.-~", *from) != (char*)0)
		{
			*to = *from;
			++to;
			++tolen;
		}
		else
		{
			sprintf(to, "%%%02x", (int) *from & 0xff);
			to += 3;
			tolen += 3;
		}
	}
	*to = '\0';
}
//发送一个文件夹目录
//dirname:文件夹名字
int send_dir(struct bufferevent *bev,const char *dirname)
{
	char encoded_name[1024];
	char path[1024];
	char timestr[64];
	struct stat sb;
	struct dirent **dirinfo;
	int i;

	char *buf = malloc(10240);
	sprintf(buf, "<html><head><meta charset=\"utf-8\"><title>%s</title></head>", dirname);
	sprintf(buf+strlen(buf), "<body><h1>当前目录:%s</h1><table>", dirname);
	//添加目录内容
	int num = scandir(dirname, &dirinfo, NULL, alphasort);
	for(i=0; i<num; ++i)
	{
		// 编码
		strencode(encoded_name, sizeof(encoded_name), dirinfo[i]->d_name);

		sprintf(path, "%s%s", dirname, dirinfo[i]->d_name);
		printf("############# path = %s\n", path);
		if (lstat(path, &sb) < 0)
		{
			sprintf(buf+strlen(buf), 
					"<tr><td><a href=\"%s\">%s</a></td></tr>\n", 
					encoded_name, dirinfo[i]->d_name);
		}
		else
		{
			strftime(timestr, sizeof(timestr), 
					"  %d  %b   %Y  %H:%M", localtime(&sb.st_mtime));
			if(S_ISDIR(sb.st_mode))
			{
                //这是一个文件夹
				sprintf(buf+strlen(buf), 
						"<tr><td><a href=\"%s/\">%s/</a></td><td>%s</td><td>%ld</td></tr>\n",
						encoded_name, dirinfo[i]->d_name, timestr, sb.st_size);
			}
			else
			{
                //这是一个普通文件
				sprintf(buf+strlen(buf), 
						"<tr><td><a href=\"%s\">%s</a></td><td>%s</td><td>%ld</td></tr>\n", 
						encoded_name, dirinfo[i]->d_name, timestr, sb.st_size);
			}

		}
		if(strlen(buf)>10000)
		{
			break;	
		}
		//bufferevent_write(bev, buf, strlen(buf));
		//    memset(buf, 0, sizeof(buf));
	}
	sprintf(buf+strlen(buf), "</table></body></html>");
	send_respond(200, "OK", "text/html", strlen(buf), bev);
	bufferevent_write(bev, buf, strlen(buf));
	printf("################# Dir Read OK !!!!!!!!!!!!!!\n");

	return 0;
}
//发送一个错误页面
void send_404(struct bufferevent *bev){
	//发送一个404页面
	struct stat sbuf;
	stat("404.html", &sbuf);
	send_respond(404, "Not Found", "text/html", sbuf.st_size, bev);
	send_file("404.html", bev);
}
//16进制数转化为10进制, return 0不会出现
int hexit(char c)
{
	if (c >= '0' && c <= '9')
		return c - '0';
	if (c >= 'a' && c <= 'f')
		return c - 'a' + 10;
	if (c >= 'A' && c <= 'F')
		return c - 'A' + 10;

	return 0;
}

void strdecode(char *to, char *from);
void http_request(const char *filename1, struct bufferevent *bev){
	struct stat sbuf;
	char filename[1024];
	strdecode(filename,(char *) filename1);
	int ret = stat(filename, &sbuf);
	if(ret != 0){
		perror("stat error");
		send_404(bev);
		return;
	}
	char buf[128];
	get_file_type(filename, buf);
	//判断是不是目录
	if(S_ISDIR(sbuf.st_mode)){
		send_dir(bev, filename);
	}else{
		//打开文件
		//send_respond(200, "OK", "text/html", sbuf.st_size, bev);
		send_respond(200, "OK", buf, sbuf.st_size, bev);
		send_file(filename, bev);
	}	
	printf("read cb over");
}


void read_cb(struct bufferevent *bev, void *arg){
	char line[1024];
	int len = get_line(bev, line, sizeof(line));
	if(len <= 0){
		printf("get line error\n");
		bufferevent_free(bev);
		return;
	}
	printf("http header: %s", line);
	//判断是不是空行
	if(strcmp(line, "\n") == 0 || strcmp(line, "\r\n") == 0){
		printf("空行\n");
		//断开连接
		bufferevent_free(bev);
		return;
	}
	//判断是不是请求行
	char path[1024] = {0}, protocol[20] = {0};
	sscanf(line, "%*s %s %s", path, protocol);
	//读取剩余数据
	char buf[1024] = {0};
	while(1){
		len = get_line(bev, buf, sizeof(buf));
		if(len <= 0){
			break;
		}
		if(strcmp(buf, "\n") == 0 || strcmp(buf, "\r\n") == 0){
			break;
		}
	}
	if(strncasecmp(line, "GET", 3) == 0){
		char *file = path + 1;
		if(strcmp(path, "/")==0){
			file = "./";
		}
		http_request(file, bev);
		signal_over = 1;
	}else{
		printf("POST\n");
	}
}
//写回调, 这一没啥用
void write_cb(struct bufferevent *bev, void *arg){
	printf("write_cb\n");
}
//事件callback函数, 某一次连接被打断的时候会调用这一个函数
void event_cb(struct bufferevent *bev, short events, void *arg){
	if(events & BEV_EVENT_EOF){
		printf("connection closed\n");
	}else if(events & BEV_EVENT_ERROR){
		printf("some other error\n");
	}
	bufferevent_free(bev);
}

//监听的回调函数
void listener_cb(struct evconnlistener *listener, evutil_socket_t fd, struct sockaddr *addr, int socklen, void *arg){
	struct event_base *base = (struct event_base *)arg;
	struct sockaddr_in *sin = (struct sockaddr_in *)addr;
	//获取客户端ip和端口
	char ip[16];
	inet_ntop(AF_INET, &sin->sin_addr.s_addr, ip, sizeof(ip));
	printf("accept a client %s:%d\n", ip, ntohs(sin->sin_port));
	//创建bufferevent, 之后使用bufferevent的回调函数处理连接事件
	struct bufferevent *bev = bufferevent_socket_new(base, fd, 
			BEV_OPT_CLOSE_ON_FREE);
	if(bev == NULL){
		printf("bufferevent error");
		return;
	}
	//设置读写回调
	bufferevent_flush(bev, EV_READ | EV_WRITE, BEV_NORMAL);
	bufferevent_setcb(bev, read_cb, write_cb, event_cb, arg);
	bufferevent_enable(bev, EV_READ | EV_WRITE);
}
//处理信号的回调函数
void signal_cb(evutil_socket_t sig, short events, void *user_data)
{
	struct event_base *base = user_data;
	struct timeval delay = { 1, 0 };

	printf("Caught an interrupt signal; exiting cleanly in one seconds.\n");
	event_base_loopexit(base, &delay);
}
//把一个url
void strdecode(char *to, char *from)
{
	for ( ; *from != '\0'; ++to, ++from)
	{
        //检测一下下面的三个字符是不是%xx格式的
		if (from[0] == '%' && isxdigit(from[1]) && isxdigit(from[2]))
		{
			//依次判断from中 %20 三个字符, 把这三个字符转换为10进制的数字
			*to = hexit(from[1])*16 + hexit(from[2]);
			// 移过已经处理的两个字符(%21指针指向1),表达式3的++from还会再向后移一个字符
			from += 2;
		}
		else
		{
			*to = *from;
		}
	}
	*to = '\0';
}

int main(int argc, char *argv[]){
	int pid;
	//看一下参数的个数对不对
    if(argc < 3){
		printf("./serv port path");
		return 0;
	}
    //建立一个守护进程
	pid = fork();
	if(pid > 0){
		exit(1);
	}
	//切换工作目录, 使用第三个参数
	const char *path = argv[2];
	int ret = chdir(path);
	if(ret == -1){
		perror("chdir error");
		return -1;
	}
	pid = setsid();
	umask(0022);
	close(STDIN_FILENO);
	int fd;
    //一个日志文件
	fd = open("http.log", O_RDWR|O_CREAT);
	if(fd==-1){
		perror("open error");
		exit(1);
	}
	dup2(fd, STDOUT_FILENO);
	dup2(fd, STDERR_FILENO);
	//获取连接的端口
	int port = atoi(argv[1]);

	//创建服务器地址
	struct sockaddr_in serv;
	//初始化服务器地址
	memset(&serv, 0, sizeof(serv));
	serv.sin_family = AF_INET;
	serv.sin_port = htons(port);
	serv.sin_addr.s_addr = htonl(INADDR_ANY);
	//创建event_base
	struct event_base *base = event_base_new();
	if(base == NULL){
		printf("event base error");
		return -1;
	}
	//创建监听器
	struct evconnlistener *listener = evconnlistener_new_bind(
			base, listener_cb, base, 
			LEV_OPT_CLOSE_ON_FREE | LEV_OPT_REUSEABLE, -1, 
			(struct sockaddr *)&serv, sizeof(serv));
	if(listener == NULL){
		printf("listener error");
		return -1;
	}
	struct event *signal_event;
    //绑定信号回调
	signal_event = evsignal_new(base, SIGINT, signal_cb, (void *)base);
	if (!signal_event || event_add(signal_event, NULL)<0) 
	{
		fprintf(stderr, "Could not create/add a signal event!\n");
		return 1;
	}
    //开启循环
	event_base_dispatch(base);
	evconnlistener_free(listener);
	event_base_free(base);
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/560309.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

NLP_知识图谱_三元组实战

文章目录 三元组含义如何构建知识图谱模型的整体结构基于transformers框架的三元组抽取baselinehow to use预训练模型下载地址训练数据下载地址 结构图代码及数据bertconfig.jsonvocab.txt datadev.jsonschemas.jsontrain.jsonvocab.json 与bert跟data同个目录model.pytrain.py…

华为ensp中rip和ospf路由重分发 原理及配置命令

作者主页&#xff1a;点击&#xff01; ENSP专栏&#xff1a;点击&#xff01; 创作时间&#xff1a;2024年4月20日20点21分 路由重分发&#xff08;Route Redistribution&#xff09;是指路由器将从一种路由协议学习到的路由信息&#xff0c;通过另一种路由协议通告出去的功…

Linux环境变量深度解析

文章目录 一、引言二、环境变量的基本概念1、环境变量的定义2、环境变量的作用与意义 三、环境变量的导入1、导入所需文件2、登陆时的导入 四、环境变量的设置方法1、查看环境变量的方式2、使用export命令临时设置环境变量3、修改配置文件以永久设置环境变量 五、命令行参数与环…

Python编程与算法面试-编程面试的重点

在求职面试的过程中&#xff0c;编程能力也是面试官非常看重的一项能力。而对于编程这项能力主要的考察点也有三个维度&#xff1a; 初级&#xff1a;编程的基本功 编程的基本功主要考察的编程语言的基本语法&#xff0c;原理知识&#xff0c;以及一些在编程过程中的常见问题…

基于unity+c#的随机点名系统(简单UI界面+列表+数组)

目录 一、功能界面显示 二、UI 1、视频的使用 &#xff08;1&#xff09;渲染纹理 &#xff08;2&#xff09; 视频铺全屏 &#xff08;3&#xff09;视频的调用 2、 下拉文本框的使用&#xff08;旧版&#xff09; 3、输入文本框的使用&#xff08;旧版&#xff09; …

OpenHarmony 视图加载——ImageViewZoom

简介 ImageViewZoom 支持加载 Resource 或 PixelMap 图片&#xff0c;支持设置图像显示类型功能&#xff0c;支持缩放功能&#xff0c;支持平移功能&#xff0c;双击放大功能&#xff0c;可以监听图片大小&#xff0c;资源变化事件&#xff0c;支持清除显示图片功能。 效果展示…

pg内核之日志管理器(五)WAL日志

概念 WAL日志 数据库运行过程中&#xff0c;数据一般是会保存在内存和磁盘中&#xff0c;为保证数据的安全性&#xff0c;防止数据库崩溃时数据不丢失&#xff0c;一般都是要保证数据实时落盘的&#xff0c;但是又由于磁盘随机IO读写速率与内存相比慢很多&#xff0c;如果每个…

RocketMQ5.x的pop模式如何解决消费堆积问题

RocketMQ4.X现存问题 消费能力不能随POD增加而增加。 理想情况下&#xff0c;POD数量小于QUEUE的数量&#xff0c;增加机器是能提高消能力的。 现实情况下&#xff0c;如果POD数量大于QUEUE的数量&#xff0c;那么多的POD机器就不会处理消费&#xff0c;是一种资源的浪费。 单…

苹果 IPA 应用部署软件 iMazing 3 Windows 版获 3.0.0.4 Beta 4

在数字化时代&#xff0c;我们的iOS设备已经成为生活中不可或缺的一部分。为了更加高效、便捷地管理这些设备&#xff0c;iMazing 3.0.0.3 应运而生&#xff0c;它以其独特的功能和卓越的性能&#xff0c;为用户带来了前所未有的全新体验。 首先&#xff0c;iMazing 3.0.0.3 提…

集简云数据表新增批量操作功能,一键实现批量触发执行对应自动化流程

在使用数据表时&#xff0c;某些情况下可能希望人工触发自动化流程执行&#xff0c;例如&#xff1a;开发票、提交工单、同步帐套信息等场景。 通过数据表按钮字段&#xff0c;可手动触发执行对应自动化流程&#xff0c;实现将数据推送到其他表单、应用系统&#xff0c;或从其…

C++必修:从C语言到C++的过渡(上)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C学习 贝蒂的主页&#xff1a;Betty’s blog 1. 什么是C C&#xff08;c plus plus&#xff09;是一种计算机高级程序设计语言&…

新型大数据架构之湖仓一体(Lakehouse)架构特性说明——Lakehouse 架构(一)

文章目录 为什么需要新的数据架构&#xff1f;湖仓一体&#xff08;Lakehouse&#xff09;——新的大数据架构模式同时具备数仓与数据湖的优点湖仓一体架构存储层计算层 湖仓一体特性单一存储拥有数据仓库的查询性能存算分离开放式架构支持各种数据源类型支持各种使用方式架构简…

csdn的编写教程(官方给的)

自定义的目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个注脚…

ubuntu 16.04.7连不上网的解决方案

首先在编辑选项卡里找到虚拟网络编辑器&#xff0c; 点击更改设置 点击添加网络 点击确定 选择桥接模式&#xff0c;自动后点击应用&#xff0c;最后点击确定即可。

4.20+C语言感想,有趣的思考题,case的省略操作,统计位数,终止循环,break和continue语句, 准备下一篇见。

鹏哥C语言感想 一.高级 这可不是什么煎饼&#xff0c;这种食物叫做蓝莓&#xff0c;俗称苹果。生长在撒哈拉沙漠的雨林地带。因外形酷似企鹅&#xff0c;所以我们又喜欢叫他北极熊。你们这些人&#xff0c;连仙人掌都不知道&#xff0c;就不要乱说他是西瓜好吗&#xff1f;再…

嵌入式4-20

客户端 #include <myhead.h> #define SER_IP "192.168.125.244" #define SER_PORT 8888 typedef struct Node { char username[20];struct sockaddr_in cin;struct Node *next; }Node,*Node_p; typedef struct {int flag;char username[20];char data[1024]…

java中File类和输入输出流的用法

目录 针对文件系统进行操作 针对文件内容进行操作 java针对文件操作可以分为两种&#xff1a;1&#xff09;针对文件系统进行操作&#xff0c;如创建文件&#xff0c;删除文件&#xff0c;创建目录&#xff0c;重命名文件等。 2&#xff09;针对文件内容进行操作&#xff0c…

【webrtc】m98 RoundRobinPacketQueue的优先级处理

m98 代码 PacedSender::EnqueuePackets 的调用者可能是多个地方,所以这个要加锁保护。RoundRobinPacketQueue 本身是没有锁的发现m98和新版本不同,参考:【webrtc】m114自己实现的PrioritizedPacketQueuepush和pop都是RtpPacketToSend 但是实际上,内部是封装为QueuedPacket 处…

Darknet框架优化介绍

一、DarkNet框架简介 1.DarkNet的简介 Darknet是一个完全使用C语言编写的人工智能框架&#xff0c;可以使用CUDA的开源框架。主要应用于图像识别领域。 它具有可移植性好&#xff0c;安装间接&#xff0c;查看源码方便等优势&#xff0c;提供了OpenCV等附加选项&#xff0c;还…

基于SSM+Jsp+Mysql的电子商城系统

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…