osb/source/game/StarTenantDatabase.cpp

104 lines
3.6 KiB
C++
Raw Normal View History

2023-06-20 14:33:09 +10:00
#include "StarTenantDatabase.hpp"
#include "StarJsonExtra.hpp"
namespace Star {
bool Tenant::criteriaSatisfied(StringMap<unsigned> const& colonyTags) const {
// Check whether colonyTags is a supermultiset of colonyTagCriteria
for (pair<String, unsigned> const& entry : colonyTagCriteria.pairs()) {
if (colonyTags.value(entry.first, 0) < entry.second)
return false;
}
return true;
}
TenantDatabase::TenantDatabase() {
auto assets = Root::singleton().assets();
2024-03-15 21:28:11 +11:00
auto& files = assets->scanExtension("tenant");
2023-06-20 14:33:09 +10:00
assets->queueJsons(files);
2024-03-15 21:28:11 +11:00
for (auto& file : files) {
2023-06-20 14:33:09 +10:00
try {
String name = assets->json(file).getString("name");
if (m_paths.contains(name))
2023-06-27 20:23:44 +10:00
Logger::error("Tenant {} defined twice, second time from {}", name, file);
2023-06-20 14:33:09 +10:00
else
m_paths[name] = file;
} catch (std::exception const& e) {
2023-06-27 20:23:44 +10:00
Logger::error("Error loading tenant file {}: {}", file, outputException(e, true));
2023-06-20 14:33:09 +10:00
}
}
}
void TenantDatabase::cleanup() {
MutexLocker locker(m_cacheMutex);
m_tenantCache.cleanup();
}
TenantPtr TenantDatabase::getTenant(String const& name) const {
MutexLocker locker(m_cacheMutex);
return m_tenantCache.get(name,
[this](String const& name) -> TenantPtr {
if (auto path = m_paths.maybe(name))
return readTenant(*path);
2023-06-27 20:23:44 +10:00
throw TenantException::format("No such tenant named '{}'", name);
2023-06-20 14:33:09 +10:00
});
}
List<TenantPtr> TenantDatabase::getMatchingTenants(StringMap<unsigned> const& colonyTags) const {
// This implementation loops over every tenant. Smarter implementations could
// be written if it becomes a bottleneck, depending on how colonyTags end up
// being
// used, how many there are, how many tenants have similar criteria, etc.
List<TenantPtr> matchingTenants;
for (String const& name : m_paths.keys()) {
TenantPtr tenant = getTenant(name);
if (tenant->criteriaSatisfied(colonyTags))
matchingTenants.append(tenant);
}
return matchingTenants;
}
TenantPtr TenantDatabase::readTenant(String const& path) {
try {
auto assets = Root::singleton().assets();
Json config = assets->json(path);
String name = config.getString("name");
float priority = config.getFloat("priority");
StringMap<unsigned> colonyTagCriteria;
for (pair<String, Json> const& entry : config.getObject("colonyTagCriteria").pairs()) {
colonyTagCriteria[entry.first] = entry.second.toUInt();
}
List<TenantSpawnable> tenants = config.getArray("tenants").transformed([](Json const& json) -> TenantSpawnable {
String spawn = json.getString("spawn");
if (spawn == "monster") {
return TenantMonsterSpawnable{json.getString("type"), json.optFloat("level"), json.opt("overrides")};
} else {
starAssert(json.getString("spawn") == "npc");
Json speciesJson = json.get("species");
List<String> species;
if (speciesJson.isType(Json::Type::Array))
species = speciesJson.toArray().transformed(mem_fn(&Json::toString));
else
species.append(speciesJson.toString());
return TenantNpcSpawnable{species, json.getString("type"), json.optFloat("level"), json.opt("overrides")};
}
});
Maybe<TenantRent> rent = config.opt("rent").apply([](Json const& json) {
return TenantRent{jsonToVec2F(json.get("periodRange")), json.getString("pool")};
});
return make_shared<Tenant>(Tenant{name, priority, colonyTagCriteria, tenants, rent, config});
} catch (std::exception const& e) {
2023-06-27 20:23:44 +10:00
throw TenantException::format("Error loading tenant '{}': {}", path, outputException(e, false));
2023-06-20 14:33:09 +10:00
}
}
}