Compare commits
	
		
			No commits in common. "190cdf25b39784a3d38d01afde8dd53928f2709b" and "c25719167aa9a4a6f2dd8c5c9fbb766cf63c66ff" have entirely different histories.
		
	
	
		
			190cdf25b3
			...
			c25719167a
		
	
		
							
								
								
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,11 +0,0 @@ | |||||||
| # Never use CMAKE in production |  | ||||||
| CMakeLists.txt |  | ||||||
| cmake-build-debug/ |  | ||||||
| # Output of build system |  | ||||||
| built/ |  | ||||||
| # This is a compilated build system script |  | ||||||
| building/main |  | ||||||
| building/*.png |  | ||||||
| building/*.svg |  | ||||||
| 
 |  | ||||||
| .idea/ |  | ||||||
| @ -1,6 +1,6 @@ | |||||||
| # ИУ9-21Б Вэб-чат C.A | # Collarbone Anihilation | ||||||
| 
 | 
 | ||||||
| Сделан на летней практике 5-ю первокурсниками ИУ9  | Веб-чат | ||||||
| 
 | 
 | ||||||
| # Список участников | # Список участников | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,9 +0,0 @@ | |||||||
| .aaa {font-size: 50px} |  | ||||||
| 
 |  | ||||||
| .ccc .aaa { |  | ||||||
|     color: yellow; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .ccc #bbb { |  | ||||||
|     color: green; |  | ||||||
| } |  | ||||||
| @ -1,16 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html lang="en"> |  | ||||||
| <head> |  | ||||||
|     <meta charset="UTF-8"> |  | ||||||
|     <title>This is a test</title> |  | ||||||
|     <link rel="stylesheet" href="/assets/css/test.css" > |  | ||||||
| </head> |  | ||||||
| <body> |  | ||||||
|     <h1> Test Test Test</h1> |  | ||||||
|     <p class="aaa">                        Test Test      asdasdsa         Test</p> |  | ||||||
|     <div class="ccc"> |  | ||||||
|         <p class="aaa"> Inside aaaa </p> |  | ||||||
|         <p id="bbb"> Iside bbbb </p> |  | ||||||
|     </div> |  | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
| @ -1,10 +0,0 @@ | |||||||
| #!/usr/bin/env sh |  | ||||||
| 
 |  | ||||||
| BUILDING_DIR="./building" |  | ||||||
| [ -d "$BUILDING_DIR" ] || exit 1 |  | ||||||
| MAIN_FILE="$BUILDING_DIR/main.cpp" |  | ||||||
| [ -f "$MAIN_FILE" ] || exit 1 |  | ||||||
| 
 |  | ||||||
| COOL_FLAGS="$(pkg-config --cflags regexis024-build-system)" |  | ||||||
| 
 |  | ||||||
| g++ $COOL_FLAGS -o "$BUILDING_DIR/main" "$MAIN_FILE" || exit 1 |  | ||||||
| @ -1,139 +0,0 @@ | |||||||
| #include <utility> |  | ||||||
| 
 |  | ||||||
| #include "regexis024_build_system.h" |  | ||||||
| 
 |  | ||||||
| std::vector<std::string> getFromPkgConfig(const std::string& req, const std::string& name){ |  | ||||||
|     std::string pc_stdout, pc_stderr; |  | ||||||
|     CommandReturnCode rc = executeCommand_and_save_output({"pkg-config", "--" + req, name}, pc_stdout, pc_stderr); |  | ||||||
|     ASSERT(rc.isOk(), "failed to use pkg-config beacause of:\n" + pc_stderr); |  | ||||||
|     // todo: learn how pkg-config actually stores these options
 |  | ||||||
|     std::vector<std::string> result; |  | ||||||
|     for (char ch: pc_stdout) { |  | ||||||
|         if (result.empty()) |  | ||||||
|             result.emplace_back(); |  | ||||||
|         if (ch == ' ') |  | ||||||
|             result.emplace_back(); |  | ||||||
|         else |  | ||||||
|             result.back() += ch; |  | ||||||
|     } |  | ||||||
|     return result; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| ExternalLibraryTarget formExternalLibraryTargetWithNativeName(const std::string& name) { |  | ||||||
|     return {name, {getFromPkgConfig("cflags", name), getFromPkgConfig("libs", name)}}; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| struct CAWebChat { |  | ||||||
|     /* Building runlevel */ |  | ||||||
|     BuildUnitsArray runlevel_1; |  | ||||||
|     /* Installation runlevel */ |  | ||||||
|     BuildUnitsArray runlevel_2; |  | ||||||
| 
 |  | ||||||
|     std::string build_type; |  | ||||||
| 
 |  | ||||||
|     std::vector<std::string> warning_flags = {"-Wall", "-Wno-unused-variable", "-Werror=return-type","-pedantic", |  | ||||||
|         "-Wno-unused-but-set-variable", "-Wno-reorder"}; |  | ||||||
|     std::vector<std::string> version_flags = {"--std", "c++14", "-D", "_POSIX_C_SOURCE=200809L"}; |  | ||||||
|     std::vector<std::string> debug_defines_release = {"_GLIBCXX_DEBUG"}; |  | ||||||
|     std::vector<std::string> debug_defines_debug = {"_GLIBCXX_DEBUG", "DEBUG_ALLOW_LOUD"}; |  | ||||||
|     std::vector<std::string> opt_flags_release = {"-g", "-O2"}; |  | ||||||
|     std::vector<std::string> opt_flags_debug = {"-g", "-ggdb", "-O0"}; |  | ||||||
| 
 |  | ||||||
