一个用于快速使用c++ 开发web服务器的框架
高性能易用的http框架,基于c++11标准开发
- 统一而简单的接口
- header-only
- 跨平台(Windows, Linux, Macos)
- 支持拦截器功能
- 支持session/cookie (持久化,默认保存到磁盘。用户可以自定义持久化介质)
- 支持websocket(无锁异步write,高效的数据写出能力)
- 易用的客户端(http client/ https client 可以通过CMakeList.txt中的ENABLE_CLIENT_SSL ON开启)
使用支持c++11标准的c++ 编译器即可编译运行,开箱即用,无需其他依赖(linux 下只需要uuid库,大部分linux系统默认都有)
sudo yum install libuuid-devel.x86_64
sudo apt-get install uuid
sudo apt-get install uuid-dev
mkdir build
cd build
cmake ..
即可生成相应平台下的编译项目
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/hello",[](request& req,response& res){
res.write_string("hello,world");
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/param",[](request& req,response& res){
auto id = req.param("id");
res.write_string(view2str(id));
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/redirect",[](request& req,response& res){
res.redirect("/abc",false); /*接口的第二个参数用来告诉框架是临时重定向 还是永久重定向 */
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/form",[](request& req,response& res){
auto name = req.query("name"); /*query接口用于查询 name="xxx" 的值*/
res.write_string(view2str(name));
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/form",[](request& req,response& res){
auto name = req.query("name"); /*query接口用于查询 name="xxx" 的值 */
auto& file = req.file("img"); /*file 接口用于查询 name="xxx" 的文件 */
res.write_string(std::to_string(file.size()));
});
serve.serve();
serve.run();
}
request::query
参数类型为string 返回值类型为string, 根据索引返回表单提交的值(url-encode-form 和 multipart-form)
request::formdata_form_map
返回通过url-encode-form提交的所有键值对集合
multipart_form_map
返回通过multipart-form提交的所有非文件键值对集合
request::param
- 当参数类型为string,返回url中的键值对中key为参数值所对应的值
- 当参数类型为std::size_t,返回url中与通配符匹配的对应索引的值
request::path_params_string
返回通配符对应的path路径的原始文本
request::pair_params_string
返回URL路径中?后的原始文本
request::path_params_list
返回通配符对应的每一个参数组成的集合
request::pair_params_map
返回URL中?后的参数键值对集合
request::raw_url
返回请求头中URL的原始文本
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/form",[](request& req,response& res){
auto& file = req.file(); /*octet-stream 只能提交单个二进制流数据 所以这里的file没有参数 */
res.write_string(std::to_string(file.size()));
});
serve.serve();
serve.run();
}
假设注册路由的集合是 { /* , /abc/* , /abc/ccc/* , /abc } 请求示例
- url为
http://127.0.0.1:8080/abc/ccc/ddd
,则最佳匹配的是 /abc/ccc/*- url为
http://127.0.0.1:8080/abc/cccc/ddd
, 则匹配 /abc/* ,- url为
http://127.0.0.1:8080/ddd/kkk
,则只匹配 /*- url为
http://127.0.0.1:8080/abc
,则最佳匹配的就是 /abc
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
auto const thread_number = std::thread::hardware_concurrency();
http_server server(thread_number);
bool r = server.listen("0.0.0.0", "8080");
if (!r) {
return 0;
}
server.set_url_redirect(false);
server.router<GET>("/*", [](request& req, response& res) {
auto url = req.raw_url();
json root;
root["router"] = "/*";
root["url"] = view2str(url);
auto params = req.url_params();
for (auto& iter : params) {
root["params"].push_back(view2str(iter));
}
root["param_size"] = req.url_params().size();
res.write_json(root);
});
server.router<GET>("/abc", [](request& req, response& res) {
res.write_string("abc");
});
server.router<GET>("/abc/*", [](request& req, response& res) {
auto url = req.raw_url();
json root;
root["router"] = "/abc/*";
root["url"] = view2str(url);
auto params = req.url_params();
for (auto& iter : params) {
root["params"].push_back(view2str(iter));
}
root["param_size"] = req.url_params().size();
res.write_json(root);
});
server.router<GET>("/abc/ccc/*", [](request& req, response& res) {
auto url = req.raw_url();
json root;
root["router"] = "/abc/ccc/*";
root["url"] = view2str(url);
auto params = req.url_params();
for (auto& iter : params) {
root["params"].push_back(view2str(iter));
}
root["param_size"] = req.url_params().size();
res.write_json(root);
});
serve.serve();
server.run();
return 0;
}
使用了第三方json库 更多用法参考 nlohmann/json
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/json",[](request& req,response& res){
json data;
data["name"] = "xfinal";
data["language"] = "morden c++";
res.write_json(data); /*第一个参数也可以直接传入json字符串,第二个接口用来控制返回方式*/
});
serve.serve();
serve.run();
}
视图层更多使用方法 参考 xmh0511/inja (在inja库的基础上修改并增添了功能)
///test.html
<!DOCTYPE html>
<html lang="zh" xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta charset="utf-8" />
<title></title>
</head>
<body>
<div style="color:aqua">
<p><span>name:</span>@{name}</p>
<p><span>language:</span>@{language}</p>
<p>#{ @{language} }#</p> <!--原样输出-->
<a href="https://www.baidu.com">@{ str2int("30")}</a>
<p>@{ str2int(str)}</p>
</div>
</body>
</html>
///test.html
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
server.add_view_method("str2int", 1, [](inja::Arguments const& args) {
auto i = std::atoi(args[0]->get<std::string>().data());
return std::string("transform:") + std::to_string(i);
});
serve.router<GET>("/view",[](request& req,response& res){
res.set_attr("name", "xfinal");
res.set_attr("language", "c++");
res.write_file_view("./static/test.html");/*也可以写内存中的模板 res.write_data_view("@{name}")*/
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
server.add_view_method("str2int", 1, [](inja::Arguments const& args) {
auto i = std::atoi(args[0]->get<std::string>().data());
return std::string("transform:") + std::to_string(i);
});
serve.router<GET>("/file",[](request& req,response& res){
res.write_file("./static/need.bmp",/*这里有个bool类型参数 可以用来告诉接口 是直接返回文件内容(false) 还是以chunked的方式返回文件
(true)*/);
/*chunked 方式支持断点续传 */
/*框架内置的静态文件处理提供了通用的处理能力,这里提供了让用户处理更多资源文件请求的可能性,比如通过参数进行图片视频等多媒体文件的裁剪*/
});
serve.serve();
serve.run();
}
简化回应api的调用方式
server.router<GET, POST>("/httppackage", [](request& req, response& res) {
struct CustomeResult :public response::http_package {
CustomeResult(std::string&& content) {
body_souce_ = std::move(content);
}
void dump(){
state_ = http_status::ok;
write_type_ = response::write_type::string;
}
};
res.write_http_package(CustomeResult{ "OK" });
});
server.router<GET, POST>("/httppackagefile", [](request& req, response& res) {
struct CustomeResult :public response::http_package {
CustomeResult(std::string&& content) {
body_souce_ = std::move(content);
}
void dump(){
state_ = http_status::ok;
write_type_ = response::write_type::file;
is_chunked_ = true;
}
};
res.write_http_package(CustomeResult{ "./static/a.mp4" });
});
server.router<GET>("/deferresponse", [](request& req, response& res) {
auto time = req.param("time");
auto time_i = std::atoi(time.data());
auto guard = res.defer(); //拥有connection所有权
std::thread t1 = std::thread([guard,&res, time_i]() {
std::this_thread::sleep_for(std::chrono::seconds(time_i));
res.write_string("OK");
});
t1.detach();
});
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
/*可以通过serve.static_path("./static") 改变目录 同时会自动改变静态资源的url的标识*/
/*内置处理方式默认设置了static 用户可以通过localhost:8080/static/abc.png 来访问静态资源文件*/
/*对于多媒体等文件默认支持断点续传功能*/
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/login",[](request& req,response& res){
req.create_session().set_data("islogin",true); /*可以通过设置参数指定session名称 req.create_session("myname")*/
res.write_string("OK");
});
serve.router<GET>("/checklogin",[](request& req,response& res){
auto& session = req.session(); /* req.session("myname") */
if(session.get_data<bool>("islogin")){
res.write_string("yes");
}else{
res.write_string("no");
}
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
class sql_storage:public session_storage{
public:
bool init(){
/*初始化操作,从介质读取数据*/
}
void save(session& session){
/*用来保存session*/
}
void remove(session& session){
/*用来移除session*/
}
void remove(std::string const& uuid){
/*通过uuid 删除session*/
}
};
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.set_session_storager(std::unique_ptr<sql_storage>(new sql_storage{})); /*设置session存储方式*/
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
struct http_interceptor{
bool before(request& req,response& res){ //这里的返回值用来告诉框架是否还继续执行用户接下来的拦截器以及注册的路由逻辑
auto id = req.query("id");
if(id=="0"){
return true;
}else{
return false;
}
}
bool after(request& req,response& res){
res.add_header("Access-Control-Allow-Origin","*");
return true;
}
};
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<POST>("/interceptor",[](request& req,response& res){
res.write_string("OK");
},http_interceptor{}); //可以注册多个拦截器 定义如示例中的http_interceptor
serve.serve();
serve.run();
}
请求body接受前会调用prehandle,返回true则继续处理当前http请求,如果为false则丢弃当前http请求体数据
#include <xfinal.hpp>
using namespace xfinal;
struct pre_interceptor{
bool prehandle(request& req,response& res){ //这里的返回值用来告诉框架是否继续处理当前http请求,如果返回false则返回用户通过write_xxx的信息,并断开连接
if(req.content_type() == content_type::multipart_form){ //如果是上传文件 则断开连接
res.write_string("invalid request",http_status::bad_request );
return false;
}
return true;
}
bool posthandle(request& req,response& res){
return true;
}
};
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<POST>("/interceptor",[](request& req,response& res){
res.write_string("OK");
},pre_interceptor{}); //可以注册多个拦截器 定义如示例中的http_interceptor
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main(){
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.on_error([](std::exception const& ec) { //提供用户记录错误日志
std::cout << ec.what() << std::endl;
});
serve.serve();
serve.run();
}
////Test.hpp
#pragma once
#include <xfinal.hpp>
class Test{
public:
void shop(request& req,response& res){
res.write_string("shop !");
}
} ;
////Test.hpp
#include "http_server.hpp"
#include "Test.hpp"
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
/*不保存接口状态信息*/
serve.router<GET,POST>("/shop",&Test::shop,nullptr);
/*这里可以记录下每次请求的一些信息,用于下次请求使用*/
Test t;
serve.router<GET,POST>("/shop",&Test::shop,t);
serve.serve();
serve.run();
}
///shop.hpp
#pragma once
#include <xfinal.hpp>
class Shop:public Controller{
public:
void go(){
this->get_response().write_string("go shopping!");
}
};
///shop.hpp
#include "http_server.hpp"
#include "shop.hpp"
using namespace xfinal;
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
/*不保存接口状态信息*/
serve.router<GET,POST>("/shop",&Shop::go,nullptr);
/*这里可以记录下每次请求的一些信息,用于下次请求使用*/
Shop t;
serve.router<GET,POST>("/shop",&Shop::go,t);
serve.serve();
serve.run();
}
同样支持AOP 和前置拦截器
#include <xfinal.hpp>
int main()
{
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
websocket_event event; //定义websocket 事件
event.on("message", [](websocket& ws) {
std::cout << ws.messages() << std::endl; //消息内容
std::cout << (int)ws.message_code() << std::endl; //消息类型
std::string message;
for (auto i = 0; i <= 400; ++i) {
message.append(std::to_string(i));
}
ws.write_string(message); //发送消息 ws.write_binary("xxx")用来写二进制类型的数据 || ws.write("xxx",opcode);
}).on("open", [](websocket& ws) {
std::cout << "open" << std::endl;
}).on("close",[](websocket& ws){
//balabala
});
//可能做聊天工具等 通过ws.shared_from_this() 保存当前会话 自行发挥 ^_^
server.router("/ws", event); //定义webscoket 路由
}
可选最后一个参数,其可调用类型为 void(bool, std::error_code)
- 参数1: 是否成功写出
- 参数2: 可以获取错误码
如果没有特殊说明,默认在run方法调用前进行设置
调用此接口后,请求的地址 http://127.0.0.1:8080/x/y/somefile.abc 就会自动到该目录下读取对应的文件
server.set_static_path("./x/y");
在set_static_path后进行设置
server.set_upload_path("./myupload");
单位: 秒 超时关闭当前socket连接
server.set_wait_read_time(10);
server.set_wait_write_time(10);
单位:秒 超时自动断开
server.set_keep_alive_wait_time(5);
单位:秒
server.set_check_session_rate(10);
单位:字节
server.set_websocket_frame_size(1024 * 1024); //最大65535
单位:秒 超时自动断开
server.set_websocket_check_read_alive_time(3600)
server.set_websocket_check_write_alive_time(3600)
server.set_disable_auto_create_directories(true) // 禁止自动创建
server.set_default_session_save_path("./session");
server.set_disable_register_static_handler(true);
针对除multipart_form的post请求 (单位:字节)
server.set_max_body_size(1024*1024);
(单位:秒)
server.set_defer_write_max_wait_time(60);
参数:
- 保存目录
- 文件名前缀
- 单个文件大小(mb)
server.init_nanolog("./directory/","mylog",2);
需要在listen方法调用前设置
server.on_error([](std::string const& message) { //提供用户记录错误日志
std::cout << message << std::endl;
});
可以提供用户自己处理400需要返回的内容,如400个性化页面等
server.set_not_found_callback([](request& req,response& res) {
res.write_string("custom not found", true, http_status::bad_request);
});
server.add_view_method("str2int", 1, [](inja::Arguments const& args) {
auto i = std::atoi(args[0]->get<std::string>().data());
return std::string("transform:") + std::to_string(i);
});
// 比如在html 中可以使用 @{str2int("1024")}
#include <xfinal.hpp>
using namespace xfinal;
int main(){
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
serve.router<GET>("/client",[](request& req,response& res){
http_client client("www.baidu.com"); //或者IP:端口号
auto con = client.request<GET>("/"); //请求的url
auto header = con.get_header("Content-Length"); //获取响应头信息
auto state = con.get_status_code(); //获取响应状态码
res.write_string(con.get_content()); //获取响应的body
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main(){
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
http_client client("www.baidu.com");
server.router<GET>("/asyclient", [](request& req, response& res) {
http_client client("www.baidu.com");
client.request<GET>("/", [](http_client::client_response const& res,std::error_code const& ec) {
if (ec) {
return;
}
std::cout << res.get_content() << std::endl;
});
client.run();
res.write_string("OK");
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main(){
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
server.router<GET>("/client_post", [](request& req, response& res) {
try {
http_client client("127.0.0.1:8080");
client.add_header("name", "hello");
client.add_header("Content-Type", "application/x-www-form-urlencoded");
std::string form = "id=1234";
auto con = client.request<POST>("/test", form);
res.write_string(con.get_content());
}
catch (...) {
res.write_string("error");
}
});
serve.serve();
serve.run();
}
#include <xfinal.hpp>
using namespace xfinal;
int main(){
http_server serve(4); //线程数
serve.listen("0.0.0.0","8080");
server.router<GET>("/client_multipart", [](request& req, response& res) {
try {
http_client client("127.0.0.1:8020");
multipart_form form;
form.append("img", multipart_file{ "./static/upload/v2.jpg" });
form.append("text", "hello,world");
auto con = client.request<POST>("/upload", form);
res.write_string(con.get_content());
}
catch (...) {
res.write_string("error");
}
});
serve.serve();
serve.run();
}
同上 request 传入回调函数
1.提升xfinal的稳定性
2.优化xfinal性能
QQ : 970252187
Mail : [email protected]
如果项目对你有所帮助,请给作者一点鼓励和支持