Skip to content

Commit

Permalink
Merge pull request #305 from DFE-Digital/trust-performance
Browse files Browse the repository at this point in the history
improve performance of trust search by batching sql commands
  • Loading branch information
jordan-beard authored Feb 22, 2023
2 parents 1eb5b57 + 9f0fce0 commit 4100013
Show file tree
Hide file tree
Showing 9 changed files with 87 additions and 30 deletions.
8 changes: 4 additions & 4 deletions TramsDataApi.Test/Controllers/TrustsControllerTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@ public void SearchTrusts_ReturnsEmptySetOfTrustSummaries_WhenNoTrustsFound()
var companiesHouseNumber = "Mockcompanieshousenumber";

var searchTrusts = new Mock<ISearchTrusts>();
searchTrusts.Setup(s => s.Execute(1, 10, groupName, ukprn, companiesHouseNumber))
searchTrusts.Setup(s => s.Execute(1, 10, groupName, ukprn, companiesHouseNumber, true))
.Returns((new List<TrustSummaryResponse>(), 0));

var controller = new TrustsController(new Mock<IGetTrustByUkprn>().Object, searchTrusts.Object, mockLogger.Object);
Expand All @@ -99,7 +99,7 @@ public void SearchTrusts_ByGroupNameAndCompaniesHouseNumber_ReturnsListOfTrustSu
.Build();

var searchTrusts = new Mock<ISearchTrusts>();
searchTrusts.Setup(s => s.Execute(1, 10, groupName, null, companiesHouseNumber))
searchTrusts.Setup(s => s.Execute(1, 10, groupName, null, companiesHouseNumber, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var controller = new TrustsController(new Mock<IGetTrustByUkprn>().Object, searchTrusts.Object, mockLogger.Object);
Expand All @@ -120,7 +120,7 @@ public void SearchTrusts_ByUrn_ReturnsListOfTrustSummaries_WhenTrustsAreFound()
.Build();

var searchTrusts = new Mock<ISearchTrusts>();
searchTrusts.Setup(s => s.Execute(1, 10, null, ukprn, null))
searchTrusts.Setup(s => s.Execute(1, 10, null, ukprn, null, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var controller = new TrustsController(new Mock<IGetTrustByUkprn>().Object, searchTrusts.Object, mockLogger.Object);
Expand All @@ -135,7 +135,7 @@ public void SearchTrusts_WithNoParams_ReturnsAllTrusts()
var expectedTrustSummaries = Builder<TrustSummaryResponse>.CreateListOfSize(5).Build();

var searchTrusts = new Mock<ISearchTrusts>();
searchTrusts.Setup(s => s.Execute(1, 10, null, null, null))
searchTrusts.Setup(s => s.Execute(1, 10, null, null, null, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var controller = new TrustsController(new Mock<IGetTrustByUkprn>().Object, searchTrusts.Object, mockLogger.Object);
Expand Down
10 changes: 5 additions & 5 deletions TramsDataApi.Test/Controllers/TrustsControllerV2Tests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,7 @@ public void SearchTrusts_ReturnsEmptySetOfTrustSummaries_WhenNoTrustsFound()
NextPageUrl = null
};

_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, groupName, ukprn, companiesHouseNumber))
_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, groupName, ukprn, companiesHouseNumber, true))
.Returns((new List<TrustSummaryResponse>(), 0));

var result = _controller.SearchTrusts(groupName, ukprn, companiesHouseNumber, 1, 10);
Expand Down Expand Up @@ -126,7 +126,7 @@ public void SearchTrusts_ByGroupNameAndCompaniesHouseNumber_ReturnsListOfTrustSu
NextPageUrl = null
};

_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, groupName, null, companiesHouseNumber))
_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, groupName, null, companiesHouseNumber, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var result = _controller.SearchTrusts(groupName, null, companiesHouseNumber, 1, 10);
Expand All @@ -153,7 +153,7 @@ public void SearchTrusts_ByUrn_ReturnsListOfTrustSummaries_WhenTrustsAreFound()
NextPageUrl = null
};

