/*
* This file is part of Checkpoint
* Copyright (C) 2017-2019 Bernardo Giordano, FlagBrew
*
* This program 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.
*
* This program 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 this program. If not, see .
*
* Additional Terms 7.b and 7.c of GPLv3 apply to this file:
* * Requiring preservation of specified reasonable legal notices or
* author attributions in that material or in the Appropriate Legal
* Notices displayed by works containing it.
* * Prohibiting misrepresentation of the origin of that material,
* or requiring that modified versions of such material be marked in
* reasonable ways as different from the original version.
*/
#include "configuration.hpp"
static struct mg_mgr mgr;
static struct mg_connection *nc;
static struct mg_serve_http_opts s_http_server_opts;
static const char *s_http_port = "8000";
static void handle_populate(struct mg_connection *nc, struct http_message *hm)
{
// populate gets called at startup, assume a new connection has been started
blinkLed(2);
auto json = Configuration::getInstance().getJson();
auto map = getCompleteTitleList();
json["title_list"] = map;
std::string body = json.dump();
mg_printf(nc, "HTTP/1.1 200 OK\r\nContent-Length: %lu\r\n\r\n%.*s",
(unsigned long)body.length(), (int)body.length(), body.c_str());
}
static void handle_save(struct mg_connection *nc, struct http_message *hm)
{
FILE* f = fopen(Configuration::getInstance().BASEPATH.c_str(), "w");
if (f != NULL)
{
fwrite(hm->body.p, 1, hm->body.len, f);
fclose(f);
}
Configuration::getInstance().load();
Configuration::getInstance().parse();
// Send response
mg_printf(nc, "HTTP/1.1 200 OK\r\nContent-Length: %lu\r\n\r\n%.*s",
(unsigned long) hm->body.len, (int) hm->body.len, hm->body.p);
}
static void ev_handler(struct mg_connection *nc, int ev, void *ev_data)
{
struct http_message *hm = (struct http_message *) ev_data;
switch (ev) {
case MG_EV_HTTP_REQUEST:
if (mg_vcmp(&hm->uri, "/save") == 0) {
handle_save(nc, hm);
} else if (mg_vcmp(&hm->uri, "/populate") == 0) {
handle_populate(nc, hm);
} else {
mg_serve_http(nc, hm, s_http_server_opts);
}
break;
default:
break;
}
}
Configuration::Configuration(void)
{
// check for existing config.json files on the sd card, BASEPATH
if (!io::fileExists(BASEPATH))
{
store();
}
// load json config file
load();
bool updateJson = false;
if (mJson.find("version") == mJson.end())
{
// if config is present but is < 3.4.2, override it
store();
}
else
{
// 3.4.2 -> 3.5.0
if (mJson["version"] < 2)
{
mJson["favorites"] = nlohmann::json::array();
updateJson = true;
}
// 3.5.0 -> 3.5.1
if (mJson["version"] < 3)
{
mJson["pksm-bridge"] = true;
updateJson = true;
}
}
if (updateJson)
{
mJson["version"] = CONFIG_VERSION;
save();
}
parse();
// load server
mg_mgr_init(&mgr, NULL);
nc = mg_bind(&mgr, s_http_port, ev_handler);
mg_set_protocol_http_websocket(nc);
s_http_server_opts.document_root = "romfs:/web_root";
s_http_server_opts.auth_domain = "flagbrew.org";
}
Configuration::~Configuration(void)
{
mg_mgr_free(&mgr);
}
void Configuration::store(void)
{
FILE* in = fopen("romfs:/config.json", "rt");
if (in != NULL)
{
nlohmann::json src = nlohmann::json::parse(in, nullptr, false);
fclose(in);
std::string writeData = src.dump(2);
writeData.shrink_to_fit();
size_t size = writeData.size();
FILE* out = fopen(BASEPATH.c_str(), "wt");
if (out != NULL)
{
fwrite(writeData.c_str(), 1, size, out);
fclose(out);
}
}
}
bool Configuration::filter(u64 id)
{
return mFilterIds.find(id) != mFilterIds.end();
}
bool Configuration::favorite(u64 id)
{
return mFavoriteIds.find(id) != mFavoriteIds.end();
}
std::vector Configuration::additionalSaveFolders(u64 id)
{
std::vector emptyvec;
auto folders = mAdditionalSaveFolders.find(id);
return folders == mAdditionalSaveFolders.end() ? emptyvec : folders->second;
}
bool Configuration::isPKSMBridgeEnabled(void)
{
return PKSMBridgeEnabled;
}
void Configuration::pollServer(void)
{
mg_mgr_poll(&mgr, 1000/60);
}
void Configuration::save(void)
{
std::string writeData = mJson.dump(2);
writeData.shrink_to_fit();
size_t size = writeData.size();
FILE* out = fopen(BASEPATH.c_str(), "wt");
if (out != NULL)
{
fwrite(writeData.c_str(), 1, size, out);
fclose(out);
}
}
void Configuration::load(void)
{
FILE* in = fopen(BASEPATH.c_str(), "rt");
if (in != NULL)
{
mJson = nlohmann::json::parse(in, nullptr, false);
fclose(in);
}
}
void Configuration::parse(void)
{
mFilterIds.clear();
mFavoriteIds.clear();
mAdditionalSaveFolders.clear();
// parse filters
std::vector filter = mJson["filter"];
for (auto& id : filter)
{
mFilterIds.emplace(strtoull(id.c_str(), NULL, 16));
}
// parse favorites
std::vector favorites = mJson["favorites"];
for (auto& id : favorites)
{
mFavoriteIds.emplace(strtoull(id.c_str(), NULL, 16));
}
// parse additional save folders
auto js = mJson["additional_save_folders"];
for (auto it = js.begin(); it != js.end(); ++it)
{
std::vector folders = it.value()["folders"];
std::vector sfolders;
for (auto& folder : folders)
{
sfolders.push_back(folder);
}
mAdditionalSaveFolders.emplace(strtoull(it.key().c_str(), NULL, 16), sfolders);
}
// parse PKSM Bridge flag
PKSMBridgeEnabled = mJson["pksm-bridge"];
}
const char* Configuration::c_str(void)
{
return mJson.dump().c_str();
}
nlohmann::json Configuration::getJson(void)
{
return mJson;
}