The Battle for Wesnoth  1.19.10+dev
multiplayer.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2007 - 2025
3  by David White <dave@whitevine.net>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
17 
18 #include "build_info.hpp"
19 #include "commandline_options.hpp"
20 #include "connect_engine.hpp"
21 #include "events.hpp"
23 #include "formula/string_utils.hpp"
24 #include "game_config_manager.hpp"
26 #include "gettext.hpp"
28 #include "gui/dialogs/message.hpp"
34 #include "log.hpp"
35 #include "map_settings.hpp"
38 #include "replay.hpp"
39 #include "resources.hpp"
40 #include "saved_game.hpp"
41 #include "sound.hpp"
43 #include "wesnothd_connection.hpp"
44 
45 #include <functional>
46 #include "utils/optional_fwd.hpp"
47 #include <thread>
48 
49 static lg::log_domain log_mp("mp/main");
50 #define DBG_MP LOG_STREAM(debug, log_mp)
51 #define LOG_MP LOG_STREAM(info, log_mp)
52 #define WRN_MP LOG_STREAM(warn, log_mp)
53 #define ERR_MP LOG_STREAM(err, log_mp)
54 
55 namespace mp
56 {
57 namespace
58 {
59 /** Pointer to the current mp_manager instance. */
60 class mp_manager* manager = nullptr;
61 
62 /** The main controller of the MP workflow. */
63 class mp_manager
64 {
65 public:
66  // Declare this as a friend to allow direct access to enter_create_mode
67  friend void mp::start_local_game();
68  friend void mp::send_to_server(const config&);
70 
71  mp_manager(const utils::optional<std::string> host);
72 
73  ~mp_manager()
74  {
75  assert(manager);
76  manager = nullptr;
77 
78  if(network_worker.joinable()) {
79  stop = true;
80  network_worker.join();
81  }
82  }
83 
84  /**
85  * Enters the mp loop. It consists of four screens:
86  *
87  * Host POV: LOBBY <---> CREATE GAME ---> STAGING -----> GAME BEGINS
88  * Player POV: LOBBY <--------------------> JOIN GAME ---> GAME BEGINS
89  */
90  void run_lobby_loop();
91 
92  bool post_scenario_staging(ng::connect_engine& engine);
93 
94  bool post_scenario_wait(bool observe);
95 
96 private:
97  /** Represents the contents of the [join_lobby] response. */
98  struct session_metadata
99  {
100  session_metadata() = default;
101 
102  session_metadata(const config& cfg)
103  : is_moderator(cfg["is_moderator"].to_bool(false))
104  , profile_url_prefix(cfg["profile_url_prefix"].str())
105  {
106  }
107 
108  /** Whether you are logged in as a server moderator. */
109  bool is_moderator = false;
110 
111  /** The external URL prefix for player profiles (empty if the server doesn't have an attached database). */
112  std::string profile_url_prefix;
113  };
114 
115  /** Opens a new server connection and prompts the client for login credentials, if necessary. */
116  std::unique_ptr<wesnothd_connection> open_connection(const std::string& host);
117 
118  /** Opens the MP lobby. */
119  bool enter_lobby_mode();
120 
121  /**
122  * Opens the MP Create screen for hosts to configure a new game.
123  * @param preset_scenario contains a scenario id if present
124  */
125  void enter_create_mode(utils::optional<std::string> preset_scenario = utils::nullopt);
126 
127  /** Opens the MP Staging screen for hosts to wait for players. */
128  void enter_staging_mode(bool preset);
129 
130  /** Opens the MP Join Game screen for non-host players and observers. */
131  void enter_wait_mode(int game_id, bool observe);
132 
133  /** Worker thread to handle receiving and processing network data. */
134  std::thread network_worker;
135 
136  /** Flag to signal the worker thread terminate. */
137  std::atomic_bool stop;
138 
139  /** The connection to the server. */
140  std::unique_ptr<wesnothd_connection> connection;
141 
142  /** The current session's info sent by the server on login. */
143  session_metadata session_info;
144 
145  /** This single instance is reused for all games played during the current connection to the server. */
146  saved_game state;
147 
148  mp::lobby_info lobby_info;
149 
150  std::list<mp::network_registrar::handler> process_handlers;
151 
152 public:
153  const session_metadata& get_session_info() const
154  {
155  return session_info;
156  }
157 
158  auto add_network_handler(const decltype(process_handlers)::value_type& func)
159  {
160  return [this, iter = process_handlers.insert(process_handlers.end(), func)]() { process_handlers.erase(iter); };
161  }
162 };
163 
164 mp_manager::mp_manager(const utils::optional<std::string> host)
165  : network_worker()
166  , stop(false)
167  , connection(nullptr)
168  , session_info()
169  , state()
170  , lobby_info()
171  , process_handlers()
172 {
173  state.classification().type = campaign_type::type::multiplayer;
174 
175  if(host) {
177  connection = open_connection(*host);
178 
179  // If for whatever reason our connection is null at this point (dismissing the password prompt, for
180  // instance), treat it as a normal condition and exit. Any actual error conditions throw exceptions
181  // which can be handled higher up the stack.
182  if(connection == nullptr) {
183  return;
184  }
185 
187 
188  config data;
189 
190  while(!stop) {
191  connection->wait_and_receive_data(data);
192 
193  if(const auto error = data.optional_child("error")) {
194  throw wesnothd_error((*error)["message"]);
195  }
196 
197  else if(data.has_child("gamelist")) {
198  this->lobby_info.process_gamelist(data);
199  break;
200  }
201 
202  else if(const auto gamelist_diff = data.optional_child("gamelist_diff")) {
203  this->lobby_info.process_gamelist_diff(*gamelist_diff);
204  }
205 
206  else {
207  // No special actions to take. Pass the data on to the network handlers.
208  for(const auto& handler : process_handlers) {
209  handler(data);
210  }
211  }
212  }
213  });
214  }
215 
216  // Avoid setting this until the connection has been fully established. open_connection may throw,
217  // in which case we don't want to point to an object instance that has not properly connected.
218  assert(!manager);
219  manager = this;
220 }
221 
222 std::unique_ptr<wesnothd_connection> mp_manager::open_connection(const std::string& host)
223 {
224  DBG_MP << "opening connection";
225 
226  if(host.empty()) {
227  return nullptr;
228  }
229 
230  // shown_hosts is used to prevent the client being locked in a redirect loop.
231  std::set<std::pair<std::string, std::string>> shown_hosts;
232  auto addr = shown_hosts.end();
233 
234  try {
235  std::tie(addr, std::ignore) = shown_hosts.insert(parse_network_address(host, "15000"));
236  } catch(const std::runtime_error&) {
237  throw wesnothd_error(_("Invalid address specified for multiplayer server"));
238  }
239 
240  // Start stage
242 
243  // Initializes the connection to the server.
244  auto conn = std::make_unique<wesnothd_connection>(addr->first, addr->second);
245 
246  // First, spin until we get a handshake from the server.
247  conn->wait_for_handshake();
248 
250 
251  config data;
252 
253  // Then, log in and wait for the lobby/game join prompt.
254  while(true) {
255  data.clear();
256  conn->wait_and_receive_data(data);
257 
258  if(const auto reject = data.optional_child("reject"); reject || data.has_attribute("version")) {
259  std::string version;
260 
261  if(reject) {
262  version = (*reject)["accepted_versions"].str();
263  } else {
264  // Backwards-compatibility "version" attribute
265  version = data["version"].str();
266  }
267 
268  utils::string_map i18n_symbols;
269  i18n_symbols["required_version"] = version;
270  i18n_symbols["your_version"] = game_config::wesnoth_version.str();
271 
272  const std::string errorstring = VGETTEXT("The server accepts versions ‘$required_version’, but you are using version ‘$your_version’", i18n_symbols);
273  throw wesnothd_error(errorstring);
274  }
275 
276  // Check for "redirect" messages
277  if(const auto redirect = data.optional_child("redirect")) {
278  auto redirect_host = (*redirect)["host"].str();
279  auto redirect_port = (*redirect)["port"].str("15000");
280 
281  bool recorded_host;
282  std::tie(std::ignore, recorded_host) = shown_hosts.emplace(redirect_host, redirect_port);
283 
284  if(!recorded_host) {
285  throw wesnothd_error(_("Server-side redirect loop"));
286  }
287 
289 
290  // Open a new connection with the new host and port.
291  conn.reset();
292  conn = std::make_unique<wesnothd_connection>(redirect_host, redirect_port);
293 
294  // Wait for new handshake.
295  conn->wait_for_handshake();
296 
298  continue;
299  }
300 
301  if(data.has_child("version")) {
302  config res;
303  config& cfg = res.add_child("version");
304  cfg["version"] = game_config::wesnoth_version.str();
305  cfg["client_source"] = game_config::dist_channel_id();
306  conn->send_data(res);
307  }
308 
309  if(const auto error = data.optional_child("error")) {
310  throw wesnothd_rejected_client_error((*error)["message"].str());
311  }
312 
313  // Continue if we did not get a direction to login
314  if(!data.has_child("mustlogin")) {
315  continue;
316  }
317 
318  // Enter login loop
319  while(true) {
320  std::string login = prefs::get().login();
321 
322  config response;
323  config& sp = response.add_child("login");
324  sp["username"] = login;
325 
326  conn->send_data(response);
327  conn->wait_and_receive_data(data);
328 
330 
331  if(const auto warning = data.optional_child("warning")) {
332  std::string warning_msg;
333 
334  if((*warning)["warning_code"] == MP_NAME_INACTIVE_WARNING) {
335  warning_msg = VGETTEXT("The nickname ‘$nick’ is inactive. "
336  "You cannot claim ownership of this nickname until you "
337  "activate your account via email or ask an "
338  "administrator to do it for you.", {{"nick", login}});
339  } else {
340  warning_msg = (*warning)["message"].str();
341  }
342 
343  warning_msg += "\n\n";
344  warning_msg += _("Do you want to continue?");
345 
347  return nullptr;
348  } else {
349  continue;
350  }
351  }
352 
353  auto error = data.optional_child("error");
354 
355  // ... and get us out of here if the server did not complain
356  if(!error) break;
357 
358  do {
359  std::string password = prefs::get().password(host, login);
360 
361  const bool fall_through = (*error)["force_confirmation"].to_bool()
363  : false;
364 
365  // If:
366  // * the server asked for a password
367  // * the password isn't empty
368  // * the user didn't press Cancel
369  // * the connection is secure or the client was started with the option to use insecure connections
370  // send the password to the server
371  // otherwise go directly to the username/password dialog
372  if(!(*error)["password_request"].empty() && !password.empty() && !fall_through && (conn->using_tls() || game_config::allow_insecure)) {
373  // the possible cases here are that either:
374  // 1) TLS encryption is enabled, thus sending the plaintext password is still secure
375  // 2) TLS encryption is not enabled, in which case the server should not be requesting a password in the first place
376  // 3) This is being used for local testing/development, so using an insecure connection is enabled manually
377 
378  sp["password"] = password;
379 
380  // Once again send our request...
381  conn->send_data(response);
382  conn->wait_and_receive_data(data);
383 
385 
386  error = data.optional_child("error");
387 
388  // ... and get us out of here if the server is happy now
389  if(!error) break;
390  }
391 
392  // Providing a password either was not attempted because we did not
393  // have any or failed:
394  // Now show a dialog that displays the error and allows to
395  // enter a new user name and/or password
396 
397  std::string error_message;
398  utils::string_map i18n_symbols;
399  i18n_symbols["nick"] = login;
400 
401  const auto extra_data = error->optional_child("data");
402  if(extra_data) {
403  using namespace std::chrono_literals;
404  i18n_symbols["duration"] = utils::format_timespan(chrono::parse_duration((*extra_data)["duration"], 0s));
405  }
406 
407  const std::string ec = (*error)["error_code"];
408 
409  if(!(*error)["password_request"].empty() && !conn->using_tls() && !game_config::allow_insecure) {
410  error_message = _("The remote server requested a password while using an insecure connection.");
411  } else if(ec == MP_MUST_LOGIN) {
412  error_message = _("You must login first.");
413  } else if(ec == MP_NAME_TAKEN_ERROR) {
414  error_message = VGETTEXT("The nickname ‘$nick’ is already taken.", i18n_symbols);
415  } else if(ec == MP_INVALID_CHARS_IN_NAME_ERROR) {
416  error_message = VGETTEXT("The nickname ‘$nick’ contains invalid "
417  "characters. Only alpha-numeric characters (one at minimum), underscores and "
418  "hyphens are allowed.", i18n_symbols);
419  } else if(ec == MP_NAME_TOO_LONG_ERROR) {
420  error_message = VGETTEXT("The nickname ‘$nick’ is too long. Nicks must be 20 characters or less.", i18n_symbols);
421  } else if(ec == MP_NAME_RESERVED_ERROR) {
422  error_message = VGETTEXT("The nickname ‘$nick’ is reserved and cannot be used by players.", i18n_symbols);
423  } else if(ec == MP_NAME_UNREGISTERED_ERROR) {
424  error_message = VGETTEXT("The nickname ‘$nick’ is not registered on this server.", i18n_symbols)
425  + _(" This server disallows unregistered nicknames.");
426  } else if(ec == MP_SERVER_IP_BAN_ERROR) {
427  if(extra_data) {
428  error_message = VGETTEXT("Your IP address is banned on this server for $duration|.", i18n_symbols);
429  } else {
430  error_message = _("Your IP address is banned on this server.");
431  }
432  } else if(ec == MP_NAME_AUTH_BAN_USER_ERROR) {
433  if(extra_data) {
434  error_message = VGETTEXT("The nickname ‘$nick’ is banned on this server’s forums for $duration|.", i18n_symbols);
435  } else {
436  error_message = VGETTEXT("The nickname ‘$nick’ is banned on this server’s forums.", i18n_symbols);
437  }
438  } else if(ec == MP_NAME_AUTH_BAN_IP_ERROR) {
439  if(extra_data) {
440  error_message = VGETTEXT("Your IP address is banned on this server’s forums for $duration|.", i18n_symbols);
441  } else {
442  error_message = _("Your IP address is banned on this server’s forums.");
443  }
444  } else if(ec == MP_NAME_AUTH_BAN_EMAIL_ERROR) {
445  if(extra_data) {
446  error_message = VGETTEXT("The email address for the nickname ‘$nick’ is banned on this server’s forums for $duration|.", i18n_symbols);
447  } else {
448  error_message = VGETTEXT("The email address for the nickname ‘$nick’ is banned on this server’s forums.", i18n_symbols);
449  }
450  } else if(ec == MP_PASSWORD_REQUEST) {
451  error_message = VGETTEXT("The nickname ‘$nick’ is registered on this server.", i18n_symbols);
452  } else if(ec == MP_PASSWORD_REQUEST_FOR_LOGGED_IN_NAME) {
453  error_message = VGETTEXT("The nickname ‘$nick’ is registered on this server.", i18n_symbols)
454  + "\n\n" + _("WARNING: There is already a client using this nickname, "
455  "logging in will cause that client to be kicked!");
456  } else if(ec == MP_INCORRECT_PASSWORD_ERROR) {
457  error_message = _("The password you provided was incorrect.");
458  } else if(ec == MP_TOO_MANY_ATTEMPTS_ERROR) {
459  error_message = _("You have made too many login attempts.");
460  } else if(ec == MP_HASHING_PASSWORD_FAILED) {
461  error_message = _("Password hashing failed.");
462  } else {
463  error_message = (*error)["message"].str();
464  }
465 
466  gui2::dialogs::mp_login dlg(host, error_message, !((*error)["password_request"].empty()));
467 
468  // Need to show the dialog from the main thread or it won't appear.
469  events::call_in_main_thread([&dlg]() { dlg.show(); });
470 
471  switch(dlg.get_retval()) {
472  // Log in with password
473  case gui2::retval::OK:
474  break;
475  // Cancel
476  default:
477  return nullptr;
478  }
479 
480  // If we have got a new username we have to start all over again
481  } while(login == prefs::get().login());
482 
483  // Somewhat hacky...
484  // If we broke out of the do-while loop above error is still going to be nullopt
485  if(!error) break;
486  } // end login loop
487 
488  if(const auto join_lobby = data.optional_child("join_lobby")) {
489  // Note any session data sent with the response. This should be the only place session_info is set.
490  session_info = { join_lobby.value() };
491 
492  // All done!
493  break;
494  }
495  }
496 
497  return conn;
498 }
499 
500 void mp_manager::run_lobby_loop()
501 {
502  // This should only work if we have a connection. If we're in a local mode,
503  // enter_create_mode should be accessed directly.
504  if(!connection) {
505  return;
506  }
507 
508  // A return of false means a config reload was requested, so do that and then loop.
509  while(!enter_lobby_mode()) {
512  gcm->load_game_config_for_create(true); // NOTE: Using reload_changed_game_config only doesn't seem to work here
513 
514  lobby_info.refresh_installed_addons_cache();
515 
516  connection->send_data(config("refresh_lobby"));
517  }
518 }
519 
520 bool mp_manager::enter_lobby_mode()
521 {
522  DBG_MP << "entering lobby mode";
523 
524  // Connection should never be null in the lobby.
525  assert(connection);
526 
527  // We use a loop here to allow returning to the lobby if you, say, cancel game creation.
528  while(true) {
529  if(auto cfg = game_config_manager::get()->game_config().optional_child("lobby_music")) {
530  for(const config& i : cfg->child_range("music")) {
532  }
533 
535  } else {
538  }
539 
540  int dlg_retval = 0;
541  int dlg_joined_game_id = 0;
542  std::string preset_scenario = "";
543  {
544  gui2::dialogs::mp_lobby dlg(lobby_info, *connection, dlg_joined_game_id);
545  dlg.show();
546  dlg_retval = dlg.get_retval();
547  preset_scenario = dlg.queue_game_scenario_id();
548  }
549 
550  try {
551  switch(dlg_retval) {
553  enter_create_mode(utils::make_optional(preset_scenario));
554  break;
556  enter_create_mode();
557  break;
559  [[fallthrough]];
561  enter_wait_mode(dlg_joined_game_id, dlg_retval == gui2::dialogs::mp_lobby::OBSERVE);
562  break;
564  // Let this function's caller reload the config and re-call.
565  return false;
566  default:
567  // Needed to handle the Quit signal and exit the loop
568  return true;
569  }
570  } catch(const config::error& error) {
571  if(!error.message.empty()) {
573  }
574 
575  // Update lobby content
576  connection->send_data(config("refresh_lobby"));
577  }
578  }
579 
580  return true;
581 }
582 
583 void mp_manager::enter_create_mode(utils::optional<std::string> preset_scenario)
584 {
585  DBG_MP << "entering create mode";
586 
587  if(preset_scenario) {
588  for(const config& game : game_config_manager::get()->game_config().mandatory_child("game_presets").child_range("game")) {
589  if(game["scenario"].str() == preset_scenario.value()) {
591  enter_staging_mode(true);
592  break;
593  }
594  }
595  } else if(gui2::dialogs::mp_create_game::execute(state, connection == nullptr)) {
596  enter_staging_mode(false);
597  } else if(connection) {
598  connection->send_data(config("refresh_lobby"));
599  }
600 }
601 
602 void mp_manager::enter_staging_mode(bool preset)
603 {
604  DBG_MP << "entering connect mode";
605 
606  std::unique_ptr<mp_game_metadata> metadata;
607 
608  // If we have a connection, set the appropriate info. No connection means we're in local game mode.
609  if(connection) {
610  metadata = std::make_unique<mp_game_metadata>(*connection);
611  metadata->connected_players.insert(prefs::get().login());
612  metadata->is_host = true;
613  metadata->is_queue_game = preset;
614  }
615 
616  bool dlg_ok = false;
617  {
618  ng::connect_engine connect_engine(state, true, metadata.get());
619  dlg_ok = gui2::dialogs::mp_staging::execute(connect_engine, connection.get());
620  } // end connect_engine
621 
622  if(dlg_ok) {
624  controller.set_mp_info(metadata.get());
625  controller.play_game();
626  }
627 
628  if(connection) {
629  connection->send_data(config("leave_game"));
630  }
631 }
632 
633 void mp_manager::enter_wait_mode(int game_id, bool observe)
634 {
635  DBG_MP << "entering wait mode";
636 
637  // The connection should never be null here, since one should never reach this screen in local game mode.
638  assert(connection);
639 
640  mp_game_metadata metadata(*connection);
641  metadata.is_host = false;
642 
643  if(const mp::game_info* gi = lobby_info.get_game_by_id(game_id)) {
644  metadata.current_turn = gi->current_turn;
645  }
646 
647  if(prefs::get().skip_mp_replay() || prefs::get().blindfold_replay()) {
648  metadata.skip_replay = true;
649  metadata.skip_replay_blindfolded = prefs::get().blindfold_replay();
650  }
651 
652  bool dlg_ok = false;
653  {
654  gui2::dialogs::mp_join_game dlg(state, *connection, true, observe);
655 
656  if(!dlg.fetch_game_config()) {
657  connection->send_data(config("leave_game"));
658  return;
659  }
660 
661  dlg_ok = dlg.show();
662  }
663 
664  if(dlg_ok) {
666  controller.set_mp_info(&metadata);
667  controller.play_game();
668  }
669 
670  connection->send_data(config("leave_game"));
671 }
672 
673 bool mp_manager::post_scenario_staging(ng::connect_engine& engine)
674 {
675  return gui2::dialogs::mp_staging::execute(engine, connection.get());
676 }
677 
678 bool mp_manager::post_scenario_wait(bool observe)
679 {
680  gui2::dialogs::mp_join_game dlg(state, *connection, false, observe);
681 
682  if(!dlg.fetch_game_config()) {
683  connection->send_data(config("leave_game"));
684  return false;
685  }
686 
687  if(dlg.started()) {
688  return true;
689  }
690 
691  return dlg.show();
692 }
693 
694 } // end anon namespace
695 
696 /** Pubic entry points for the MP workflow */
697 
698 void start_client(const std::string& host)
699 {
700  DBG_MP << "starting client";
701  mp_manager(host).run_lobby_loop();
702 }
703 
705 {
706  DBG_MP << "starting local game";
707 
709 
710  mp_manager(utils::nullopt).enter_create_mode();
711 }
712 
714 {
715  DBG_MP << "starting local MP game from commandline";
716 
718 
719  // The setup is done equivalently to lobby MP games using as much of existing
720  // code as possible. This means that some things are set up that are not
721  // needed in commandline mode, but they are required by the functions called.
723 
724  DBG_MP << "entering create mode";
725 
726  // Set the default parameters
727  saved_game state;
728  state.classification().type = campaign_type::type::multiplayer;
729 
730  mp_game_settings& parameters = state.mp_settings();
731 
732  // Hardcoded default values
733  state.classification().era_id = "era_default";
734  parameters.name = "multiplayer_The_Freelands";
735 
736  // Default values for which at getter function exists
737  parameters.num_turns = settings::get_turns("");
738  parameters.village_gold = settings::get_village_gold("");
740  parameters.xp_modifier = settings::get_xp_modifier("");
741 
742  // Do not use map settings if --ignore-map-settings commandline option is set
743  if(cmdline_opts.multiplayer_ignore_map_settings) {
744  DBG_MP << "ignoring map settings";
745  parameters.use_map_settings = false;
746  } else {
747  parameters.use_map_settings = true;
748  }
749 
750  // None of the other parameters need to be set, as their creation values above are good enough for CL mode.
751  // In particular, we do not want to use the preferences values.
752 
753  state.classification().type = campaign_type::type::multiplayer;
754 
755  // [era] define.
756  if(cmdline_opts.multiplayer_era) {
757  state.classification().era_id = *cmdline_opts.multiplayer_era;
758  }
759 
760  if(auto cfg_era = game_config.find_child("era", "id", state.classification().era_id)) {
761  state.classification().era_define = cfg_era["define"].str();
762  } else {
763  PLAIN_LOG << "Could not find era '" << state.classification().era_id << "'";
764  return;
765  }
766 
767  // [multiplayer] define.
768  if(cmdline_opts.multiplayer_scenario) {
769  parameters.name = *cmdline_opts.multiplayer_scenario;
770  }
771 
772  if(auto cfg_multiplayer = game_config.find_child("multiplayer", "id", parameters.name)) {
773  state.classification().scenario_define = cfg_multiplayer["define"].str();
774  } else {
775  PLAIN_LOG << "Could not find [multiplayer] '" << parameters.name << "'";
776  return;
777  }
778 
780  config {"next_scenario", parameters.name}
781  );
782 
784 
785  state.expand_random_scenario();
786  state.expand_mp_events();
787  state.expand_mp_options();
788 
789  // Should number of turns be determined from scenario data?
790  if(parameters.use_map_settings && state.get_starting_point().has_attribute("turns")) {
791  DBG_MP << "setting turns from scenario data: " << state.get_starting_point()["turns"];
792  parameters.num_turns = state.get_starting_point()["turns"].to_int();
793  }
794 
795  DBG_MP << "entering connect mode";
796 
797  {
798  ng::connect_engine connect_engine(state, true, nullptr);
799 
800  // Update the parameters to reflect game start conditions
801  connect_engine.start_game_commandline(cmdline_opts, game_config);
802  }
803 
804  if(resources::recorder && cmdline_opts.multiplayer_label) {
805  std::string label = *cmdline_opts.multiplayer_label;
806  resources::recorder->add_log_data("ai_log","ai_label",label);
807  }
808 
809  unsigned int repeat = (cmdline_opts.multiplayer_repeat) ? *cmdline_opts.multiplayer_repeat : 1;
810  for(unsigned int i = 0; i < repeat; i++){
811  saved_game state_copy(state);
812  campaign_controller controller(state_copy);
813  controller.play_game();
814  }
815 }
816 
818 {
819  return manager && manager->post_scenario_staging(engine);
820 }
821 
822 bool goto_mp_wait(bool observe)
823 {
824  return manager && manager->post_scenario_wait(observe);
825 }
826 
828 {
829  return manager && manager->get_session_info().is_moderator;
830 }
831 
832 std::string get_profile_link(int user_id)
833 {
834  if(manager) {
835  const std::string& prefix = manager->get_session_info().profile_url_prefix;
836 
837  if(!prefix.empty()) {
838  return prefix + std::to_string(user_id);
839  }
840  }
841 
842  return "";
843 }
844 
846 {
847  if(manager && manager->connection) {
848  manager->connection->send_data(data);
849  }
850 }
851 
853 {
854  if(manager /*&& manager->connection*/) {
855  remove_handler = manager->add_network_handler(func);
856  }
857 }
858 
860 {
861  if(remove_handler) {
862  remove_handler();
863  }
864 }
865 
867 {
868  return manager ? &manager->lobby_info : nullptr;
869 }
870 
871 } // end namespace mp
utils::optional< std::string > multiplayer_scenario
Non-empty if –scenario was given on the command line.
utils::optional< unsigned int > multiplayer_repeat
Repeats specified by –multiplayer-repeat option.
utils::optional< std::string > multiplayer_label
Non-empty if –label was given on the command line.
utils::optional< std::string > multiplayer_era
Non-empty if –era was given on the command line.
bool multiplayer_ignore_map_settings
True if –ignore-map-settings was given at the command line.
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
bool has_attribute(config_key_type key) const
Definition: config.cpp:157
config & add_child(config_key_type key)
Definition: config.cpp:436
std::string scenario_define
If there is a define the scenario uses to customize data.
std::string era_define
If there is a define the era uses to customize data.
campaign_type::type type
static game_config_manager * get()
void load_game_config_for_game(const game_classification &classification, const std::string &scenario_id)
void load_game_config_for_create(bool is_mp, bool is_test=false)
const game_config_view & game_config() const
A class grating read only view to a vector of config objects, viewed as one config with all children ...
static void progress(loading_stage stage=loading_stage::none)
Report what is being loaded to the loading screen.
static void display(const std::function< void()> &f)
@ yes_no_buttons
Shows a yes and no button.
Definition: message.hpp:81
@ ok_cancel_buttons
Shows an ok and cancel button.
Definition: message.hpp:77
bool show(const unsigned auto_close_time=0)
Shows the window.
static void quick_mp_setup(saved_game &state, const config presets)
presets needs to be a copy! Otherwise you'll get segfaults when clicking the Join button since it res...
const std::string queue_game_scenario_id() const
Definition: lobby.hpp:69
@ RELOAD_CONFIG
player clicked the Create button
Definition: lobby.hpp:65
@ CREATE_PRESET
player clicked Join button on an [mp_queue] game, but there was no existing game to join
Definition: lobby.hpp:66
int get_retval()
Definition: window.hpp:402
This class represents the collective information the client has about the players and games on the se...
Definition: lobby_info.hpp:31
network_registrar(const handler &func)
std::function< void()> remove_handler
Definition: multiplayer.hpp:87
std::function< void(const config &)> handler
Definition: multiplayer.hpp:81
void start_game_commandline(const commandline_options &cmdline_opts, const game_config_view &game_config)
static prefs & get()
void set_message_private(bool value)
std::string login()
std::string password(const std::string &server, const std::string &login)
void add_log_data(const std::string &key, const std::string &var)
Definition: replay.cpp:299
game_classification & classification()
Definition: saved_game.hpp:56
void expand_mp_options()
adds values of [option]s into [carryover_sides_start][variables] so that they are applied in the next...
Definition: saved_game.cpp:449
void set_carryover_sides_start(config carryover_sides_start)
Definition: saved_game.cpp:165
std::string get_scenario_id() const
Definition: saved_game.cpp:698
mp_game_settings & mp_settings()
Multiplayer parameters for this game.
Definition: saved_game.hpp:60
config & get_starting_point()
Definition: saved_game.cpp:631
void expand_mp_events()
adds [event]s from [era] and [modification] into this scenario does NOT expand [option]s because vari...
Definition: saved_game.cpp:408
void expand_random_scenario()
takes care of generate_map=, generate_scenario=, map= attributes This should be called before expandi...
Definition: saved_game.cpp:524
std::string str() const
Serializes the version number into string form.
#define VGETTEXT(msgid,...)
Handy wrappers around interpolate_variables_into_string and gettext.
std::size_t i
Definition: function.cpp:1022
static std::string _(const char *str)
Definition: gettext.hpp:103
std::string label
What to show in the filter's drop-down list.
Definition: manager.cpp:201
Standard logging facilities (interface).
#define PLAIN_LOG
Definition: log.hpp:296
General settings and defaults for scenarios.
static lg::log_domain log_mp("mp/main")
#define DBG_MP
Definition: multiplayer.cpp:50
Define the errors the server may send during the login procedure.
#define MP_INCORRECT_PASSWORD_ERROR
#define MP_NAME_AUTH_BAN_USER_ERROR
#define MP_MUST_LOGIN
#define MP_NAME_RESERVED_ERROR
#define MP_PASSWORD_REQUEST_FOR_LOGGED_IN_NAME
#define MP_NAME_AUTH_BAN_EMAIL_ERROR
#define MP_TOO_MANY_ATTEMPTS_ERROR
#define MP_HASHING_PASSWORD_FAILED
#define MP_SERVER_IP_BAN_ERROR
#define MP_NAME_TOO_LONG_ERROR
#define MP_NAME_AUTH_BAN_IP_ERROR
#define MP_PASSWORD_REQUEST
#define MP_NAME_INACTIVE_WARNING
#define MP_NAME_UNREGISTERED_ERROR
#define MP_INVALID_CHARS_IN_NAME_ERROR
#define MP_NAME_TAKEN_ERROR
auto parse_duration(const config_attribute_value &val, const Duration &def=Duration{0})
Definition: chrono.hpp:71
void call_in_main_thread(const std::function< void(void)> &f)
Definition: events.cpp:777
Game configuration data as global variables.
Definition: build_info.cpp:61
const version_info wesnoth_version(VERSION)
bool allow_insecure
Definition: game_config.cpp:78
std::string dist_channel_id()
Return the distribution channel identifier, or "Default" if missing.
Definition: build_info.cpp:396
void show_error_message(const std::string &msg, bool message_use_markup)
Shows an error message to the user.
Definition: message.cpp:201
void show_message(const std::string &title, const std::string &msg, const std::string &button_caption, const bool auto_close, const bool message_use_markup, const bool title_use_markup)
Shows a message to the user.
Definition: message.cpp:148
@ OK
Dialog was closed with the OK button.
Definition: retval.hpp:35
@ CANCEL
Dialog was closed with the CANCEL button.
Definition: retval.hpp:38
Main entry points of multiplayer mode.
Definition: lobby_data.cpp:49
lobby_info * get_lobby_info()
Returns the lobby_info object for the given session.
void send_to_server(const config &data)
Attempts to send given data to server if a connection is open.
void start_local_game()
Starts a multiplayer game in single-user mode.
bool goto_mp_staging(ng::connect_engine &engine)
Opens the MP Staging screen and sets the game state according to the changes made.
void start_local_game_commandline(const commandline_options &cmdline_opts)
Starts a multiplayer game in single-user mode using command line settings.
std::string get_profile_link(int user_id)
Gets the forum profile link for the given user.
void start_client(const std::string &host)
Pubic entry points for the MP workflow.
bool logged_in_as_moderator()
Gets whether the currently logged-in user is a moderator.
bool goto_mp_wait(bool observe)
Opens the MP Join Game screen and sets the game state according to the changes made.
replay * recorder
Definition: resources.cpp:28
int get_village_support(const std::string &value)
Gets the village unit level support.
int get_turns(const std::string &value)
Gets the number of turns.
int get_xp_modifier(const std::string &value)
Gets the xp modifier.
int get_village_gold(const std::string &value, const game_classification *classification)
Gets the village gold.
void empty_playlist()
Definition: sound.cpp:613
void play_music_config(const config &music_node, bool allow_interrupt_current_track, int i)
Definition: sound.cpp:716
void stop_music()
Definition: sound.cpp:558
void commit_music_changes()
Definition: sound.cpp:843
static std::string format_timespan(const std::chrono::duration< Rep, Period > &span, bool detailed=false)
Formats a timespan into human-readable text for player authentication functions.
std::map< std::string, t_string > string_map
std::pair< std::string, std::string > parse_network_address(const std::string &address, const std::string &default_port)
Parse a host:port style network address, supporting [] notation for ipv6 addresses.
std::string_view data
Definition: picture.cpp:178
Replay control code.
std::string message
Definition: exceptions.hpp:30
This class represents the info a client has about a game on the server.
Definition: lobby_data.hpp:62
unsigned current_turn
bool skip_replay_blindfolded
An error occurred during when trying to communicate with the wesnothd server.
Error used when the client is rejected by the MP server.
static map_location::direction s