Files
sqltoolsservice/test/Microsoft.SqlTools.ServiceLayer.UnitTests/QueryExecution/SaveResults/SerializationServiceTests.cs
David Shiflet 839acf67cd Convert most tools service tests to nunit (#1037)
* Remove xunit dependency from testdriver

* swap expected/actual as needed

* Convert Test.Common to nunit

* port hosting unit tests to nunit

* port batchparser integration tests to nunit

* port testdriver.tests to nunit

* fix target to copy dependency

* port servicelayer unittests to nunit

* more unit test fixes

* port integration tests to nunit

* fix test method type

* try using latest windows build for PRs

* reduce test memory use
2020-08-05 13:43:14 -04:00

351 lines
14 KiB
C#

//
// Copyright (c) Microsoft. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
//
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
using System.Xml;
using Microsoft.SqlTools.Extensibility;
using Microsoft.SqlTools.Hosting.Protocol;
using Microsoft.SqlTools.ServiceLayer.QueryExecution;
using Microsoft.SqlTools.ServiceLayer.QueryExecution.Contracts;
using Microsoft.SqlTools.ServiceLayer.Test.Common;
using Microsoft.SqlTools.ServiceLayer.Test.Common.RequestContextMocking;
using Moq;
using Newtonsoft.Json;
using NUnit.Framework;
namespace Microsoft.SqlTools.ServiceLayer.UnitTests.QueryExecution.SaveResults
{
public class SerializationServiceTests
{
private static readonly DbCellValue[][] DefaultData = new DbCellValue[3][] {
new DbCellValue[] {
new DbCellValue() { DisplayValue = "1", IsNull = false },
new DbCellValue() { DisplayValue = "Hello", IsNull = false },
new DbCellValue() { DisplayValue = "false", IsNull = false },
},
new DbCellValue[] {
new DbCellValue() { DisplayValue = "2", IsNull = false },
new DbCellValue() { DisplayValue = null, IsNull = true },
new DbCellValue() { DisplayValue = "true", IsNull = false },
},
new DbCellValue[] {
new DbCellValue() { DisplayValue = "3", IsNull = false },
new DbCellValue() { DisplayValue = "World", IsNull = false },
new DbCellValue() { DisplayValue = "True", IsNull = false },
}
};
private static readonly ColumnInfo[] DefaultColumns = {
new ColumnInfo("IntCol", "Int"),
new ColumnInfo("StringCol", "NVarChar"),
new ColumnInfo("BitCol", "Bit")
};
public SerializationServiceTests()
{
HostMock = new Mock<IProtocolEndpoint>();
ServiceProvider = ExtensionServiceProvider.CreateDefaultServiceProvider();
HostLoader.InitializeHostedServices(ServiceProvider, HostMock.Object);
SerializationService = ServiceProvider.GetService<SerializationService>();
}
protected ExtensionServiceProvider ServiceProvider { get; private set; }
protected Mock<IProtocolEndpoint> HostMock { get; private set; }
protected SerializationService SerializationService { get; private set; }
[Test]
public async Task SaveResultsAsCsvNoHeaderSuccess()
{
await TestSaveAsCsvSuccess(false);
}
[Test]
public async Task SaveResultsAsCsvWithHeaderSuccess()
{
await TestSaveAsCsvSuccess(true);
}
private async Task TestSaveAsCsvSuccess(bool includeHeaders)
{
await this.RunFileSaveTest(async (filePath) =>
{
// Given:
// ... A simple data set that requires 1 message
SerializeDataStartRequestParams saveParams = new SerializeDataStartRequestParams()
{
FilePath = filePath,
Columns = DefaultColumns,
Rows = DefaultData,
IsLastBatch = true,
SaveFormat = "csv",
IncludeHeaders = includeHeaders
};
// When: I attempt to save this to a file
var efv = new EventFlowValidator<SerializeDataResult>()
.AddStandardResultValidator()
.Complete();
await SerializationService.RunSerializeStartRequest(saveParams, efv.Object);
// Then:
// ... There should not have been an error
efv.Validate();
// ... And the file should look as expected
VerifyContents.VerifyCsvMatchesData(saveParams.Rows, saveParams.Columns, saveParams.IncludeHeaders, saveParams.FilePath);
});
}
[Test]
public async Task SaveResultsAsCsvNoHeaderMultiRequestSuccess()
{
await TestSaveAsCsvMultiRequestSuccess(false);
}
[Test]
public async Task SaveResultsAsCsvWithHeaderMultiRequestSuccess()
{
await TestSaveAsCsvMultiRequestSuccess(true);
}
private async Task TestSaveAsCsvMultiRequestSuccess(bool includeHeaders)
{
Action<SerializeDataStartRequestParams> setParams = (serializeParams) => {
serializeParams.SaveFormat = "csv";
serializeParams.IncludeHeaders = includeHeaders;
};
Action<string> validation = (filePath) => {
VerifyContents.VerifyCsvMatchesData(DefaultData, DefaultColumns, includeHeaders, filePath);
};
await this.TestSerializeDataMultiRequestSuccess(setParams, validation);
}
[Test]
public async Task SaveAsJsonMultiRequestSuccess()
{
Action<SerializeDataStartRequestParams> setParams = (serializeParams) => {
serializeParams.SaveFormat = "json";
};
Action<string> validation = (filePath) => {
VerifyContents.VerifyJsonMatchesData(DefaultData, DefaultColumns, filePath);
};
await this.TestSerializeDataMultiRequestSuccess(setParams, validation);
}
[Test]
public async Task SaveAsXmlMultiRequestSuccess()
{
Action<SerializeDataStartRequestParams> setParams = (serializeParams) => {
serializeParams.SaveFormat = "xml";
};
Action<string> validation = (filePath) => {
VerifyContents.VerifyXmlMatchesData(DefaultData, DefaultColumns, filePath);
};
await this.TestSerializeDataMultiRequestSuccess(setParams, validation);
}
private async Task TestSerializeDataMultiRequestSuccess(Action<SerializeDataStartRequestParams> setStandardParams, Action<string> verify)
{
await this.RunFileSaveTest(async (filePath) =>
{
// Given:
// ... A simple data set that requires 3 messages
var startParams = new SerializeDataStartRequestParams()
{
FilePath = filePath,
Columns = DefaultColumns,
Rows = new DbCellValue[][] { DefaultData[0] },
IsLastBatch = false
};
setStandardParams(startParams);
// When I send all 3 messages
await SendAndVerifySerializeStartRequest(startParams);
var continueParams = new SerializeDataContinueRequestParams()
{
FilePath = filePath,
Rows = new DbCellValue[][] { DefaultData[1] },
IsLastBatch = false
};
await SendAndVerifySerializeContinueRequest(continueParams);
continueParams.Rows = new DbCellValue[][] { DefaultData[2] };
continueParams.IsLastBatch = true;
await SendAndVerifySerializeContinueRequest(continueParams);
// ... Then the file should look as expected
verify(filePath);
});
}
private async Task SendAndVerifySerializeStartRequest(SerializeDataStartRequestParams request1)
{
// When: I attempt to save this to a file
var efv = new EventFlowValidator<SerializeDataResult>()
.AddStandardResultValidator()
.Complete();
await SerializationService.RunSerializeStartRequest(request1, efv.Object);
// Then:
// ... There should not have been an error
efv.Validate();
}
private async Task SendAndVerifySerializeContinueRequest(SerializeDataContinueRequestParams request1)
{
// When: I attempt to save this to a file
var efv = new EventFlowValidator<SerializeDataResult>()
.AddStandardResultValidator()
.Complete();
await SerializationService.RunSerializeContinueRequest(request1, efv.Object);
// Then:
// ... There should not have been an error
efv.Validate();
}
private Task RunFileSaveTest(Func<string, Task> doSave)
{
using (SelfCleaningTempFile tempFile = new SelfCleaningTempFile())
{
return doSave(tempFile.FilePath);
}
}
}
public static class SerializeDataEventFlowValidatorExtensions
{
public static EventFlowValidator<SerializeDataResult> AddStandardResultValidator(
this EventFlowValidator<SerializeDataResult> efv)
{
return efv.AddResultValidation(r =>
{
Assert.NotNull(r);
Assert.Null(r.Messages);
Assert.True(r.Succeeded);
});
}
}
public static class VerifyContents
{
public static void VerifyCsvMatchesData(DbCellValue[][] data, ColumnInfo[] columns, bool includeHeaders, string filePath)
{
Assert.True(File.Exists(filePath), "Expected file to have been written");
string[] lines = File.ReadAllLines(filePath);
int expectedLength = includeHeaders ? data.Length + 1 : data.Length;
Assert.AreEqual(expectedLength, lines.Length);
int lineIndex = 0;
if (includeHeaders)
{
AssertLineEquals(lines[lineIndex], columns.Select((c) => c.Name).ToArray());
lineIndex++;
}
for (int dataIndex =0; dataIndex < data.Length && lineIndex < lines.Length; dataIndex++, lineIndex++)
{
AssertLineEquals(lines[lineIndex], data[dataIndex].Select((d) => GetCsvPrintValue(d)).ToArray());
}
}
private static string GetCsvPrintValue(DbCellValue d)
{
return d.IsNull ? "NULL" : d.DisplayValue;
}
private static void AssertLineEquals(string line, string[] expected)
{
var actual = line.Split(',');
Assert.True(actual.Length == expected.Length, $"Line '{line}' does not match values {string.Join(",", expected)}");
for (int i = 0; i < actual.Length; i++)
{
Assert.True(expected[i] == actual[i], $"Line '{line}' does not match values '{string.Join(",", expected)}' as '{expected[i]}' does not equal '{actual[i]}'");
}
}
public static void VerifyJsonMatchesData(DbCellValue[][] data, ColumnInfo[] columns, string filePath)
{
// ... Upon deserialization to an array of dictionaries
Assert.True(File.Exists(filePath), "Expected file to have been written");
string output = File.ReadAllText(filePath);
Dictionary<string, object>[] outputObject =
JsonConvert.DeserializeObject<Dictionary<string, object>[]>(output);
// ... There should be 2 items in the array,
// ... The item should have three fields, and three values, assigned appropriately
// ... The deserialized values should match the display value
Assert.AreEqual(data.Length, outputObject.Length);
for (int rowIndex = 0; rowIndex < outputObject.Length; rowIndex++)
{
Dictionary<string,object> item = outputObject[rowIndex];
Assert.AreEqual(columns.Length, item.Count);
for (int columnIndex = 0; columnIndex < columns.Length; columnIndex++)
{
var key = columns[columnIndex].Name;
Assert.True(item.ContainsKey(key));
DbCellValue value = data[rowIndex][columnIndex];
object expectedValue = GetJsonExpectedValue(value, columns[columnIndex]);
Assert.AreEqual(expectedValue, item[key]);
}
}
}
private static object GetJsonExpectedValue(DbCellValue value, ColumnInfo column)
{
if (value.IsNull)
{
return null;
}
else if (column.DataTypeName == "Int")
{
return Int64.Parse(value.DisplayValue.ToLower());
}
else if (column.DataTypeName == "Bit")
{
return Boolean.Parse(value.DisplayValue.ToLower());
}
return value.DisplayValue;
}
public static void VerifyXmlMatchesData(DbCellValue[][] data, ColumnInfo[] columns, string filePath)
{
// ... Upon deserialization to an array of dictionaries
Assert.True(File.Exists(filePath), "Expected file to have been written");
string output = File.ReadAllText(filePath);
XmlDocument xmlDoc = new XmlDocument();
xmlDoc.LoadXml(output);
// ... There should be 2 items in the array,
// ... The item should have three fields, and three values, assigned appropriately
// ... The deserialized values should match the display value
string xpath = "data/row";
var rows = xmlDoc.SelectNodes(xpath);
Assert.AreEqual(data.Length, rows.Count);
for (int rowIndex = 0; rowIndex < rows.Count; rowIndex++)
{
var rowValue = rows.Item(rowIndex);
var xmlCols = rowValue.ChildNodes.Cast<XmlNode>().ToArray();
Assert.AreEqual(columns.Length, xmlCols.Length);
for (int columnIndex = 0; columnIndex < columns.Length; columnIndex++)
{
var columnName = columns[columnIndex].Name;
var xmlColumn = xmlCols.FirstOrDefault(x => x.Name == columnName);
Assert.NotNull(xmlColumn);
DbCellValue value = data[rowIndex][columnIndex];
object expectedValue = GetXmlExpectedValue(value);
Assert.AreEqual(expectedValue, xmlColumn.InnerText);
}
}
}
private static string GetXmlExpectedValue(DbCellValue d)
{
return d.IsNull ? "" : d.DisplayValue;
}
}
}