56 #if !(defined(_WIN32))
61 #define DBG_CS LOG_STREAM(debug, log_campaignd)
62 #define LOG_CS LOG_STREAM(info, log_campaignd)
63 #define WRN_CS LOG_STREAM(warn, log_campaignd)
64 #define ERR_CS LOG_STREAM(err, log_campaignd)
67 #define ERR_CONFIG LOG_STREAM(err, log_config)
68 #define WRN_CONFIG LOG_STREAM(warn, log_config)
71 #define ERR_SERVER LOG_STREAM(err, log_server)
83 const std::set<std::string> cap_defaults = {
93 const std::string default_web_url =
"https://add-ons.wesnoth.org/";
104 const std::string default_license_notice = R
"(<span size='x-large'>General Rules</span>
106 The current version of the server rules can be found at: https://r.wesnoth.org/t51347
108 <span color='#f88'>Any content that does not conform to the rules listed at the link above, as well as the licensing terms below, may be removed at any time without prior notice.</span>
110 <span size='x-large'>Licensing</span>
112 All content within add-ons uploaded to this server must be licensed under the terms of the GNU General Public License (GPL), version 2 or later, with the sole exception of graphics and audio explicitly denoted as released under a Creative Commons license either in:
114 a) a combined toplevel file, e.g. “<span font_family='monospace'>My_Addon/ART_LICENSE</span>”; <b>or</b>
115 b) a file with the same path as the asset with “<span font_family='monospace'>.license</span>” appended, e.g. “<span font_family='monospace'>My_Addon/images/units/axeman.png.license</span>”.
117 <b>By uploading content to this server, you certify that you have the right to:</b>
119 a) release all included art and audio explicitly denoted with a Creative Commons license in the prescribed manner under that license; <b>and</b>
120 b) release all other included content under the terms of the chosen versions of the GNU GPL.)";
122 bool timing_reports_enabled =
false;
126 if(timing_reports_enabled) {
128 LOG_CS << req <<
"Time elapsed: " << tim <<
" ms";
130 LOG_CS << req <<
"Time elapsed [" <<
label <<
"]: " << tim <<
" ms";
154 if(!addon[
"forum_auth"].to_bool()) {
167 inline void set_passphrase(
config& addon,
const std::string& passphrase)
170 if(!addon[
"forum_auth"].to_bool()) {
180 inline std::string make_update_pack_filename(
const std::string& old_version,
const std::string& new_version)
190 inline std::string make_full_pack_filename(
const std::string& version)
200 inline std::string make_index_filename(
const std::string& version)
210 inline std::string index_from_full_pack_filename(std::string pack_fn)
212 auto dot_pos = pack_fn.find_last_of(
'.');
213 if(dot_pos != std::string::npos) {
214 pack_fn.replace(dot_pos, std::string::npos,
".hash.gz");
224 return cfg && !cfg->empty();
232 template<
typename... Vals>
233 utils::optional<std::vector<std::string>> multi_find_illegal_names(
const Vals&... args)
235 std::vector<std::string>
names;
238 return !
names.empty() ? utils::optional(
names) : utils::nullopt;
246 template<
typename... Vals>
247 utils::optional<std::vector<std::string>> multi_find_case_conflicts(
const Vals&... args)
249 std::vector<std::string>
names;
252 return !
names.empty() ? utils::optional(
names) : utils::nullopt;
260 std::string simple_wml_escape(
const std::string& text)
263 auto it = text.begin();
265 while(it != text.end()) {
266 res.append(*it ==
'"' ? 2 : 1, *it);
277 , user_handler_(nullptr)
278 , capabilities_(cap_defaults)
282 , cfg_file_(cfg_file)
285 , update_pack_lifespan_(0)
286 , strict_versions_(true)
290 , feedback_url_format_()
295 , stats_exempt_ips_()
296 , flush_timer_(io_service_)
301 std::memset( &sa, 0,
sizeof(sa) );
302 #pragma GCC diagnostic ignored "-Wold-style-cast"
303 sa.sa_handler = SIG_IGN;
304 int res = sigaction( SIGPIPE, &sa,
nullptr);
339 LOG_CS <<
"READ-ONLY MODE ACTIVE";
353 web_url_ = svinfo_cfg[
"web_url"].str(default_web_url);
354 license_notice_ = svinfo_cfg[
"license_notice"].str(default_license_notice);
362 hooks_.emplace(std::string(
"hook_post_upload"),
cfg_[
"hook_post_upload"]);
363 hooks_.emplace(std::string(
"hook_post_erase"),
cfg_[
"hook_post_erase"]);
367 if(!
cfg_[
"control_socket"].empty()) {
368 const std::string&
path =
cfg_[
"control_socket"].str();
371 const int res = mkfifo(
path.c_str(),0660);
372 if(res != 0 && errno != EEXIST) {
373 ERR_CS <<
"could not make fifo at '" <<
path <<
"' (" << strerror(errno) <<
")";
376 int fifo = open(
path.c_str(), O_RDWR|O_NONBLOCK);
378 LOG_CS <<
"opened fifo at '" <<
path <<
"'. Server commands may be written to this file.";
402 std::vector<std::string> legacy_addons, dirs;
405 for(
const std::string& addon_dir : dirs) {
409 addons_.emplace(meta[
"name"].str(), meta);
418 WRN_CS <<
"Old format addons have been detected in the config! They will be converted to the new file format! "
419 << campaigns.
child_count(
"campaign") <<
" entries to be processed.";
421 const std::string& addon_id = campaign[
"name"].str();
422 const std::string& addon_file = campaign[
"filename"].str();
425 +
"' already exists in the new form! Possible code or filesystem interference!\n");
427 if(std::find(legacy_addons.begin(), legacy_addons.end(), addon_id) == legacy_addons.end()) {
429 +
"'. Check the file structure!\n");
438 config version_cfg =
config(
"version", campaign[
"version"].str());
439 version_cfg[
"filename"] = make_full_pack_filename(campaign[
"version"]);
440 campaign.add_child(
"version", version_cfg);
442 data.remove_attributes(
"title",
"campaign_name",
"author",
"description",
"version",
"timestamp",
"original_timestamp",
"icon",
"type",
"tags");
455 writer.
write(data_hash);
456 campaign_hash_file.
commit();
459 addons_.emplace(addon_id, campaign);
463 LOG_CS <<
"Legacy addons processing finished.";
467 LOG_CS <<
"Loaded addons metadata. " <<
addons_.size() <<
" addons found.";
472 ERR_CS <<
"The server id must be set when database support is used.";
476 LOG_CS <<
"User handler initialized.";
485 o << '[' << (utils::holds_alternative<tls_socket_ptr>(r.
sock) ?
"+" :
"") << r.
addr <<
' ' << r.
cmd <<
"] ";
491 boost::asio::spawn(
io_service_, [
this, socket](boost::asio::yield_context yield) {
498 boost::asio::spawn(
io_service_, [
this, socket](boost::asio::yield_context yield) {
503 template<
class Socket>
509 socket->lowest_layer().close();
518 if(
i !=
data.ordered_end()) {
522 request_handlers_table::const_iterator j
527 request req{
c.key,
c.cfg, socket, yield};
528 auto st = service_timer(req);
529 j->second(
this, req);
531 send_error(
"Unrecognized [" +
c.key +
"] request.",socket);
542 if(error == boost::asio::error::operation_aborted)
545 ERR_CS <<
"Error reading from fifo: " << error.message();
551 std::getline(is, cmd);
555 if(ctl ==
"shut_down") {
556 LOG_CS <<
"Shut down requested by admin, shutting down...";
558 }
else if(ctl ==
"readonly") {
564 }
else if(ctl ==
"flush") {
565 LOG_CS <<
"Flushing config to disk...";
567 }
else if(ctl ==
"reload") {
569 if(ctl[1] ==
"blacklist") {
570 LOG_CS <<
"Reloading blacklist...";
573 ERR_CS <<
"Unrecognized admin reload argument: " << ctl[1];
576 LOG_CS <<
"Reloading all configuration...";
578 LOG_CS <<
"Reloaded configuration";
580 }
else if(ctl ==
"delete") {
582 ERR_CS <<
"Incorrect number of arguments for 'delete'";
584 const std::string& addon_id = ctl[1];
586 LOG_CS <<
"deleting add-on '" << addon_id <<
"' requested from control FIFO";
589 }
else if(ctl ==
"hide" || ctl ==
"unhide") {
591 ERR_CS <<
"Incorrect number of arguments for '" << ctl.
cmd() <<
"'";
593 const std::string& addon_id = ctl[1];
597 ERR_CS <<
"Add-on '" << addon_id <<
"' not found, cannot " << ctl.
cmd();
599 addon[
"hidden"] = (ctl.
cmd() ==
"hide");
602 LOG_CS <<
"Add-on '" << addon_id <<
"' is now " << (ctl.
cmd() ==
"hide" ?
"hidden" :
"unhidden");
605 }
else if(ctl ==
"setpass") {
607 ERR_CS <<
"Incorrect number of arguments for 'setpass'";
609 const std::string& addon_id = ctl[1];
610 const std::string& newpass = ctl[2];
614 ERR_CS <<
"Add-on '" << addon_id <<
"' not found, cannot set passphrase";
615 }
else if(newpass.empty()) {
617 ERR_CS <<
"Add-on passphrases may not be empty!";
618 }
else if(addon[
"forum_auth"].to_bool()) {
619 ERR_CS <<
"Can't set passphrase for add-on using forum_auth.";
621 set_passphrase(*addon, newpass);
624 LOG_CS <<
"New passphrase set for '" << addon_id <<
"'";
627 }
else if(ctl ==
"setattr") {
629 ERR_CS <<
"Incorrect number of arguments for 'setattr'";
631 const std::string& addon_id = ctl[1];
632 const std::string& key = ctl[2];
644 ERR_CS <<
"Add-on '" << addon_id <<
"' not found, cannot set attribute";
645 }
else if(key ==
"name" || key ==
"version") {
646 ERR_CS <<
"setattr cannot be used to rename add-ons or change their version";
647 }
else if(key ==
"passhash"|| key ==
"passsalt") {
648 ERR_CS <<
"setattr cannot be used to set auth data -- use setpass instead";
649 }
else if(!addon->has_attribute(key)) {
656 ERR_CS <<
"Attribute '" << key <<
"' is not a recognized add-on attribute";
661 LOG_CS <<
"Set attribute on add-on '" << addon_id <<
"':\n"
662 << key <<
"=\"" << value <<
"\"";
665 }
else if(ctl ==
"log") {
666 static const std::map<std::string, lg::severity> log_levels = {
675 ERR_CS <<
"Incorrect number of arguments for 'log'";
676 }
else if(ctl[1] ==
"precise") {
679 LOG_CS <<
"Precise timestamps enabled";
680 }
else if(ctl[2] ==
"off") {
682 LOG_CS <<
"Precise timestamps disabled";
684 ERR_CS <<
"Invalid argument for 'log precise': " << ctl[2];
686 }
else if(log_levels.find(ctl[1]) == log_levels.end()) {
687 ERR_CS <<
"Invalid log level '" << ctl[1] <<
"'";
689 auto sev = log_levels.find(ctl[1])->second;
692 ERR_CS <<
"Unknown log domain '" << domain <<
"'";
694 LOG_CS <<
"Set log level for domain '" << domain <<
"' to " << ctl[1];
698 }
else if(ctl ==
"timings") {
700 ERR_CS <<
"Incorrect number of arguments for 'timings'";
701 }
else if(ctl[1] ==
"on") {
702 campaignd::timing_reports_enabled =
true;
703 LOG_CS <<
"Request servicing timing reports enabled";
704 }
else if(ctl[1] ==
"off") {
705 campaignd::timing_reports_enabled =
false;
706 LOG_CS <<
"Request servicing timing reports disabled";
708 ERR_CS <<
"Invalid argument for 'timings': " << ctl[1];
711 ERR_CS <<
"Unrecognized admin command: " << ctl.
full();
719 LOG_CS <<
"SIGHUP caught, reloading config.";
723 LOG_CS <<
"Reloaded configuration";
732 flush_timer_.expires_from_now(std::chrono::minutes(10));
739 ERR_CS <<
"Error from reload timer: " << error.message();
740 throw boost::system::system_error(error);
771 DBG_CS <<
"writing configuration and add-ons list to disk...";
778 if(addon && !addon[
"filename"].empty()) {
789 void server::fire(
const std::string& hook, [[maybe_unused]]
const std::string& addon)
791 const std::map<std::string, std::string>::const_iterator itor =
hooks_.find(hook);
792 if(itor ==
hooks_.end()) {
796 const std::string& script = itor->second;
802 ERR_CS <<
"Tried to execute a script on an unsupported platform";
807 if((childpid = fork()) == -1) {
808 ERR_CS <<
"fork failed while updating add-on " << addon;
816 execlp(script.c_str(), script.c_str(), addon.c_str(),
static_cast<char *
>(
nullptr));
819 PLAIN_LOG <<
"ERROR: exec failed with errno " << errno <<
" for addon " << addon;
842 const auto& escaped_msg = simple_wml_escape(
msg);
855 const auto& escaped_msg = simple_wml_escape(
msg);
863 const std::string& status_hex =
formatter()
864 <<
"0x" << std::setfill(
'0') << std::setw(2*
sizeof(
unsigned int)) << std::hex
865 << std::uppercase << status_code;
868 const auto& escaped_status_str = simple_wml_escape(std::to_string(status_code));
869 const auto& escaped_msg = simple_wml_escape(
msg);
870 const auto& escaped_extra_data = simple_wml_escape(extra_data);
876 err_cfg.
set_attr_dup(
"extra_data", escaped_extra_data.c_str());
877 err_cfg.
set_attr_dup(
"status_code", escaped_status_str.c_str());
886 return addon->second;
897 ERR_CS <<
"Cannot delete unrecognized add-on '" <<
id <<
"'";
901 if(cfg[
"forum_auth"].to_bool()) {
905 std::string fn = cfg[
"filename"].str();
908 ERR_CS <<
"Add-on '" <<
id <<
"' does not have an associated filename, cannot delete";
912 ERR_CS <<
"Could not delete the directory for addon '" <<
id
913 <<
"' (" << fn <<
"): " << strerror(errno);
919 fire(
"hook_post_erase",
id);
921 LOG_CS <<
"Deleted add-on '" <<
id <<
"'";
924 #define REGISTER_CAMPAIGND_HANDLER(req_id) \
925 handlers_[#req_id] = std::bind(&server::handle_##req_id, \
926 std::placeholders::_1, std::placeholders::_2)
942 DBG_CS << req <<
"Sending server identification";
944 std::ostringstream ostr;
953 const auto& wml = ostr.str();
962 LOG_CS << req <<
"Sending add-ons list";
964 std::time_t epoch = std::time(
nullptr);
968 if(req.
cfg[
"times_relative_to"] !=
"now") {
972 bool before_flag =
false;
973 std::time_t before = epoch;
975 before += req.
cfg[
"before"].to_time_t();
979 bool after_flag =
false;
980 std::time_t after = epoch;
982 after += req.
cfg[
"after"].to_time_t();
986 const std::string& name = req.
cfg[
"name"];
987 const std::string& lang = req.
cfg[
"language"];
989 for(
const auto& addon :
addons_)
991 if(!name.empty() && name != addon.first) {
997 if(
i[
"hidden"].to_bool()) {
1001 const auto& tm =
i[
"timestamp"];
1003 if(before_flag && (tm.empty() || tm.to_time_t(0) >= before)) {
1006 if(after_flag && (tm.empty() || tm.to_time_t(0) <= after)) {
1013 for(
const config& j :
i.child_range(
"translation"))
1015 if(j[
"language"] == lang && j[
"supported"].to_bool(
true)) {
1033 j.remove_attributes(
"passphrase",
"passhash",
"passsalt",
"upload_ip",
"email");
1043 j.clear_children(
"feedback");
1046 j.clear_children(
"update_pack");
1052 std::ostringstream ostr;
1053 write(ostr, response);
1054 std::string wml = ostr.str();
1065 if(!addon || addon[
"hidden"].to_bool()) {
1070 const auto& name = req.
cfg[
"name"].str();
1073 if(version_map.empty()) {
1074 send_error(
"No versions of the add-on '" + name +
"' are available on the server.", req.
sock);
1079 const auto& from = req.
cfg[
"from_version"].str();
1080 const auto& to = req.
cfg[
"version"].str(version_map.rbegin()->first);
1085 auto to_version_iter = version_map.find(to_parsed);
1086 if(to_version_iter == version_map.end()) {
1087 send_error(
"Could not find requested version " + to +
" of the addon '" + name +
1092 auto full_pack_path = addon[
"filename"].str() +
'/' + to_version_iter->second[
"filename"].str();
1101 if(!from.empty() && from_parsed < to_parsed && version_map.count(from_parsed) != 0) {
1110 int delivery_size = 0;
1111 bool force_use_full =
false;
1113 auto start_point = version_map.find(from_parsed);
1114 auto end_point = std::next(to_version_iter, 1);
1116 if(std::distance(start_point, end_point) <= 1) {
1118 ERR_CS <<
"Bad update sequence bounds in version " << from <<
" -> " << to <<
" update sequence for the add-on '" << name <<
"', sending a full pack instead";
1119 force_use_full =
true;
1122 for(
auto iter = start_point; !force_use_full && std::distance(iter, end_point) > 1;) {
1123 const auto& prev_version_cfg = iter->second;
1124 const auto& next_version_cfg = (++iter)->second;
1127 if(pack[
"from"].str() != prev_version_cfg[
"version"].str() ||
1128 pack[
"to"].str() != next_version_cfg[
"version"].str()) {
1133 const auto& update_pack_path = addon[
"filename"].str() +
'/' + pack[
"filename"].str();
1138 if(!step_delta.
empty()) {
1140 delta.
append(std::move(step_delta));
1143 ERR_CS <<
"Broken update sequence from version " << from <<
" to "
1144 << to <<
" for the add-on '" << name <<
"', sending a full pack instead";
1145 force_use_full =
true;
1153 if(delivery_size > full_pack_size && full_pack_size > 0) {
1154 force_use_full =
true;
1160 if(!force_use_full && !delta.
empty()) {
1161 std::ostringstream ostr;
1163 const auto& wml_text = ostr.str();
1168 LOG_CS << req <<
"Sending add-on '" << name <<
"' version: " << from <<
" -> " << to <<
" (delta)";
1170 utils::visit([
this, &req, &doc](
auto && sock) {
1174 full_pack_path.clear();
1181 if(!full_pack_path.empty()) {
1182 if(full_pack_size < 0) {
1183 send_error(
"Add-on '" + name +
"' could not be read by the server.", req.
sock);
1187 LOG_CS << req <<
"Sending add-on '" << name <<
"' version: " << to <<
" size: " << full_pack_size / 1024 <<
" KiB";
1188 utils::visit([
this, &req, &full_pack_path](
auto&& socket) {
1197 addon[
"downloads"] = 1 + addon[
"downloads"].to_int();
1209 if(!addon || addon[
"hidden"].to_bool()) {
1214 std::string
path = addon[
"filename"].str() +
'/';
1218 if(version_map.empty()) {
1219 send_error(
"No versions of the add-on '" + req.
cfg[
"name"].str() +
"' are available on the server.", req.
sock);
1222 const auto& version_str = addon[
"version"].str();
1224 auto version = version_map.find(version_parsed);
1225 if(version != version_map.end()) {
1226 path += version->second[
"filename"].str();
1229 if(version_str.empty()) {
1230 path += version_map.rbegin()->second[
"filename"].str();
1232 path += (--version_map.upper_bound(version_parsed))->second[
"filename"].str();
1236 path = index_from_full_pack_filename(
path);
1240 send_error(
"Missing index file for the add-on '" + req.
cfg[
"name"].str() +
"'.", req.
sock);
1244 LOG_CS << req <<
"Sending add-on hash index for '" << req.
cfg[
"name"] <<
"' size: " <<
file_size / 1024 <<
" KiB";
1245 utils::visit([
this, &
path, &req](
auto&& socket) {
1256 LOG_CS <<
"in read-only mode, request for upload terms denied";
1257 send_error(
"The server is currently in read-only mode, add-on uploads are disabled.", req.
sock);
1261 LOG_CS << req <<
"Sending license terms";
1268 LOG_CS <<
"Validation error: uploads not permitted in read-only mode.";
1278 const bool is_upload_pack = have_wml(removelist) || have_wml(addlist);
1280 const std::string& name = upload[
"name"].str();
1282 existing_addon =
nullptr;
1285 bool passed_name_utf8_check =
false;
1289 passed_name_utf8_check =
true;
1293 existing_addon = &
c.second;
1298 if(!passed_name_utf8_check) {
1299 LOG_CS <<
"Validation error: bad UTF-8 in add-on name";
1302 ERR_CS <<
"Validation error: add-ons list has bad UTF-8 somehow, this is a server side issue, it's bad, and you should probably fix it ASAP";
1309 if(upload[
"passphrase"].empty()) {
1310 LOG_CS <<
"Validation error: no passphrase specified";
1314 if(existing_addon && upload[
"forum_auth"].to_bool() != (*existing_addon)[
"forum_auth"].to_bool()) {
1315 LOG_CS <<
"Validation error: forum_auth is " << upload[
"forum_auth"].to_bool() <<
" but was previously uploaded set to " << (*existing_addon)[
"forum_auth"].to_bool();
1317 }
else if(upload[
"forum_auth"].to_bool()) {
1319 LOG_CS <<
"Validation error: client requested forum authentication but server does not support it";
1323 LOG_CS <<
"Validation error: forum auth requested for an author who doesn't exist";
1327 for(
const std::string& secondary_author :
utils::split(upload[
"secondary_authors"].str(),
',')) {
1329 LOG_CS <<
"Validation error: forum auth requested for a secondary author who doesn't exist";
1335 LOG_CS <<
"Validation error: forum passphrase does not match";
1339 }
else if(existing_addon && !authenticate(*existing_addon, upload[
"passphrase"])) {
1340 LOG_CS <<
"Validation error: campaignd passphrase does not match";
1344 if(existing_addon && (*existing_addon)[
"hidden"].to_bool()) {
1345 LOG_CS <<
"Validation error: add-on is hidden";
1351 upload[
"title"].str(),
1352 upload[
"description"].str(),
1353 upload[
"author"].str(),
1355 upload[
"email"].str()))
1357 LOG_CS <<
"Validation error: blacklisted uploader or publish information";
1361 LOG_CS <<
"Validation error: invalid UTF-8 sequence in publish information while checking against the blacklist";
1367 if(!is_upload_pack && !have_wml(
data)) {
1368 LOG_CS <<
"Validation error: no add-on data.";
1372 if(is_upload_pack && !have_wml(removelist) && !have_wml(addlist)) {
1373 LOG_CS <<
"Validation error: no add-on data.";
1378 LOG_CS <<
"Validation error: invalid add-on name.";
1383 LOG_CS <<
"Validation error: add-on name starts with an illegal formatting character.";
1387 if(upload[
"title"].empty()) {
1388 LOG_CS <<
"Validation error: no add-on title specified";
1393 LOG_CS <<
"Validation error: add-on title starts with an illegal formatting character.";
1398 LOG_CS <<
"Validation error: unknown add-on type specified";
1402 if(upload[
"author"].empty()) {
1403 LOG_CS <<
"Validation error: no add-on author specified";
1407 if(upload[
"version"].empty()) {
1408 LOG_CS <<
"Validation error: no add-on version specified";
1412 if(existing_addon) {
1416 if(
strict_versions_ ? new_version <= old_version : new_version < old_version) {
1417 LOG_CS <<
"Validation error: add-on version not incremented";
1422 if(upload[
"description"].empty()) {
1423 LOG_CS <<
"Validation error: no add-on description specified";
1428 if(upload[
"email"].empty() && !upload[
"forum_auth"].to_bool()) {
1429 LOG_CS <<
"Validation error: no add-on email specified";
1433 if(
const auto badnames = multi_find_illegal_names(
data, addlist, removelist)) {
1435 LOG_CS <<
"Validation error: invalid filenames in add-on pack (" << badnames->size() <<
" entries)";
1439 if(
const auto badnames = multi_find_case_conflicts(
data, addlist, removelist)) {
1441 LOG_CS <<
"Validation error: case conflicts in add-on pack (" << badnames->size() <<
" entries)";
1445 if(is_upload_pack && !existing_addon) {
1446 LOG_CS <<
"Validation error: attempted to send an update pack for a non-existent add-on";
1452 int topic_id = std::stoi(url_params[
"topic_id"].str(
"0"));
1455 LOG_CS <<
"Validation error: feedback topic ID does not exist in forum database";
1460 LOG_CS <<
"Validation error: feedback topic ID is not a valid number";
1470 const std::time_t upload_ts = std::time(
nullptr);
1472 const auto& name = upload[
"name"].str();
1474 LOG_CS << req <<
"Validating add-on '" << name <<
"'...";
1476 config* addon_ptr =
nullptr;
1477 std::string val_error_data;
1478 const auto val_status =
validate_addon(req, addon_ptr, val_error_data);
1481 LOG_CS <<
"Upload of '" << name <<
"' aborted due to a failed validation check";
1483 send_error(
msg, val_error_data,
static_cast<unsigned int>(val_status), req.
sock);
1487 LOG_CS << req <<
"Processing add-on '" << name <<
"'...";
1493 const bool is_delta_upload = have_wml(delta_remove) || have_wml(delta_add);
1494 const bool is_existing_upload = addon_ptr !=
nullptr;
1496 if(!is_existing_upload) {
1498 auto entry =
addons_.emplace(name,
config(
"original_timestamp", upload_ts));
1499 addon_ptr = &(*entry.first).second;
1502 config& addon = *addon_ptr;
1504 LOG_CS << req <<
"Upload type: "
1505 << (is_delta_upload ?
"delta" :
"full") <<
", "
1506 << (is_existing_upload ?
"update" :
"new");
1511 "title",
"name",
"uploader",
"author",
"secondary_authors",
"description",
"version",
"icon",
1512 "translate",
"dependencies",
"core",
"type",
"tags",
"email",
"forum_auth"
1515 const std::string& pathstem =
"data/" + name;
1516 addon[
"filename"] = pathstem;
1517 addon[
"upload_ip"] = req.
addr;
1519 if(!is_existing_upload && !addon[
"forum_auth"].to_bool()) {
1520 set_passphrase(addon, upload[
"passphrase"]);
1523 if(addon[
"downloads"].empty()) {
1524 addon[
"downloads"] = 0;
1527 addon[
"timestamp"] = upload_ts;
1528 addon[
"uploads"] = 1 + addon[
"uploads"].to_int();
1533 addon.
add_child(
"feedback", *url_params);
1535 topic_id = url_params[
"topic_id"].to_int();
1539 if(addon[
"forum_auth"].to_bool()) {
1540 addon[
"email"] =
user_handler_->get_user_email(upload[
"uploader"].str());
1546 if(!do_authors_exist || is_primary) {
1556 user_handler_->db_insert_addon_info(
server_id_, name, addon[
"title"].str(), addon[
"type"].str(), addon[
"version"].str(), addon[
"forum_auth"].to_bool(), topic_id, upload[
"uploader"].str());
1567 if(!locale_params[
"language"].empty()) {
1569 locale[
"language"] = locale_params[
"language"].str();
1570 locale[
"supported"] =
false;
1572 if(!locale_params[
"title"].empty()) {
1573 locale[
"title"] = locale_params[
"title"].str();
1575 if(!locale_params[
"description"].empty()) {
1576 locale[
"description"] = locale_params[
"description"].str();
1587 if(have_wml(full_pack)) {
1589 rw_full_pack = std::move(
const_cast<config&
>(*full_pack));
1594 const auto& new_version = addon[
"version"].str();
1597 if(is_delta_upload) {
1602 if(version_map.empty()) {
1604 ERR_CS <<
"Add-on '" << name <<
"' has an empty version table, this should not happen";
1609 auto prev_version = upload[
"from"].str();
1611 if(prev_version.empty()) {
1612 prev_version = version_map.rbegin()->first;
1617 auto vm_entry = version_map.find(prev_version_parsed);
1618 if(vm_entry == version_map.end()) {
1619 prev_version = (--version_map.upper_bound(prev_version_parsed))->first;
1627 std::set<std::string> delete_packs;
1628 for(
const auto& pack : addon.
child_range(
"update_pack")) {
1629 if(pack[
"to"].str() == new_version) {
1630 const auto& pack_filename = pack[
"filename"].
str();
1632 delete_packs.insert(pack_filename);
1636 if(!delete_packs.empty()) {
1638 return delete_packs.find(
p[
"filename"].str()) != delete_packs.end();
1642 const auto& update_pack_fn = make_update_pack_filename(prev_version, new_version);
1646 pack_info[
"from"] = prev_version;
1647 pack_info[
"to"] = new_version;
1649 pack_info[
"filename"] = update_pack_fn;
1654 LOG_CS <<
"Saving provided update pack for " << prev_version <<
" -> " << new_version <<
"...";
1658 static const config empty_config;
1660 writer.open_child(
"removelist");
1661 writer.write(have_wml(delta_remove) ? *delta_remove : empty_config);
1662 writer.close_child(
"removelist");
1664 writer.open_child(
"addlist");
1665 writer.write(have_wml(delta_add) ? *delta_add : empty_config);
1666 writer.close_child(
"addlist");
1676 auto it = version_map.find(prev_version_parsed);
1677 if(it == version_map.end()) {
1679 ERR_CS <<
"Previous version dropped off the version map?";
1685 rw_full_pack.
clear();
1688 if(have_wml(delta_remove)) {
1692 if(have_wml(delta_add)) {
1708 config version_cfg{
"version", new_version};
1709 version_cfg[
"filename"] = make_full_pack_filename(new_version);
1711 version_map.erase(new_version_parsed);
1714 return old_cfg[
"version"].str() == new_version;
1718 version_map.emplace(new_version_parsed, version_cfg);
1719 addon.
add_child(
"version", version_cfg);
1723 rw_full_pack[
"name"] =
"";
1727 const auto& full_pack_path = pathstem +
'/' + version_cfg[
"filename"].str();
1728 const auto& index_path = pathstem +
'/' + make_index_filename(new_version);
1731 config pack_index{
"name",
""};
1736 addon_pack_file.commit();
1740 addon_index_file.commit();
1747 std::set<std::string> expire_packs;
1750 if(upload_ts > pack[
"expire"].to_time_t() || pack[
"from"].str() == new_version || (!is_delta_upload && pack[
"to"].str() == new_version)) {
1751 LOG_CS <<
"Expiring upate pack for " << pack[
"from"].str() <<
" -> " << pack[
"to"].str();
1752 const auto& pack_filename = pack[
"filename"].str();
1754 expire_packs.insert(pack_filename);
1758 if(!expire_packs.empty()) {
1760 return expire_packs.find(
p[
"filename"].str()) != expire_packs.end();
1767 for(
auto iter = version_map.begin(); std::distance(iter, version_map.end()) > 1;) {
1768 const config& prev_version = iter->second;
1769 const config& next_version = (++iter)->second;
1771 const auto& prev_version_name = prev_version[
"version"].str();
1772 const auto& next_version_name = next_version[
"version"].str();
1776 for(
const auto& pack : addon.
child_range(
"update_pack")) {
1777 if(pack[
"from"].str() == prev_version_name && pack[
"to"].str() == next_version_name) {
1788 LOG_CS <<
"Automatically generating update pack for " << prev_version_name <<
" -> " << next_version_name <<
"...";
1790 const auto& prev_path = pathstem +
'/' + prev_version[
"filename"].str();
1791 const auto& next_path = pathstem +
'/' + next_version[
"filename"].str();
1794 ERR_CS <<
"Unable to automatically generate an update pack for '" << name
1795 <<
"' for version " << prev_version_name <<
" to " << next_version_name
1800 const auto& update_pack_fn = make_update_pack_filename(prev_version_name, next_version_name);
1803 pack_info[
"from"] = prev_version_name;
1804 pack_info[
"to"] = next_version_name;
1806 pack_info[
"filename"] = update_pack_fn;
1829 LOG_CS << req <<
"Finished uploading add-on '" << upload[
"name"] <<
"'";
1833 fire(
"hook_post_upload", name);
1839 const std::string&
id =
erase[
"name"].str();
1842 LOG_CS << req <<
"in read-only mode, request to delete '" <<
id <<
"' denied";
1843 send_error(
"Cannot delete add-on: The server is currently in read-only mode.", req.
sock);
1847 LOG_CS << req <<
"Deleting add-on '" <<
id <<
"'";
1864 if(!addon[
"forum_auth"].to_bool()) {
1865 if(!authenticate(*addon, pass)) {
1876 if(addon[
"hidden"].to_bool()) {
1877 LOG_CS <<
"Add-on removal denied - hidden add-on.";
1878 send_error(
"Add-on deletion denied. Please contact the server administration for assistance.", req.
sock);
1892 LOG_CS <<
"in read-only mode, request to change passphrase denied";
1893 send_error(
"Cannot change passphrase: The server is currently in read-only mode.", req.
sock);
1901 }
else if(addon[
"forum_auth"].to_bool()) {
1902 send_error(
"Changing the password for add-ons using forum_auth is not supported.", req.
sock);
1903 }
else if(!authenticate(*addon, cpass[
"passphrase"])) {
1905 }
else if(addon[
"hidden"].to_bool()) {
1906 LOG_CS <<
"Passphrase change denied - hidden add-on.";
1907 send_error(
"Add-on passphrase change denied. Please contact the server administration for assistance.", req.
sock);
1908 }
else if(cpass[
"new_passphrase"].empty()) {
1911 set_passphrase(*addon, cpass[
"new_passphrase"]);
1923 std::string uploader = addon[
"uploader"].str();
1924 std::string
id = addon[
"name"].str();
1932 if((do_authors_exist && !is_primary && !is_secondary) || (is_secondary && is_delete)) {
1936 std::string author = addon[
"uploader"].str();
1938 std::string hashed_password =
hash_password(passphrase, salt, author);
1949 std::string config_file =
"server.cfg";
1950 unsigned short port = 0;
1956 for(
auto domain : {
"campaignd",
"campaignd/blacklist",
"server" }) {
1967 std::cout << cmdline.help_text();
1971 if(cmdline.version) {
1976 if(cmdline.config_file) {
1982 if(cmdline.server_dir) {
1987 port = *cmdline.port;
1992 PLAIN_LOG <<
"Invalid network port: " << port;
1997 if(cmdline.show_log_domains) {
2002 for(
const auto& ldl : cmdline.log_domain_levels) {
2004 PLAIN_LOG <<
"Unknown log domain: " << ldl.first;
2009 if(cmdline.log_precise_timestamps) {
2013 if(cmdline.report_timings) {
2014 campaignd::timing_reports_enabled =
true;
2020 PLAIN_LOG <<
"Server directory '" << *cmdline.server_dir <<
"' does not exist or is not a directory.";
2025 PLAIN_LOG <<
"Server configuration file '" << config_file <<
"' is not a file.";
2033 PLAIN_LOG <<
"Bad server directory '" << server_path <<
"'.";
2049 }
catch(
const boost::program_options::error&
e) {
2050 PLAIN_LOG <<
"Error in command line: " <<
e.what();
2053 PLAIN_LOG <<
"Could not parse config file: " <<
e.message;
2058 }
catch(
const std::bad_function_call& ) {
2059 PLAIN_LOG <<
"Bad request handler function call";
campaignd authentication API.
std::vector< std::string > names
#define REGISTER_CAMPAIGND_HANDLER(req_id)
int main(int argc, char **argv)
static lg::log_domain log_campaignd("campaignd")
static int run_campaignd(int argc, char **argv)
static lg::log_domain log_server("server")
static lg::log_domain log_config("config")
void read(const config &cfg)
Initializes the blacklist from WML.
bool is_blacklisted(const std::string &name, const std::string &title, const std::string &description, const std::string &author, const std::string &ip, const std::string &email) const
Whether an add-on described by these fields is blacklisted.
Represents a server control line written to a communication socket.
const std::string & cmd() const
Returns the control command.
std::string full() const
Return the full command line string.
std::size_t args_count() const
Returns the total number of arguments, not including the command itself.
request_handlers_table handlers_
ADDON_CHECK_STATUS validate_addon(const server::request &req, config *&existing_addon, std::string &error_data)
Performs validation on an incoming add-on.
void handle_upload(const request &)
void delete_addon(const std::string &id)
std::set< std::string > capabilities_
config cfg_
Server config.
void handle_request_terms(const request &)
void send_error(const std::string &msg, const any_socket_ptr &sock)
Send a client an error message.
std::string license_notice_
void load_blacklist()
Reads the add-ons upload blacklist from WML.
std::map< std::string, std::string > hooks_
void handle_new_client(socket_ptr socket)
const config & server_info() const
Retrieves the contents of the [server_info] WML node.
std::string blacklist_file_
std::unique_ptr< user_handler > user_handler_
void handle_server_id(const request &)
void handle_delete(const request &)
void write_config()
Writes the server configuration WML back to disk.
void mark_dirty(const std::string &addon)
void send_message(const std::string &msg, const any_socket_ptr &sock)
Send a client an informational message.
optional_config get_addon(const std::string &id)
Retrieves an addon by id if found, or a null config otherwise.
void handle_request_campaign_hash(const request &)
void handle_flush(const boost::system::error_code &error)
std::string feedback_url_format_
void flush_cfg()
Starts timer to write config to disk every ten minutes.
std::unordered_map< std::string, config > addons_
The hash map of addons metadata.
std::unordered_set< std::string > dirty_addons_
The set of unique addon names with pending metadata updates.
bool ignore_address_stats(const std::string &addr) const
Checks if the specified address should never bump download counts.
static const std::size_t default_document_size_limit
Default upload size limit in bytes.
int compress_level_
Used for add-on archives.
boost::asio::basic_waitable_timer< std::chrono::steady_clock > flush_timer_
void handle_read_from_fifo(const boost::system::error_code &error, std::size_t bytes_transferred)
const std::string cfg_file_
void serve_requests(Socket socket, boost::asio::yield_context yield)
void fire(const std::string &hook, const std::string &addon)
Fires a hook script.
bool authenticate_forum(const config &addon, const std::string &passphrase, bool is_delete)
Check whether the provided passphrase matches the add-on and its author by checked against the forum ...
time_t update_pack_lifespan_
void handle_change_passphrase(const request &)
std::vector< std::string > stats_exempt_ips_
void handle_request_campaign(const request &)
void load_config()
Reads the server configuration from WML.
server(const std::string &cfg_file, unsigned short port=0)
void handle_sighup(const boost::system::error_code &error, int signal_number)
void register_handlers()
Registers client request handlers.
void handle_request_campaign_list(const request &)
Variant for storing WML attributes.
bool empty() const
Tests for an attribute that either was never set or was set to "".
Class for writing a config out to a file in pieces.
void write(const config &cfg)
A config object defines a single node in a WML file, with access to child nodes.
void copy_or_remove_attributes(const config &from, T... keys)
Copies or deletes attributes to match the source config.
void append(const config &cfg)
Append data from another config object to this one.
const config & child_or_empty(config_key_type key) const
Returns the first child with the given key, or an empty config if there is none.
config & mandatory_child(config_key_type key, int n=0)
Returns the nth child with the given key, or throws an error if there is none.
std::size_t child_count(config_key_type key) const
void clear_children(T... keys)
bool has_child(config_key_type key) const
Determine whether a config has a child or not.
child_itors child_range(config_key_type key)
void remove_children(config_key_type key, std::function< bool(const config &)> p=[](config){return true;})
Removes all children with tag key for which p returns true.
optional_config_impl< config > optional_child(config_key_type key, int n=0)
Equivalent to mandatory_child, but returns an empty optional if the nth child was not found.
config & add_child(config_key_type key)
Wrapper class that guarantees that file commit atomicity.
void commit()
Commits the new file contents to disk atomically.
scoped_ostream & ostream()
Returns the write stream associated with the file.
A class to handle the non-SQL logic for connecting to the phpbb forum database.
severity get_severity() const
Base class for implementing servers that use gzipped-WML network protocol.
std::string hash_password(const std::string &pw, const std::string &salt, const std::string &username)
Handles hashing the password provided by the player before comparing it to the hashed password in the...
void coro_send_file(socket_ptr socket, const std::string &filename, boost::asio::yield_context yield)
Send contents of entire file directly to socket from within a coroutine.
boost::asio::signal_set sighup_
boost::asio::streambuf admin_cmd_
std::unique_ptr< simple_wml::document > coro_receive_doc(SocketPtr socket, boost::asio::yield_context yield)
Receive WML document from a coroutine.
void async_send_doc_queued(SocketPtr socket, simple_wml::document &doc)
High level wrapper for sending a WML document.
void coro_send_doc(SocketPtr socket, simple_wml::document &doc, boost::asio::yield_context yield)
Send a WML document from within a coroutine.
boost::asio::io_service io_service_
boost::asio::posix::stream_descriptor input_
void load_tls_config(const config &cfg)
static std::size_t document_size_limit
node & add_child(const char *name)
node & set_attr_dup(const char *key, const char *value)
An interface class to handle nick registration To activate it put a [user_handler] section into the s...
Thrown by operations encountering invalid UTF-8 data.
virtual std::string hex_digest() const override
A simple wrapper class for optional reference types.
Represents version numbers.
std::string str() const
Serializes the version number into string form.
optional_config_impl< config > optional_config
Declarations for File-IO.
Atomic filesystem commit functions.
unsigned in
If equal to search_counter, the node is off the list.
Interfaces for manipulating version numbers of engine, add-ons, etc.
std::string label
What to show in the filter's drop-down list.
std::map< std::string, addon_info > addons_list
New lexcical_cast header.
Standard logging facilities (interface).
std::pair< std::string, std::string > generate_hash(const std::string &passphrase)
Generates a salted hash from the specified passphrase.
bool verify_passphrase(const std::string &passphrase, const std::string &salt, const std::string &hash)
Verifies the specified plain text passphrase against a salted hash.
std::ostream & operator<<(std::ostream &o, const server::request &r)
std::string format_addon_feedback_url(const std::string &format, const config ¶ms)
Format a feedback URL for an add-on.
void data_apply_addlist(config &data, const config &addlist)
void find_translations(const config &base_dir, config &addon)
Scans an add-on archive directory for translations.
bool data_apply_removelist(config &data, const config &removelist)
bool is_text_markup_char(char c)
std::map< version_info, config > get_version_map(config &addon)
std::string client_address(const any_socket_ptr &sock)
void add_license(config &cfg)
Adds a COPYING.txt file with the full text of the GNU GPL to an add-on.
filesystem::scoped_istream istream_file(const std::string &fname, bool treat_failure_as_error)
void get_files_in_dir(const std::string &dir, std::vector< std::string > *files, std::vector< std::string > *dirs, name_mode mode, filter_mode filter, reorder_mode reorder, file_tree_checksum *checksum)
Get a list of all files and/or directories in a given directory.
bool delete_file(const std::string &filename)
bool is_directory(const std::string &fname)
Returns true if the given file is a directory.
bool delete_directory(const std::string &dirname, const bool keep_pbl)
int file_size(const std::string &fname)
Returns the size of a file, or -1 if the file doesn't exist.
std::unique_ptr< std::istream > scoped_istream
bool set_cwd(const std::string &dir)
std::string normalize_path(const std::string &fpath, bool normalize_separators, bool resolve_dot_entries)
Returns the absolute path of a file.
const std::string revision
std::string list_log_domains(const std::string &filter)
void precise_timestamps(bool pt)
bool set_log_domain_severity(const std::string &name, severity severity)
std::string lowercase(const std::string &s)
Returns a lowercased version of the string.
std::string & erase(std::string &str, const std::size_t start, const std::size_t len)
Erases a portion of a UTF-8 string.
bool wildcard_string_match(const std::string &str, const std::string &match)
Match using '*' as any number of characters (including none), '+' as one or more characters,...
bool string_bool(const std::string &str, bool def)
Convert no, false, off, 0, 0.0 to false, empty to def, and others to true.
std::string join(const T &v, const std::string &s=",")
Generates a new string joining container items in a list.
std::vector< std::string > split(const config_attribute_value &val)
static void msg(const char *act, debug_info &i, const char *to="", const char *result="")
campaignd command line options parsing.
void read(config &cfg, std::istream &in, abstract_validator *validator)
void write(std::ostream &out, const configr_of &cfg, unsigned int level)
void read_gz(config &cfg, std::istream &file, abstract_validator *validator)
Might throw a std::ios_base::failure especially a gzip_error.
std::shared_ptr< boost::asio::ssl::stream< socket_ptr::element_type > > tls_socket_ptr
utils::variant< socket_ptr, tls_socket_ptr > any_socket_ptr
std::shared_ptr< boost::asio::ip::tcp::socket > socket_ptr
Client request information object.
const any_socket_ptr sock
boost::asio::yield_context yield
context of the coroutine the request is executed in async operations on sock can use it instead of a ...
An exception object used when an IO error occurs.
Reports time elapsed at the end of an object scope.
bool addon_name_legal(const std::string &name)
Checks whether an add-on id/name is legal or not.
void make_updatepack(config &pack, const config &from, const config &to)
&from, &to are the top directories of their structures; addlist/removelist tag is treated as [dir]
ADDON_TYPE get_addon_type(const std::string &str)
bool check_names_legal(const config &dir, std::vector< std::string > *badlist)
Scans an add-on archive for illegal names.
void write_hashlist(config &hashlist, const config &data)
const unsigned short default_campaignd_port
Default port number for the addon server.
std::string addon_check_status_desc(unsigned int code)
bool check_case_insensitive_duplicates(const config &dir, std::vector< std::string > *badlist)
Scans an add-on archive for case-conflicts.
@ SERVER_ADDONS_LIST
Corrupted server add-ons list.
@ UNAUTHORIZED
Authentication failed.
@ BAD_FEEDBACK_TOPIC_ID
The provided topic ID for the addon's feedback forum thread is invalid.
@ SERVER_DELTA_NO_VERSIONS
No versions to deltify against.
@ NO_TITLE
No title specified.
@ AUTH_TYPE_MISMATCH
The addon's forum_auth value does not match its previously set value.
@ ILLEGAL_FILENAME
Bad filename.
@ NO_PASSPHRASE
No passphrase specified.
@ UNEXPECTED_DELTA
Delta for a non-existent add-on.
@ VERSION_NOT_INCREMENTED
Version number is not an increment.
@ FILENAME_CASE_CONFLICT
Filename case conflict.
@ TITLE_HAS_MARKUP
Markup in add-on title.
@ SERVER_UNSPECIFIED
Unspecified server error.
@ BAD_TYPE
Bad add-on type.
@ NO_AUTHOR
No author specified.
@ NO_DESCRIPTION
No description specified.
@ NO_EMAIL
No email specified.
@ FEEDBACK_TOPIC_ID_NOT_FOUND
The provided topic ID for the addon's feedback forum thread wasn't found in the forum database.
@ INVALID_UTF8_NAME
Invalid UTF-8 sequence in add-on name.
@ USER_DOES_NOT_EXIST
Requested forum authentication for a user that doesn't exist on the forums.
@ INVALID_UTF8_ATTRIBUTE
Invalid UTF-8 sequence in add-on metadata.
@ BAD_NAME
Bad add-on name.
@ NAME_HAS_MARKUP
Markup in add-on name.
@ SERVER_FORUM_AUTH_DISABLED
The remote add-ons server does not support forum authorization.
@ SERVER_READ_ONLY
Server read-only mode on.
@ NO_VERSION
No version specified.