2024-02-25 15:46:47 +01:00
|
|
|
#pragma once
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
#include "StarBTreeDatabase.hpp"
|
|
|
|
#include "StarVersioningDatabase.hpp"
|
|
|
|
#include "StarEntity.hpp"
|
|
|
|
#include "StarOrderedSet.hpp"
|
|
|
|
#include "StarWorldTiles.hpp"
|
|
|
|
#include "StarRpcPromise.hpp"
|
|
|
|
#include "StarBiomePlacement.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
STAR_EXCEPTION(WorldStorageException, StarException);
|
|
|
|
|
|
|
|
STAR_CLASS(EntityMap);
|
|
|
|
STAR_STRUCT(WorldGeneratorFacade);
|
|
|
|
STAR_CLASS(WorldStorage);
|
|
|
|
|
|
|
|
typedef HashMap<ByteArray, Maybe<ByteArray>> WorldChunks;
|
|
|
|
|
|
|
|
enum class SectorLoadLevel : uint8_t {
|
|
|
|
None = 0,
|
|
|
|
Tiles = 1,
|
|
|
|
Entities = 2,
|
|
|
|
|
|
|
|
Loaded = 2
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class SectorGenerationLevel : uint8_t {
|
|
|
|
None = 0,
|
|
|
|
BaseTiles = 1,
|
|
|
|
MicroDungeons = 2,
|
|
|
|
CaveLiquid = 3,
|
|
|
|
Finalize = 4,
|
|
|
|
|
|
|
|
Complete = 4,
|
|
|
|
|
|
|
|
Terraform = 5
|
|
|
|
};
|
|
|
|
|
|
|
|
struct WorldGeneratorFacade {
|
|
|
|
typedef ServerTileSectorArray::Sector Sector;
|
|
|
|
|
|
|
|
WorldGeneratorFacade() {}
|
|
|
|
virtual ~WorldGeneratorFacade() {}
|
|
|
|
|
|
|
|
// Should bring a given sector from generationLevel - 1 to generationLevel.
|
|
|
|
virtual void generateSectorLevel(WorldStorage* storage, Sector const& sector, SectorGenerationLevel generationLevel) = 0;
|
|
|
|
|
|
|
|
virtual void sectorLoadLevelChanged(WorldStorage* storage, Sector const& sector, SectorLoadLevel loadLevel) = 0;
|
|
|
|
|
|
|
|
// Perform terraforming operations (biome reapplication) on the given sector
|
|
|
|
virtual void terraformSector(WorldStorage* storage, Sector const& sector) = 0;
|
|
|
|
|
|
|
|
// Called after an entity is loaded, but before the entity is added to the
|
|
|
|
// EntityMap.
|
|
|
|
virtual void initEntity(WorldStorage* storage, EntityId newEntityId, EntityPtr const& entity) = 0;
|
|
|
|
|
|
|
|
// Called after the entity is removed from the entity map but before it is
|
|
|
|
// stored.
|
|
|
|
virtual void destructEntity(WorldStorage* storage, EntityPtr const& entity) = 0;
|
|
|
|
|
|
|
|
// Should return true if this entity should maintain the sector, false
|
|
|
|
// otherwise.
|
|
|
|
virtual bool entityKeepAlive(WorldStorage* storage, EntityPtr const& entity) const = 0;
|
|
|
|
|
|
|
|
// Should return true if this entity should be stored along with the world,
|
|
|
|
// false otherwise.
|
|
|
|
virtual bool entityPersistent(WorldStorage* storage, EntityPtr const& entity) const = 0;
|
|
|
|
|
|
|
|
// Queues up a microdungeon. Fulfills the rpc promise with the position the
|
|
|
|
// microdungeon was placed at
|
|
|
|
virtual RpcPromise<Vec2I> enqueuePlacement(List<BiomeItemDistribution> placements, Maybe<DungeonId> id) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Handles paging entity and tile data in / out of disk backed storage for
|
|
|
|
// WorldServer and triggers initial generation. Ties tile sectors to entity
|
|
|
|
// sectors, and allows for multiple stage generation of those sectors. Sector
|
|
|
|
// generation is done in stages, so that lower generation stages are done in a
|
|
|
|
// one sector border around the higher generation stages.
|
|
|
|
//
|
|
|
|
// WorldStorage is designed so that once constructed, any exceptions triggered
|
|
|
|
// during loading, unloading, or generation that would result in an
|
|
|
|
// indeterminate world state cause the underlying database to be rolled back
|
|
|
|
// and then immediately closed. The underlying database committed only when
|
|
|
|
// destructed without error, or a manual call to sync().
|
|
|
|
class WorldStorage {
|
|
|
|
public:
|
|
|
|
typedef ServerTileSectorArray::Sector Sector;
|
|
|
|
typedef ServerTileSectorArray::Array TileArray;
|
|
|
|
typedef ServerTileSectorArray::ArrayPtr TileArrayPtr;
|
|
|
|
|
|
|
|
static WorldChunks getWorldChunksUpdate(WorldChunks const& oldChunks, WorldChunks const& newChunks);
|
|
|
|
static void applyWorldChunksUpdateToFile(String const& file, WorldChunks const& update);
|
|
|
|
static WorldChunks getWorldChunksFromFile(String const& file);
|
|
|
|
|
|
|
|
// Create a new world of the given size.
|
|
|
|
WorldStorage(Vec2U const& worldSize, IODevicePtr const& device, WorldGeneratorFacadePtr const& generatorFacade);
|
|
|
|
// Read an existing world.
|
|
|
|
WorldStorage(IODevicePtr const& device, WorldGeneratorFacadePtr const& generatorFacade);
|
|
|
|
// Read an in-memory world.
|
|
|
|
WorldStorage(WorldChunks const& chunks, WorldGeneratorFacadePtr const& generatorFacade);
|
|
|
|
~WorldStorage();
|
|
|
|
|
|
|
|
VersionedJson worldMetadata();
|
|
|
|
void setWorldMetadata(VersionedJson const& metadata);
|
|
|
|
|
|
|
|
ServerTileSectorArrayPtr const& tileArray() const;
|
|
|
|
EntityMapPtr const& entityMap() const;
|
|
|
|
|
|
|
|
Maybe<Sector> sectorForPosition(Vec2I const& position) const;
|
|
|
|
List<Sector> sectorsForRegion(RectI const& region) const;
|
|
|
|
Maybe<RectI> regionForSector(Sector sector) const;
|
|
|
|
|
|
|
|
SectorLoadLevel sectorLoadLevel(Sector sector) const;
|
|
|
|
// Returns the sector generation level if it is currently loaded, nothing
|
|
|
|
// otherwise.
|
|
|
|
Maybe<SectorGenerationLevel> sectorGenerationLevel(Sector sector) const;
|
|
|
|
// Returns true if the sector is both loaded and fully generated.
|
|
|
|
bool sectorActive(Sector) const;
|
|
|
|
|
|
|
|
// Fully load the given sector and reset its TTL without triggering any
|
|
|
|
// generation.
|
|
|
|
void loadSector(Sector sector);
|
|
|
|
// Fully load, reset the TTL, and if necessary, fully generate the given
|
|
|
|
// sector.
|
|
|
|
void activateSector(Sector sector);
|
|
|
|
// Queue the given sector for activation, if it is not already active. If
|
|
|
|
// the sector is loaded at all, also resets the TTL.
|
|
|
|
void queueSectorActivation(Sector sector);
|
|
|
|
|
|
|
|
// Immediately (synchronously) fully generates the sector, then flags it as requiring
|
|
|
|
// terraforming (biome reapplication) which will be handled by the normal generation process
|
|
|
|
void triggerTerraformSector(Sector sector);
|
|
|
|
|
|
|
|
// Queues up a microdungeon. Fulfills the rpc promise with the position the
|
|
|
|
// microdungeon was placed at
|
|
|
|
RpcPromise<Vec2I> enqueuePlacement(List<BiomeItemDistribution> placements, Maybe<DungeonId> id);
|
|
|
|
|
|
|
|
// Return the remaining time to live for a sector, if loaded. A sector's
|
|
|
|
// time to live is reset when loaded or generated, and when the time to live
|
|
|
|
// reaches zero, the sector is automatically unloaded.
|
|
|
|
Maybe<float> sectorTimeToLive(Sector sector) const;
|
|
|
|
// Set the given sector's time to live, if it is loaded at all. Returns
|
|
|
|
// false if the sector was not loaded so no action was taken.
|
|
|
|
bool setSectorTimeToLive(Sector sector, float newTimeToLive);
|
|
|
|
|
|
|
|
// Returns the position for a given unique entity if it exists in this world,
|
|
|
|
// loaded or not.
|
|
|
|
Maybe<Vec2F> findUniqueEntity(String const& uniqueId);
|
|
|
|
|
|
|
|
// If the given unique entity is not loaded, loads its sector and then if the
|
|
|
|
// unique entity is found, returns the entity id, otherwise NullEntityId.
|
|
|
|
EntityId loadUniqueEntity(String const& uniqueId);
|
|
|
|
|
|
|
|
// Does any queued generation work, potentially limiting the total number of
|
|
|
|
// increases of SectorGenerationLevel by the sectorGenerationLevelLimit, if
|
|
|
|
// given. If sectorOrdering is given, then it will be used to prioritize the
|
|
|
|
// queued sectors.
|
|
|
|
void generateQueue(Maybe<size_t> sectorGenerationLevelLimit, function<bool(Sector, Sector)> sectorOrdering = {});
|
|
|
|
// Ticks down the TTL on sectors and generation queue entries, stores old
|
|
|
|
// sectors, expires old generation queue entries, and unloads any zombie
|
|
|
|
// entities.
|
2024-07-28 10:53:14 +10:00
|
|
|
void tick(float dt, String const* worldId = nullptr);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
// Unload all sectors that can be unloaded (if force is specified, ALWAYS
|
|
|
|
// unloads all sectors)
|
|
|
|
void unloadAll(bool force = false);
|
|
|
|
|
|
|
|
// Sync all active sectors without unloading them, and commits the underlying
|
|
|
|
// database.
|
|
|
|
void sync();
|
|
|
|
|
|
|
|
// Syncs all active sectors to disk and stores the full content of the world
|
|
|
|
// into memory.
|
|
|
|
WorldChunks readChunks();
|
|
|
|
|
|
|
|
// if this is set, all terrain generation is assumed to be handled by dungeon placement
|
|
|
|
// and steps such as microdungeons, biome objects and grass mods will be skipped
|
|
|
|
bool floatingDungeonWorld() const;
|
|
|
|
void setFloatingDungeonWorld(bool floatingDungeonWorld);
|
|
|
|
|
|
|
|
private:
|
|
|
|
enum class StoreType : uint8_t {
|
|
|
|
Metadata = 0,
|
|
|
|
TileSector = 1,
|
|
|
|
EntitySector = 2,
|
|
|
|
UniqueIndex = 3,
|
|
|
|
SectorUniques = 4
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef pair<Sector, Vec2F> SectorAndPosition;
|
|
|
|
|
|
|
|
struct WorldMetadataStore {
|
|
|
|
Vec2U worldSize;
|
|
|
|
VersionedJson userMetadata;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef List<VersionedJson> EntitySectorStore;
|
|
|
|
// Map of uuid to entity's position and sector they were stored in.
|
|
|
|
typedef HashMap<String, SectorAndPosition> UniqueIndexStore;
|
|
|
|
// Set of unique ids that are stored in a given sector
|
|
|
|
typedef HashSet<String> SectorUniqueStore;
|
|
|
|
|
|
|
|
struct TileSectorStore {
|
|
|
|
TileSectorStore();
|
|
|
|
|
|
|
|
// Also store generation level along with tiles, simply because tiles are
|
|
|
|
// the first things to be loaded and the last to be stored.
|
|
|
|
SectorGenerationLevel generationLevel;
|
|
|
|
|
|
|
|
VersionNumber tileSerializationVersion;
|
|
|
|
TileArrayPtr tiles;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct SectorMetadata {
|
|
|
|
SectorMetadata();
|
|
|
|
|
|
|
|
SectorLoadLevel loadLevel;
|
|
|
|
SectorGenerationLevel generationLevel;
|
|
|
|
float timeToLive;
|
|
|
|
};
|
|
|
|
|
|
|
|
static ByteArray metadataKey();
|
|
|
|
static WorldMetadataStore readWorldMetadata(ByteArray const& data);
|
|
|
|
static ByteArray writeWorldMetadata(WorldMetadataStore const& metadata);
|
|
|
|
|
|
|
|
static ByteArray entitySectorKey(Sector const& sector);
|
|
|
|
static EntitySectorStore readEntitySector(ByteArray const& data);
|
|
|
|
static ByteArray writeEntitySector(EntitySectorStore const& store);
|
|
|
|
|
|
|
|
static ByteArray tileSectorKey(Sector const& sector);
|
|
|
|
static TileSectorStore readTileSector(ByteArray const& data);
|
|
|
|
static ByteArray writeTileSector(TileSectorStore const& store);
|
|
|
|
|
|
|
|
static ByteArray uniqueIndexKey(String const& uniqueId);
|
|
|
|
static UniqueIndexStore readUniqueIndexStore(ByteArray const& data);
|
|
|
|
static ByteArray writeUniqueIndexStore(UniqueIndexStore const& store);
|
|
|
|
|
|
|
|
static ByteArray sectorUniqueKey(Sector const& sector);
|
|
|
|
static SectorUniqueStore readSectorUniqueStore(ByteArray const& data);
|
|
|
|
static ByteArray writeSectorUniqueStore(SectorUniqueStore const& store);
|
|
|
|
|
|
|
|
static void openDatabase(BTreeDatabase& db, IODevicePtr device);
|
|
|
|
|
|
|
|
WorldStorage();
|
|
|
|
|
|
|
|
bool belongsInSector(Sector const& sector, Vec2F const& position) const;
|
|
|
|
|
|
|
|
// Generate a random TTL value in the configured range
|
|
|
|
float randomizedSectorTTL() const;
|
|
|
|
|
|
|
|
// Generate the given sector to the given generation level. If
|
|
|
|
// sectorGenerationLevelLimit is given, stops work as soon as the given
|
|
|
|
// number of generation level changes has occurred. Returns whether the
|
|
|
|
// given sector was fully generated, and the total number of generation
|
|
|
|
// levels increased. If any sector's generation level is brought up at all,
|
|
|
|
// it will also reset the TTL for that sector.
|
|
|
|
pair<bool, size_t> generateSectorToLevel(Sector const& sector, SectorGenerationLevel targetGenerationLevel, size_t sectorGenerationLevelLimit = NPos);
|
|
|
|
|
|
|
|
// Bring the sector up to the given load level, and all surrounding sectors
|
|
|
|
// as appropriate. If the load level is brought up, also resets the TTL.
|
|
|
|
void loadSectorToLevel(Sector const& sector, SectorLoadLevel targetLoadLevel);
|
|
|
|
|
|
|
|
// Store and unload the given sector to the given level, given the state of
|
|
|
|
// the surrounding sectors. If force is true, will always unload to the
|
|
|
|
// given level.
|
2024-07-28 10:53:14 +10:00
|
|
|
bool unloadSectorToLevel(Sector const& sector, SectorLoadLevel targetLoadLevel, bool force = false);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
// Sync this sector to disk without unloading it.
|
|
|
|
void syncSector(Sector const& sector);
|
|
|
|
|
|
|
|
// Returns the sectors within WorldSectorSize of the given sector. This is
|
|
|
|
// *not exactly the same* as the surrounding 9 sectors in a square pattern,
|
|
|
|
// because first this does not return invalid sectors, and second, If a world
|
|
|
|
// is not evenly divided by the sector size, this may return extra sectors on
|
|
|
|
// one side because they are within range.
|
|
|
|
List<Sector> adjacentSectors(Sector const& sector) const;
|
|
|
|
|
|
|
|
// Replace the sector uniques for this sector with the given set
|
|
|
|
void updateSectorUniques(Sector const& sector, UniqueIndexStore const& sectorUniques);
|
|
|
|
// Merge the stored sector uniques for this sector with the given set
|
|
|
|
void mergeSectorUniques(Sector const& sector, UniqueIndexStore const& sectorUniques);
|
|
|
|
|
|
|
|
Maybe<SectorAndPosition> getUniqueIndexEntry(String const& uniqueId);
|
|
|
|
void setUniqueIndexEntry(String const& uniqueId, SectorAndPosition const& sectorAndPosition);
|
|
|
|
// Remove the index entry for this unique id, if the index entry found points
|
|
|
|
// to the given sector
|
|
|
|
void removeUniqueIndexEntry(String const& uniqueId, Sector const& sector);
|
|
|
|
|
|
|
|
Vec2F m_sectorTimeToLive;
|
|
|
|
float m_generationQueueTimeToLive;
|
|
|
|
|
|
|
|
ServerTileSectorArrayPtr m_tileArray;
|
|
|
|
EntityMapPtr m_entityMap;
|
|
|
|
WorldGeneratorFacadePtr m_generatorFacade;
|
|
|
|
|
|
|
|
bool m_floatingDungeonWorld;
|
|
|
|
|
|
|
|
StableHashMap<Sector, SectorMetadata> m_sectorMetadata;
|
|
|
|
OrderedHashMap<Sector, float> m_generationQueue;
|
|
|
|
BTreeDatabase m_db;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|