using AVEVA.Historian.Client.Models; using AVEVA.Historian.Client.Protocol; using AVEVA.Historian.Client.Wcf; namespace AVEVA.Historian.Client.Tests; /// /// Golden tests for the ExeC/GetR SQL result codec. The fixture is the real, clean GetR /// pResultBuff captured from the live server for the benign query /// SELECT 1 AS ProbeValue (no sensitive data) — an NRBF stream wrapping a /// System.Data.DataTable serialized with SerializationFormat.Xml. It is the /// contract-level byte[] the WCF channel reassembles (free of MDAS transport chunk markers). /// public sealed class WcfSqlResultProtocolTests { // GetR pResultBuff for "SELECT 1 AS ProbeValue" (1232 bytes), base64. Captured via the // AVEVA_HISTORIAN_SQL_DUMP diagnostic in HistorianWcfSqlClient. private const string GetRStreamBase64 = "AAEAAAD/////AQAAAAAAAAAMAgAAAE5TeXN0ZW0uRGF0YSwgVmVyc2lvbj00LjAuMC4wLCBDdWx0dXJlPW5ldXRyYWwsIFB1Ymxp" + "Y0tleVRva2VuPWI3N2E1YzU2MTkzNGUwODkFAQAAABVTeXN0ZW0uRGF0YS5EYXRhVGFibGUDAAAAGURhdGFUYWJsZS5SZW1vdGlu" + "Z1ZlcnNpb24JWG1sU2NoZW1hC1htbERpZmZHcmFtAwEBDlN5c3RlbS5WZXJzaW9uAgAAAAkDAAAABgQAAACJBTw/eG1sIHZlcnNp" + "b249IjEuMCIgZW5jb2Rpbmc9InV0Zi0xNiI/Pg0KPHhzOnNjaGVtYSB4bWxucz0iIiB4bWxuczp4cz0iaHR0cDovL3d3dy53My5v" + "cmcvMjAwMS9YTUxTY2hlbWEiIHhtbG5zOm1zZGF0YT0idXJuOnNjaGVtYXMtbWljcm9zb2Z0LWNvbTp4bWwtbXNkYXRhIj4NCiAg" + "PHhzOmVsZW1lbnQgbmFtZT0iVGFibGUxIj4NCiAgICA8eHM6Y29tcGxleFR5cGU+DQogICAgICA8eHM6c2VxdWVuY2U+DQogICAg" + "ICAgIDx4czplbGVtZW50IG5hbWU9IlByb2JlVmFsdWUiIG1zZGF0YTpSZWFkT25seT0idHJ1ZSIgdHlwZT0ieHM6aW50IiBtc2Rh" + "dGE6dGFyZ2V0TmFtZXNwYWNlPSIiIC8+DQogICAgICA8L3hzOnNlcXVlbmNlPg0KICAgIDwveHM6Y29tcGxleFR5cGU+DQogIDwv" + "eHM6ZWxlbWVudD4NCiAgPHhzOmVsZW1lbnQgbmFtZT0idG1wRGF0YVNldCIgbXNkYXRhOklzRGF0YVNldD0idHJ1ZSIgbXNkYXRh" + "Ok1haW5EYXRhVGFibGU9IlRhYmxlMSIgbXNkYXRhOlVzZUN1cnJlbnRMb2NhbGU9InRydWUiPg0KICAgIDx4czpjb21wbGV4VHlw" + "ZT4NCiAgICAgIDx4czpjaG9pY2UgbWluT2NjdXJzPSIwIiBtYXhPY2N1cnM9InVuYm91bmRlZCIgLz4NCiAgICA8L3hzOmNvbXBs" + "ZXhUeXBlPg0KICA8L3hzOmVsZW1lbnQ+DQo8L3hzOnNjaGVtYT4GBQAAAJoCPGRpZmZncjpkaWZmZ3JhbSB4bWxuczptc2RhdGE9" + "InVybjpzY2hlbWFzLW1pY3Jvc29mdC1jb206eG1sLW1zZGF0YSIgeG1sbnM6ZGlmZmdyPSJ1cm46c2NoZW1hcy1taWNyb3NvZnQt" + "Y29tOnhtbC1kaWZmZ3JhbS12MSI+DQogIDx0bXBEYXRhU2V0Pg0KICAgIDxUYWJsZTEgZGlmZmdyOmlkPSJUYWJsZTExIiBtc2Rh" + "dGE6cm93T3JkZXI9IjAiPg0KICAgICAgPFByb2JlVmFsdWU+MTwvUHJvYmVWYWx1ZT4NCiAgICA8L1RhYmxlMT4NCiAgPC90bXBE" + "YXRhU2V0Pg0KPC9kaWZmZ3I6ZGlmZmdyYW0+BAMAAAAOU3lzdGVtLlZlcnNpb24EAAAABl9NYWpvcgZfTWlub3IGX0J1aWxkCV9S" + "ZXZpc2lvbgAAAAAICAgIAgAAAAAAAAD//////////ws="; [Fact] public void Parse_DataTableStream_ExtractsColumnAndTypedRow() { byte[] stream = Convert.FromBase64String(GetRStreamBase64); HistorianSqlResult result = HistorianSqlResultProtocol.Parse(stream, returnValue: 0); HistorianSqlColumn column = Assert.Single(result.Columns); Assert.Equal("ProbeValue", column.Name); Assert.Equal("xs:int", column.SchemaType); IReadOnlyList row = Assert.Single(result.Rows); object? value = Assert.Single(row); Assert.Equal(1, value); // typed to int per the xs:int schema Assert.IsType(value); Assert.Equal(0, result.ReturnValue); } [Fact] public void Parse_PreservesReturnValue() { byte[] stream = Convert.FromBase64String(GetRStreamBase64); HistorianSqlResult result = HistorianSqlResultProtocol.Parse(stream, returnValue: 7); Assert.Equal(7, result.ReturnValue); } [Fact] public void Parse_EmptyBuffer_ReturnsEmptyResult() { HistorianSqlResult result = HistorianSqlResultProtocol.Parse(ReadOnlyMemory.Empty, returnValue: 3); Assert.Empty(result.Columns); Assert.Empty(result.Rows); Assert.Equal(3, result.ReturnValue); } [Fact] public void Parse_NonNrbfBuffer_Throws() { byte[] garbage = [0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08]; Assert.Throws( () => HistorianSqlResultProtocol.Parse(garbage, returnValue: 0)); } }