Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WIP: utils query tests, especially around failure cases #146

Closed
wants to merge 3 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 11 additions & 1 deletion src/CommonLib/ConnectionPoolManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,17 @@
using SharpHoundCommonLib.Processors;

namespace SharpHoundCommonLib {
public class ConnectionPoolManager : IDisposable{
public interface ILdapConnectionProvider {
Task<(bool Success, string Message)> TestDomainConnection(string identifier, bool globalCatalog);
Task<(bool Success, LdapConnectionWrapper ConnectionWrapper, string Message)> GetLdapConnection(
string identifier, bool globalCatalog);
Task<(bool Success, LdapConnectionWrapper connectionWrapper, string Message)> GetLdapConnectionForServer(
string identifier, string server, bool globalCatalog);
void ReleaseConnection(LdapConnectionWrapper connectionWrapper, bool connectionFaulted = false);
void Dispose();
}

public class ConnectionPoolManager : ILdapConnectionProvider, IDisposable{
private readonly ConcurrentDictionary<string, LdapConnectionPool> _pools = new();
private readonly LdapConfig _ldapConfig;
private readonly string[] _translateNames = { "Administrator", "admin" };
Expand Down
9 changes: 8 additions & 1 deletion src/CommonLib/LdapUtils.cs
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ private readonly ConcurrentDictionary<string, string>
private readonly string[] _translateNames = { "Administrator", "admin" };
private LdapConfig _ldapConfig = new();

private ConnectionPoolManager _connectionPool;
private ILdapConnectionProvider _connectionPool;

private static readonly TimeSpan MinBackoffDelay = TimeSpan.FromSeconds(2);
private static readonly TimeSpan MaxBackoffDelay = TimeSpan.FromSeconds(20);
Expand Down Expand Up @@ -82,6 +82,13 @@ public LdapUtils() {
_connectionPool = new ConnectionPoolManager(_ldapConfig, _log);
}

public LdapUtils(ILdapConnectionProvider ldapConnectionProvider) {
_nativeMethods = new NativeMethods();
_portScanner = new PortScanner();
_log = Logging.LogProvider.CreateLogger("LDAPUtils");
_connectionPool = ldapConnectionProvider;
}

public LdapUtils(NativeMethods nativeMethods = null, PortScanner scanner = null, ILogger log = null) {
_nativeMethods = nativeMethods ?? new NativeMethods();
_portScanner = scanner ?? new PortScanner();
Expand Down
180 changes: 180 additions & 0 deletions test/unit/LdapUtilsQueryTest.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,180 @@
using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using Xunit;
using Moq;
using System.DirectoryServices.Protocols;
using SharpHoundCommonLib;

public class LdapUtilsQueryTest
{
// [Fact]
// public async Task RangedRetrieval_SuccessfulRetrieval_ReturnsExpectedResults()
// {
// // Arrange
// var distinguishedName = "CN=TestUser,DC=example,DC=com";
// var attributeName = "member";
// var domain = "example.com";

// var connectionWrapper = new Mock<LdapConnectionWrapper>();
// var connection = new Mock<LdapConnection>();
// connectionWrapper.SetupGet(x => x.Connection).Returns(connection.Object);

// var mockConnectionPool = new Mock<ILdapConnectionProvider>();
// mockConnectionPool.Setup(x => x.GetLdapConnection(domain, false))
// .ReturnsAsync((true, connectionWrapper.Object, null));

// var utils = new LdapUtils(mockConnectionPool.Object);

// var searchResponse = new Mock<SearchResponse>();
// var entry = new SearchResultEntry
// {
// Attributes =
// {
// new DirectoryAttribute("member;range=0-*", "CN=Member1,DC=example,DC=com", "CN=Member2,DC=example,DC=com")
// }
// };
// searchResponse.Entries.Add(entry);

// connection.Setup(x => x.SendRequest(It.IsAny<SearchRequest>()))
// .Returns(searchResponse);

// // Act
// var results = new List<Result<string>>();
// await foreach (var result in utils.RangedRetrieval(distinguishedName, attributeName))
// {
// results.Add(result);
// }

// // Assert
// Assert.Equal(2, results.Count);
// Assert.True(results[0].IsSuccess);
// Assert.Equal("CN=Member1,DC=example,DC=com", results[0].Value);
// Assert.True(results[1].IsSuccess);
// Assert.Equal("CN=Member2,DC=example,DC=com", results[1].Value);
// }

[Fact]
public async Task RangedRetrieval_ConnectionFailure_ReturnsFailResult()
{
// Arrange
var distinguishedName = "CN=TestUser,DC=example,DC=com";
var attributeName = "member";

var utils = new LdapUtils();

// Act
var results = new List<Result<string>>();
await foreach (var result in utils.RangedRetrieval(distinguishedName, attributeName))
{
results.Add(result);
}

// Assert
Assert.Single(results);
Assert.False(results[0].IsSuccess);
Assert.Equal("All attempted connections failed", results[0].Error);
}

// [Fact]
// public async Task RangedRetrieval_ServerDown_RetriesAndRecovers()
// {
// // Arrange
// var distinguishedName = "CN=TestUser,DC=example,DC=com";
// var attributeName = "member";
// var domain = "example.com";

// var connectionWrapper = new Mock<LdapConnectionWrapper>();
// var connection = new Mock<LdapConnection>();

// // TODO : setup

// // Act
// var results = new List<Result<string>>();
// await foreach (var result in _utils.RangedRetrieval(distinguishedName, attributeName))
// {
// results.Add(result);
// }

// // TODO Assert
// }

[Fact]
public async Task RangedRetrieval_CancellationRequested_StopsRetrieval()
{
// Arrange
var distinguishedName = "CN=TestUser,DC=example,DC=com";
var attributeName = "member";
var domain = "example.com";

var connectionWrapper = new Mock<LdapConnectionWrapper>(null, null, false, string.Empty);
var connection = new Mock<LdapConnection>();
var mockConnectionPool = new Mock<ILdapConnectionProvider>();

mockConnectionPool.Setup(x => x.GetLdapConnection(domain, false))
.ReturnsAsync((true, connectionWrapper.Object, null));

var utils = new LdapUtils(mockConnectionPool.Object);

var cts = new CancellationTokenSource();
cts.Cancel();

// Act
var results = new List<Result<string>>();
await foreach (var result in utils.RangedRetrieval(distinguishedName, attributeName, cts.Token))
{
results.Add(result);
}

// Assert
Assert.False(results[0].IsSuccess);
}

[Fact]
public async Task Query_ConnectionFailure_ReturnsEmptyResult()
{
// Arrange
var queryParams = new LdapQueryParameters();
var utils = new LdapUtils();

// Act
var results = new List<LdapResult<IDirectoryObject>>();
await foreach (var result in utils.Query(queryParams))
{
results.Add(result);
}

// Assert
Assert.Empty(results);
}

[Fact]
public async Task Query_CancellationRequested_StopsRetrieval()
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Todo: This doesn't really test anything just yet

{
// Arrange
var queryParams = new LdapQueryParameters();
var domain = "example.com";

var connectionWrapper = new Mock<LdapConnectionWrapper>(null, null, false, string.Empty);
var connection = new Mock<LdapConnection>();
var mockConnectionPool = new Mock<ILdapConnectionProvider>();

mockConnectionPool.Setup(x => x.GetLdapConnection(domain, false))
.ReturnsAsync((true, connectionWrapper.Object, null));

var utils = new LdapUtils(mockConnectionPool.Object);

var cts = new CancellationTokenSource();
cts.Cancel();

// Act
var results = new List<LdapResult<IDirectoryObject>>();
await foreach (var result in utils.Query(queryParams, cts.Token))
{
results.Add(result);
}

// Assert
Assert.Empty(results);
}
}
Loading