// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. using namespace ircd; std::map files {}; void init_files() { // TODO: XXX for(const auto &file : fs::ls_recursive("/home/jason/charybdis/charybdis/modules/static")) { const auto name(tokens_after(file, '/', 5)); files.emplace(std::string(name), file); } } static string_view content_type(const mutable_buffer &out, const string_view &filename, const string_view &content); resource::response get_root(client &client, const resource::request &request) try { const auto &path { !request.head.path? "index.html": request.head.path == "/"? "index.html": request.head.path }; auto it { files.find(lstrip(path, '/')) }; if(it == end(files)) throw http::error{http::NOT_FOUND}; const unique_buffer chunk_buffer { 48_KiB }; const auto &filename{it->second}; const string_view first_chunk { fs::read(filename, chunk_buffer) }; char content_type_buf[64]; resource::response { client, http::OK, content_type(content_type_buf, filename, first_chunk), size(first_chunk) < size(chunk_buffer)? size(first_chunk) : -1 }; if(size(first_chunk) < size(chunk_buffer)) { client.write_all(first_chunk); return {}; } char headbuf[64]; const unwind::exceptional terminate{[&headbuf, &client] { //TODO: find out if it's not a client problem so we don't have to eject client.close(net::dc::RST, net::close_ignore); }}; client.write_all(http::writechunk(headbuf, size(first_chunk))); client.write_all(first_chunk); client.write_all("\r\n"_sv); for(size_t offset(size(first_chunk));;) { const string_view chunk { fs::read(filename, chunk_buffer, offset) }; if(empty(chunk)) break; const string_view head { http::writechunk(headbuf, size(chunk)) }; client.write_all(head); client.write_all(chunk); client.write_all("\r\n"_sv); offset += size(chunk); if(size(chunk) < size(chunk_buffer)) break; } client.write_all(http::writechunk(headbuf, 0)); client.write_all("\r\n"_sv); return {}; } catch(const fs::filesystem_error &e) { throw http::error { http::NOT_FOUND //TODO: interp error_code }; } string_view content_type(const mutable_buffer &out, const string_view &filename, const string_view &content) { const auto extension { rsplit(filename, '.').second }; string_view content_type; switch(hash(extension)) { case hash("css"): content_type = "text/css; charset=utf-8"; break; case hash("js"): content_type = "application/javascript; charset=utf-8"; break; case hash("html"): content_type = "text/html; charset=utf-8"; break; case hash("ico"): content_type = "image/x-icon"; break; case hash("svg"): content_type = "image/svg+xml"; break; case hash("png"): content_type = "image/png"; break; case hash("gif"): content_type = "image/gif"; break; case hash("jpeg"): case hash("jpg"): content_type = "image/jpeg"; break; case hash("woff2"): content_type = "application/font-woff2"; break; case hash("woff"): content_type = "application/font-woff"; break; case hash("eot"): content_type = "application/vnd.ms-fontobject"; break; case hash("otf"): case hash("ttf"): content_type = "application/font-sfnt"; break; case hash("ogg"): content_type = "application/ogg"; break; case hash("json"): content_type = "application/json; charset=utf-8"; break; case hash("txt"): content_type = "text/plain; charset=utf-8"; break; default: { content_type = magic::mime(out, content); break; } } return content_type; } resource root_resource { "/", { "Webroot resource", root_resource.DIRECTORY } }; resource::method root_get { root_resource, "GET", get_root }; mapi::header IRCD_MODULE { "Web root content resource", init_files };