using System.Collections.Generic; using System.IO; using System.Linq; using Content.Server.GameTicking; using Content.Server.Maps; using Content.Server.Shuttles.Components; using Content.Server.Shuttles.Systems; using Content.Server.Spawners.Components; using Content.Server.Station.Components; using Content.Shared.CCVar; using Content.Shared.Roles; using Robust.Shared.Configuration; using Robust.Shared.ContentPack; using Robust.Shared.GameObjects; using Robust.Shared.Map; using Robust.Shared.Map.Components; using Robust.Shared.Prototypes; using Content.Shared.Station.Components; using Robust.Shared.EntitySerialization; using Robust.Shared.EntitySerialization.Systems; using Robust.Shared.Utility; using YamlDotNet.RepresentationModel; namespace Content.IntegrationTests.Tests { [TestFixture] public sealed class PostMapInitTest { private const bool SkipTestMaps = true; private const string TestMapsPath = "/Maps/Test/"; private static readonly string[] NoSpawnMaps = { "CentComm", "Dart" }; private static readonly string[] Grids = { "/Maps/centcomm.yml", "/Maps/Shuttles/cargo.yml", "/Maps/Shuttles/emergency.yml", "/Maps/Shuttles/infiltrator.yml", }; private static readonly string[] GameMaps = { "Dev", "TestTeg", "Fland", "Meta", "Packed", "Omega", "Bagel", "CentComm", "Box", "Core", "Marathon", "MeteorArena", "Saltern", "Reach", "Train", "Oasis", "Cog", "Gate", "Amber", "Loop" }; /// /// Asserts that specific files have been saved as grids and not maps. /// [Test, TestCaseSource(nameof(Grids))] public async Task GridsLoadableTest(string mapFile) { await using var pair = await PoolManager.GetServerClient(); var server = pair.Server; var entManager = server.ResolveDependency(); var mapLoader = entManager.System(); var mapSystem = entManager.System(); var cfg = server.ResolveDependency(); Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False); var path = new ResPath(mapFile); await server.WaitPost(() => { mapSystem.CreateMap(out var mapId); try { Assert.That(mapLoader.TryLoadGrid(mapId, path, out var grid)); } catch (Exception ex) { throw new Exception($"Failed to load map {mapFile}, was it saved as a map instead of a grid?", ex); } mapSystem.DeleteMap(mapId); }); await server.WaitRunTicks(1); await pair.CleanReturnAsync(); } [Test] public async Task NoSavedPostMapInitTest() { await using var pair = await PoolManager.GetServerClient(); var server = pair.Server; var resourceManager = server.ResolveDependency(); var loader = server.System(); var mapFolder = new ResPath("/Maps"); var maps = resourceManager .ContentFindFiles(mapFolder) .Where(filePath => filePath.Extension == "yml" && !filePath.Filename.StartsWith(".", StringComparison.Ordinal)) .ToArray(); var v7Maps = new List(); foreach (var map in maps) { var rootedPath = map.ToRootedPath(); // ReSharper disable once RedundantLogicalConditionalExpressionOperand if (SkipTestMaps && rootedPath.ToString().StartsWith(TestMapsPath, StringComparison.Ordinal)) { continue; } if (!resourceManager.TryContentFileRead(rootedPath, out var fileStream)) { Assert.Fail($"Map not found: {rootedPath}"); } using var reader = new StreamReader(fileStream); var yamlStream = new YamlStream(); yamlStream.Load(reader); var root = yamlStream.Documents[0].RootNode; var meta = root["meta"]; var version = meta["format"].AsInt(); if (version >= 7) { v7Maps.Add(map); continue; } var postMapInit = meta["postmapinit"].AsBool(); Assert.That(postMapInit, Is.False, $"Map {map.Filename} was saved postmapinit"); } var deps = server.ResolveDependency().DependencyCollection; foreach (var map in v7Maps) { if (!loader.TryReadFile(map, out var data)) { Assert.Fail($"Failed to read {map}"); continue; } var reader = new EntityDeserializer(deps, data, DeserializationOptions.Default); if (!reader.TryProcessData()) { Assert.Fail($"Failed to process {map}"); continue; } foreach (var mapId in reader.MapYamlIds) { var mapData = reader.YamlEntities[mapId]; Assert.That(!mapData.PostInit, $"Map {map.Filename} contains a postmapinit map with yaml id: {mapId}"); } } await pair.CleanReturnAsync(); } [Test, TestCaseSource(nameof(GameMaps))] public async Task GameMapsLoadableTest(string mapProto) { await using var pair = await PoolManager.GetServerClient(new PoolSettings { Dirty = true // Stations spawn a bunch of nullspace entities and maps like centcomm. }); var server = pair.Server; var mapManager = server.ResolveDependency(); var entManager = server.ResolveDependency(); var mapLoader = entManager.System(); var mapSystem = entManager.System(); var protoManager = server.ResolveDependency(); var ticker = entManager.EntitySysManager.GetEntitySystem(); var shuttleSystem = entManager.EntitySysManager.GetEntitySystem(); var cfg = server.ResolveDependency(); Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False); await server.WaitPost(() => { MapId mapId; try { var opts = DeserializationOptions.Default with {InitializeMaps = true}; ticker.LoadGameMap(protoManager.Index(mapProto), out mapId, opts); } catch (Exception ex) { throw new Exception($"Failed to load map {mapProto}", ex); } mapSystem.CreateMap(out var shuttleMap); var largest = 0f; EntityUid? targetGrid = null; var memberQuery = entManager.GetEntityQuery(); var grids = mapManager.GetAllGrids(mapId).ToList(); var gridUids = grids.Select(o => o.Owner).ToList(); targetGrid = gridUids.First(); foreach (var grid in grids) { var gridEnt = grid.Owner; if (!memberQuery.HasComponent(gridEnt)) continue; var area = grid.Comp.LocalAABB.Width * grid.Comp.LocalAABB.Height; if (area > largest) { largest = area; targetGrid = gridEnt; } } // Test shuttle can dock. // This is done inside gamemap test because loading the map takes ages and we already have it. var station = entManager.GetComponent(targetGrid!.Value).Station; if (entManager.TryGetComponent(station, out var stationEvac)) { var shuttlePath = stationEvac.EmergencyShuttlePath; Assert.That(mapLoader.TryLoadGrid(shuttleMap, shuttlePath, out var shuttle), $"Failed to load {shuttlePath}"); Assert.That( shuttleSystem.TryFTLDock(shuttle!.Value.Owner, entManager.GetComponent(shuttle!.Value.Owner), targetGrid.Value), $"Unable to dock {shuttlePath} to {mapProto}"); } mapSystem.DeleteMap(shuttleMap); if (entManager.HasComponent(station)) { // Test that the map has valid latejoin spawn points or container spawn points if (!NoSpawnMaps.Contains(mapProto)) { var lateSpawns = 0; lateSpawns += GetCountLateSpawn(gridUids, entManager); lateSpawns += GetCountLateSpawn(gridUids, entManager); Assert.That(lateSpawns, Is.GreaterThan(0), $"Found no latejoin spawn points on {mapProto}"); } // Test all availableJobs have spawnPoints // This is done inside gamemap test because loading the map takes ages and we already have it. var comp = entManager.GetComponent(station); var jobs = new HashSet>(comp.SetupAvailableJobs.Keys); var spawnPoints = entManager.EntityQuery() .Where(x => x.SpawnType == SpawnPointType.Job && x.Job != null) .Select(x => x.Job.Value); jobs.ExceptWith(spawnPoints); spawnPoints = entManager.EntityQuery() .Where(x => x.SpawnType is SpawnPointType.Job or SpawnPointType.Unset && x.Job != null) .Select(x => x.Job.Value); jobs.ExceptWith(spawnPoints); Assert.That(jobs, Is.Empty, $"There is no spawnpoints for {string.Join(", ", jobs)} on {mapProto}."); } try { mapSystem.DeleteMap(mapId); } catch (Exception ex) { throw new Exception($"Failed to delete map {mapProto}", ex); } }); await server.WaitRunTicks(1); await pair.CleanReturnAsync(); } private static int GetCountLateSpawn(List gridUids, IEntityManager entManager) where T : ISpawnPoint, IComponent { var resultCount = 0; var queryPoint = entManager.AllEntityQueryEnumerator(); #nullable enable while (queryPoint.MoveNext(out T? comp, out var xform)) { var spawner = (ISpawnPoint) comp; if (spawner.SpawnType is not SpawnPointType.LateJoin || xform.GridUid == null || !gridUids.Contains(xform.GridUid.Value)) { continue; } #nullable disable resultCount++; break; } return resultCount; } [Test] public async Task AllMapsTested() { await using var pair = await PoolManager.GetServerClient(); var server = pair.Server; var protoMan = server.ResolveDependency(); var gameMaps = protoMan.EnumeratePrototypes() .Where(x => !pair.IsTestPrototype(x)) .Select(x => x.ID) .ToHashSet(); Assert.That(gameMaps.Remove(PoolManager.TestMap)); Assert.That(gameMaps, Is.EquivalentTo(GameMaps.ToHashSet()), "Game map prototype missing from test cases."); await pair.CleanReturnAsync(); } [Test] public async Task NonGameMapsLoadableTest() { await using var pair = await PoolManager.GetServerClient(); var server = pair.Server; var mapLoader = server.ResolveDependency().GetEntitySystem(); var resourceManager = server.ResolveDependency(); var protoManager = server.ResolveDependency(); var cfg = server.ResolveDependency(); Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False); var gameMaps = protoManager.EnumeratePrototypes().Select(o => o.MapPath).ToHashSet(); var mapFolder = new ResPath("/Maps"); var maps = resourceManager .ContentFindFiles(mapFolder) .Where(filePath => filePath.Extension == "yml" && !filePath.Filename.StartsWith(".", StringComparison.Ordinal)) .ToArray(); var mapPaths = new List(); foreach (var map in maps) { if (gameMaps.Contains(map)) continue; var rootedPath = map.ToRootedPath(); if (SkipTestMaps && rootedPath.ToString().StartsWith(TestMapsPath, StringComparison.Ordinal)) { continue; } mapPaths.Add(rootedPath); } await server.WaitPost(() => { Assert.Multiple(() => { // This bunch of files contains a random mixture of both map and grid files. // TODO MAPPING organize files var opts = MapLoadOptions.Default with { DeserializationOptions = DeserializationOptions.Default with { InitializeMaps = true, LogOrphanedGrids = false } }; HashSet> maps; foreach (var path in mapPaths) { try { Assert.That(mapLoader.TryLoadEntities(path, out maps, out _, opts)); } catch (Exception ex) { throw new Exception($"Failed to load map {path}", ex); } try { foreach (var map in maps) { server.EntMan.DeleteEntity(map); } } catch (Exception ex) { throw new Exception($"Failed to delete map {path}", ex); } } }); }); await server.WaitRunTicks(1); await pair.CleanReturnAsync(); } } }