_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, null, ukprn, null))
_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, null, ukprn, null, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var result = _controller.SearchTrusts(null, ukprn, null, 1, 10);
Expand All @@ -172,7 +172,7 @@ public void SearchTrusts_WithNoParams_ReturnsAllTrusts()
RecordCount = 5,
NextPageUrl = null
};
_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, null, null, null))
_mockSearchTrustsUseCase.Setup(s => s.Execute(1, 10, null, null, null, true))
.Returns((expectedTrustSummaries, expectedTrustSummaries.Count));

var result = _controller.SearchTrusts(null, null, null, 1, 10);
Expand All @@ -194,7 +194,7 @@ public void SearchTrusts_WithMultiplePagesOfResults_ReturnsAllTrustsWithPagingRe
};

_mockSearchTrustsUseCase
.Setup(s => s.Execute(1, 10, null, null, null))
.Setup(s => s.Execute(1, 10, null, null, null, true))
.Returns((expectedTrustSummaries.Take(10).ToList(), expectedTrustSummaries.Count));

_controller.ControllerContext = new ControllerContext { HttpContext = new DefaultHttpContext() };
Expand Down
48 changes: 41 additions & 7 deletions TramsDataApi.Test/UseCases/SearchTrustsTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -50,19 +50,22 @@ public void SearchTrusts_ReturnsListOfTrustSummaryResponses_WhenTrustsFound()

var trustsGateway = new Mock<ITrustGateway>();
var establishmentsGateway = new Mock<IEstablishmentGateway>();
trustsGateway.Setup(g => g.SearchGroups(1, 10, groupName, null, null))

trustsGateway.Setup(g => g.SearchGroups(1, 10, groupName, string.Empty, string.Empty))
.Returns((expectedTrusts, expectedTrusts.Count));

establishmentsGateway.Setup(g => g.GetByTrustUid(It.IsAny<string>()))
trustsGateway.Setup(m => m.GetIfdTrustsByTrustRef(It.IsAny<string[]>()))
.Returns(new List<Trust>());

establishmentsGateway.Setup(g => g.GetByTrustUids(It.IsAny<string[]>()))
.Returns(new List<Establishment>());

var expected = expectedTrusts.
Select(e => TrustSummaryResponseFactory.Create(e, new List<Establishment>(), null))
.ToList();

var searchTrusts = new SearchTrusts(trustsGateway.Object, establishmentsGateway.Object);
var (result, _) = searchTrusts.Execute(1, 10, groupName, null, null);
var (result, _) = searchTrusts.Execute(1, 10, groupName);

result.Should().BeEquivalentTo(expected);
}
Expand All @@ -81,12 +84,16 @@ public void SearchTrusts_ShouldGetTrustsWithEstablishments_WhenTrustsAndEstablis
.With(e => e.TrustsCode = expectedTrust.GroupUid)
.Build();

var trusts = Builder<Trust>.CreateListOfSize(3).Build();

var trustGateway = new Mock<ITrustGateway>();
var establishmentGateway = new Mock<IEstablishmentGateway>();

trustGateway.Setup(g => g.SearchGroups(1, 10, null, ukprn, null))
trustGateway.Setup(g => g.SearchGroups(1, 10, string.Empty, ukprn, string.Empty))
.Returns((new List<Group> {expectedTrust}, 1));
establishmentGateway.Setup(g => g.GetByTrustUid(expectedTrust.GroupUid))
trustGateway.Setup(m => m.GetIfdTrustsByTrustRef(It.IsAny<string[]>()))
.Returns(trusts);
establishmentGateway.Setup(g => g.GetByTrustUids(It.IsAny<string[]>()))
.Returns(expectedEstablishments);

