//#define CROW_ENABLE_LOGGING #include #include #include "routing.h" #include "utility.h" #include "crow.h" #include "json.h" using namespace std; using namespace crow; struct Test { Test(); virtual void test() = 0; }; vector tests; Test::Test() { tests.push_back(this); } bool failed__ = false; void error_print() { cerr << endl; } template void error_print(const A& a, Args...args) { cerr< void fail(Args...args) { error_print(args...);failed__ = true; } #define ASSERT_TRUE(x) if (!(x)) fail("Assert fail: expected ", #x, " is true, at " __FILE__ ":",__LINE__) #define ASSERT_EQUAL(a, b) if ((a) != (b)) fail("Assert fail: expected ", (a), " actual " , (b), ", " #a " == " #b ", at " __FILE__ ":",__LINE__) #define ASSERT_NOTEQUAL(a, b) if ((a) == (b)) fail("Assert fail: not expected ", (a), ", " #a " != " #b ", at " __FILE__ ":",__LINE__) #define ASSERT_THROW(x) \ try \ { \ x; \ fail("Assert fail: exception should be thrown"); \ } \ catch(std::exception&) \ { \ } #define TEST(x) struct test##x:public Test{void test();}x##_; \ void test##x::test() #define DISABLE_TEST(x) struct test##x{void test();}x##_; \ void test##x::test() TEST(Rule) { TaggedRule<> r("/http/"); r.name("abc"); // empty handler - fail to validate try { r.validate(); fail("empty handler should fail to validate"); } catch(runtime_error& e) { } int x = 0; // registering handler r([&x]{x = 1;return "";}); r.validate(); // executing handler ASSERT_EQUAL(0, x); r.handle(request(), routing_params()); ASSERT_EQUAL(1, x); // registering handler with request argument r([&x](const crow::request&){x = 2;return "";}); r.validate(); // executing handler ASSERT_EQUAL(1, x); r.handle(request(), routing_params()); ASSERT_EQUAL(2, x); } TEST(ParameterTagging) { static_assert(black_magic::is_valid(""), "valid url"); static_assert(!black_magic::is_valid(""), "invalid url"); static_assert(!black_magic::is_valid("nt>"), "invalid url"); ASSERT_EQUAL(1, black_magic::get_parameter_tag("")); ASSERT_EQUAL(2, black_magic::get_parameter_tag("")); ASSERT_EQUAL(3, black_magic::get_parameter_tag("")); ASSERT_EQUAL(3, black_magic::get_parameter_tag("")); ASSERT_EQUAL(4, black_magic::get_parameter_tag("")); ASSERT_EQUAL(4, black_magic::get_parameter_tag("")); ASSERT_EQUAL(5, black_magic::get_parameter_tag("")); ASSERT_EQUAL(6*6+6+1, black_magic::get_parameter_tag("")); ASSERT_EQUAL(6*6+6+2, black_magic::get_parameter_tag("")); ASSERT_EQUAL(6*6+6*3+2, black_magic::get_parameter_tag("")); // url definition parsed in compile time, build into *one number*, and given to template argument static_assert(std::is_same, black_magic::arguments<6*6+6*3+2>::type>::value, "tag to type container"); } TEST(RoutingTest) { Crow app; int A{}; uint32_t B{}; double C{}; string D{}; string E{}; CROW_ROUTE(app, "/0/") ([&](uint32_t b){ B = b; return "OK"; }); CROW_ROUTE(app, "/1//") ([&](int a, uint32_t b){ A = a; B = b; return "OK"; }); CROW_ROUTE(app, "/4////") ([&](int a, uint32_t b, double c, string d){ A = a; B = b; C = c; D = d; return "OK"; }); CROW_ROUTE(app, "/5/////") ([&](int a, uint32_t b, double c, string d, string e){ A = a; B = b; C = c; D = d; E = e; return "OK"; }); app.validate(); //app.debug_print(); { request req; req.url = "/0/1001999"; auto res = app.handle(req); ASSERT_EQUAL(200, res.code); ASSERT_EQUAL(1001999, B); } { request req; req.url = "/1/-100/1999"; auto res = app.handle(req); ASSERT_EQUAL(200, res.code); ASSERT_EQUAL(-100, A); ASSERT_EQUAL(1999, B); } { request req; req.url = "/4/5000/3/-2.71828/hellhere"; req.headers["TestHeader"] = "Value"; auto res = app.handle(req); ASSERT_EQUAL(200, res.code); ASSERT_EQUAL(5000, A); ASSERT_EQUAL(3, B); ASSERT_EQUAL(-2.71828, C); ASSERT_EQUAL("hellhere", D); } { request req; req.url = "/5/-5/999/3.141592/hello_there/a/b/c/d"; req.headers["TestHeader"] = "Value"; auto res = app.handle(req); ASSERT_EQUAL(200, res.code); ASSERT_EQUAL(-5, A); ASSERT_EQUAL(999, B); ASSERT_EQUAL(3.141592, C); ASSERT_EQUAL("hello_there", D); ASSERT_EQUAL("a/b/c/d", E); } } TEST(simple_response_routing_params) { ASSERT_EQUAL(100, response(100).code); ASSERT_EQUAL(200, response("Hello there").code); ASSERT_EQUAL(500, response(500, "Internal Error?").code); routing_params rp; rp.int_params.push_back(1); rp.int_params.push_back(5); rp.uint_params.push_back(2); rp.double_params.push_back(3); rp.string_params.push_back("hello"); ASSERT_EQUAL(1, rp.get(0)); ASSERT_EQUAL(5, rp.get(1)); ASSERT_EQUAL(2, rp.get(0)); ASSERT_EQUAL(3, rp.get(0)); ASSERT_EQUAL("hello", rp.get(0)); } TEST(server_handling_error_request) { static char buf[2048]; Crow app; CROW_ROUTE(app, "/")([]{return "A";}); Server server(&app, 45451); auto _ = async(launch::async, [&]{server.run();}); std::string sendmsg = "POX"; asio::io_service is; { asio::ip::tcp::socket c(is); c.connect(asio::ip::tcp::endpoint(asio::ip::address::from_string("127.0.0.1"), 45451)); c.send(asio::buffer(sendmsg)); try { c.receive(asio::buffer(buf, 2048)); fail(); } catch(std::exception& e) { std::cerr << e.what() << std::endl; } } server.stop(); } TEST(multi_server) { static char buf[2048]; Crow app1, app2; CROW_ROUTE(app1, "/")([]{return "A";}); CROW_ROUTE(app2, "/")([]{return "B";}); Server server1(&app1, 45451); Server server2(&app2, 45452); auto _ = async(launch::async, [&]{server1.run();}); auto _2 = async(launch::async, [&]{server2.run();}); std::string sendmsg = "POST /\r\nContent-Length:3\r\nX-HeaderTest: 123\r\n\r\nA=B\r\n"; asio::io_service is; { asio::ip::tcp::socket c(is); c.connect(asio::ip::tcp::endpoint(asio::ip::address::from_string("127.0.0.1"), 45451)); c.send(asio::buffer(sendmsg)); size_t recved = c.receive(asio::buffer(buf, 2048)); ASSERT_EQUAL('A', buf[recved-1]); } { asio::ip::tcp::socket c(is); c.connect(asio::ip::tcp::endpoint(asio::ip::address::from_string("127.0.0.1"), 45452)); for(auto ch:sendmsg) { char buf[1] = {ch}; std::cerr << ch << '(' << (int)ch<<')'<test(); } catch(std::exception& e) { fail(e.what()); } if (failed__) { cerr << "F"; failed = true; } else cerr << "."; } cerr<