aboutsummaryrefslogtreecommitdiff
path: root/src/openvic-simulation/politics/Government.cpp
blob: 97b1d686e78269158bd912ab258a70f9a27d61ae (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
#include "Government.hpp"

#include "openvic-simulation/GameManager.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();
}

std::vector<Ideology const*> const& GovernmentType::get_ideologies() const {
   return ideologies;
}

bool GovernmentType::holds_elections() const {
   return elections;
}

bool GovernmentType::can_appoint_ruling_party() const {
   return appoint_ruling_party;
}

Timespan GovernmentType::get_term_duration() const {
   return term_duration;
}

std::string_view GovernmentType::get_flag_type() const {
   return flag_type_identifier;
}

GovernmentTypeManager::GovernmentTypeManager() : government_types { "government types" } {}

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;
   }

   return government_types.add_item({ identifier, std::move(ideologies), elections, appoint_ruling_party, term_duration, flag_type });
}

/* 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 = "republic";

         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.find(key) != reserved_keys.end()) 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;
}