var expected = new List<TrustSummaryResponse>
Expand All @@ -95,7 +102,34 @@ public void SearchTrusts_ShouldGetTrustsWithEstablishments_WhenTrustsAndEstablis
};

var searchTrusts = new SearchTrusts(trustGateway.Object, establishmentGateway.Object);
var (result, _) = searchTrusts.Execute(1, 10, null, ukprn, null);
var (result, _) = searchTrusts.Execute(1, 10, ukPrn: ukprn);
result.Should().BeEquivalentTo(expected);
}

[Fact]
public void SearchTrusts_WithoutEstablishments()
{
var ukprn = "mockurn";
var expectedTrust = Builder<Group>
.CreateNew()
.With(g => g.Ukprn = ukprn)
.Build();

var trusts = Builder<Trust>.CreateListOfSize(3).Build();
var trustGateway = new Mock<ITrustGateway>();

trustGateway.Setup(g => g.SearchGroups(1, 10, string.Empty, ukprn, string.Empty))
.Returns((new List<Group> {expectedTrust}, 1));
trustGateway.Setup(m => m.GetIfdTrustsByTrustRef(It.IsAny<string[]>()))
.Returns(trusts);

var expected = new List<TrustSummaryResponse>
{
TrustSummaryResponseFactory.Create(expectedTrust, Enumerable.Empty<Establishment>(), null)
};

var searchTrusts = new SearchTrusts(trustGateway.Object, null);
var (result, _) = searchTrusts.Execute(1, 10, ukPrn: ukprn, includeEstablishments: false);
result.Should().BeEquivalentTo(expected);
}
}
Expand Down
2 changes: 1 addition & 1 deletion TramsDataApi/Controllers/TrustsController.cs
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ public ActionResult<List<TrustSummaryResponse>> SearchTrusts(string groupName, s
groupName, ukPrn, companiesHouseNumber, page, count);

var trusts = _searchTrusts
.Execute(page, count, groupName, ukPrn, companiesHouseNumber)
.Execute(page, count, groupName, ukPrn, companiesHouseNumber, true)
.Item1.ToList();

