mirror of
https://github.com/ckaczor/sqltoolsservice.git
synced 2026-01-14 01:25:40 -05:00
* Added endpoint for fetching all notebook jobs * Refractored NotebookJobInfo to AgentNotebookInfo to make it more consistent with the rest of the codebase * Added Notebook History endpoint in contracts. * Added Create, Update, Delete notebook endpoints. Also added separate fetch template, materialized notebook endpoints. This will make the Notebook Request and Notebook History responses lighter. * AgentNotebookInfo is now derived from AgentJobInfo * added fetch noteook history endpoint * Added fetching materialized notebook endpoint * Added code for cleaning up the directory * Added create notebook api * Added Update and delete notebook job * Fixed notebook history API * Added last run info to the script and template folder * Added execute database feature for notebook Jobs * SQL commands are now using sqlparameters to prevent any injection attacks * Changed rundate and runtime to string to preserve leading zeros * integration test for agentnotebooks api * Made some changes mentioned in PR * Refactored the code, removed enpoint logic from the notebook handler and wrote test cases * changes select statements, fixed a bug in the test job cleanup and fixed other stuff mentioned in the PR. * added notebook_error column in notebook history select statement * Added get template notebook endpoint
288 lines
15 KiB
C#
288 lines
15 KiB
C#
using System;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.SqlTools.Hosting.Protocol;
|
|
using Microsoft.SqlTools.ServiceLayer.Agent;
|
|
using Microsoft.SqlTools.ServiceLayer.Agent.Contracts;
|
|
using Microsoft.SqlTools.ServiceLayer.Test.Common;
|
|
using Microsoft.SqlTools.ServiceLayer.Utility;
|
|
using Microsoft.SqlTools.ServiceLayer.Management;
|
|
using Microsoft.SqlTools.ServiceLayer.IntegrationTests.Utility;
|
|
using Moq;
|
|
using Xunit;
|
|
|
|
namespace Microsoft.SqlTools.ServiceLayer.IntegrationTests.Agent
|
|
{
|
|
public class AgentNotebookTests
|
|
{
|
|
/// <summary>
|
|
/// Test case for fetch notebook jobs Request Handler
|
|
/// </summary>
|
|
[Fact]
|
|
public async Task TestHandleAgentNotebooksRequest()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
var fetchNotebooksContext = new Mock<RequestContext<AgentNotebooksResult>>();
|
|
|
|
fetchNotebooksContext.Setup(x => x.SendResult(It.IsAny<AgentNotebooksResult>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleAgentNotebooksRequest(new AgentNotebooksParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri
|
|
}, fetchNotebooksContext.Object);
|
|
|
|
fetchNotebooksContext.Verify(x => x.SendResult(It.Is<AgentNotebooksResult>(p => p.Success == true)));
|
|
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tests the create job helper function
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestAgentNotebookCreateHelper()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
AgentNotebookInfo notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
Assert.Equal(false, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
notebook = AgentTestUtils.SetupNotebookJob(connectionResult).Result;
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
await AgentTestUtils.CleanupNotebookJob(connectionResult, notebook);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tests the create job request handler with an invalid file path
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestHandleCreateAgentNotebookRequestWithInvalidTemplatePath()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
AgentNotebookInfo notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
var createNotebookContext = new Mock<RequestContext<CreateAgentNotebookResult>>();
|
|
createNotebookContext.Setup(x => x.SendResult(It.IsAny<CreateAgentNotebookResult>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleCreateAgentNotebookRequest(new CreateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = "garbargepath"
|
|
}, createNotebookContext.Object);
|
|
|
|
createNotebookContext.Verify(x => x.SendResult(It.Is<CreateAgentNotebookResult>(p => p.Success == false)));
|
|
Assert.Equal(false, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// creating a job with duplicate name
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestDuplicateJobCreation()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
AgentNotebookInfo notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
var createNotebookContext = new Mock<RequestContext<CreateAgentNotebookResult>>();
|
|
createNotebookContext.Setup(x => x.SendResult(It.IsAny<CreateAgentNotebookResult>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleCreateAgentNotebookRequest(new CreateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = AgentTestUtils.CreateTemplateNotebookFile()
|
|
}, createNotebookContext.Object);
|
|
|
|
createNotebookContext.Verify(x => x.SendResult(It.Is<CreateAgentNotebookResult>(p => p.Success == true)));
|
|
await service.HandleCreateAgentNotebookRequest(new CreateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = AgentTestUtils.CreateTemplateNotebookFile()
|
|
}, createNotebookContext.Object);
|
|
createNotebookContext.Verify(x => x.SendResult(It.Is<CreateAgentNotebookResult>(p => p.Success == false)));
|
|
await AgentTestUtils.CleanupNotebookJob(connectionResult, notebook);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tests the create notebook job handler
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestCreateAgentNotebookHandler()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
AgentNotebookInfo notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
var createNotebookContext = new Mock<RequestContext<CreateAgentNotebookResult>>();
|
|
createNotebookContext.Setup(x => x.SendResult(It.IsAny<CreateAgentNotebookResult>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleCreateAgentNotebookRequest(new CreateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = AgentTestUtils.CreateTemplateNotebookFile()
|
|
}, createNotebookContext.Object);
|
|
createNotebookContext.Verify(x => x.SendResult(It.Is<CreateAgentNotebookResult>(p => p.Success == true)));
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
var createdNotebook = AgentTestUtils.GetNotebook(connectionResult, notebook.Name);
|
|
await AgentTestUtils.CleanupNotebookJob(connectionResult, createdNotebook);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tests the delete notebook job handler
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestDeleteAgentNotebookHandler()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
//creating a notebook job
|
|
AgentNotebookInfo notebook = AgentTestUtils.SetupNotebookJob(connectionResult).Result;
|
|
//verifying it's getting created
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
//deleting the notebook job
|
|
var deleteNotebookContext = new Mock<RequestContext<ResultStatus>>();
|
|
deleteNotebookContext.Setup(x => x.SendResult(It.IsAny<ResultStatus>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleDeleteAgentNotebooksRequest(new DeleteAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook
|
|
}, deleteNotebookContext.Object);
|
|
deleteNotebookContext.Verify(x => x.SendResult(It.Is<ResultStatus>(p => p.Success == true)));
|
|
//verifying if the job is deleted
|
|
Assert.Equal(false, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// deleting a existing notebook job
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestDeleteNonExistentJob()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
//getting a test notebook object
|
|
var notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
|
|
var deleteNotebookContext = new Mock<RequestContext<ResultStatus>>();
|
|
deleteNotebookContext.Setup(x => x.SendResult(It.IsAny<ResultStatus>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleDeleteAgentNotebooksRequest(new DeleteAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook
|
|
}, deleteNotebookContext.Object);
|
|
//endpoint should error out
|
|
deleteNotebookContext.Verify(x => x.SendResult(It.Is<ResultStatus>(p => p.Success == false)));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// updating a non existing notebook job
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestUpdateNonExistentJob()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
//getting a test notebook object
|
|
AgentNotebookInfo notebook = AgentTestUtils.GetTestNotebookInfo("myTestNotebookJob" + Guid.NewGuid().ToString(), "master");
|
|
|
|
var updateNotebookContext = new Mock<RequestContext<UpdateAgentNotebookResult>>();
|
|
updateNotebookContext.Setup(x => x.SendResult(It.IsAny<UpdateAgentNotebookResult>())).Returns(Task.FromResult(new object()));
|
|
await service.HandleUpdateAgentNotebookRequest(new UpdateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = AgentTestUtils.CreateTemplateNotebookFile()
|
|
}, updateNotebookContext.Object);
|
|
// enpoint should error out
|
|
updateNotebookContext.Verify(x => x.SendResult(It.Is<UpdateAgentNotebookResult>(p => p.Success == false)));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// update notebook handler with garbage path
|
|
/// </summary>
|
|
[Fact]
|
|
internal async Task TestUpdateWithGarbagePath()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
|
|
//seting up a temp notebook job
|
|
var notebook = AgentTestUtils.SetupNotebookJob(connectionResult).Result;
|
|
//verifying that the notebook is created
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
|
|
var updateNotebookContext = new Mock<RequestContext<UpdateAgentNotebookResult>>();
|
|
updateNotebookContext.Setup(x => x.SendResult(It.IsAny<UpdateAgentNotebookResult>())).Returns(Task.FromResult(new object()));
|
|
//calling the endpoint with a garbage path
|
|
await service.HandleUpdateAgentNotebookRequest(new UpdateAgentNotebookParams()
|
|
{
|
|
OwnerUri = connectionResult.ConnectionInfo.OwnerUri,
|
|
Notebook = notebook,
|
|
TemplateFilePath = "garbargepath"
|
|
}, updateNotebookContext.Object);
|
|
//the enpoint should return false
|
|
updateNotebookContext.Verify(x => x.SendResult(It.Is<UpdateAgentNotebookResult>(p => p.Success == false)));
|
|
|
|
//cleaning up the job
|
|
await AgentTestUtils.CleanupNotebookJob(connectionResult, notebook);
|
|
Assert.Equal(false, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
}
|
|
}
|
|
|
|
[Fact]
|
|
internal async Task TestDeletingUpdatedJob()
|
|
{
|
|
using (SelfCleaningTempFile queryTempFile = new SelfCleaningTempFile())
|
|
{
|
|
var service = new AgentService();
|
|
var connectionResult = await LiveConnectionHelper.InitLiveConnectionInfoAsync("master", queryTempFile.FilePath);
|
|
|
|
//seting up a temp notebook job
|
|
var notebook = AgentTestUtils.SetupNotebookJob(connectionResult).Result;
|
|
//verifying that the notebook is created
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
|
|
var originalName = notebook.Name;
|
|
//Changing the notebookName
|
|
notebook.Name = "myTestNotebookJob" + Guid.NewGuid().ToString();
|
|
|
|
Assert.Equal(false, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
|
|
await AgentNotebookHelper.UpdateNotebook(
|
|
service,
|
|
connectionResult.ConnectionInfo.OwnerUri,
|
|
originalName,
|
|
notebook,
|
|
null,
|
|
ManagementUtils.asRunType(0)
|
|
);
|
|
|
|
Assert.Equal(true, AgentTestUtils.VerifyNotebook(connectionResult, notebook));
|
|
|
|
//cleaning up the job
|
|
await AgentTestUtils.CleanupNotebookJob(connectionResult, notebook);
|
|
}
|
|
}
|
|
}
|
|
} |