aboutsummaryrefslogtreecommitdiff
path: root/src/openvic-simulation/politics/Government.cpp
blob: 49e0f2f930ca6a40dceb5d54adc654618b9292cf (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#include "Government.hpp"

using namespace OpenVic;
using namespace OpenVic::NodeTools;

GovernmentType::GovernmentType(
   std::string_view new_identifier, std::vector<Ideology const*>&& new_ideologies, bool new_elections,
   bool new_appoint_ruling_party, Timespan new_term_duration, std::string_view new_flag_type_identifier
) : HasIdentifier { new_identifier }, ideologies { std::move(new_ideologies) }, elections { new_elections },
   appoint_ruling_party { new_appoint_ruling_party }, term_duration { new_term_duration },
   flag_type_identifier { new_flag_type_identifier } {}

bool GovernmentType::is_ideology_compatible(Ideology const* ideology) const {
   return std::find(ideologies.begin(), ideologies.end(), ideology) != ideologies.end();
}

bool GovernmentTypeManager::add_government_type(
   std::string_view identifier, std::vector<Ideology const*>&& ideologies, bool elections, bool appoint_ruling_party,
   Timespan term_duration, std::string_view flag_type
) {
   if (identifier.empty()) {
      Logger::error("Invalid government type identifier - empty!");
      return false;
   }

   if (ideologies.empty()) {
      Logger::error("No compatible ideologies defined for government type ", identifier);
      return false;
   }

   if (elections && term_duration < 0) {
      Logger::error("No or invalid term duration for government type ", identifier);
      return false;
   }

   const bool ret = government_types.add_item({
      identifier, std::move(ideologies), elections, appoint_ruling_party, term_duration, flag_type
   });

   /* flag_type can be empty here for default/non-ideological flag */
   if (ret && std::find(flag_types.begin(), flag_types.end(), flag_type) == flag_types.end()) {
      flag_types.emplace_back(flag_type);
   }

   return ret;
}

/* REQUIREMENTS: FS-525, SIM-27 */
bool GovernmentTypeManager::load_government_types_file(IdeologyManager const& ideology_manager, ast::NodeCPtr root) {
   bool ret = expect_dictionary_reserve_length(government_types,
      [this, &ideology_manager](std::string_view government_type_identifier, ast::NodeCPtr government_type_value) -> bool {
         std::vector<Ideology const*> ideologies;
         bool elections = false, appoint_ruling_party = false;
         Timespan term_duration = 0;
         std::string_view flag_type_identifier;

         size_t total_expected_ideologies = 0;
         bool ret = expect_dictionary_keys_and_default(
            increment_callback(total_expected_ideologies),
            "election", ONE_EXACTLY, expect_bool(assign_variable_callback(elections)),
            "duration", ZERO_OR_ONE, expect_months(assign_variable_callback(term_duration)),
            "appoint_ruling_party", ONE_EXACTLY, expect_bool(assign_variable_callback(appoint_ruling_party)),
            "flagType", ZERO_OR_ONE, expect_identifier(assign_variable_callback(flag_type_identifier))
         )(government_type_value);
         ideologies.reserve(total_expected_ideologies);

         ret &= expect_dictionary(
            [this, &ideology_manager, &ideologies, government_type_identifier](
               std::string_view key, ast::NodeCPtr value
            ) -> bool {
               static const string_set_t reserved_keys = { "election", "duration", "appoint_ruling_party", "flagType" };
               if (reserved_keys.contains(key)) {
                  return true;
               }
               Ideology const* ideology = ideology_manager.get_ideology_by_identifier(key);
               if (ideology == nullptr) {
                  Logger::error(
                     "When loading government type ", government_type_identifier, ", specified ideology ", key,
                     " is invalid!"
                  );
                  return false;
               }
               return expect_bool([&ideologies, ideology, government_type_identifier](bool val) -> bool {
                  if (val) {
                     if (std::find(ideologies.begin(), ideologies.end(), ideology) == ideologies.end()) {
                        ideologies.push_back(ideology);
                        return true;
                     }
                     Logger::error(
                        "Government type ", government_type_identifier, " marked as supporting ideology ",
                        ideology->get_identifier()
                     );
                     return false;
                  }
                  Logger::error(
                     "Government type ", government_type_identifier, " redundantly marked as not supporting ideology ",
                     ideology->get_identifier(), " multiple times"
                  );
                  return false;
               })(value);
            }
         )(government_type_value);

         ret &= add_government_type(
            government_type_identifier, std::move(ideologies), elections, appoint_ruling_party, term_duration,
            flag_type_identifier
         );
         return ret;
      }
   )(root);
   lock_government_types();

   return ret;
}

bool GovernmentTypeManager::is_valid_flag_type(std::string_view type) const {
   return std::any_of(flag_types.begin(), flag_types.end(), [type](std::string const& flag_type) -> bool {
      return flag_type == type;
   });
}