_logger.LogInformation(
Expand Down
8 changes: 5 additions & 3 deletions TramsDataApi/Controllers/V2/TrustsController.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,17 @@ public TrustsController(IGetTrustByUkprn getTrustByUkPrn, ISearchTrusts searchTr

[HttpGet("trusts")]
[MapToApiVersion("2.0")]
public ActionResult<ApiResponseV2<TrustSummaryResponse>> SearchTrusts(string groupName, string ukPrn, string companiesHouseNumber, int page = 1, int count = 50)
public ActionResult<ApiResponseV2<TrustSummaryResponse>> SearchTrusts(string groupName, string ukPrn, string companiesHouseNumber,
int page = 1, int count = 50, bool includeEstablishments = true)
{
_logger.LogInformation(
"Searching for trusts by groupName \"{name}\", UKPRN \"{prn}\", companiesHouseNumber \"{number}\", page {page}, count {count}",
groupName, ukPrn, companiesHouseNumber, page, count);

var (trusts, recordCount) = _searchTrusts
.Execute(page, count, groupName, ukPrn, companiesHouseNumber);

.Execute(page, count, groupName, ukPrn, companiesHouseNumber, includeEstablishments);
trusts = trusts.ToList();

_logger.LogInformation(
"Found {count} trusts for groupName \"{name}\", UKPRN \"{prn}\", companiesHouseNumber \"{number}\", page {page}, count {count}",
trusts.Count(), groupName, ukPrn, companiesHouseNumber, page, count);
Expand Down
2 changes: 1 addition & 1 deletion TramsDataApi/Gateways/ITrustGateway.cs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ public interface ITrustGateway
Group GetGroupByUkPrn(string ukPrn);
Trust GetIfdTrustByGroupId(string groupId);
Trust GetIfdTrustByRID(string RID);
IQueryable<Trust> GetIfdTrustsByTrustRef(string[] trustRefs);
IList<Trust> GetIfdTrustsByTrustRef(string[] trustRefs);
(IList<Group>, int) SearchGroups(int page, int count, string groupName, string ukPrn, string companiesHouseNumber);
IEnumerable<Group> GetMultipleGroupsByUkprn(IEnumerable<string> ukprns);
IEnumerable<Trust> GetMultipleTrustsByGroupId(IEnumerable<string> groupIds);
Expand Down
7 changes: 2 additions & 5 deletions TramsDataApi/Gateways/TrustGateway.cs
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,9 @@ public Trust GetIfdTrustByRID(string RID)
return _dbContext.Trust.FirstOrDefault(x => x.Rid.Equals(RID));
}

public IQueryable<Trust> GetIfdTrustsByTrustRef(string[] trustRefs)
public IList<Trust> GetIfdTrustsByTrustRef(string[] trustRefs)
{
Expression<Func<Trust, bool>> predicate = PredicateBuilder.False<Trust>();
foreach (var trustRef in trustRefs) predicate = predicate.Or(t => t.TrustRef == trustRef);

return _dbContext.Trust.Where(predicate);
return _dbContext.Trust.Where(t => trustRefs.Contains(t.TrustRef)).ToList();
}

public (IList<Group>, int) SearchGroups(int page, int count, string groupName, string ukPrn,
Expand Down
8 changes: 7 additions & 1 deletion TramsDataApi/UseCases/ISearchTrusts.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,12 @@ namespace TramsDataApi.UseCases
{
public interface ISearchTrusts
{
public (IEnumerable<TrustSummaryResponse>, int) Execute(int page, int count, string groupName, string urn, string companiesHouseNumber);
public (IEnumerable<TrustSummaryResponse>, int) Execute(
int page = 1,
int count = 50,
string groupName = "",
string ukPrn = "",
string companiesHouseNumber = "",
bool includeEstablishments = true);
}
}
24 changes: 21 additions & 3 deletions TramsDataApi/UseCases/SearchTrusts.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using System.Collections.Generic;
using System.Linq;
using TramsDataApi.DatabaseModels;
using TramsDataApi.Factories;
using TramsDataApi.Gateways;
using TramsDataApi.ResponseModels;
Expand All @@ -17,15 +18,32 @@ public SearchTrusts(ITrustGateway trustGateway, IEstablishmentGateway establishm
_establishmentGateway = establishmentGateway;
}

public (IEnumerable<TrustSummaryResponse>, int) Execute(int page, int count, string groupName, string ukPrn, string companiesHouseNumber)
public (IEnumerable<TrustSummaryResponse>, int) Execute(
int page = 1,
int count = 50,
string groupName = "",
string ukPrn = "",
string companiesHouseNumber = "",
bool includeEstablishments = true)
{
var (groups, recordCount) = _trustGateway.SearchGroups(page, count, groupName, ukPrn, companiesHouseNumber);

var groupIds = groups.Select(g => g.GroupId).ToArray();
var trustsForGroup = _trustGateway.GetIfdTrustsByTrustRef(groupIds);

IEnumerable<Establishment> establishmentsForGroup = Enumerable.Empty<Establishment>();

if (includeEstablishments)
{
var groupUids = groups.Select(g => g.GroupUid).ToArray();
establishmentsForGroup = _establishmentGateway.GetByTrustUids(groupUids);
}

return (
groups.Select(group =>
{
var trust = _trustGateway.GetIfdTrustByGroupId(group.GroupId);
var establishments = _establishmentGateway.GetByTrustUid(group.GroupUid);
var establishments = establishmentsForGroup.Where(e => e.TrustsCode == group.GroupUid);
var trust = trustsForGroup.FirstOrDefault(e => e.TrustRef == group.GroupUid);
return TrustSummaryResponseFactory.Create(group, establishments, trust);
}).ToArray(),
recordCount
Expand Down

0 comments on commit 4100013

Please sign in to comment.