|     std::vector<std::string> getSomeRadFlags() const { |  | ||||||
|         std::vector<std::string> my_flag_collection; |  | ||||||
|         gxx_add_cli_options(my_flag_collection, warning_flags); |  | ||||||
|         gxx_add_cli_options(my_flag_collection, version_flags); |  | ||||||
|         if (build_type == "release") { |  | ||||||
|             gxx_add_cli_defines(my_flag_collection, debug_defines_release); |  | ||||||
|             gxx_add_cli_options(my_flag_collection, opt_flags_release); |  | ||||||
|         } else if (build_type == "debug") { |  | ||||||
|             gxx_add_cli_defines(my_flag_collection, debug_defines_debug); |  | ||||||
|             gxx_add_cli_options(my_flag_collection, opt_flags_debug); |  | ||||||
|         } |  | ||||||
|         return my_flag_collection; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     CAWebChat(std::string  _build_type, const NormalCBuildSystemCommandMeaning& cmd) |  | ||||||
|         : build_type(std::move(_build_type)) |  | ||||||
|     { |  | ||||||
|         ASSERT(build_type == "release" || build_type == "debug", "Unknown build type"); |  | ||||||
| 
 |  | ||||||
|         std::vector<ExternalLibraryTarget> ext_targets = { |  | ||||||
|             formExternalLibraryTargetWithNativeName("libjsonincpp"), |  | ||||||
|             formExternalLibraryTargetWithNativeName("sqlite3"), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         std::vector<CTarget> my_targets; |  | ||||||
| 
 |  | ||||||
|         { CTarget T("engine_engine_number_9", "shared_library"); |  | ||||||
|             T.additional_compilation_flags = getSomeRadFlags(); |  | ||||||
|             T.proj_deps = {}; |  | ||||||
|             T.units = { |  | ||||||
|                 "baza.cpp", |  | ||||||
|                 "thread_synchronization.cpp", |  | ||||||
|                 "os_utils.cpp", |  | ||||||
|                 "http_structures/client_request_parse.cpp", |  | ||||||
|                 "http_structures/response_gen.cpp", |  | ||||||
|                 "connecting_assets/static_asset_manager.cpp", |  | ||||||
|                 "running_mainloop.cpp", |  | ||||||
|             }; |  | ||||||
|             for (std::string& u: T.units) |  | ||||||
|                 u = "http_server/engine_engine_number_9/" + u; |  | ||||||
|             T.include_pr = "http_server"; |  | ||||||
|             T.include_ir = ""; |  | ||||||
|             T.exported_headers = { |  | ||||||
|                 "baza.h", |  | ||||||
|                 "baza_throw.h", |  | ||||||
|                 "thread_synchronization.h", |  | ||||||
|                 "os_utils.h", |  | ||||||
|                 "connecting_assets/static_asset_manager.h", |  | ||||||
|                 "http_structures/client_request.h", |  | ||||||
|                 "http_structures/response_gen.h", |  | ||||||
|                 "running_mainloop.h", |  | ||||||
|             }; |  | ||||||
|             for (std::string& u: T.exported_headers) |  | ||||||
|                 u = "engine_engine_number_9/" + u; |  | ||||||
| 
 |  | ||||||
|             T.installation_dir = ""; |  | ||||||
|             my_targets.push_back(T); |  | ||||||
|         } |  | ||||||
|         { CTarget T("iu9-ca-web-chat", "executable"); |  | ||||||
|             T.additional_compilation_flags = getSomeRadFlags(); |  | ||||||
|             T.proj_deps = {CTargetDependenceOnProjectsLibrary("engine_engine_number_9")}; |  | ||||||
|             T.units = {"main.cpp"}; |  | ||||||
|             for (std::string& u: T.units) |  | ||||||
|                 u = "web_chat/" + u; |  | ||||||
|             T.include_pr = "web_chat"; |  | ||||||
|             T.installation_dir = ""; |  | ||||||
|             my_targets.push_back(T); |  | ||||||
|         } |  | ||||||
|         regular_ctargets_to_2bus_conversion(ext_targets, my_targets, runlevel_1, runlevel_2, |  | ||||||
|             cmd.project_root, cmd.installation_root); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| int main(int argc, char** argv) { |  | ||||||
|     try { |  | ||||||
|         ASSERT_pl(argc > 0); |  | ||||||
|         assert(argc > 0); |  | ||||||
|         std::vector<std::string> args(argc - 1); |  | ||||||
|         for (int i = 0; i + 1 < argc; i++) { |  | ||||||
|             args[i] = argv[i + 1]; |  | ||||||
|         } |  | ||||||
|         NormalCBuildSystemCommandMeaning cmd; |  | ||||||
|         regular_bs_cli_cmd_interpret(args, cmd); |  | ||||||
|         CAWebChat bs("debug", cmd); |  | ||||||
|         // std::string map = "Runlevel 1\n";
 |  | ||||||
|         // draw_bu_arr_in_dot(bs.runlevel_1, map);
 |  | ||||||
|         // map += "Runlevel 2\n";
 |  | ||||||
|         // draw_bu_arr_in_dot(bs.runlevel_2, map);
 |  | ||||||
|         // printf("%s", map.c_str());
 |  | ||||||
|         if (cmd.need_to_build) |  | ||||||
|             complete_tasks_of_build_units(bs.runlevel_1); |  | ||||||
|         umask(~0755); |  | ||||||
|         if (cmd.need_to_install) |  | ||||||
|             complete_tasks_of_build_units(bs.runlevel_2); |  | ||||||
|     } catch (const buildSystemFailure& e) { |  | ||||||
|         printf("Build system failure\n""%s\n", e.toString().c_str()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,41 +0,0 @@ | |||||||
| #include "baza.h" |  | ||||||
| 
 |  | ||||||
| #include <errno.h> |  | ||||||
| #include <string.h> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     ServerError::ServerError(const std::string &err, const std::string &file, const std::string &func, int line): err(err), |  | ||||||
|        FILE(file), |  | ||||||
|        func(func), |  | ||||||
|        LINE(line) { |  | ||||||
|         char buf[4096]; |  | ||||||
|         snprintf(buf, 4096, "Error occured in function %s (line %d of %s)\n" |  | ||||||
|                  "Error: %s", |  | ||||||
|                  func.c_str(), LINE, FILE.c_str(), err.c_str()); |  | ||||||
|         WHAT = buf; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     const char * ServerError::what() const noexcept { |  | ||||||
|         return WHAT.c_str(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     std::string prettyprint_errno(const std::string &pref) { |  | ||||||
|         const char* d = strerrorname_np(errno); |  | ||||||
|         return pref.empty() ? std::string(d) : std::string(pref) + ": " + d; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /* This function is VITAL */ |  | ||||||
|     bool strIn(const std::string &str, const char *arr[]) { |  | ||||||
|         for (const char** elPtr = arr; *elPtr != NULL; elPtr += 1) { |  | ||||||
|             if (str == (*elPtr)) |  | ||||||
|                 return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool endsIn(const std::string &a, const std::string &b) { |  | ||||||
|         if (b.size() > a.size()) |  | ||||||
|             return false; |  | ||||||
|         return std::equal(a.end() - (ssize_t)b.size(), a.end(), b.begin()); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,27 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_BAZA_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_BAZA_H |  | ||||||
| 
 |  | ||||||
| #include <string> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     class ServerError : public std::exception{ |  | ||||||
|         std::string err; |  | ||||||
|         std::string FILE; |  | ||||||
|         std::string func; |  | ||||||
|         int LINE; |  | ||||||
|         std::string WHAT; |  | ||||||
| 
 |  | ||||||
|     public: |  | ||||||
|         ServerError(const std::string &err, const std::string &file, const std::string &func, int line); |  | ||||||
| 
 |  | ||||||
|         const char *what() const noexcept override; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     std::string prettyprint_errno(const std::string& pref); |  | ||||||
| 
 |  | ||||||
|     bool strIn(const std::string& str, const char* arr[]); |  | ||||||
| 
 |  | ||||||
|     bool endsIn(const std::string& a, const std::string& b); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,16 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_BAZA_INTER_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_BAZA_INTER_H |  | ||||||
| 
 |  | ||||||
| /* Do not export this file */ |  | ||||||
| 
 |  | ||||||
| #include "baza.h" |  | ||||||
| 
 |  | ||||||
| #define THROW(err) throw ServerError(err, __FILE__, __func__, __LINE__) |  | ||||||
| #define THROW_on_errno(err) THROW(prettyprint_errno(err)) |  | ||||||
| #define THROW_on_errno_pl() THROW(prettyprint_errno("")) |  | ||||||
| #define ASSERT(cond, err) do { if (!(cond)) { THROW(err); } } while (0); |  | ||||||
| #define ASSERT_pl(cond) ASSERT(cond, "Failed assertion `" #cond "`") |  | ||||||
| #define ASSERT_on_iret(iret, err) ASSERT((iret) >= 0, prettyprint_errno(err)); |  | ||||||
| #define ASSERT_on_iret_pl(iret) ASSERT(iret >= 0, prettyprint_errno("")); |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,15 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_BAZA_THROW_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_BAZA_THROW_H |  | ||||||
| 
 |  | ||||||
| #include "baza.h" |  | ||||||
| 
 |  | ||||||
| #define een9_THROW(err) throw een9::ServerError(err, __FILE__, __func__, __LINE__) |  | ||||||
| #define een9_THROW_on_errno(err) een9_THROW(een9::prettyprint_errno(err)) |  | ||||||
| #define een9_THROW_on_errno_pl() een9_THROW(een9::prettyprint_errno("")) |  | ||||||
| #define een9_ASSERT(cond, err) do { if (!(cond)) { een9_THROW(err); } } while (0); |  | ||||||
| #define een9_ASSERT_pl(cond) een9_ASSERT(cond, "Failed assertion `" #cond "`") |  | ||||||
| #define een9_ASSERT_on_iret(iret, err) een9_ASSERT((iret) >= 0, een9::prettyprint_errno(err)); |  | ||||||
| #define een9_ASSERT_on_iret_pl(iret) een9_ASSERT(iret >= 0, een9::prettyprint_errno("")); |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,86 +0,0 @@ | |||||||
| #include "static_asset_manager.h" |  | ||||||
| #include "../os_utils.h" |  | ||||||
| #include "../baza_inter.h" |  | ||||||
| #include <memory> |  | ||||||
| #include <utility> |  | ||||||
| #include <unistd.h> |  | ||||||
| #include <sys/stat.h> |  | ||||||
| #include <dirent.h> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     std::vector<std::string> detour_over_regular_folder(const std::string& path) { |  | ||||||
|         std::vector<std::string> result; |  | ||||||
|         int ret; |  | ||||||
| 
 |  | ||||||
|         std::vector<std::string> todo; |  | ||||||
|         todo.emplace_back(); |  | ||||||
|         while (!todo.empty()) { |  | ||||||
|             std::string cur = std::move(todo.back()); |  | ||||||
|             todo.pop_back(); |  | ||||||
|             std::string path_to_cur_ent = path + "/" + cur; |  | ||||||
|             struct stat info; |  | ||||||
|             ret = stat(path_to_cur_ent.c_str(), &info); |  | ||||||
|             ASSERT_on_iret(ret, "stat(\"" + cur + "\")"); |  | ||||||
|             if (S_ISDIR(info.st_mode)) { |  | ||||||
|                 DIR* D = opendir(path_to_cur_ent.c_str()); |  | ||||||
|                 cur += "/"; |  | ||||||
|                 ASSERT(D != NULL, prettyprint_errno("opendir(\"" + cur +"\")")); |  | ||||||
|                 struct Guard1{ DIR*& D; ~Guard1(){ closedir(D); } } g1{D}; |  | ||||||
|                 while (true) { |  | ||||||
|                     errno = 0; |  | ||||||
|                     struct dirent* Dent = readdir(D); |  | ||||||
|                     if (Dent == NULL) { |  | ||||||
|                         if (errno == 0) |  | ||||||
|                             break; |  | ||||||
|                         THROW_on_errno("dirent in \"" + cur + "\""); |  | ||||||
|                     } |  | ||||||
|                     std::string child_entry = Dent->d_name; |  | ||||||
|                     if (child_entry != "." && child_entry != "..") |  | ||||||
|                         todo.push_back(cur + child_entry); |  | ||||||
|                 } |  | ||||||
|             } else if (S_ISREG(info.st_mode)) { |  | ||||||
|                 result.push_back(cur); |  | ||||||
|             } else { |  | ||||||
|                 THROW("unknown fs entry type \"" + cur + "\""); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void updateStaticAssetManager(StaticAssetManager& sam) { |  | ||||||
|         sam.url_to_asset.clear(); |  | ||||||
|         for (const StaticAssetManagerRule& dir_rule: sam.rules) { |  | ||||||
|             std::vector<std::string> c_files = detour_over_regular_folder(dir_rule.directory); |  | ||||||
|             for (const std::string& file: c_files) { |  | ||||||
|                 for (const StaticAssetManagerRulePostfixFilter& ext: dir_rule.postfix_rules_type_assign) { |  | ||||||
|                     if (endsIn(file, ext.required_postfix)) { |  | ||||||
|                         /* Found it! */ |  | ||||||
|                         StaticAsset etot{ext.assigned_type, }; |  | ||||||
|                         readFile(dir_rule.directory + "/" + file, etot.content); |  | ||||||
|                         sam.url_to_asset[dir_rule.url_prefix + file] = etot; |  | ||||||
|                         break; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     int StaticAssetManagerSlaveModule::get_asset(const std::string& url, StaticAsset& ret) { |  | ||||||
|         RwlockReadGuard lg(mut); |  | ||||||
|         if (sam.url_to_asset.count(url) == 0) |  | ||||||
|             return -1; |  | ||||||
|         ret = sam.url_to_asset[url]; |  | ||||||
|         return 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void StaticAssetManagerSlaveModule::update() { |  | ||||||
|         RwlockWriteGuard lg(mut); |  | ||||||
|         updateStaticAssetManager(sam); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void StaticAssetManagerSlaveModule::update(std::vector<StaticAssetManagerRule> new_rules) { |  | ||||||
|         RwlockWriteGuard lg(mut); |  | ||||||
|         sam.rules = std::move(new_rules); |  | ||||||
|         updateStaticAssetManager(sam); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,50 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_CONNECTING_ASSETS_STATIC_ASSET_MANAGER_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_CONNECTING_ASSETS_STATIC_ASSET_MANAGER_H |  | ||||||
| 
 |  | ||||||
| #include <vector> |  | ||||||
| #include <string> |  | ||||||
| #include <map> |  | ||||||
| #include "../thread_synchronization.h" |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     struct StaticAssetManagerRulePostfixFilter { |  | ||||||
|         std::string required_postfix; |  | ||||||
|         std::string assigned_type; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct StaticAssetManagerRule { |  | ||||||
|         std::string directory; |  | ||||||
|         std::string url_prefix;  // Better end with /
 |  | ||||||
|         /* These are rules that filter name ending. First is a required name postfix, Second is a type of document
 |  | ||||||
|          * that gets assigned to matching files |  | ||||||
|          */ |  | ||||||
|         std::vector<StaticAssetManagerRulePostfixFilter> postfix_rules_type_assign; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct StaticAsset { |  | ||||||
|         std::string type; |  | ||||||
|         std::string content; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct StaticAssetManager { |  | ||||||
|         std::vector<StaticAssetManagerRule> rules; |  | ||||||
| 
 |  | ||||||
|         std::map<std::string, StaticAsset> url_to_asset; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     void updateStaticAssetManager(StaticAssetManager& sam); |  | ||||||
| 
 |  | ||||||
|     struct StaticAssetManagerSlaveModule { |  | ||||||
|         RwlockObj mut; |  | ||||||
|         StaticAssetManager sam; |  | ||||||
| 
 |  | ||||||
|         /* Returns newgative on failure. Still can throw execptions derived from std::execption */ |  | ||||||
|         int get_asset(const std::string& url, StaticAsset& ret); |  | ||||||
| 
 |  | ||||||
|         void update(); |  | ||||||
| 
 |  | ||||||
|         void update(std::vector<StaticAssetManagerRule> new_rules); |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,19 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_HTTP_STRUCTURES_CLIENT_REQUEST_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_HTTP_STRUCTURES_CLIENT_REQUEST_H |  | ||||||
| 
 |  | ||||||
| #include <vector> |  | ||||||
| #include <string> |  | ||||||
| #include <utility> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     struct ClientRequest { |  | ||||||
|         std::string method; |  | ||||||
|         std::string url; |  | ||||||
|         std::string http_version; |  | ||||||
|         std::vector<std::pair<std::string, std::string>> headers; |  | ||||||
|         bool has_body = false; |  | ||||||
|         std::string body; |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,101 +0,0 @@ | |||||||
| #include "client_request_parse.h" |  | ||||||
| #include "../baza_inter.h" |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     constexpr size_t max_allowed_request_content_length = 1000000000; |  | ||||||
| 
 |  | ||||||
|     static const char* supported_http_versions[] = {"HTTP/0.9", "HTTP/1.0", "HTTP/1.1", NULL}; |  | ||||||
|     static const char* supported_http_methods[] = {"GET", "POST", NULL}; |  | ||||||
| 
 |  | ||||||
|     void ClientRequestParser::feedByte(char b) { |  | ||||||
|         if (finished) |  | ||||||
|             THROW("Excess tailing bytes"); |  | ||||||
|         if (collecting_body) { |  | ||||||
|             res.body += b; |  | ||||||
|             if (res.body.size() >= content_lenth) { |  | ||||||
|                 finished = true; |  | ||||||
|             } |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (line_end[line_end_progress] == b) { |  | ||||||
|             line_end_progress++; |  | ||||||
|             if (line_end_progress == line_end.size()) { |  | ||||||
|                 line_end_progress = 0; |  | ||||||
|                 /* Evaluating meaning of complete request line */ |  | ||||||
|                 if (i == 0) { |  | ||||||
|                     parseFirstLine(); |  | ||||||
|                 } else if (cur_line.empty()) { |  | ||||||
|                     processEndOfHeader(); |  | ||||||
|                 } else { |  | ||||||
|                     parseHeaderLine(); |  | ||||||
|                 } |  | ||||||
|                 cur_line = ""; |  | ||||||
|                 i++; |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             ASSERT_pl(line_end_progress == 0); |  | ||||||
|             cur_line += b; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool ClientRequestParser::completed() const { |  | ||||||
|         return finished; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void ClientRequestParser::parseFirstLine() { |  | ||||||
|         std::vector<std::string> huyushki = {""}; |  | ||||||
|         for (char ch: cur_line) { |  | ||||||
|             if (ch == ' ') { |  | ||||||
|                 huyushki.emplace_back(); |  | ||||||
|             } else { |  | ||||||
|                 huyushki.back() += ch; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         ASSERT_pl(huyushki.size() == 3); |  | ||||||
|         res.method = huyushki[0]; |  | ||||||
|         // todo: decypher and check url
 |  | ||||||
|         res.url = huyushki[1]; |  | ||||||
|         res.http_version = huyushki[2]; |  | ||||||
|         ASSERT_pl(strIn(res.method, supported_http_methods)); |  | ||||||
|         ASSERT_pl(strIn(res.http_version, supported_http_versions)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void ClientRequestParser::parseHeaderLine() { |  | ||||||
|         std::pair<std::string, std::string> np; |  | ||||||
|         static const std::string sep = ": "; |  | ||||||
|         size_t sep_progress = 0; |  | ||||||
|         bool reading_value = false; |  | ||||||
|         for (char ch: cur_line) { |  | ||||||
|             if (reading_value) { |  | ||||||
|                 np.second += ch; |  | ||||||
|             } else if (sep[sep_progress] == ch) { |  | ||||||
|                 sep_progress++; |  | ||||||
|                 if (sep_progress == sep.size()) { |  | ||||||
|                     reading_value = true; |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 ASSERT_pl(sep_progress == 0); |  | ||||||
|                 np.first += ch; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         res.headers.push_back(np); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void ClientRequestParser::processEndOfHeader() { |  | ||||||
|         for (auto& p: res.headers) { |  | ||||||
|             if (p.first == "Content-Length") { |  | ||||||
|                 res.has_body = true; |  | ||||||
|                 long cl = std::stol(p.second); |  | ||||||
|                 ASSERT_pl(cl > 0 && cl <= max_allowed_request_content_length); |  | ||||||
|                 content_lenth = cl; |  | ||||||
|                 collecting_body = true; |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         if (!collecting_body) { |  | ||||||
|             finished = true; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ClientRequestParser::ClientRequestParser(ClientRequest &res) : res(res) {} |  | ||||||
| } |  | ||||||
| @ -1,41 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_HTTP_CLIENT_REQUEST_PARSE_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_HTTP_CLIENT_REQUEST_PARSE_H |  | ||||||
| 
 |  | ||||||
| /* Do not export this file */ |  | ||||||
| 
 |  | ||||||
| #include "../baza.h" |  | ||||||
| #include "client_request.h" |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| // todo: parse unicode % blocks in url
 |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     struct ClientRequestParser { |  | ||||||
|         ClientRequest& res; |  | ||||||
| 
 |  | ||||||
|         explicit ClientRequestParser(ClientRequest& res); |  | ||||||
| 
 |  | ||||||
|         void feedByte(char b); |  | ||||||
| 
 |  | ||||||
|         bool completed() const; |  | ||||||
| 
 |  | ||||||
|         bool finished = false; |  | ||||||
|         /* internal parse data */ |  | ||||||
|         ssize_t i = 0; |  | ||||||
|         std::string cur_line; |  | ||||||
|         bool collecting_body = false; |  | ||||||
|         size_t content_lenth = 0; |  | ||||||
| 
 |  | ||||||
|         const std::string line_end = "\r\n"; |  | ||||||
|         size_t line_end_progress = 0; |  | ||||||
| 
 |  | ||||||
|         /* argument stored in cur_line */ |  | ||||||
|         void parseFirstLine(); |  | ||||||
| 
 |  | ||||||
|         void parseHeaderLine(); |  | ||||||
| 
 |  | ||||||
|         void processEndOfHeader(); |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,41 +0,0 @@ | |||||||
| #include "response_gen.h" |  | ||||||
| #include "../baza_inter.h" |  | ||||||
| #include <assert.h> |  | ||||||
| #include <string.h> |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     std::string form_http_server_response_header(const char* code, const std::map<std::string, std::string>& headers) { |  | ||||||
|         assert(strlen(code) == 3); |  | ||||||
|         std::string result = std::string("HTTP/1.0 ") + code + " " + (code[0] < '4' ? "OK" : "ERROR") + "\r\n"; |  | ||||||
|         for (auto& p: headers) |  | ||||||
|             result += (p.first + ": " + p.second + "\r\n"); |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_reponse_header_only(const char* code, const std::map<std::string, std::string>& headers) { |  | ||||||
|         return form_http_server_response_header(code, headers) + "\r\n"; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_response_with_body(const char* code, |  | ||||||
|         const std::map<std::string, std::string>& headers, |  | ||||||
|         const std::string& body) |  | ||||||
|     { |  | ||||||
|         std::string result = form_http_server_response_header(code, headers) |  | ||||||
|         + "Content-Length: " + std::to_string(body.size()) + "\r\n\r\n" + body; |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /* Message from server to client */ |  | ||||||
|     std::string form_http_server_response_200(const std::string& Content_Type, const std::string& body) { |  | ||||||
|         return form_http_server_response_with_body("200", { |  | ||||||
|             {"Content-Type", Content_Type} |  | ||||||
|         }, body); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_response_404(const std::string& Content_Type, const std::string& body) { |  | ||||||
|         return form_http_server_response_with_body("404", { |  | ||||||
|             {"Content-Type", Content_Type} |  | ||||||
|         }, body); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,21 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_HTTP_STRUCTURES_RESPONSE_GEN_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_HTTP_STRUCTURES_RESPONSE_GEN_H |  | ||||||
| 
 |  | ||||||
| #include <map> |  | ||||||
| #include <string> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     std::string form_http_server_response_header(const char* code, const std::map<std::string, std::string>& headers); |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_reponse_header_only(const char* code, const std::map<std::string, std::string>& headers); |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_response_with_body(const char* code, |  | ||||||
|         const std::map<std::string, std::string>& headers, |  | ||||||
|         const std::string& body); |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_response_200(const std::string& Content_Type, const std::string& body); |  | ||||||
| 
 |  | ||||||
|     std::string form_http_server_response_404(const std::string& Content_Type, const std::string& body); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,68 +0,0 @@ | |||||||
| #include "os_utils.h" |  | ||||||
| #include <utility> |  | ||||||
| #include "baza_inter.h" |  | ||||||
| #include <unistd.h> |  | ||||||
| #include <string.h> |  | ||||||
| #include <fcntl.h> |  | ||||||
| #include <sys/stat.h> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     UniqueFdWrapper::UniqueFdWrapper(int fd_): fd(fd_) {} |  | ||||||
| 
 |  | ||||||
|     UniqueFdWrapper::UniqueFdWrapper(UniqueFdWrapper &&formerOwner) noexcept { |  | ||||||
|         fd = formerOwner.fd; |  | ||||||
|         formerOwner.fd = -1; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     UniqueFdWrapper& UniqueFdWrapper::operator=(UniqueFdWrapper &&formerOwner) noexcept { |  | ||||||
|         std::swap(fd, formerOwner.fd); |  | ||||||
|         return *this; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     int UniqueFdWrapper::operator()() const { |  | ||||||
|         return fd; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     UniqueFdWrapper::~UniqueFdWrapper() { |  | ||||||
|         // printf("DEBUG!!! Closing fd = %d\n", fd);
 |  | ||||||
|         if (fd >= 0) |  | ||||||
|             close(fd); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool isNeededFsEntity(const std::string &path, mode_t t_mode) { |  | ||||||
|         struct stat info; |  | ||||||
|         errno = 0; |  | ||||||
|         int ret = stat(path.c_str(), &info); |  | ||||||
|         if (errno == 0) { |  | ||||||
|             return (info.st_mode & S_IFMT) == t_mode; |  | ||||||
|         } if (errno == ENOENT) |  | ||||||
|             return false; |  | ||||||
|         THROW_on_errno("stat\"" + path + "\""); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool isRegularFile(const std::string &path) { |  | ||||||
|         return isNeededFsEntity(path, S_IFREG); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool isDirectory(const std::string &path) { |  | ||||||
|         return isNeededFsEntity(path, S_IFDIR); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void readFromFileDescriptor(int fd, std::string &result, const std::string &description) { |  | ||||||
|         int ret; |  | ||||||
|         char buf[2048]; |  | ||||||
|         while ((ret = (int)read(fd, buf, 2048)) > 0) { |  | ||||||
|             size_t oldN = result.size(); |  | ||||||
|             result.resize(oldN + ret); |  | ||||||
|             memcpy(&result[oldN], buf, ret); |  | ||||||
|         } |  | ||||||
|         ASSERT_on_iret(ret, "Reading from " + description); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void readFile(const std::string &path, std::string &result) { |  | ||||||
|         int fd = open(path.c_str(), O_RDONLY); |  | ||||||
|         ASSERT_on_iret(fd, "Opening \"" + path + "\""); |  | ||||||
|         UniqueFdWrapper fdw(fd); |  | ||||||
|         readFromFileDescriptor(fdw(), result, "file \"" + path + "\""); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,34 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_OS_UTILS_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_OS_UTILS_H |  | ||||||
| 
 |  | ||||||
| #include "baza.h" |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     class UniqueFdWrapper { |  | ||||||
|         int fd = -1; |  | ||||||
|     public: |  | ||||||
|         explicit UniqueFdWrapper() = default; |  | ||||||
|         explicit UniqueFdWrapper(int fd_); |  | ||||||
| 
 |  | ||||||
|         UniqueFdWrapper(const UniqueFdWrapper&) = delete; |  | ||||||
|         UniqueFdWrapper& operator=(const UniqueFdWrapper&) = delete; |  | ||||||
| 
 |  | ||||||
|         UniqueFdWrapper(UniqueFdWrapper&& formerOwner) noexcept; |  | ||||||
|         UniqueFdWrapper& operator=(UniqueFdWrapper&& formerOwner) noexcept; |  | ||||||
| 
 |  | ||||||
|         int operator()() const; |  | ||||||
| 
 |  | ||||||
|         ~UniqueFdWrapper(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     bool isRegularFile(const std::string& path); |  | ||||||
| 
 |  | ||||||
|     bool isDirectory(const std::string& path); |  | ||||||
| 
 |  | ||||||
|     /* result += read(fd); Argument description is for error handling */ |  | ||||||
|     void readFromFileDescriptor(int fd, std::string& result, const std::string& description = ""); |  | ||||||
| 
 |  | ||||||
|     void readFile(const std::string& path, std::string& result); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,257 +0,0 @@ | |||||||
| #include "running_mainloop.h" |  | ||||||
| 
 |  | ||||||
| #include <string.h> |  | ||||||
| #include <sys/socket.h> |  | ||||||
| #include <arpa/inet.h> |  | ||||||
| #include <poll.h> |  | ||||||
| #include <assert.h> |  | ||||||
| #include <map> |  | ||||||
| #include <queue> |  | ||||||
| #include <utility> |  | ||||||
| #include "thread_synchronization.h" |  | ||||||
| #include "os_utils.h" |  | ||||||
| #include "http_structures/client_request_parse.h" |  | ||||||
| #include "http_structures/response_gen.h" |  | ||||||
| #include "baza_inter.h" |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     // todo: add timeout for multiple bytes, add more settings
 |  | ||||||
|     ClientRequest process_connection_input(int fd, const EEN9_ServerTips& s_tips) { |  | ||||||
|         ClientRequest res; |  | ||||||
|         ClientRequestParser parser(res); |  | ||||||
|         int ret; |  | ||||||
|         char buf[2048]; |  | ||||||
|         ASSERT_pl(!parser.completed()) |  | ||||||
|         while ((ret = (int)recv(fd, buf, 2048, 0)) > 0) { |  | ||||||
|             for (size_t i = 0; i < ret; i++) |  | ||||||
|                 parser.feedByte(buf[i]); |  | ||||||
|             if (parser.completed()) |  | ||||||
|                 break; |  | ||||||
|         } |  | ||||||
|         ASSERT_on_iret(ret, "recv"); |  | ||||||
|         ASSERT_pl(parser.completed()); |  | ||||||
|         // printf("Log: worker received clients request\n%s\n", client_request.toString().c_str());
 |  | ||||||
|         return res; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void process_connection_output(int fd, const std::string& server_response) { |  | ||||||
|         size_t N = server_response.size(), i = 0; |  | ||||||
|         while (i < N) { |  | ||||||
|             int written = (int)send(fd, &server_response[i], std::min(2048lu, N - i), 0); |  | ||||||
|             ASSERT_on_iret(written, "sending"); |  | ||||||
|             ASSERT_pl(written > 0); |  | ||||||
|             i += written; |  | ||||||
|         } |  | ||||||
|         printf("Log: worker: succesfully asnwered with response\n"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void process_connection(const SlaveTask& task, |  | ||||||
|                             const guest_core_t& guest_core) |  | ||||||
|     { |  | ||||||
|         ClientRequest client_request = process_connection_input(task.fd(), task.s_tips); |  | ||||||
|         std::string server_response = guest_core(task, client_request); |  | ||||||
|         process_connection_output(task.fd(), server_response); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     struct QElementHttpConnections { |  | ||||||
|         SlaveTask task; |  | ||||||
|         QElementHttpConnections* nxt = NULL; |  | ||||||
| 
 |  | ||||||
|         explicit QElementHttpConnections(SlaveTask task): task(std::move(task)) {} |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct WorkersTaskQueue { |  | ||||||
|         QElementHttpConnections* first = NULL; |  | ||||||
|         QElementHttpConnections** afterLastPtr; |  | ||||||
|         size_t sz = 0; |  | ||||||
| 
 |  | ||||||
|         WorkersTaskQueue() { |  | ||||||
|             afterLastPtr = &first; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         bool empty() const { |  | ||||||
|             return sz == 0; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         size_t size() const { |  | ||||||
|             return sz; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         void push_back(SlaveTask task) { |  | ||||||
|             /* Throws a goddamn execption. Because why not. Ofcourse everything has to throw an exception */ |  | ||||||
|             /* CLion says. Allocated memory is leaking. YOUR MOTHER IS LEAKING YOU FOOL!! MY CODE IS FINE!! */ |  | ||||||
|             QElementHttpConnections* el = new QElementHttpConnections(std::move(task)); |  | ||||||
|             /* Exception does not leave queue in incorrect state */ |  | ||||||
|             *afterLastPtr = el; |  | ||||||
|             afterLastPtr = &(el->nxt); |  | ||||||
|             sz++; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         void pop_first(SlaveTask& ret_task) { |  | ||||||
|             assert(!empty()); |  | ||||||
|             ret_task = std::move(first->task); |  | ||||||
|             if (sz == 1) { |  | ||||||
|                 delete first; |  | ||||||
|                 first = NULL; |  | ||||||
|                 afterLastPtr = &first; |  | ||||||
|                 sz = 0; |  | ||||||
|             } else { |  | ||||||
|                 /* Before I popped the first, this element was second, but now it took place of the first */ |  | ||||||
|                 QElementHttpConnections* old_deut = first->nxt; |  | ||||||
|                 delete first; |  | ||||||
|                 first = old_deut; |  | ||||||
|                 sz--; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct WorkersTaskEnv { |  | ||||||
|         /* This alarm notifies about new tasks and termination signal. Because we are polite people, we don't cancel threads */ |  | ||||||
|         CondVarBedObj corvee_bed; |  | ||||||
|         WorkersTaskQueue queue; |  | ||||||
|         bool& termination; |  | ||||||
|         guest_core_t guest_core; |  | ||||||
| 
 |  | ||||||
|         WorkersTaskEnv(bool& term, guest_core_t g_c): termination(term), guest_core(std::move(g_c)){} |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     void* worker_func(void* wte_ptr) { |  | ||||||
|         WorkersTaskEnv& wte = *((WorkersTaskEnv*)wte_ptr); |  | ||||||
|         printf("Worker started\n"); |  | ||||||
|         while (true) { |  | ||||||
|             try { |  | ||||||
|                 MutexLockGuard cb_lg(wte.corvee_bed, __func__); |  | ||||||
|                 woke: |  | ||||||
|                 if (wte.termination) |  | ||||||
|                     break; |  | ||||||
|                 if (wte.queue.empty()) { |  | ||||||
|                     wte.corvee_bed.sleep(__func__); |  | ||||||
|                     goto woke; |  | ||||||
|                 } |  | ||||||
|                 SlaveTask task; |  | ||||||
|                 wte.queue.pop_first(task); |  | ||||||
|                 process_connection(task, wte.guest_core); |  | ||||||
|             } catch (const std::exception& e) { |  | ||||||
|                 printf("Client request procession failure in worker\n"); |  | ||||||
|                 printf("%s\n", e.what()); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         printf("Worker finished\n"); |  | ||||||
|         return NULL; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void configure_socket_rcvsndtimeo(int fd, timeval tv) { |  | ||||||
|         int ret; |  | ||||||
|         ret = setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(timeval)); |  | ||||||
|         ASSERT_on_iret_pl(ret); |  | ||||||
|         ret = setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(timeval)); |  | ||||||
|         ASSERT_on_iret_pl(ret); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // todo: retrieve address of connected client
 |  | ||||||
| 
 |  | ||||||
|     void electric_boogaloo(const MainloopParameters& params, bool& termination_trigger) { |  | ||||||
|         WorkersTaskEnv wte(termination_trigger, params.guest_core); |  | ||||||
|         ASSERT(params.slave_number > 0, "No workers spawned"); |  | ||||||
|         size_t Nip = params.ports_to_listen.size(); |  | ||||||
|         ASSERT(Nip > 0, "No open listeting addresses"); |  | ||||||
| 
 |  | ||||||
|         std::vector<pthread_t> workers(params.slave_number); |  | ||||||
|         for (size_t i = 0; i < params.slave_number; i++) { |  | ||||||
|             pthread_create(&workers[i], NULL, worker_func, &wte); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         try { |  | ||||||
|             int ret; |  | ||||||
|             std::vector<UniqueFdWrapper> listening_socks(Nip); |  | ||||||
|             for (size_t i = 0; i < Nip; i++) { |  | ||||||
|                 printf("Creating listening socket\n"); |  | ||||||
|                 uint16_t port = params.ports_to_listen[i]; |  | ||||||
|                 int listening_socket_fd = socket(AF_INET, SOCK_STREAM, 0); |  | ||||||
|                 ASSERT_on_iret(listening_socket_fd, "Listening socket creation"); |  | ||||||
|                 UniqueFdWrapper listening_socket(listening_socket_fd); |  | ||||||
|                 printf("Listening socket created\n"); |  | ||||||
|                 sockaddr_in listening_address; |  | ||||||
|                 listening_address.sin_family = AF_INET; |  | ||||||
|                 listening_address.sin_port = htons(port); |  | ||||||
|                 uint32_t lca = (127u << 24) | 1; |  | ||||||
|                 listening_address.sin_addr.s_addr = htonl(lca); |  | ||||||
|                 int reuseaddr_nozero_option_value = 1; |  | ||||||
|                 ret = setsockopt(listening_socket_fd, SOL_SOCKET, SO_REUSEADDR, &reuseaddr_nozero_option_value, sizeof(int)); |  | ||||||
|                 ASSERT_on_iret(ret, "setting SO_REUSEADDR befire binding to address"); |  | ||||||
|                 ret = bind(listening_socket(), (const sockaddr*)&listening_address, sizeof(listening_address)); |  | ||||||
|                 ASSERT_on_iret(ret, "binding to INADDR_ANY:" + std::to_string(port)); |  | ||||||
|                 printf("Binded socket to address\n"); |  | ||||||
|                 ret = listen(listening_socket(), 128); |  | ||||||
|                 ASSERT_on_iret(ret, "listening for connections"); |  | ||||||
|                 printf("Listening socket succesfully started listening\n"); |  | ||||||
|                 listening_socks[i] = std::move(listening_socket); |  | ||||||
|             } |  | ||||||
|             std::vector<pollfd> pollfds(Nip); |  | ||||||
|             for (size_t i = 0; i < Nip; i++) { |  | ||||||
|                 pollfds[i].fd = listening_socks[i](); |  | ||||||
|                 pollfds[i].events = POLLRDNORM; |  | ||||||
|             } |  | ||||||
|             printf("Entering mainloop\n"); |  | ||||||
|             ASSERT(params.mainloop_recheck_interval_us > 0, "Incorrect poll timeout"); |  | ||||||
|             while (true) { |  | ||||||
|                 MutexLockGuard lg1(wte.corvee_bed, "poller termination check"); |  | ||||||
|                 if (wte.termination) |  | ||||||
|                     break; |  | ||||||
|                 lg1.unlock(); |  | ||||||
|                 for (size_t i = 0; i < Nip; i++) { |  | ||||||
|                     pollfds[i].revents = 0; |  | ||||||
|                 } |  | ||||||
|                 errno = 0; |  | ||||||
|                 ret = poll(pollfds.data(), Nip, params.mainloop_recheck_interval_us); |  | ||||||
|                 if (errno == EINTR) |  | ||||||
|                     break; |  | ||||||
|                 ASSERT_on_iret(ret, "polling"); |  | ||||||
|                 for (size_t i = 0; i < Nip; i++) { |  | ||||||
|                     if ((pollfds[i].revents & POLLRDNORM)) { |  | ||||||
|                         try { |  | ||||||
|                             sockaddr client_address; |  | ||||||
|                             socklen_t client_addr_len = sizeof(client_address); |  | ||||||
|                             int session_sock = accept(pollfds[i].fd, &client_address, &client_addr_len); |  | ||||||
|                             ASSERT_on_iret(session_sock, "Failed to accept incoming connection"); |  | ||||||
|                             printf("Log: successful connection\n"); |  | ||||||
|                             UniqueFdWrapper session_sock_fdw(session_sock); |  | ||||||
|                             configure_socket_rcvsndtimeo(session_sock_fdw(), params.s_conf.request_timeout); |  | ||||||
|                             { MutexLockGuard lg2(wte.corvee_bed, "poller adds connection"); |  | ||||||
|                                 SlaveTask task{ConnectionInfo{}, std::move(session_sock_fdw), |  | ||||||
|                                     EEN9_ServerTips{wte.queue.size(), |  | ||||||
|                                         params.s_conf.critical_load_1, params.s_conf.request_timeout}}; |  | ||||||
|                                 if (wte.queue.size() < params.s_conf.critical_load_2) |  | ||||||
|                                     wte.queue.push_back(std::move(task)); |  | ||||||
|                             } |  | ||||||
|                             wte.corvee_bed.din_don(); |  | ||||||
|                         } catch (const std::exception& e) { |  | ||||||
|                             printf("Error aceepting connection\n"); |  | ||||||
|                             printf("%s\n", e.what()); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } catch (const std::exception& e) { |  | ||||||
|             printf("System failure 2\n"); |  | ||||||
|             printf("%s\n", e.what()); |  | ||||||
|             /* There is no need to tiptoe around this multi-access field. It is write-onle-and-for-good-kind  */ |  | ||||||
|             wte.termination = true; |  | ||||||
|             wte.corvee_bed.wake_them_all(); |  | ||||||
|         } |  | ||||||
|         wte.termination = true; |  | ||||||
|         wte.corvee_bed.wake_them_all(); |  | ||||||
|         for (size_t i = 0; i < params.slave_number; i++) { |  | ||||||
|             pthread_join(workers[i], NULL); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void safe_electric_boogaloo(const MainloopParameters& params, bool& termination_trigger) { |  | ||||||
|         try { |  | ||||||
|             electric_boogaloo(params, termination_trigger); |  | ||||||
|         } catch (const std::exception& e) { |  | ||||||
|             printf("System failure\n"); |  | ||||||
|             printf("%s\n", e.what()); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,59 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_MAINLOOP_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_MAINLOOP_H |  | ||||||
| 
 |  | ||||||
| #include "baza.h" |  | ||||||
| #include <functional> |  | ||||||
| #include <sys/time.h> |  | ||||||
| #include "os_utils.h" |  | ||||||
| #include "http_structures/client_request.h" |  | ||||||
| #include <stdint.h> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     struct ConnectionInfo { |  | ||||||
|         // todo: add server address field
 |  | ||||||
|         // todo: add client address field
 |  | ||||||
|         int type; // O_o why??
 |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     /* This structure is passed to guest function. It contains server info that might be or might be not used
 |  | ||||||
|      * by guest */ |  | ||||||
|     struct EEN9_ServerTips { |  | ||||||
|         size_t server_load; |  | ||||||
|         size_t critical_load_1; |  | ||||||
|         timeval recommended_timeout; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct SlaveTask { |  | ||||||
|         ConnectionInfo conn_info; |  | ||||||
|         UniqueFdWrapper fd; |  | ||||||
|         EEN9_ServerTips s_tips; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     /* guest_core function must not throw anything that is not derived from std::exception */ |  | ||||||
|     typedef std::function<std::string(const SlaveTask&, const ClientRequest&)> guest_core_t; |  | ||||||
| 
 |  | ||||||
|     struct ServersConfiguration { |  | ||||||
|         size_t critical_load_1 = 90; |  | ||||||
|         size_t critical_load_2 = 100; |  | ||||||
| 
 |  | ||||||
|         timeval request_timeout{20, 0}; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     struct MainloopParameters { |  | ||||||
|         bool do_logging = true; |  | ||||||
|         bool open_admin_listener = true; |  | ||||||
|         // todo: add support for unix socket address
 |  | ||||||
|         uint16_t admin_listener_port = 12345; |  | ||||||
|         guest_core_t guest_core; |  | ||||||
|         size_t slave_number = 2; |  | ||||||
|         std::vector<uint16_t> ports_to_listen; |  | ||||||
|         int mainloop_recheck_interval_us = 100; |  | ||||||
| 
 |  | ||||||
|         ServersConfiguration s_conf; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     void electric_boogaloo(const MainloopParameters& params, bool& termination_trigger); |  | ||||||
|     void safe_electric_boogaloo(const MainloopParameters& params, bool& termination_trigger); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,161 +0,0 @@ | |||||||
| #include "thread_synchronization.h" |  | ||||||
| #include "baza_inter.h" |  | ||||||
| #include <assert.h> |  | ||||||
| namespace een9 { |  | ||||||
|     /*==================================================== RwlockObj ============ */ |  | ||||||
| 
 |  | ||||||
|     void RwlockObj::unlock(int el, const char *who) { |  | ||||||
|         assert(lock == el); |  | ||||||
|         lock = 0; |  | ||||||
|         if (me) |  | ||||||
|             printf("Rwl %s unlocked by %s\n", me, who); |  | ||||||
|         int ret = pthread_rwlock_unlock(&mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     RwlockObj::RwlockObj(const char *me): me(me) { |  | ||||||
|         int ret = pthread_rwlock_init(&mut, NULL); |  | ||||||
|         ASSERT_on_iret(ret, "pthread rwlock init"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockObj::read_lock(const char *who) { |  | ||||||
|         int ret = pthread_rwlock_rdlock(&mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|         if (me) |  | ||||||
|             printf("Rwl %s read-locked by %s\n", me, who); |  | ||||||
|         lock = 1; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockObj::write_lock(const char *who) { |  | ||||||
|         int ret = pthread_rwlock_wrlock(&mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|         if (me) |  | ||||||
|             printf("Rwl %s write-locked by %s\n", me, who); |  | ||||||
|         lock = 2; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockObj::read_unclock(const char *who) { |  | ||||||
|         unlock(1, who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockObj::write_unclock(const char *who) { |  | ||||||
|         unlock(2, who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     RwlockObj::~RwlockObj() { |  | ||||||
|         pthread_rwlock_destroy(&mut); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*==================================================== MutexObj ============ */ |  | ||||||
| 
 |  | ||||||
|     MutextObj::MutextObj(const char *me): me(me) { |  | ||||||
|         int ret = pthread_mutex_init(&mut, NULL); |  | ||||||
|         ASSERT_on_iret(ret == 0, "pthread mutex init"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void MutextObj::lock(const char *who) { |  | ||||||
|         int ret = pthread_mutex_lock(&mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|         if (me) |  | ||||||
|             printf("Mut %s locked by %s\n", me, who); |  | ||||||
|         locked = 1; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void MutextObj::unlock(const char *who) { |  | ||||||
|         assert(locked == 1); |  | ||||||
|         locked = 0; |  | ||||||
|         if (me) |  | ||||||
|             printf("Mut %s unlocked by %s\n", me, who); |  | ||||||
|         int ret = pthread_mutex_unlock(&mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     MutextObj::~MutextObj() { |  | ||||||
|         pthread_mutex_destroy(&mut); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*==================================================== CondVarBedObj ============ */ |  | ||||||
| 
 |  | ||||||
|     CondVarBedObj::CondVarBedObj(const char *me): MutextObj(me) { |  | ||||||
|         int ret = pthread_cond_init(&alarm, NULL); |  | ||||||
|         ASSERT_on_iret(ret, "pthread cond variable init"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void CondVarBedObj::sleep(const char *who) { |  | ||||||
|         assert(locked == 1); |  | ||||||
|         locked = 0; |  | ||||||
|         if (me) |  | ||||||
|             printf("Mut %s unlocked. Sleeping (%s)\n", me, who); |  | ||||||
|         int ret = pthread_cond_wait(&alarm, &mut); |  | ||||||
|         assert(ret == 0); |  | ||||||
|         if (me) |  | ||||||
|             printf("Mut %s locked. Woke up (%s)\n", me, who); |  | ||||||
|         locked = 1; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void CondVarBedObj::din_don() { |  | ||||||
|         int ret = pthread_cond_signal(&alarm); |  | ||||||
|         assert(ret == 0); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void CondVarBedObj::wake_them_all() { |  | ||||||
|         int ret = pthread_cond_broadcast(&alarm); |  | ||||||
|         assert(ret == 0); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     CondVarBedObj::~CondVarBedObj() { |  | ||||||
|         pthread_mutex_destroy(&mut); |  | ||||||
|         pthread_cond_destroy(&alarm); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*==================================================== MutexLockGuard ============ */ |  | ||||||
| 
 |  | ||||||
|     MutexLockGuard::MutexLockGuard(MutextObj &mut, const char *who): mut(mut), who(who) { |  | ||||||
|         mut.lock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void MutexLockGuard::unlock() { |  | ||||||
|         assert(!premature_unlock); |  | ||||||
|         premature_unlock = true; |  | ||||||
|         mut.unlock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     MutexLockGuard::~MutexLockGuard() { |  | ||||||
|         if (!premature_unlock) |  | ||||||
|             mut.unlock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*==================================================== RwLockReadLockGuard ============ */ |  | ||||||
| 
 |  | ||||||
|     RwlockReadGuard::RwlockReadGuard(RwlockObj &mut, const char *who): mut(mut), who(who) { |  | ||||||
|         mut.read_lock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockReadGuard::unlock() { |  | ||||||
|         assert(!premature_unlock); |  | ||||||
|         premature_unlock = true; |  | ||||||
|         mut.read_unclock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     RwlockReadGuard::~RwlockReadGuard() { |  | ||||||
|         if (!premature_unlock) |  | ||||||
|             mut.read_unclock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /*==================================================== RwLockWriteLockGuard ============ */ |  | ||||||
| 
 |  | ||||||
|     RwlockWriteGuard::RwlockWriteGuard(RwlockObj &mut, const char *who): mut(mut), who(who) { |  | ||||||
|         mut.write_lock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void RwlockWriteGuard::unlock() { |  | ||||||
|         assert(!premature_unlock); |  | ||||||
|         premature_unlock = true; |  | ||||||
|         mut.write_unclock(who); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     RwlockWriteGuard::~RwlockWriteGuard() { |  | ||||||
|         if (!premature_unlock) |  | ||||||
|             mut.write_unclock(who); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,109 +0,0 @@ | |||||||
| #ifndef ENGINE_ENGINE_NUMBER_9_THREAD_SYNCHRONIZATION_H |  | ||||||
| #define ENGINE_ENGINE_NUMBER_9_THREAD_SYNCHRONIZATION_H |  | ||||||
| 
 |  | ||||||
| #include "baza.h" |  | ||||||
| #include <pthread.h> |  | ||||||
| 
 |  | ||||||
| namespace een9 { |  | ||||||
|     class RwlockObj { |  | ||||||
|     protected: |  | ||||||
|         pthread_rwlock_t mut; |  | ||||||
|         int lock = 0; |  | ||||||
|         const char* me; |  | ||||||
| 
 |  | ||||||
|         void unlock(int el, const char* who); |  | ||||||
| 
 |  | ||||||
|     public: |  | ||||||
|         RwlockObj(const RwlockObj& ) = delete; |  | ||||||
|         RwlockObj& operator=(const RwlockObj& ) = delete; |  | ||||||
| 
 |  | ||||||
|         explicit RwlockObj(const char* me = NULL); |  | ||||||
| 
 |  | ||||||
|         void read_lock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void write_lock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void read_unclock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void write_unclock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         ~RwlockObj(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     class MutextObj { |  | ||||||
|     protected: |  | ||||||
|         pthread_mutex_t mut; |  | ||||||
|         int locked = 0; |  | ||||||
|         const char* me; |  | ||||||
| 
 |  | ||||||
|     public: |  | ||||||
|         MutextObj(const MutextObj&) = delete; |  | ||||||
|         MutextObj& operator= (const MutextObj&) = delete; |  | ||||||
| 
 |  | ||||||
|         explicit MutextObj(const char* me = NULL); |  | ||||||
| 
 |  | ||||||
|         void lock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void unlock(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         ~MutextObj(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     class CondVarBedObj : public MutextObj{ |  | ||||||
|         pthread_cond_t alarm; |  | ||||||
|         /* In this case, variable `locked` check won't ensure proper mutex usage, but it can help sometimes */ |  | ||||||
| 
 |  | ||||||
|     public: |  | ||||||
|         CondVarBedObj(const CondVarBedObj& ) = delete; |  | ||||||
|         CondVarBedObj& operator=(const CondVarBedObj& ) = delete; |  | ||||||
| 
 |  | ||||||
|         explicit CondVarBedObj(const char* me = NULL); |  | ||||||
| 
 |  | ||||||
|         void sleep(const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void din_don(); |  | ||||||
| 
 |  | ||||||
|         void wake_them_all(); |  | ||||||
| 
 |  | ||||||
|         ~CondVarBedObj(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     class MutexLockGuard { |  | ||||||
|         MutextObj& mut; |  | ||||||
|         const char* who; |  | ||||||
|         bool premature_unlock = false; |  | ||||||
|     public: |  | ||||||
|         explicit MutexLockGuard(MutextObj &mut, const char* who = ""); |  | ||||||
| 
 |  | ||||||
|         void unlock(); |  | ||||||
| 
 |  | ||||||
|         ~MutexLockGuard(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     class RwlockReadGuard { |  | ||||||
|         RwlockObj& mut; |  | ||||||
|         const char* who; |  | ||||||
|         bool premature_unlock = false; |  | ||||||
|     public: |  | ||||||
|         explicit RwlockReadGuard(RwlockObj &mut, const char *who = ""); |  | ||||||
| 
 |  | ||||||
|         void unlock(); |  | ||||||
| 
 |  | ||||||
|         ~RwlockReadGuard(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     class RwlockWriteGuard { |  | ||||||
|         RwlockObj& mut; |  | ||||||
|         const char* who; |  | ||||||
|         bool premature_unlock = false; |  | ||||||
|     public: |  | ||||||
|         explicit RwlockWriteGuard(RwlockObj &mut, const char *who = ""); |  | ||||||
| 
 |  | ||||||
|         void unlock(); |  | ||||||
| 
 |  | ||||||
|         ~RwlockWriteGuard(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| @ -1,56 +0,0 @@ | |||||||
| #include <engine_engine_number_9/baza_throw.h> |  | ||||||
| #include <engine_engine_number_9/running_mainloop.h> |  | ||||||
| #include <engine_engine_number_9/http_structures/response_gen.h> |  | ||||||
| #include <signal.h> |  | ||||||
| #include <engine_engine_number_9/connecting_assets/static_asset_manager.h> |  | ||||||
| #include <assert.h> |  | ||||||
| 
 |  | ||||||
| bool termination = false; |  | ||||||
| 
 |  | ||||||
| void sigterm_action(int ) { |  | ||||||
|     termination = true; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int main(int argc, char** argv){ |  | ||||||
|     een9_ASSERT_pl(argc > 0); |  | ||||||
|     if (argc < 1 + 2) { |  | ||||||
|         printf("Usage: %s <file with settings> <assets folder>\n", argv[0]); |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
|     // if ()
 |  | ||||||
|     if (!een9::isDirectory(argv[2])) { |  | ||||||
|         printf("\"%s\" is not a directory\n", argv[2]); |  | ||||||
|     } |  | ||||||
|     std::string assets_dir = argv[2]; |  | ||||||
| 
 |  | ||||||
|     een9::StaticAssetManagerSlaveModule samI; |  | ||||||
|     samI.update({ |  | ||||||
|         een9::StaticAssetManagerRule{assets_dir + "/html", "/assets/html", {{".html", "text/html"}} }, |  | ||||||
|         een9::StaticAssetManagerRule{assets_dir + "/css", "/assets/css", {{".css", "text/css"}} }, |  | ||||||
|         een9::StaticAssetManagerRule{assets_dir + "/js", "/assets/js", {{".js", "text/js"}} }, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     een9::MainloopParameters params; |  | ||||||
|     params.guest_core = [&samI](const een9::SlaveTask& task, const een9::ClientRequest& req) -> std::string { |  | ||||||
|         een9::StaticAsset sa; |  | ||||||
|         int ret; |  | ||||||
|         ret = samI.get_asset(req.url, sa); |  | ||||||
|         if (ret >= 0) { |  | ||||||
|             return een9::form_http_server_response_200(sa.type, sa.content); |  | ||||||
|         } |  | ||||||
|         if (req.url == "/" || req.url == "/index.html") { |  | ||||||
|             ret = samI.get_asset("/assets/html/test.html", sa); |  | ||||||
|             een9_ASSERT_pl(ret == 0); |  | ||||||
|             return een9::form_http_server_response_200(sa.type, sa.content); |  | ||||||
|         } |  | ||||||
|         return een9::form_http_server_response_404("text/html", "<h1> Not found! </h1>"); |  | ||||||
|     }; |  | ||||||
|     params.ports_to_listen = {1025}; |  | ||||||
|     params.slave_number = 8; |  | ||||||
|     params.open_admin_listener = false; |  | ||||||
| 
 |  | ||||||
|     signal(SIGINT, sigterm_action); |  | ||||||
| 
 |  | ||||||
|     een9::safe_electric_boogaloo(params, termination); |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user