2021-02-18 20:45:45 -08:00
|
|
|
using System.IO;
|
2020-06-22 04:43:47 -04:00
|
|
|
using System.Linq;
|
2023-04-25 11:48:29 +12:00
|
|
|
using Content.Shared.CCVar;
|
|
|
|
|
using Robust.Shared.Configuration;
|
2021-02-11 01:13:03 -08:00
|
|
|
using Robust.Shared.ContentPack;
|
2024-12-22 15:13:10 +13:00
|
|
|
using Robust.Shared.EntitySerialization.Systems;
|
2022-11-13 17:47:48 +11:00
|
|
|
using Robust.Shared.GameObjects;
|
2019-09-19 13:46:01 +02:00
|
|
|
using Robust.Shared.Map;
|
2024-12-22 15:13:10 +13:00
|
|
|
using Robust.Shared.Map.Events;
|
|
|
|
|
using Robust.Shared.Serialization.Markdown.Mapping;
|
2019-09-19 13:46:01 +02:00
|
|
|
using Robust.Shared.Utility;
|
|
|
|
|
|
2020-01-20 22:14:44 +01:00
|
|
|
namespace Content.IntegrationTests.Tests
|
2019-09-19 13:46:01 +02:00
|
|
|
{
|
|
|
|
|
/// <summary>
|
2024-12-22 15:13:10 +13:00
|
|
|
/// Tests that a grid's yaml does not change when saved consecutively.
|
2019-09-19 13:46:01 +02:00
|
|
|
/// </summary>
|
|
|
|
|
[TestFixture]
|
2022-06-19 20:22:28 -07:00
|
|
|
public sealed class SaveLoadSaveTest
|
2019-09-19 13:46:01 +02:00
|
|
|
{
|
|
|
|
|
[Test]
|
2024-12-22 15:13:10 +13:00
|
|
|
public async Task CreateSaveLoadSaveGrid()
|
2019-09-19 13:46:01 +02:00
|
|
|
{
|
2023-08-25 02:56:51 +02:00
|
|
|
await using var pair = await PoolManager.GetServerClient();
|
|
|
|
|
var server = pair.Server;
|
2023-07-05 21:54:25 -07:00
|
|
|
var entManager = server.ResolveDependency<IEntityManager>();
|
|
|
|
|
var mapLoader = entManager.System<MapLoaderSystem>();
|
2024-07-02 20:01:37 -04:00
|
|
|
var mapSystem = entManager.System<SharedMapSystem>();
|
2019-09-19 13:46:01 +02:00
|
|
|
var mapManager = server.ResolveDependency<IMapManager>();
|
2023-04-25 11:48:29 +12:00
|
|
|
var cfg = server.ResolveDependency<IConfigurationManager>();
|
2023-05-31 11:13:02 +10:00
|
|
|
Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False);
|
2022-11-13 17:47:48 +11:00
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
var testSystem = server.System<SaveLoadSaveTestSystem>();
|
|
|
|
|
testSystem.Enabled = true;
|
|
|
|
|
|
|
|
|
|
var rp1 = new ResPath("/save load save 1.yml");
|
|
|
|
|
var rp2 = new ResPath("/save load save 2.yml");
|
|
|
|
|
|
2022-06-19 20:22:28 -07:00
|
|
|
await server.WaitPost(() =>
|
2019-09-19 13:46:01 +02:00
|
|
|
{
|
2024-07-02 20:01:37 -04:00
|
|
|
mapSystem.CreateMap(out var mapId0);
|
|
|
|
|
var grid0 = mapManager.CreateGridEntity(mapId0);
|
2024-12-22 15:13:10 +13:00
|
|
|
entManager.RunMapInit(grid0.Owner, entManager.GetComponent<MetaDataComponent>(grid0));
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveGrid(grid0.Owner, rp1));
|
2024-07-02 20:01:37 -04:00
|
|
|
mapSystem.CreateMap(out var mapId1);
|
2024-12-22 15:13:10 +13:00
|
|
|
Assert.That(mapLoader.TryLoadGrid(mapId1, rp1, out var grid1));
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveGrid(grid1!.Value, rp2));
|
2019-09-19 13:46:01 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await server.WaitIdleAsync();
|
|
|
|
|
var userData = server.ResolveDependency<IResourceManager>().UserData;
|
|
|
|
|
|
|
|
|
|
string one;
|
|
|
|
|
string two;
|
|
|
|
|
|
2022-11-13 17:47:48 +11:00
|
|
|
await using (var stream = userData.Open(rp1, FileMode.Open))
|
2019-09-19 13:46:01 +02:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
2022-11-13 17:47:48 +11:00
|
|
|
one = await reader.ReadToEndAsync();
|
2019-09-19 13:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
2022-11-13 17:47:48 +11:00
|
|
|
await using (var stream = userData.Open(rp2, FileMode.Open))
|
2019-09-19 13:46:01 +02:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
2022-11-13 17:47:48 +11:00
|
|
|
two = await reader.ReadToEndAsync();
|
2019-09-19 13:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-05 21:54:25 -07:00
|
|
|
Assert.Multiple(() =>
|
|
|
|
|
{
|
2021-01-15 17:19:52 +01:00
|
|
|
Assert.That(two, Is.EqualTo(one));
|
2020-06-22 04:43:47 -04:00
|
|
|
var failed = TestContext.CurrentContext.Result.Assertions.FirstOrDefault();
|
|
|
|
|
if (failed != null)
|
|
|
|
|
{
|
2021-01-14 20:41:13 +01:00
|
|
|
var oneTmp = Path.GetTempFileName();
|
|
|
|
|
var twoTmp = Path.GetTempFileName();
|
|
|
|
|
|
|
|
|
|
File.WriteAllText(oneTmp, one);
|
|
|
|
|
File.WriteAllText(twoTmp, two);
|
|
|
|
|
|
|
|
|
|
TestContext.AddTestAttachment(oneTmp, "First save file");
|
|
|
|
|
TestContext.AddTestAttachment(twoTmp, "Second save file");
|
2020-06-22 04:43:47 -04:00
|
|
|
TestContext.Error.WriteLine("Complete output:");
|
2021-01-14 20:41:13 +01:00
|
|
|
TestContext.Error.WriteLine(oneTmp);
|
|
|
|
|
TestContext.Error.WriteLine(twoTmp);
|
2020-06-22 04:43:47 -04:00
|
|
|
}
|
|
|
|
|
});
|
2024-12-22 15:13:10 +13:00
|
|
|
testSystem.Enabled = false;
|
2023-08-25 02:56:51 +02:00
|
|
|
await pair.CleanReturnAsync();
|
2019-09-19 13:46:01 +02:00
|
|
|
}
|
2019-10-13 16:26:39 +02:00
|
|
|
|
2023-07-05 21:54:25 -07:00
|
|
|
private const string TestMap = "Maps/bagel.yml";
|
2023-03-26 14:17:27 +13:00
|
|
|
|
2019-10-13 16:26:39 +02:00
|
|
|
/// <summary>
|
|
|
|
|
/// Loads the default map, runs it for 5 ticks, then assert that it did not change.
|
|
|
|
|
/// </summary>
|
|
|
|
|
[Test]
|
2022-11-19 21:05:32 -07:00
|
|
|
public async Task LoadSaveTicksSaveBagel()
|
2019-10-13 16:26:39 +02:00
|
|
|
{
|
2023-08-25 02:56:51 +02:00
|
|
|
await using var pair = await PoolManager.GetServerClient();
|
|
|
|
|
var server = pair.Server;
|
2022-11-13 17:47:48 +11:00
|
|
|
var mapLoader = server.ResolveDependency<IEntitySystemManager>().GetEntitySystem<MapLoaderSystem>();
|
2024-12-22 15:13:10 +13:00
|
|
|
var mapSys = server.System<SharedMapSystem>();
|
|
|
|
|
var testSystem = server.System<SaveLoadSaveTestSystem>();
|
|
|
|
|
testSystem.Enabled = true;
|
|
|
|
|
|
|
|
|
|
var rp1 = new ResPath("/load save ticks save 1.yml");
|
|
|
|
|
var rp2 = new ResPath("/load save ticks save 2.yml");
|
2019-10-13 16:26:39 +02:00
|
|
|
|
2022-04-20 06:09:50 -06:00
|
|
|
MapId mapId = default;
|
2023-04-25 11:48:29 +12:00
|
|
|
var cfg = server.ResolveDependency<IConfigurationManager>();
|
2023-05-31 11:13:02 +10:00
|
|
|
Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False);
|
2019-10-13 16:26:39 +02:00
|
|
|
|
2023-03-26 14:17:27 +13:00
|
|
|
// Load bagel.yml as uninitialized map, and save it to ensure it's up to date.
|
2019-10-13 16:26:39 +02:00
|
|
|
server.Post(() =>
|
|
|
|
|
{
|
2024-12-22 15:13:10 +13:00
|
|
|
var path = new ResPath(TestMap);
|
|
|
|
|
Assert.That(mapLoader.TryLoadMap(path, out var map, out _), $"Failed to load test map {TestMap}");
|
|
|
|
|
mapId = map!.Value.Comp.MapId;
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveMap(mapId, rp1));
|
2019-10-13 16:26:39 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Run 5 ticks.
|
|
|
|
|
server.RunTicks(5);
|
|
|
|
|
|
2022-06-19 20:22:28 -07:00
|
|
|
await server.WaitPost(() =>
|
2019-10-13 16:26:39 +02:00
|
|
|
{
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveMap(mapId, rp2));
|
2019-10-13 16:26:39 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await server.WaitIdleAsync();
|
|
|
|
|
var userData = server.ResolveDependency<IResourceManager>().UserData;
|
|
|
|
|
|
|
|
|
|
string one;
|
|
|
|
|
string two;
|
|
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
await using (var stream = userData.Open(rp1, FileMode.Open))
|
2019-10-13 16:26:39 +02:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
2022-11-13 17:47:48 +11:00
|
|
|
one = await reader.ReadToEndAsync();
|
2019-10-13 16:26:39 +02:00
|
|
|
}
|
|
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
await using (var stream = userData.Open(rp2, FileMode.Open))
|
2019-10-13 16:26:39 +02:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
2022-11-13 17:47:48 +11:00
|
|
|
two = await reader.ReadToEndAsync();
|
2019-10-13 16:26:39 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-05 21:54:25 -07:00
|
|
|
Assert.Multiple(() =>
|
|
|
|
|
{
|
2021-07-04 18:11:52 +02:00
|
|
|
Assert.That(two, Is.EqualTo(one));
|
|
|
|
|
var failed = TestContext.CurrentContext.Result.Assertions.FirstOrDefault();
|
|
|
|
|
if (failed != null)
|
|
|
|
|
{
|
|
|
|
|
var oneTmp = Path.GetTempFileName();
|
|
|
|
|
var twoTmp = Path.GetTempFileName();
|
|
|
|
|
|
|
|
|
|
File.WriteAllText(oneTmp, one);
|
|
|
|
|
File.WriteAllText(twoTmp, two);
|
|
|
|
|
|
|
|
|
|
TestContext.AddTestAttachment(oneTmp, "First save file");
|
|
|
|
|
TestContext.AddTestAttachment(twoTmp, "Second save file");
|
|
|
|
|
TestContext.Error.WriteLine("Complete output:");
|
|
|
|
|
TestContext.Error.WriteLine(oneTmp);
|
|
|
|
|
TestContext.Error.WriteLine(twoTmp);
|
|
|
|
|
}
|
|
|
|
|
});
|
2023-03-26 14:17:27 +13:00
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
testSystem.Enabled = false;
|
|
|
|
|
await server.WaitPost(() => mapSys.DeleteMap(mapId));
|
2023-08-25 02:56:51 +02:00
|
|
|
await pair.CleanReturnAsync();
|
2023-03-26 14:17:27 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Loads the same uninitialized map at slightly different times, and then checks that they are the same
|
|
|
|
|
/// when getting saved.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Should ensure that entities do not perform randomization prior to initialization and should prevents
|
|
|
|
|
/// bugs like the one discussed in github.com/space-wizards/RobustToolbox/issues/3870. This test is somewhat
|
|
|
|
|
/// similar to <see cref="LoadSaveTicksSaveBagel"/> and <see cref="SaveLoadSave"/>, but neither of these
|
|
|
|
|
/// caught the mentioned bug.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
[Test]
|
|
|
|
|
public async Task LoadTickLoadBagel()
|
|
|
|
|
{
|
2023-08-25 02:56:51 +02:00
|
|
|
await using var pair = await PoolManager.GetServerClient();
|
|
|
|
|
var server = pair.Server;
|
2023-03-26 14:17:27 +13:00
|
|
|
|
2024-07-02 20:01:37 -04:00
|
|
|
var mapLoader = server.System<MapLoaderSystem>();
|
2024-12-22 15:13:10 +13:00
|
|
|
var mapSys = server.System<SharedMapSystem>();
|
2023-03-26 14:17:27 +13:00
|
|
|
var userData = server.ResolveDependency<IResourceManager>().UserData;
|
2023-04-25 11:48:29 +12:00
|
|
|
var cfg = server.ResolveDependency<IConfigurationManager>();
|
2023-05-31 11:13:02 +10:00
|
|
|
Assert.That(cfg.GetCVar(CCVars.GridFill), Is.False);
|
2024-12-22 15:13:10 +13:00
|
|
|
var testSystem = server.System<SaveLoadSaveTestSystem>();
|
|
|
|
|
testSystem.Enabled = true;
|
2023-03-26 14:17:27 +13:00
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
MapId mapId1 = default;
|
|
|
|
|
MapId mapId2 = default;
|
2024-12-24 18:57:52 +13:00
|
|
|
var fileA = new ResPath("/load tick load a.yml");
|
|
|
|
|
var fileB = new ResPath("/load tick load b.yml");
|
2023-03-26 14:17:27 +13:00
|
|
|
string yamlA;
|
|
|
|
|
string yamlB;
|
|
|
|
|
|
|
|
|
|
// Load & save the first map
|
|
|
|
|
server.Post(() =>
|
|
|
|
|
{
|
2024-12-22 15:13:10 +13:00
|
|
|
var path = new ResPath(TestMap);
|
|
|
|
|
Assert.That(mapLoader.TryLoadMap(path, out var map, out _), $"Failed to load test map {TestMap}");
|
|
|
|
|
mapId1 = map!.Value.Comp.MapId;
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveMap(mapId1, fileA));
|
2023-03-26 14:17:27 +13:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await server.WaitIdleAsync();
|
2024-12-24 18:57:52 +13:00
|
|
|
await using (var stream = userData.Open(fileA, FileMode.Open))
|
2023-03-26 14:17:27 +13:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
|
|
|
|
yamlA = await reader.ReadToEndAsync();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
server.RunTicks(5);
|
|
|
|
|
|
|
|
|
|
// Load & save the second map
|
|
|
|
|
server.Post(() =>
|
|
|
|
|
{
|
2024-12-22 15:13:10 +13:00
|
|
|
var path = new ResPath(TestMap);
|
|
|
|
|
Assert.That(mapLoader.TryLoadMap(path, out var map, out _), $"Failed to load test map {TestMap}");
|
|
|
|
|
mapId2 = map!.Value.Comp.MapId;
|
2024-12-24 18:57:52 +13:00
|
|
|
Assert.That(mapLoader.TrySaveMap(mapId2, fileB));
|
2023-03-26 14:17:27 +13:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await server.WaitIdleAsync();
|
|
|
|
|
|
2024-12-24 18:57:52 +13:00
|
|
|
await using (var stream = userData.Open(fileB, FileMode.Open))
|
2023-03-26 14:17:27 +13:00
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
|
|
|
|
yamlB = await reader.ReadToEndAsync();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Assert.That(yamlA, Is.EqualTo(yamlB));
|
|
|
|
|
|
2024-12-22 15:13:10 +13:00
|
|
|
testSystem.Enabled = false;
|
|
|
|
|
await server.WaitPost(() => mapSys.DeleteMap(mapId1));
|
|
|
|
|
await server.WaitPost(() => mapSys.DeleteMap(mapId2));
|
2023-08-25 02:56:51 +02:00
|
|
|
await pair.CleanReturnAsync();
|
2019-10-13 16:26:39 +02:00
|
|
|
}
|
2024-12-22 15:13:10 +13:00
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Simple system that modifies the data saved to a yaml file by removing the timestamp.
|
|
|
|
|
/// Required by some tests that validate that re-saving a map does not modify it.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private sealed class SaveLoadSaveTestSystem : EntitySystem
|
|
|
|
|
{
|
|
|
|
|
public bool Enabled;
|
|
|
|
|
public override void Initialize()
|
|
|
|
|
{
|
2024-12-24 18:57:52 +13:00
|
|
|
SubscribeLocalEvent<AfterSerializationEvent>(OnAfterSave);
|
2024-12-22 15:13:10 +13:00
|
|
|
}
|
|
|
|
|
|
2024-12-24 18:57:52 +13:00
|
|
|
private void OnAfterSave(AfterSerializationEvent ev)
|
2024-12-22 15:13:10 +13:00
|
|
|
{
|
|
|
|
|
if (!Enabled)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// Remove timestamp.
|
|
|
|
|
((MappingDataNode)ev.Node["meta"]).Remove("time");
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-19 13:46:01 +02:00
|
|
|
}
|
|
|
|
|
}
|