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