blob: 5eabffedb9672c8a1b359a51273d61a20cd49f3d [file] [log] [blame]
Ed Tanous911ac312017-08-15 09:37:42 -07001#pragma once
2
3#include <experimental/filesystem>
4#include <fstream>
5#include <string>
6#include <crow/app.h>
7#include <crow/http_request.h>
8#include <crow/http_response.h>
9#include <crow/routing.h>
10#include <boost/algorithm/string/replace.hpp>
11#include <boost/container/flat_set.hpp>
12
13namespace crow {
14namespace webassets {
15
16namespace filesystem = std::experimental::filesystem;
Ed Tanousba9f9a62017-10-11 16:40:35 -070017
Ed Tanous55c7b7a2018-05-22 15:27:24 -070018struct CmpStr {
Ed Tanousba9f9a62017-10-11 16:40:35 -070019 bool operator()(const char* a, const char* b) const {
20 return std::strcmp(a, b) < 0;
21 }
22};
Ed Tanous911ac312017-08-15 09:37:42 -070023
24static boost::container::flat_set<std::string> routes;
25
26template <typename... Middlewares>
Ed Tanous55c7b7a2018-05-22 15:27:24 -070027void requestRoutes(Crow<Middlewares...>& app) {
28 const static boost::container::flat_map<const char*, const char*, CmpStr>
29 contentTypes{
Ed Tanousba9f9a62017-10-11 16:40:35 -070030 {{".css", "text/css;charset=UTF-8"},
31 {".html", "text/html;charset=UTF-8"},
32 {".js", "text/html;charset=UTF-8"},
33 {".png", "image/png;charset=UTF-8"},
34 {".woff", "application/x-font-woff"},
35 {".woff2", "application/x-font-woff2"},
Ed Tanous5b6a1f92018-02-02 11:05:21 -080036 {".gif", "image/gif"},
37 {".ico", "image/x-icon"},
Ed Tanousba9f9a62017-10-11 16:40:35 -070038 {".ttf", "application/x-font-ttf"},
39 {".svg", "image/svg+xml"},
40 {".eot", "application/vnd.ms-fontobject"},
41 {".xml", "application/xml"},
Ed Tanous257f5792018-03-17 14:40:09 -070042 {".jpg", "image/jpeg"},
43 {".jpeg", "image/jpeg"},
Ed Tanousb39db7142018-08-14 11:20:42 -070044 {".json", "application/json"},
Ed Tanousba9f9a62017-10-11 16:40:35 -070045 // dev tools don't care about map type, setting to json causes
46 // browser to show as text
47 // https://stackoverflow.com/questions/19911929/what-mime-type-should-i-use-for-javascript-source-map-files
48 {".map", "application/json"}}};
Ed Tanous9dc2c4d2018-03-07 14:51:27 -080049 filesystem::path rootpath{"/usr/share/www/"};
Ed Tanous55c7b7a2018-05-22 15:27:24 -070050 filesystem::recursive_directory_iterator dirIter(rootpath);
Ed Tanousa38b0b22018-08-29 11:10:47 -070051 // In certain cases, we might have both a gzipped version of the file AND a
52 // non-gzipped version. To avoid duplicated routes, we need to make sure we
53 // get the gzipped version first. Because the gzipped path should be longer
54 // than the non gzipped path, if we sort in Ascending order, we should be
55 // guaranteed to get the gzip version first.
56 std::vector<filesystem::directory_entry> paths(filesystem::begin(dirIter),
57 filesystem::end(dirIter));
58 std::sort(paths.rbegin(), paths.rend());
Ed Tanous9dc2c4d2018-03-07 14:51:27 -080059
Ed Tanousa38b0b22018-08-29 11:10:47 -070060 for (const filesystem::directory_entry& dir : paths) {
Ed Tanous55c7b7a2018-05-22 15:27:24 -070061 filesystem::path absolutePath = dir.path();
62 filesystem::path relativePath{
63 absolutePath.string().substr(rootpath.string().size() - 1)};
Ed Tanous911ac312017-08-15 09:37:42 -070064 if (filesystem::is_directory(dir)) {
65 // don't recurse into hidden directories or symlinks
66 if (boost::starts_with(dir.path().filename().string(), ".") ||
67 filesystem::is_symlink(dir)) {
Ed Tanous55c7b7a2018-05-22 15:27:24 -070068 dirIter.disable_recursion_pending();
Ed Tanous911ac312017-08-15 09:37:42 -070069 }
70 } else if (filesystem::is_regular_file(dir)) {
Ed Tanous55c7b7a2018-05-22 15:27:24 -070071 std::string extension = relativePath.extension();
72 filesystem::path webpath = relativePath;
73 const char* contentEncoding = nullptr;
Ed Tanous5b6a1f92018-02-02 11:05:21 -080074
75 if (extension == ".gz") {
Ed Tanous911ac312017-08-15 09:37:42 -070076 webpath = webpath.replace_extension("");
Ed Tanous5b6a1f92018-02-02 11:05:21 -080077 // Use the non-gzip version for determining content type
78 extension = webpath.extension().string();
Ed Tanous55c7b7a2018-05-22 15:27:24 -070079 contentEncoding = "gzip";
Ed Tanous911ac312017-08-15 09:37:42 -070080 }
81
Ed Tanous9dc2c4d2018-03-07 14:51:27 -080082 if (boost::starts_with(webpath.filename().string(), "index.")) {
Ed Tanous911ac312017-08-15 09:37:42 -070083 webpath = webpath.parent_path();
Ed Tanous9dc2c4d2018-03-07 14:51:27 -080084 if (webpath.string().size() == 0 || webpath.string().back() != '/') {
Ed Tanous15aab542018-03-28 10:29:31 -070085 // insert the non-directory version of this path
86 routes.insert(webpath);
Ed Tanousba9f9a62017-10-11 16:40:35 -070087 webpath += "/";
88 }
Ed Tanous911ac312017-08-15 09:37:42 -070089 }
90
Ed Tanousa38b0b22018-08-29 11:10:47 -070091 std::pair<boost::container::flat_set<std::string>::iterator, bool>
92 inserted = routes.insert(webpath);
93
94 if (!inserted.second) {
95 // Got a duplicated path. This is expected in certain situations
96 BMCWEB_LOG_DEBUG << "Got duplicated path " << webpath;
97 continue;
98 }
Ed Tanous55c7b7a2018-05-22 15:27:24 -070099 const char* contentType = nullptr;
Ed Tanous911ac312017-08-15 09:37:42 -0700100
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700101 auto contentTypeIt = contentTypes.find(extension.c_str());
102 if (contentTypeIt == contentTypes.end()) {
103 BMCWEB_LOG_ERROR << "Cannot determine content-type for " << absolutePath
104 << " with extension " << extension;
Ed Tanous9dc2c4d2018-03-07 14:51:27 -0800105 } else {
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700106 contentType = contentTypeIt->second;
Ed Tanous5b6a1f92018-02-02 11:05:21 -0800107 }
108
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700109 app.routeDynamic(webpath)(
110 [absolutePath, contentType, contentEncoding](const crow::Request& req,
111 crow::Response& res) {
112 if (contentType != nullptr) {
113 res.addHeader("Content-Type", contentType);
Ed Tanous911ac312017-08-15 09:37:42 -0700114 }
Ed Tanousba9f9a62017-10-11 16:40:35 -0700115
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700116 if (contentEncoding != nullptr) {
117 res.addHeader("Content-Encoding", contentEncoding);
Ed Tanousba9f9a62017-10-11 16:40:35 -0700118 }
119
Ed Tanous911ac312017-08-15 09:37:42 -0700120 // res.set_header("Cache-Control", "public, max-age=86400");
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700121 std::ifstream inf(absolutePath);
Ed Tanous911ac312017-08-15 09:37:42 -0700122 if (!inf) {
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700123 BMCWEB_LOG_DEBUG << "failed to read file";
Ed Tanouse0d918b2018-03-27 17:41:04 -0700124 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous911ac312017-08-15 09:37:42 -0700125 res.end();
126 return;
127 }
128
Ed Tanouse0d918b2018-03-27 17:41:04 -0700129 res.body() = {std::istreambuf_iterator<char>(inf),
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700130 std::istreambuf_iterator<char>()};
Ed Tanous911ac312017-08-15 09:37:42 -0700131 res.end();
132 });
133 }
134 }
Ed Tanous911ac312017-08-15 09:37:42 -0700135} // namespace webassets
Ed Tanous5b6a1f92018-02-02 11:05:21 -0800136} // namespace webassets
137} // namespace crow