mirror of
https://github.com/ckaczor/sqltoolsservice.git
synced 2026-01-14 09:59:48 -05:00
The main feature of this pull request is a new callback that's added to the query class that is called when a batch has completed execution and retrieval of results. This callback will send an event to the extension with the batch summary information. After that, the extension can submit subset requests for the resultsets of the batch. Other smaller changes in this pull request: Refactor to assign a batch a id when its created instead of when returning the list of batch summaries Passing the SelectionData around instead of extracting the values for it Moving creation of BatchSummary into the Batch class Retrieval of results is now permitted even if the entire query has not completed, as long as the batch requested has completed. Also note, this does not break the protocol. It adds a new event that a queryRunner can listen to, but it doesn't require it to be listened to. * Refactor to remove SectionData class in favor of BufferRange * Adding callback for batch completion that will let the extension know that a batch has completed execution * Refactoring to make progressive results work as per async query execution * Allowing retrieval of batch results while query is in progress * reverting global.json, whoops * Adding a few missing comments, and fixing a couple code style bugs * Using SelectionData everywhere again * One more missing comment
297 lines
11 KiB
C#
297 lines
11 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.Data;
|
|
using System.Data.Common;
|
|
using System.Data.SqlClient;
|
|
using System.IO;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.SqlServer.Management.Common;
|
|
using Microsoft.SqlServer.Management.SmoMetadataProvider;
|
|
using Microsoft.SqlServer.Management.SqlParser.Binder;
|
|
using Microsoft.SqlTools.ServiceLayer.Connection;
|
|
using Microsoft.SqlTools.ServiceLayer.Connection.Contracts;
|
|
using Microsoft.SqlTools.ServiceLayer.LanguageServices;
|
|
using Microsoft.SqlTools.ServiceLayer.QueryExecution;
|
|
using Microsoft.SqlTools.ServiceLayer.QueryExecution.Contracts;
|
|
using Microsoft.SqlTools.ServiceLayer.QueryExecution.DataStorage;
|
|
using Microsoft.SqlTools.ServiceLayer.SqlContext;
|
|
using Microsoft.SqlTools.ServiceLayer.Test.Utility;
|
|
using Microsoft.SqlTools.ServiceLayer.Workspace;
|
|
using Microsoft.SqlTools.ServiceLayer.Workspace.Contracts;
|
|
using Moq;
|
|
using Moq.Protected;
|
|
|
|
namespace Microsoft.SqlTools.ServiceLayer.Test.QueryExecution
|
|
{
|
|
public class Common
|
|
{
|
|
#region Constants
|
|
|
|
public const string InvalidQuery = "SELECT *** FROM sys.objects";
|
|
|
|
public const string NoOpQuery = "-- No ops here, just us chickens.";
|
|
|
|
public const int Ordinal = 0;
|
|
|
|
public const string OwnerUri = "testFile";
|
|
|
|
public const int StandardColumns = 5;
|
|
|
|
public const string StandardQuery = "SELECT * FROM sys.objects";
|
|
|
|
public const int StandardRows = 5;
|
|
|
|
public const string UdtQuery = "SELECT hierarchyid::Parse('/')";
|
|
|
|
public const SelectionData WholeDocument = null;
|
|
|
|
public static readonly ConnectionDetails StandardConnectionDetails = new ConnectionDetails
|
|
{
|
|
DatabaseName = "123",
|
|
Password = "456",
|
|
ServerName = "789",
|
|
UserName = "012"
|
|
};
|
|
|
|
public static readonly SelectionData SubsectionDocument = new SelectionData(0, 0, 2, 2);
|
|
|
|
#endregion
|
|
|
|
public static Dictionary<string, string>[] StandardTestData
|
|
{
|
|
get { return GetTestData(StandardRows, StandardColumns); }
|
|
}
|
|
|
|
#region Public Methods
|
|
|
|
public static Batch GetBasicExecutedBatch()
|
|
{
|
|
Batch batch = new Batch(StandardQuery, SubsectionDocument, 1, GetFileStreamFactory());
|
|
batch.Execute(CreateTestConnection(new[] {StandardTestData}, false), CancellationToken.None).Wait();
|
|
return batch;
|
|
}
|
|
|
|
public static Query GetBasicExecutedQuery()
|
|
{
|
|
ConnectionInfo ci = CreateTestConnectionInfo(new[] {StandardTestData}, false);
|
|
Query query = new Query(StandardQuery, ci, new QueryExecutionSettings(), GetFileStreamFactory());
|
|
query.Execute();
|
|
query.ExecutionTask.Wait();
|
|
return query;
|
|
}
|
|
|
|
public static Dictionary<string, string>[] GetTestData(int columns, int rows)
|
|
{
|
|
Dictionary<string, string>[] output = new Dictionary<string, string>[rows];
|
|
for (int row = 0; row < rows; row++)
|
|
{
|
|
Dictionary<string, string> rowDictionary = new Dictionary<string, string>();
|
|
for (int column = 0; column < columns; column++)
|
|
{
|
|
rowDictionary.Add(string.Format("column{0}", column), string.Format("val{0}{1}", column, row));
|
|
}
|
|
output[row] = rowDictionary;
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region FileStreamWriteMocking
|
|
|
|
public static IFileStreamFactory GetFileStreamFactory()
|
|
{
|
|
Mock<IFileStreamFactory> mock = new Mock<IFileStreamFactory>();
|
|
mock.Setup(fsf => fsf.GetReader(It.IsAny<string>()))
|
|
.Returns(new ServiceBufferFileStreamReader(new InMemoryWrapper(), It.IsAny<string>()));
|
|
mock.Setup(fsf => fsf.GetWriter(It.IsAny<string>(), It.IsAny<int>(), It.IsAny<int>()))
|
|
.Returns(new ServiceBufferFileStreamWriter(new InMemoryWrapper(), It.IsAny<string>(), 1024,
|
|
1024));
|
|
|
|
return mock.Object;
|
|
}
|
|
|
|
public class InMemoryWrapper : IFileStreamWrapper
|
|
{
|
|
private readonly MemoryStream memoryStream;
|
|
private bool readingOnly;
|
|
private readonly byte[] storage = new byte[8192];
|
|
|
|
public InMemoryWrapper()
|
|
{
|
|
memoryStream = new MemoryStream(storage);
|
|
}
|
|
|
|
public void Close()
|
|
{
|
|
memoryStream.Dispose();
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
// We'll dispose this via a special method
|
|
}
|
|
|
|
public void Flush()
|
|
{
|
|
if (readingOnly) { throw new InvalidOperationException(); }
|
|
}
|
|
|
|
public void Init(string fileName, int bufferSize, FileAccess fAccess)
|
|
{
|
|
readingOnly = fAccess == FileAccess.Read;
|
|
}
|
|
|
|
public int ReadData(byte[] buffer, int bytes)
|
|
{
|
|
return ReadData(buffer, bytes, memoryStream.Position);
|
|
}
|
|
|
|
public int ReadData(byte[] buffer, int bytes, long fileOffset)
|
|
{
|
|
memoryStream.Seek(fileOffset, SeekOrigin.Begin);
|
|
return memoryStream.Read(buffer, 0, bytes);
|
|
}
|
|
|
|
public int WriteData(byte[] buffer, int bytes)
|
|
{
|
|
if (readingOnly) { throw new InvalidOperationException(); }
|
|
memoryStream.Write(buffer, 0, bytes);
|
|
memoryStream.Flush();
|
|
return bytes;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region DbConnection Mocking
|
|
|
|
public static DbCommand CreateTestCommand(Dictionary<string, string>[][] data, bool throwOnRead)
|
|
{
|
|
var commandMock = new Mock<DbCommand> { CallBase = true };
|
|
var commandMockSetup = commandMock.Protected()
|
|
.Setup<DbDataReader>("ExecuteDbDataReader", It.IsAny<CommandBehavior>());
|
|
|
|
// Setup the expected behavior
|
|
if (throwOnRead)
|
|
{
|
|
var mockException = new Mock<DbException>();
|
|
mockException.SetupGet(dbe => dbe.Message).Returns("Message");
|
|
commandMockSetup.Throws(mockException.Object);
|
|
}
|
|
else
|
|
{
|
|
commandMockSetup.Returns(new TestDbDataReader(data));
|
|
}
|
|
|
|
|
|
return commandMock.Object;
|
|
}
|
|
|
|
public static DbConnection CreateTestConnection(Dictionary<string, string>[][] data, bool throwOnRead)
|
|
{
|
|
var connectionMock = new Mock<DbConnection> { CallBase = true };
|
|
connectionMock.Protected()
|
|
.Setup<DbCommand>("CreateDbCommand")
|
|
.Returns(CreateTestCommand(data, throwOnRead));
|
|
connectionMock.Setup(dbc => dbc.Open())
|
|
.Callback(() => connectionMock.SetupGet(dbc => dbc.State).Returns(ConnectionState.Open));
|
|
connectionMock.Setup(dbc => dbc.Close())
|
|
.Callback(() => connectionMock.SetupGet(dbc => dbc.State).Returns(ConnectionState.Closed));
|
|
|
|
return connectionMock.Object;
|
|
}
|
|
|
|
public static ISqlConnectionFactory CreateMockFactory(Dictionary<string, string>[][] data, bool throwOnRead)
|
|
{
|
|
var mockFactory = new Mock<ISqlConnectionFactory>();
|
|
mockFactory.Setup(factory => factory.CreateSqlConnection(It.IsAny<string>()))
|
|
.Returns(CreateTestConnection(data, throwOnRead));
|
|
|
|
return mockFactory.Object;
|
|
}
|
|
|
|
public static ConnectionInfo CreateTestConnectionInfo(Dictionary<string, string>[][] data, bool throwOnRead)
|
|
{
|
|
return new ConnectionInfo(CreateMockFactory(data, throwOnRead), OwnerUri, StandardConnectionDetails);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Service Mocking
|
|
|
|
public static void GetAutoCompleteTestObjects(
|
|
out TextDocumentPosition textDocument,
|
|
out ScriptFile scriptFile,
|
|
out ConnectionInfo connInfo
|
|
)
|
|
{
|
|
textDocument = new TextDocumentPosition
|
|
{
|
|
TextDocument = new TextDocumentIdentifier {Uri = OwnerUri},
|
|
Position = new Position
|
|
{
|
|
Line = 0,
|
|
Character = 0
|
|
}
|
|
};
|
|
|
|
|
|
connInfo = CreateTestConnectionInfo(null, false);
|
|
|
|
var srvConn = GetServerConnection(connInfo);
|
|
var metadataProvider = SmoMetadataProvider.CreateConnectedProvider(srvConn);
|
|
var binder = BinderProvider.CreateBinder(metadataProvider);
|
|
connInfo = Common.CreateTestConnectionInfo(null, false);
|
|
|
|
LanguageService.Instance.ScriptParseInfoMap.Add(textDocument.TextDocument.Uri, new ScriptParseInfo());
|
|
|
|
scriptFile = new ScriptFile {ClientFilePath = textDocument.TextDocument.Uri};
|
|
|
|
}
|
|
|
|
public static ServerConnection GetServerConnection(ConnectionInfo connection)
|
|
{
|
|
string connectionString = ConnectionService.BuildConnectionString(connection.ConnectionDetails);
|
|
var sqlConnection = new SqlConnection(connectionString);
|
|
return new ServerConnection(sqlConnection);
|
|
}
|
|
|
|
public static async Task<QueryExecutionService> GetPrimedExecutionService(ISqlConnectionFactory factory, bool isConnected, WorkspaceService<SqlToolsSettings> workspaceService)
|
|
{
|
|
var connectionService = new ConnectionService(factory);
|
|
if (isConnected)
|
|
{
|
|
await connectionService.Connect(new ConnectParams
|
|
{
|
|
Connection = StandardConnectionDetails,
|
|
OwnerUri = OwnerUri
|
|
});
|
|
}
|
|
return new QueryExecutionService(connectionService, workspaceService) {BufferFileStreamFactory = GetFileStreamFactory()};
|
|
}
|
|
|
|
public static WorkspaceService<SqlToolsSettings> GetPrimedWorkspaceService()
|
|
{
|
|
// Set up file for returning the query
|
|
var fileMock = new Mock<ScriptFile>();
|
|
fileMock.SetupGet(file => file.Contents).Returns(StandardQuery);
|
|
|
|
// Set up workspace mock
|
|
var workspaceService = new Mock<WorkspaceService<SqlToolsSettings>>();
|
|
workspaceService.Setup(service => service.Workspace.GetFile(It.IsAny<string>()))
|
|
.Returns(fileMock.Object);
|
|
|
|
return workspaceService.Object;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|