1
0
mirror of synced 2025-05-08 17:41:29 +03:00
This commit is contained in:
yhirose 2018-05-06 21:16:35 -04:00
parent e46cc54d13
commit 6c5d0b2a18
8 changed files with 255 additions and 103 deletions

View File

@ -19,11 +19,11 @@ int main(void)
Server svr; Server svr;
svr.get("/hi", [](const Request& req, Response& res) { svr.Get("/hi", [](const Request& req, Response& res) {
res.set_content("Hello World!", "text/plain"); res.set_content("Hello World!", "text/plain");
}); });
svr.get(R"(/numbers/(\d+))", [&](const Request& req, Response& res) { svr.Get(R"(/numbers/(\d+))", [&](const Request& req, Response& res) {
auto numbers = req.matches[1]; auto numbers = req.matches[1];
res.set_content(numbers, "text/plain"); res.set_content(numbers, "text/plain");
}); });
@ -32,13 +32,15 @@ int main(void)
} }
``` ```
`Post`, `Put`, `Delete` and `Options` methods are also supported.
### Method Chain ### Method Chain
```cpp ```cpp
svr.get("/get", [](const auto& req, auto& res) { svr.Get("/get", [](const auto& req, auto& res) {
res.set_content("get", "text/plain"); res.set_content("get", "text/plain");
}) })
.post("/post", [](const auto& req, auto& res) { .Post("/post", [](const auto& req, auto& res) {
res.set_content(req.body(), "text/plain"); res.set_content(req.body(), "text/plain");
}) })
.listen("localhost", 1234); .listen("localhost", 1234);
@ -72,7 +74,7 @@ svr.set_error_handler([](const auto& req, auto& res) {
### 'multipart/form-data' POST data ### 'multipart/form-data' POST data
```cpp ```cpp
svr.post("/multipart", [&](const auto& req, auto& res) { svr.Post("/multipart", [&](const auto& req, auto& res) {
auto size = req.files.size(); auto size = req.files.size();
auto ret = req.has_file("name1")); auto ret = req.has_file("name1"));
const auto& file = req.get_file_value("name1"); const auto& file = req.get_file_value("name1");
@ -95,7 +97,7 @@ int main(void)
{ {
httplib::Client cli("localhost", 1234); httplib::Client cli("localhost", 1234);
auto res = cli.get("/hi"); auto res = cli.Get("/hi");
if (res && res->status == 200) { if (res && res->status == 200) {
std::cout << res->body << std::endl; std::cout << res->body << std::endl;
} }
@ -105,8 +107,8 @@ int main(void)
### POST ### POST
```c++ ```c++
res = cli.post("/post", "text", "text/plain"); res = cli.Post("/post", "text", "text/plain");
res = cli.post("/person", "name=john1&note=coder", "application/x-www-form-urlencoded"); res = cli.Post("/person", "name=john1&note=coder", "application/x-www-form-urlencoded");
``` ```
### POST with parameters ### POST with parameters
@ -115,7 +117,26 @@ res = cli.post("/person", "name=john1&note=coder", "application/x-www-form-urlen
httplib::Map params; httplib::Map params;
params["name"] = "john"; params["name"] = "john";
params["note"] = "coder"; params["note"] = "coder";
auto res = cli.post("/post", params); auto res = cli.Post("/post", params);
```
### PUT
```c++
res = cli.Post("/resource/foo", "text", "text/plain");
```
### DELETE
```c++
res = cli.Delete("/resource/foo");
```
### OPTIONS
```c++
res = cli.Options("*");
res = cli.Options("/resource/foo");
``` ```
### Connection Timeout ### Connection Timeout
@ -130,7 +151,7 @@ httplib::Client client(url, port);
// prints: 0 / 000 bytes => 50% complete // prints: 0 / 000 bytes => 50% complete
std::shared_ptr<httplib::Response> res = std::shared_ptr<httplib::Response> res =
cli.get("/", [](uint64_t len, uint64_t total) { cli.Get("/", [](uint64_t len, uint64_t total) {
printf("%lld / %lld bytes => %d%% complete\n", printf("%lld / %lld bytes => %d%% complete\n",
len, total, len, total,
(int)((len/total)*100)); (int)((len/total)*100));
@ -150,7 +171,7 @@ httplib::Client cli("httpbin.org", 80);
// 'Range: bytes=1-10' // 'Range: bytes=1-10'
httplib::Headers headers = { httplib::make_range_header(1, 10) }; httplib::Headers headers = { httplib::make_range_header(1, 10) };
auto res = cli.get("/range/32", headers); auto res = cli.Get("/range/32", headers);
// res->status should be 206. // res->status should be 206.
// res->body should be "bcdefghijk". // res->body should be "bcdefghijk".
``` ```
@ -185,4 +206,4 @@ The server applies gzip compression to the following MIME type contents:
License License
------- -------
MIT license (© 2017 Yuji Hirose) MIT license (© 2018 Yuji Hirose)

View File

@ -25,7 +25,7 @@ int main(void) {
for (int i = 0; i < 3; i++) { for (int i = 0; i < 3; i++) {
StopWatch sw(to_string(i).c_str()); StopWatch sw(to_string(i).c_str());
auto res = cli.post("/post", body, "application/octet-stream"); auto res = cli.Post("/post", body, "application/octet-stream");
assert(res->status == 200); assert(res->status == 200);
} }

View File

@ -18,7 +18,7 @@ int main(void)
httplib::Client cli("localhost", 8080); httplib::Client cli("localhost", 8080);
#endif #endif
auto res = cli.get("/hi"); auto res = cli.Get("/hi");
if (res) { if (res) {
cout << res->status << endl; cout << res->status << endl;
cout << res->get_header_value("Content-Type") << endl; cout << res->get_header_value("Content-Type") << endl;

View File

@ -12,7 +12,7 @@ int main(void)
{ {
Server svr; Server svr;
svr.get("/hi", [](const auto& /*req*/, auto& res) { svr.Get("/hi", [](const auto& /*req*/, auto& res) {
res.set_content("Hello World!", "text/plain"); res.set_content("Hello World!", "text/plain");
}); });

View File

@ -81,25 +81,25 @@ int main(void)
return -1; return -1;
} }
svr.get("/", [=](const auto& /*req*/, auto& res) { svr.Get("/", [=](const auto& /*req*/, auto& res) {
res.set_redirect("/hi"); res.set_redirect("/hi");
}); });
svr.get("/hi", [](const auto& /*req*/, auto& res) { svr.Get("/hi", [](const auto& /*req*/, auto& res) {
res.set_content("Hello World!\n", "text/plain"); res.set_content("Hello World!\n", "text/plain");
}); });
svr.get("/slow", [](const auto& /*req*/, auto& res) { svr.Get("/slow", [](const auto& /*req*/, auto& res) {
using namespace std::chrono_literals; using namespace std::chrono_literals;
std::this_thread::sleep_for(2s); std::this_thread::sleep_for(2s);
res.set_content("Slow...\n", "text/plain"); res.set_content("Slow...\n", "text/plain");
}); });
svr.get("/dump", [](const auto& req, auto& res) { svr.Get("/dump", [](const auto& req, auto& res) {
res.set_content(dump_headers(req.headers), "text/plain"); res.set_content(dump_headers(req.headers), "text/plain");
}); });
svr.get("/stop", [&](const auto& /*req*/, auto& /*res*/) { svr.Get("/stop", [&](const auto& /*req*/, auto& /*res*/) {
svr.stop(); svr.stop();
}); });

View File

@ -107,7 +107,7 @@ int main(int argc, const char** argv)
Server svr(version); Server svr(version);
#endif #endif
svr.post("/multipart", [](const auto& req, auto& res) { svr.Post("/multipart", [](const auto& req, auto& res) {
auto body = auto body =
dump_headers(req.headers) + dump_headers(req.headers) +
dump_multipart_files(req.files); dump_multipart_files(req.files);

150
httplib.h
View File

@ -194,8 +194,12 @@ public:
virtual bool is_valid() const; virtual bool is_valid() const;
Server& get(const char* pattern, Handler handler); Server& Get(const char* pattern, Handler handler);
Server& post(const char* pattern, Handler handler); Server& Post(const char* pattern, Handler handler);
Server& Put(const char* pattern, Handler handler);
Server& Delete(const char* pattern, Handler handler);
Server& Options(const char* pattern, Handler handler);
bool set_base_dir(const char* path); bool set_base_dir(const char* path);
@ -236,6 +240,9 @@ private:
std::string base_dir_; std::string base_dir_;
Handlers get_handlers_; Handlers get_handlers_;
Handlers post_handlers_; Handlers post_handlers_;
Handlers put_handlers_;
Handlers delete_handlers_;
Handlers options_handlers_;
Handler error_handler_; Handler error_handler_;
Logger logger_; Logger logger_;
@ -256,17 +263,26 @@ public:
virtual bool is_valid() const; virtual bool is_valid() const;
std::shared_ptr<Response> get(const char* path, Progress progress = nullptr); std::shared_ptr<Response> Get(const char* path, Progress progress = nullptr);
std::shared_ptr<Response> get(const char* path, const Headers& headers, Progress progress = nullptr); std::shared_ptr<Response> Get(const char* path, const Headers& headers, Progress progress = nullptr);
std::shared_ptr<Response> head(const char* path); std::shared_ptr<Response> Head(const char* path);
std::shared_ptr<Response> head(const char* path, const Headers& headers); std::shared_ptr<Response> Head(const char* path, const Headers& headers);
std::shared_ptr<Response> post(const char* path, const std::string& body, const char* content_type); std::shared_ptr<Response> Post(const char* path, const std::string& body, const char* content_type);
std::shared_ptr<Response> post(const char* path, const Headers& headers, const std::string& body, const char* content_type); std::shared_ptr<Response> Post(const char* path, const Headers& headers, const std::string& body, const char* content_type);
std::shared_ptr<Response> post(const char* path, const Params& params); std::shared_ptr<Response> Post(const char* path, const Params& params);
std::shared_ptr<Response> post(const char* path, const Headers& headers, const Params& params); std::shared_ptr<Response> Post(const char* path, const Headers& headers, const Params& params);
std::shared_ptr<Response> Put(const char* path, const std::string& body, const char* content_type);
std::shared_ptr<Response> Put(const char* path, const Headers& headers, const std::string& body, const char* content_type);
std::shared_ptr<Response> Delete(const char* path);
std::shared_ptr<Response> Delete(const char* path, const Headers& headers);
std::shared_ptr<Response> Options(const char* path);
std::shared_ptr<Response> Options(const char* path, const Headers& headers);
bool send(Request& req, Response& res); bool send(Request& req, Response& res);
@ -1411,18 +1427,36 @@ inline Server::~Server()
{ {
} }
inline Server& Server::get(const char* pattern, Handler handler) inline Server& Server::Get(const char* pattern, Handler handler)
{ {
get_handlers_.push_back(std::make_pair(std::regex(pattern), handler)); get_handlers_.push_back(std::make_pair(std::regex(pattern), handler));
return *this; return *this;
} }
inline Server& Server::post(const char* pattern, Handler handler) inline Server& Server::Post(const char* pattern, Handler handler)
{ {
post_handlers_.push_back(std::make_pair(std::regex(pattern), handler)); post_handlers_.push_back(std::make_pair(std::regex(pattern), handler));
return *this; return *this;
} }
inline Server& Server::Put(const char* pattern, Handler handler)
{
put_handlers_.push_back(std::make_pair(std::regex(pattern), handler));
return *this;
}
inline Server& Server::Delete(const char* pattern, Handler handler)
{
delete_handlers_.push_back(std::make_pair(std::regex(pattern), handler));
return *this;
}
inline Server& Server::Options(const char* pattern, Handler handler)
{
options_handlers_.push_back(std::make_pair(std::regex(pattern), handler));
return *this;
}
inline bool Server::set_base_dir(const char* path) inline bool Server::set_base_dir(const char* path)
{ {
if (detail::is_dir(path)) { if (detail::is_dir(path)) {
@ -1475,7 +1509,7 @@ inline void Server::stop()
inline bool Server::parse_request_line(const char* s, Request& req) inline bool Server::parse_request_line(const char* s, Request& req)
{ {
static std::regex re("(GET|HEAD|POST) ([^?]+)(?:\\?(.+?))? (HTTP/1\\.[01])\r\n"); static std::regex re("(GET|HEAD|POST|PUT|DELETE|OPTIONS) ([^?]+)(?:\\?(.+?))? (HTTP/1\\.[01])\r\n");
std::cmatch m; std::cmatch m;
if (std::regex_match(s, m, re)) { if (std::regex_match(s, m, re)) {
@ -1675,6 +1709,12 @@ inline bool Server::routing(Request& req, Response& res)
return dispatch_request(req, res, get_handlers_); return dispatch_request(req, res, get_handlers_);
} else if (req.method == "POST") { } else if (req.method == "POST") {
return dispatch_request(req, res, post_handlers_); return dispatch_request(req, res, post_handlers_);
} else if (req.method == "PUT") {
return dispatch_request(req, res, put_handlers_);
} else if (req.method == "DELETE") {
return dispatch_request(req, res, delete_handlers_);
} else if (req.method == "OPTIONS") {
return dispatch_request(req, res, options_handlers_);
} }
return false; return false;
} }
@ -1725,7 +1765,7 @@ inline bool Server::process_request(Stream& strm, bool last_connection)
req.set_header("REMOTE_ADDR", strm.get_remote_addr().c_str()); req.set_header("REMOTE_ADDR", strm.get_remote_addr().c_str());
// Body // Body
if (req.method == "POST") { if (req.method == "POST" || req.method == "PUT") {
if (!detail::read_content(strm, req)) { if (!detail::read_content(strm, req)) {
res.status = 400; res.status = 400;
write_response(strm, last_connection, req, res); write_response(strm, last_connection, req, res);
@ -1947,12 +1987,12 @@ inline bool Client::read_and_close_socket(socket_t sock, Request& req, Response&
}); });
} }
inline std::shared_ptr<Response> Client::get(const char* path, Progress progress) inline std::shared_ptr<Response> Client::Get(const char* path, Progress progress)
{ {
return get(path, Headers(), progress); return Get(path, Headers(), progress);
} }
inline std::shared_ptr<Response> Client::get(const char* path, const Headers& headers, Progress progress) inline std::shared_ptr<Response> Client::Get(const char* path, const Headers& headers, Progress progress)
{ {
Request req; Request req;
req.method = "GET"; req.method = "GET";
@ -1965,12 +2005,12 @@ inline std::shared_ptr<Response> Client::get(const char* path, const Headers& he
return send(req, *res) ? res : nullptr; return send(req, *res) ? res : nullptr;
} }
inline std::shared_ptr<Response> Client::head(const char* path) inline std::shared_ptr<Response> Client::Head(const char* path)
{ {
return head(path, Headers()); return Head(path, Headers());
} }
inline std::shared_ptr<Response> Client::head(const char* path, const Headers& headers) inline std::shared_ptr<Response> Client::Head(const char* path, const Headers& headers)
{ {
Request req; Request req;
req.method = "HEAD"; req.method = "HEAD";
@ -1982,13 +2022,13 @@ inline std::shared_ptr<Response> Client::head(const char* path, const Headers& h
return send(req, *res) ? res : nullptr; return send(req, *res) ? res : nullptr;
} }
inline std::shared_ptr<Response> Client::post( inline std::shared_ptr<Response> Client::Post(
const char* path, const std::string& body, const char* content_type) const char* path, const std::string& body, const char* content_type)
{ {
return post(path, Headers(), body, content_type); return Post(path, Headers(), body, content_type);
} }
inline std::shared_ptr<Response> Client::post( inline std::shared_ptr<Response> Client::Post(
const char* path, const Headers& headers, const std::string& body, const char* content_type) const char* path, const Headers& headers, const std::string& body, const char* content_type)
{ {
Request req; Request req;
@ -2004,12 +2044,12 @@ inline std::shared_ptr<Response> Client::post(
return send(req, *res) ? res : nullptr; return send(req, *res) ? res : nullptr;
} }
inline std::shared_ptr<Response> Client::post(const char* path, const Params& params) inline std::shared_ptr<Response> Client::Post(const char* path, const Params& params)
{ {
return post(path, Headers(), params); return Post(path, Headers(), params);
} }
inline std::shared_ptr<Response> Client::post(const char* path, const Headers& headers, const Params& params) inline std::shared_ptr<Response> Client::Post(const char* path, const Headers& headers, const Params& params)
{ {
std::string query; std::string query;
for (auto it = params.begin(); it != params.end(); ++it) { for (auto it = params.begin(); it != params.end(); ++it) {
@ -2021,7 +2061,63 @@ inline std::shared_ptr<Response> Client::post(const char* path, const Headers& h
query += it->second; query += it->second;
} }
return post(path, headers, query, "application/x-www-form-urlencoded"); return Post(path, headers, query, "application/x-www-form-urlencoded");
}
inline std::shared_ptr<Response> Client::Put(
const char* path, const std::string& body, const char* content_type)
{
return Put(path, Headers(), body, content_type);
}
inline std::shared_ptr<Response> Client::Put(
const char* path, const Headers& headers, const std::string& body, const char* content_type)
{
Request req;
req.method = "PUT";
req.headers = headers;
req.path = path;
req.headers.emplace("Content-Type", content_type);
req.body = body;
auto res = std::make_shared<Response>();
return send(req, *res) ? res : nullptr;
}
inline std::shared_ptr<Response> Client::Delete(const char* path)
{
return Delete(path, Headers());
}
inline std::shared_ptr<Response> Client::Delete(const char* path, const Headers& headers)
{
Request req;
req.method = "DELETE";
req.path = path;
req.headers = headers;
auto res = std::make_shared<Response>();
return send(req, *res) ? res : nullptr;
}
inline std::shared_ptr<Response> Client::Options(const char* path)
{
return Options(path, Headers());
}
inline std::shared_ptr<Response> Client::Options(const char* path, const Headers& headers)
{
Request req;
req.method = "OPTIONS";
req.path = path;
req.headers = headers;
auto res = std::make_shared<Response>();
return send(req, *res) ? res : nullptr;
} }
/* /*

View File

@ -131,7 +131,7 @@ void testChunkedEncoding(httplib::HttpVersion ver)
httplib::Client cli(host, port, sec, ver); httplib::Client cli(host, port, sec, ver);
#endif #endif
auto res = cli.get("/httpgallery/chunked/chunkedimage.aspx?0.4153841143030137"); auto res = cli.Get("/httpgallery/chunked/chunkedimage.aspx?0.4153841143030137");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
std::string out; std::string out;
@ -163,7 +163,7 @@ TEST(RangeTest, FromHTTPBin)
{ {
httplib::Headers headers; httplib::Headers headers;
auto res = cli.get("/range/32", headers); auto res = cli.Get("/range/32", headers);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(res->body, "abcdefghijklmnopqrstuvwxyzabcdef"); EXPECT_EQ(res->body, "abcdefghijklmnopqrstuvwxyzabcdef");
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
@ -171,7 +171,7 @@ TEST(RangeTest, FromHTTPBin)
{ {
httplib::Headers headers = { httplib::make_range_header(1) }; httplib::Headers headers = { httplib::make_range_header(1) };
auto res = cli.get("/range/32", headers); auto res = cli.Get("/range/32", headers);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(res->body, "bcdefghijklmnopqrstuvwxyzabcdef"); EXPECT_EQ(res->body, "bcdefghijklmnopqrstuvwxyzabcdef");
EXPECT_EQ(206, res->status); EXPECT_EQ(206, res->status);
@ -179,7 +179,7 @@ TEST(RangeTest, FromHTTPBin)
{ {
httplib::Headers headers = { httplib::make_range_header(1, 10) }; httplib::Headers headers = { httplib::make_range_header(1, 10) };
auto res = cli.get("/range/32", headers); auto res = cli.Get("/range/32", headers);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(res->body, "bcdefghijk"); EXPECT_EQ(res->body, "bcdefghijk");
EXPECT_EQ(206, res->status); EXPECT_EQ(206, res->status);
@ -200,7 +200,7 @@ TEST(ConnectionErrorTest, InvalidHost)
httplib::Client cli(host, port, sec, ver); httplib::Client cli(host, port, sec, ver);
#endif #endif
auto res = cli.get("/"); auto res = cli.Get("/");
ASSERT_TRUE(res == nullptr); ASSERT_TRUE(res == nullptr);
} }
@ -218,7 +218,7 @@ TEST(ConnectionErrorTest, InvalidPort)
httplib::Client cli(host, port, sec, ver); httplib::Client cli(host, port, sec, ver);
#endif #endif
auto res = cli.get("/"); auto res = cli.Get("/");
ASSERT_TRUE(res == nullptr); ASSERT_TRUE(res == nullptr);
} }
@ -236,7 +236,7 @@ TEST(ConnectionErrorTest, Timeout)
httplib::Client cli(host, port, sec, ver); httplib::Client cli(host, port, sec, ver);
#endif #endif
auto res = cli.get("/"); auto res = cli.Get("/");
ASSERT_TRUE(res == nullptr); ASSERT_TRUE(res == nullptr);
} }
@ -259,31 +259,31 @@ protected:
virtual void SetUp() { virtual void SetUp() {
svr_.set_base_dir("./www"); svr_.set_base_dir("./www");
svr_.get("/hi", [&](const Request& /*req*/, Response& res) { svr_.Get("/hi", [&](const Request& /*req*/, Response& res) {
res.set_content("Hello World!", "text/plain"); res.set_content("Hello World!", "text/plain");
}) })
.get("/slow", [&](const Request& /*req*/, Response& res) { .Get("/slow", [&](const Request& /*req*/, Response& res) {
msleep(2000); msleep(2000);
res.set_content("slow", "text/plain"); res.set_content("slow", "text/plain");
}) })
.get("/remote_addr", [&](const Request& req, Response& res) { .Get("/remote_addr", [&](const Request& req, Response& res) {
auto remote_addr = req.headers.find("REMOTE_ADDR")->second; auto remote_addr = req.headers.find("REMOTE_ADDR")->second;
res.set_content(remote_addr.c_str(), "text/plain"); res.set_content(remote_addr.c_str(), "text/plain");
}) })
.get("/endwith%", [&](const Request& /*req*/, Response& res) { .Get("/endwith%", [&](const Request& /*req*/, Response& res) {
res.set_content("Hello World!", "text/plain"); res.set_content("Hello World!", "text/plain");
}) })
.get("/", [&](const Request& /*req*/, Response& res) { .Get("/", [&](const Request& /*req*/, Response& res) {
res.set_redirect("/hi"); res.set_redirect("/hi");
}) })
.post("/person", [&](const Request& req, Response& res) { .Post("/person", [&](const Request& req, Response& res) {
if (req.has_param("name") && req.has_param("note")) { if (req.has_param("name") && req.has_param("note")) {
persons_[req.get_param_value("name")] = req.get_param_value("note"); persons_[req.get_param_value("name")] = req.get_param_value("note");
} else { } else {
res.status = 400; res.status = 400;
} }
}) })
.get("/person/(.*)", [&](const Request& req, Response& res) { .Get("/person/(.*)", [&](const Request& req, Response& res) {
string name = req.matches[1]; string name = req.matches[1];
if (persons_.find(name) != persons_.end()) { if (persons_.find(name) != persons_.end()) {
auto note = persons_[name]; auto note = persons_[name];
@ -292,14 +292,14 @@ protected:
res.status = 404; res.status = 404;
} }
}) })
.post("/chunked", [&](const Request& req, Response& /*res*/) { .Post("/chunked", [&](const Request& req, Response& /*res*/) {
EXPECT_EQ(req.body, "dechunked post body"); EXPECT_EQ(req.body, "dechunked post body");
}) })
.post("/largechunked", [&](const Request& req, Response& /*res*/) { .Post("/largechunked", [&](const Request& req, Response& /*res*/) {
std::string expected(6 * 30 * 1024u, 'a'); std::string expected(6 * 30 * 1024u, 'a');
EXPECT_EQ(req.body, expected); EXPECT_EQ(req.body, expected);
}) })
.post("/multipart", [&](const Request& req, Response& /*res*/) { .Post("/multipart", [&](const Request& req, Response& /*res*/) {
EXPECT_EQ(5u, req.files.size()); EXPECT_EQ(5u, req.files.size());
ASSERT_TRUE(!req.has_file("???")); ASSERT_TRUE(!req.has_file("???"));
@ -329,14 +329,24 @@ protected:
EXPECT_EQ(0u, file.length); EXPECT_EQ(0u, file.length);
} }
}) })
.Put("/put", [&](const Request& req, Response& res) {
EXPECT_EQ(req.body, "PUT");
res.set_content(req.body, "text/plain");
})
.Delete("/delete", [&](const Request& /*req*/, Response& res) {
res.set_content("DELETE", "text/plain");
})
.Options(R"(\*)", [&](const Request& /*req*/, Response& res) {
res.set_header("Allow", "GET, POST, HEAD, OPTIONS");
})
#ifdef CPPHTTPLIB_ZLIB_SUPPORT #ifdef CPPHTTPLIB_ZLIB_SUPPORT
.get("/gzip", [&](const Request& /*req*/, Response& res) { .Get("/gzip", [&](const Request& /*req*/, Response& res) {
res.set_content("1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890", "text/plain"); res.set_content("1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890", "text/plain");
}) })
.get("/nogzip", [&](const Request& /*req*/, Response& res) { .Get("/nogzip", [&](const Request& /*req*/, Response& res) {
res.set_content("1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890", "application/octet-stream"); res.set_content("1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890", "application/octet-stream");
}) })
.post("/gzipmultipart", [&](const Request& req, Response& /*res*/) { .Post("/gzipmultipart", [&](const Request& req, Response& /*res*/) {
EXPECT_EQ(2u, req.files.size()); EXPECT_EQ(2u, req.files.size());
ASSERT_TRUE(!req.has_file("???")); ASSERT_TRUE(!req.has_file("???"));
@ -388,7 +398,7 @@ protected:
TEST_F(ServerTest, GetMethod200) TEST_F(ServerTest, GetMethod200)
{ {
auto res = cli_.get("/hi"); auto res = cli_.Get("/hi");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("Content-Type")); EXPECT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -397,7 +407,7 @@ TEST_F(ServerTest, GetMethod200)
TEST_F(ServerTest, GetMethod302) TEST_F(ServerTest, GetMethod302)
{ {
auto res = cli_.get("/"); auto res = cli_.Get("/");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(302, res->status); EXPECT_EQ(302, res->status);
EXPECT_EQ("/hi", res->get_header_value("Location")); EXPECT_EQ("/hi", res->get_header_value("Location"));
@ -405,14 +415,14 @@ TEST_F(ServerTest, GetMethod302)
TEST_F(ServerTest, GetMethod404) TEST_F(ServerTest, GetMethod404)
{ {
auto res = cli_.get("/invalid"); auto res = cli_.Get("/invalid");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
TEST_F(ServerTest, HeadMethod200) TEST_F(ServerTest, HeadMethod200)
{ {
auto res = cli_.head("/hi"); auto res = cli_.Head("/hi");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("Content-Type")); EXPECT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -421,7 +431,7 @@ TEST_F(ServerTest, HeadMethod200)
TEST_F(ServerTest, HeadMethod404) TEST_F(ServerTest, HeadMethod404)
{ {
auto res = cli_.head("/invalid"); auto res = cli_.Head("/invalid");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
EXPECT_EQ("", res->body); EXPECT_EQ("", res->body);
@ -429,7 +439,7 @@ TEST_F(ServerTest, HeadMethod404)
TEST_F(ServerTest, GetMethodPersonJohn) TEST_F(ServerTest, GetMethodPersonJohn)
{ {
auto res = cli_.get("/person/john"); auto res = cli_.Get("/person/john");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("Content-Type")); EXPECT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -438,15 +448,15 @@ TEST_F(ServerTest, GetMethodPersonJohn)
TEST_F(ServerTest, PostMethod1) TEST_F(ServerTest, PostMethod1)
{ {
auto res = cli_.get("/person/john1"); auto res = cli_.Get("/person/john1");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(404, res->status); ASSERT_EQ(404, res->status);
res = cli_.post("/person", "name=john1&note=coder", "application/x-www-form-urlencoded"); res = cli_.Post("/person", "name=john1&note=coder", "application/x-www-form-urlencoded");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(200, res->status); ASSERT_EQ(200, res->status);
res = cli_.get("/person/john1"); res = cli_.Get("/person/john1");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(200, res->status); ASSERT_EQ(200, res->status);
ASSERT_EQ("text/plain", res->get_header_value("Content-Type")); ASSERT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -455,7 +465,7 @@ TEST_F(ServerTest, PostMethod1)
TEST_F(ServerTest, PostMethod2) TEST_F(ServerTest, PostMethod2)
{ {
auto res = cli_.get("/person/john2"); auto res = cli_.Get("/person/john2");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(404, res->status); ASSERT_EQ(404, res->status);
@ -463,11 +473,11 @@ TEST_F(ServerTest, PostMethod2)
params.emplace("name", "john2"); params.emplace("name", "john2");
params.emplace("note", "coder"); params.emplace("note", "coder");
res = cli_.post("/person", params); res = cli_.Post("/person", params);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(200, res->status); ASSERT_EQ(200, res->status);
res = cli_.get("/person/john2"); res = cli_.Get("/person/john2");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(200, res->status); ASSERT_EQ(200, res->status);
ASSERT_EQ("text/plain", res->get_header_value("Content-Type")); ASSERT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -476,7 +486,7 @@ TEST_F(ServerTest, PostMethod2)
TEST_F(ServerTest, GetMethodDir) TEST_F(ServerTest, GetMethodDir)
{ {
auto res = cli_.get("/dir/"); auto res = cli_.Get("/dir/");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/html", res->get_header_value("Content-Type")); EXPECT_EQ("text/html", res->get_header_value("Content-Type"));
@ -495,7 +505,7 @@ TEST_F(ServerTest, GetMethodDir)
TEST_F(ServerTest, GetMethodDirTest) TEST_F(ServerTest, GetMethodDirTest)
{ {
auto res = cli_.get("/dir/test.html"); auto res = cli_.Get("/dir/test.html");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/html", res->get_header_value("Content-Type")); EXPECT_EQ("text/html", res->get_header_value("Content-Type"));
@ -504,7 +514,7 @@ TEST_F(ServerTest, GetMethodDirTest)
TEST_F(ServerTest, GetMethodDirTestWithDoubleDots) TEST_F(ServerTest, GetMethodDirTestWithDoubleDots)
{ {
auto res = cli_.get("/dir/../dir/test.html"); auto res = cli_.Get("/dir/../dir/test.html");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/html", res->get_header_value("Content-Type")); EXPECT_EQ("text/html", res->get_header_value("Content-Type"));
@ -513,21 +523,21 @@ TEST_F(ServerTest, GetMethodDirTestWithDoubleDots)
TEST_F(ServerTest, GetMethodInvalidPath) TEST_F(ServerTest, GetMethodInvalidPath)
{ {
auto res = cli_.get("/dir/../test.html"); auto res = cli_.Get("/dir/../test.html");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
TEST_F(ServerTest, GetMethodOutOfBaseDir) TEST_F(ServerTest, GetMethodOutOfBaseDir)
{ {
auto res = cli_.get("/../www/dir/test.html"); auto res = cli_.Get("/../www/dir/test.html");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
TEST_F(ServerTest, GetMethodOutOfBaseDir2) TEST_F(ServerTest, GetMethodOutOfBaseDir2)
{ {
auto res = cli_.get("/dir/../../www/dir/test.html"); auto res = cli_.Get("/dir/../../www/dir/test.html");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
@ -540,13 +550,13 @@ TEST_F(ServerTest, InvalidBaseDir)
TEST_F(ServerTest, EmptyRequest) TEST_F(ServerTest, EmptyRequest)
{ {
auto res = cli_.get(""); auto res = cli_.Get("");
ASSERT_TRUE(res == nullptr); ASSERT_TRUE(res == nullptr);
} }
TEST_F(ServerTest, LongRequest) TEST_F(ServerTest, LongRequest)
{ {
auto res = cli_.get("/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/__ok__"); auto res = cli_.Get("/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/__ok__");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
@ -554,7 +564,7 @@ TEST_F(ServerTest, LongRequest)
TEST_F(ServerTest, TooLongRequest) TEST_F(ServerTest, TooLongRequest)
{ {
auto res = cli_.get("/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/__ng___"); auto res = cli_.Get("/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/TooLongRequest/__ng___");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
@ -610,28 +620,28 @@ TEST_F(ServerTest, TooLongHeader)
TEST_F(ServerTest, PercentEncoding) TEST_F(ServerTest, PercentEncoding)
{ {
auto res = cli_.get("/e%6edwith%"); auto res = cli_.Get("/e%6edwith%");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
} }
TEST_F(ServerTest, PercentEncodingUnicode) TEST_F(ServerTest, PercentEncodingUnicode)
{ {
auto res = cli_.get("/e%u006edwith%"); auto res = cli_.Get("/e%u006edwith%");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
} }
TEST_F(ServerTest, InvalidPercentEncoding) TEST_F(ServerTest, InvalidPercentEncoding)
{ {
auto res = cli_.get("/%endwith%"); auto res = cli_.Get("/%endwith%");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
TEST_F(ServerTest, InvalidPercentEncodingUnicode) TEST_F(ServerTest, InvalidPercentEncodingUnicode)
{ {
auto res = cli_.get("/%uendwith%"); auto res = cli_.Get("/%uendwith%");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(404, res->status); EXPECT_EQ(404, res->status);
} }
@ -663,7 +673,7 @@ TEST_F(ServerTest, MultipartFormData)
TEST_F(ServerTest, CaseInsensitiveHeaderName) TEST_F(ServerTest, CaseInsensitiveHeaderName)
{ {
auto res = cli_.get("/hi"); auto res = cli_.Get("/hi");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("content-type")); EXPECT_EQ("text/plain", res->get_header_value("content-type"));
@ -731,7 +741,7 @@ TEST_F(ServerTest, LargeChunkedPost) {
TEST_F(ServerTest, GetMethodRemoteAddr) TEST_F(ServerTest, GetMethodRemoteAddr)
{ {
auto res = cli_.get("/remote_addr"); auto res = cli_.Get("/remote_addr");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("Content-Type")); EXPECT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -740,18 +750,43 @@ TEST_F(ServerTest, GetMethodRemoteAddr)
TEST_F(ServerTest, SlowRequest) TEST_F(ServerTest, SlowRequest)
{ {
request_threads_.push_back(std::thread([=]() { auto res = cli_.get("/slow"); })); request_threads_.push_back(std::thread([=]() { auto res = cli_.Get("/slow"); }));
request_threads_.push_back(std::thread([=]() { auto res = cli_.get("/slow"); })); request_threads_.push_back(std::thread([=]() { auto res = cli_.Get("/slow"); }));
request_threads_.push_back(std::thread([=]() { auto res = cli_.get("/slow"); })); request_threads_.push_back(std::thread([=]() { auto res = cli_.Get("/slow"); }));
msleep(100); msleep(100);
} }
TEST_F(ServerTest, Put)
{
auto res = cli_.Put("/put", "PUT", "text/plain");
ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status);
EXPECT_EQ("PUT", res->body);
}
TEST_F(ServerTest, Delete)
{
auto res = cli_.Delete("/delete");
ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status);
EXPECT_EQ("DELETE", res->body);
}
TEST_F(ServerTest, Options)
{
auto res = cli_.Options("*");
ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status);
EXPECT_EQ("GET, POST, HEAD, OPTIONS", res->get_header_value("Allow"));
EXPECT_TRUE(res->body.empty());
}
#ifdef CPPHTTPLIB_ZLIB_SUPPORT #ifdef CPPHTTPLIB_ZLIB_SUPPORT
TEST_F(ServerTest, Gzip) TEST_F(ServerTest, Gzip)
{ {
Headers headers; Headers headers;
headers.emplace("Accept-Encoding", "gzip, deflate"); headers.emplace("Accept-Encoding", "gzip, deflate");
auto res = cli_.get("/gzip", headers); auto res = cli_.Get("/gzip", headers);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ("gzip", res->get_header_value("Content-Encoding")); EXPECT_EQ("gzip", res->get_header_value("Content-Encoding"));
@ -765,7 +800,7 @@ TEST_F(ServerTest, NoGzip)
{ {
Headers headers; Headers headers;
headers.emplace("Accept-Encoding", "gzip, deflate"); headers.emplace("Accept-Encoding", "gzip, deflate");
auto res = cli_.get("/nogzip", headers); auto res = cli_.Get("/nogzip", headers);
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(false, res->has_header("Content-Encoding")); EXPECT_EQ(false, res->has_header("Content-Encoding"));
@ -839,7 +874,7 @@ protected:
{} {}
virtual void SetUp() { virtual void SetUp() {
svr_.get("/hi", [&](const Request& /*req*/, Response& res) { svr_.Get("/hi", [&](const Request& /*req*/, Response& res) {
res.set_content("Hello World!", "text/plain"); res.set_content("Hello World!", "text/plain");
}); });
@ -869,7 +904,7 @@ protected:
TEST_F(ServerTestWithAI_PASSIVE, GetMethod200) TEST_F(ServerTestWithAI_PASSIVE, GetMethod200)
{ {
auto res = cli_.get("/hi"); auto res = cli_.Get("/hi");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
EXPECT_EQ(200, res->status); EXPECT_EQ(200, res->status);
EXPECT_EQ("text/plain", res->get_header_value("Content-Type")); EXPECT_EQ("text/plain", res->get_header_value("Content-Type"));
@ -914,7 +949,7 @@ TEST_F(ServerUpDownTest, QuickStartStop)
TEST(SSLClientTest, ServerNameIndication) TEST(SSLClientTest, ServerNameIndication)
{ {
SSLClient cli("httpbin.org", 443); SSLClient cli("httpbin.org", 443);
auto res = cli.get("/get"); auto res = cli.Get("/get");
ASSERT_TRUE(res != nullptr); ASSERT_TRUE(res != nullptr);
ASSERT_EQ(200, res->status); ASSERT_EQ(200, res->status);
} }