using System.IO.Compression; using LiquidCode.Tester.Worker.Services; using Microsoft.Extensions.Logging; using Moq; namespace LiquidCode.Tester.Worker.Tests; public class PolygonPackageParserTests : IDisposable { private readonly PackageParserService _service; private readonly string _testDirectory; public PolygonPackageParserTests() { var logger = new Mock>(); var xmlLogger = new Mock>(); var answerGenLogger = new Mock>(); var cppLogger = new Mock>(); var cppConfigMock = new Mock(); var polygonParser = new PolygonProblemXmlParser(xmlLogger.Object); var compilationFactory = new Mock(); var executionFactory = new Mock(); var answerGenerator = new AnswerGenerationService( compilationFactory.Object, executionFactory.Object, answerGenLogger.Object); var cppCompilation = new CppCompilationService(cppLogger.Object, cppConfigMock.Object); _service = new PackageParserService(logger.Object, polygonParser, answerGenerator, cppCompilation); _testDirectory = Path.Combine(Path.GetTempPath(), "PolygonPackageTests", Guid.NewGuid().ToString()); Directory.CreateDirectory(_testDirectory); } [Fact] public async Task ParsePackageAsync_PolygonPackageWithProblemXml_ParsesSuccessfully() { // Arrange var problemXml = @" 1000 268435456 2 tests/%02d tests/%02d.a "; var zipStream = CreatePolygonPackage(problemXml, new[] { ("tests/01", "input1"), ("tests/01.a", "output1"), ("tests/02", "input2"), ("tests/02.a", "output2") }); // Act var result = await _service.ParsePackageAsync(zipStream); // Assert Assert.NotNull(result); Assert.Equal(2, result.TestCases.Count); Assert.Equal(1000, result.DefaultTimeLimit); Assert.Equal(256, result.DefaultMemoryLimit); // Verify first test Assert.Equal(1, result.TestCases[0].Number); Assert.True(File.Exists(result.TestCases[0].InputFilePath)); Assert.True(File.Exists(result.TestCases[0].OutputFilePath)); Assert.Equal("input1", await File.ReadAllTextAsync(result.TestCases[0].InputFilePath)); Assert.Equal("output1", await File.ReadAllTextAsync(result.TestCases[0].OutputFilePath)); // Verify second test Assert.Equal(2, result.TestCases[1].Number); Assert.True(File.Exists(result.TestCases[1].InputFilePath)); Assert.True(File.Exists(result.TestCases[1].OutputFilePath)); // Cleanup if (Directory.Exists(result.WorkingDirectory)) { Directory.Delete(result.WorkingDirectory, true); } } [Fact] public async Task ParsePackageAsync_PolygonPackageMissingAnswerFiles_SkipsTests() { // Arrange var problemXml = @" 2000 536870912 3 tests/%02d tests/%02d.a "; var zipStream = CreatePolygonPackage(problemXml, new[] { ("tests/01", "input1"), ("tests/01.a", "output1"), ("tests/02", "input2"), // Missing 02.a ("tests/03", "input3"), ("tests/03.a", "output3") }); // Act var result = await _service.ParsePackageAsync(zipStream); // Assert Assert.NotNull(result); Assert.Equal(2, result.TestCases.Count); // Only tests with answer files Assert.Equal(1, result.TestCases[0].Number); Assert.Equal(3, result.TestCases[1].Number); // Test 2 skipped // Cleanup if (Directory.Exists(result.WorkingDirectory)) { Directory.Delete(result.WorkingDirectory, true); } } [Fact] public async Task ParsePackageAsync_NoProblemXml_FallsBackToLegacyFormat() { // Arrange - create package without problem.xml var zipStream = CreateLegacyPackage(new[] { ("tests/test1.in", "input1"), ("tests/test1.out", "output1"), ("tests/test2.in", "input2"), ("tests/test2.out", "output2") }); // Act var result = await _service.ParsePackageAsync(zipStream); // Assert Assert.NotNull(result); Assert.Equal(2, result.TestCases.Count); Assert.Equal(2000, result.DefaultTimeLimit); // Default values Assert.Equal(256, result.DefaultMemoryLimit); // Cleanup if (Directory.Exists(result.WorkingDirectory)) { Directory.Delete(result.WorkingDirectory, true); } } private MemoryStream CreatePolygonPackage(string problemXml, IEnumerable<(string fileName, string content)> files) { var memoryStream = new MemoryStream(); using (var archive = new ZipArchive(memoryStream, ZipArchiveMode.Create, true)) { // Add problem.xml var xmlEntry = archive.CreateEntry("problem.xml"); using var xmlStream = xmlEntry.Open(); using var xmlWriter = new StreamWriter(xmlStream); xmlWriter.Write(problemXml); // Add test files foreach (var (fileName, content) in files) { var entry = archive.CreateEntry(fileName); using var entryStream = entry.Open(); using var writer = new StreamWriter(entryStream); writer.Write(content); } } memoryStream.Position = 0; return memoryStream; } private MemoryStream CreateLegacyPackage(IEnumerable<(string fileName, string content)> files) { var memoryStream = new MemoryStream(); using (var archive = new ZipArchive(memoryStream, ZipArchiveMode.Create, true)) { foreach (var (fileName, content) in files) { var entry = archive.CreateEntry(fileName); using var entryStream = entry.Open(); using var writer = new StreamWriter(entryStream); writer.Write(content); } } memoryStream.Position = 0; return memoryStream; } public void Dispose() { if (Directory.Exists(_testDirectory)) { try { Directory.Delete(_testDirectory, true); } catch { // Ignore cleanup errors } } } }