/* Copyright 2017, Michele Santullo
* This file is part of "tawashi".
*
* "tawashi" is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* "tawashi" is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with "tawashi". If not, see .
*/
#include "tawashiConfig.h"
#include "submit_paste_response.hpp"
#include "pastie_response.hpp"
#include "index_response.hpp"
#include "error_response.hpp"
#include "response_factory.hpp"
#include "cgi_env.hpp"
#include "ini_file.hpp"
#include "safe_stack_object.hpp"
#include "pathname/pathname.hpp"
#include "duckhandy/compatibility.h"
#include "settings_bag.hpp"
#include "logging_levels.hpp"
#include
#include
#include
#include
#include
#include
//www.open-std.org/jtc1/sc22/wg21/docs/papers/2014/n4150.pdf
namespace {
std::string config_file_path() a_pure;
std::string config_file_path() {
mchlib::PathName config_path(TAWASHI_CONFIG_PATH);
mchlib::PathName full_path("");
if (config_path.is_absolute()) {
full_path = std::move(config_path);
}
else {
full_path = mchlib::PathName(TAWASHI_PATH_PREFIX);
full_path.join(config_path);
}
full_path.join(TAWASHI_CONFIG_FILE);
return full_path.path();
}
template
std::unique_ptr make_response (
const Kakoune::SafePtr& parSettings,
const Kakoune::SafePtr& parCgiEnv
) {
return static_cast>(
std::make_unique(parSettings, &std::cout, parCgiEnv)
);
}
void fill_defaults (tawashi::SettingsBag& parSettings) {
parSettings.add_default("redis_server", "127.0.0.1");
parSettings.add_default("redis_port", "6379");
parSettings.add_default("redis_mode", "sock");
parSettings.add_default("redis_sock", "/tmp/redis.sock");
parSettings.add_default("redis_db", "0");
parSettings.add_default("base_uri", "http://127.0.0.1");
parSettings.add_default("website_root", "");
parSettings.add_default("langmap_dir", "/usr/share/source-highlight");
parSettings.add_default("min_pastie_size", "10");
parSettings.add_default("max_pastie_size", "10000");
parSettings.add_default("truncate_long_pasties", "false");
parSettings.add_default("logging_level", "err");
parSettings.add_default("resubmit_wait", "10");
}
void print_buildtime_info() {
std::cout << "NDEBUG defined: ";
#if defined(NDEBUG)
std::cout << "yes (Release build)";
#else
std::cout << "no (Debug build)";
#endif
std::cout << '\n';
std::cout << "TAWASHI_CONFIG_FILE: \"" << TAWASHI_CONFIG_FILE << "\"\n";
std::cout << "TAWASHI_CONFIG_PATH: \"" << TAWASHI_CONFIG_PATH << "\"\n";
std::cout << "TAWASHI_PATH_PREFIX: \"" << TAWASHI_PATH_PREFIX << "\"\n";
std::cout << "VERSION_MAJOR: " << VERSION_MAJOR << '\n';
std::cout << "VERSION_MINOR: " << VERSION_MINOR << '\n';
std::cout << "VERSION_PATCH: " << VERSION_PATCH << '\n';
std::cout << "config_file_path(): \"" << config_file_path() << "\"\n";
}
curry::SafeStackObject load_ini() {
using curry::SafeStackObject;
using tawashi::IniFile;
using std::istream_iterator;
std::ifstream conf(config_file_path());
conf >> std::noskipws;
return SafeStackObject(istream_iterator(conf), istream_iterator());
}
void set_logging_level (const tawashi::SettingsBag& parSettings) {
auto logging_level = tawashi::LoggingLevels::_from_string_nocase(parSettings.as("logging_level").c_str());
spdlog::set_level(static_cast(logging_level._to_integral()));
}
} //unnamed namespace
int main (int parArgc, char* parArgv[], char* parEnvp[]) {
using curry::SafeStackObject;
using tawashi::IndexResponse;
using tawashi::SubmitPasteResponse;
using tawashi::PastieResponse;
using tawashi::ErrorResponse;
using tawashi::Response;
if (2 == parArgc and boost::string_ref(parArgv[1]) == "--show-paths") {
print_buildtime_info();
return 0;
}
//Prepare the logger
spdlog::set_pattern("[%Y-%m-%d %T %z] - %v");
spdlog::set_level(spdlog::level::trace); //set to maximum possible here
auto statuslog = spdlog::stderr_logger_st("statuslog");
statuslog->info("Loading config: \"{}\"", config_file_path());
SafeStackObject ini = load_ini();
auto settings = SafeStackObject(ini);
fill_defaults(*settings);
set_logging_level(*settings);
auto cgi_env = SafeStackObject(parEnvp);
tawashi::ResponseFactory resp_factory(settings, cgi_env);
SPDLOG_TRACE(statuslog, "Registering makers in the response factory");
resp_factory.register_maker("index.cgi", &make_response);
resp_factory.register_maker("", &make_response);
resp_factory.register_maker("paste.cgi", &make_response);
resp_factory.register_maker("error.cgi", &make_response);
resp_factory.register_jolly_maker(&make_response);
std::unique_ptr response = resp_factory.make_response(cgi_env->path_info().substr(1));
response->send();
return 0;
}