books/backend/Books.Api.Tests/AiBookkeeper/ChartOfAccountsProviderTests.cs
Nicolaj Hartmann 1f75c5d791 Add all backend domain, commands, repositories, and tests
This commit includes all previously untracked backend files:

Domain:
- Accounts, Attachments, BankConnections, Customers
- FiscalYears, Invoices, JournalEntryDrafts
- Orders, Products, UserAccess

Commands & Handlers:
- Full CQRS command structure for all domains

Repositories:
- PostgreSQL repositories for all read models
- Bank transaction and ledger repositories

GraphQL:
- Input types, scalars, and types for all entities
- Mutations and queries

Infrastructure:
- Banking integration (Enable Banking client)
- File storage, Invoicing, Reporting, SAF-T export
- Database migrations (003-029)

Tests:
- Integration tests for GraphQL endpoints
- Domain tests
- Invoicing and reporting tests

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 22:19:42 +01:00

220 lines
7.7 KiB
C#

using Books.Api.AiBookkeeper;
using Books.Api.EventFlow.ReadModels;
using Books.Api.EventFlow.Repositories;
using Moq;
using AwesomeAssertions;
namespace Books.Api.Tests.AiBookkeeper;
/// <summary>
/// Unit tests for ChartOfAccountsProvider.
/// Verifies that accounts are filtered and sorted correctly.
/// </summary>
[Trait("Category", "Unit")]
public class ChartOfAccountsProviderTests
{
private readonly Mock<IAccountRepository> _accountRepository;
private readonly ChartOfAccountsProvider _sut;
public ChartOfAccountsProviderTests()
{
_accountRepository = new Mock<IAccountRepository>();
_sut = new ChartOfAccountsProvider(_accountRepository.Object);
}
[Fact]
public async Task GetChartOfAccountsAsync_ReturnsOnlyExpenseTypeAccounts()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
CreateAccount("1000", "Bank", "asset", null),
CreateAccount("2000", "Leverandørgæld", "liability", null),
CreateAccount("3000", "Egenkapital", "equity", null),
CreateAccount("4000", "Omsætning", "income", null),
CreateAccount("5000", "Vareforbrug", "cogs", "I25"),
CreateAccount("6000", "Løn", "personnel", "INGEN"),
CreateAccount("7000", "Kontorudgifter", "expense", "I25"),
CreateAccount("8000", "Renteudgifter", "financial", null)
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert - Only expense-type accounts should be included
result.Accounts.Should().HaveCount(4);
result.Accounts.Select(a => a.AccountType).Should().OnlyContain(t =>
t == "cogs" || t == "personnel" || t == "expense" || t == "financial");
}
[Fact]
public async Task GetChartOfAccountsAsync_ExcludesAssetAccounts()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
CreateAccount("1000", "Bank", "asset", null),
CreateAccount("1200", "Varelager", "asset", null),
CreateAccount("7000", "Kontorudgifter", "expense", "I25")
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
result.Accounts.Should().HaveCount(1);
result.Accounts.Should().NotContain(a => a.AccountType == "asset");
}
[Fact]
public async Task GetChartOfAccountsAsync_ExcludesLiabilityAccounts()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
CreateAccount("2000", "Leverandørgæld", "liability", null),
CreateAccount("2100", "Skyldig moms", "liability", null),
CreateAccount("7000", "Kontorudgifter", "expense", "I25")
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
result.Accounts.Should().HaveCount(1);
result.Accounts.Should().NotContain(a => a.AccountType == "liability");
}
[Fact]
public async Task GetChartOfAccountsAsync_ExcludesIncomeAccounts()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
CreateAccount("4000", "Omsætning DK", "income", "U25"),
CreateAccount("4100", "Omsætning EU", "income", "U25EU"),
CreateAccount("7000", "Kontorudgifter", "expense", "I25")
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
result.Accounts.Should().HaveCount(1);
result.Accounts.Should().NotContain(a => a.AccountType == "income");
}
[Fact]
public async Task GetChartOfAccountsAsync_SortsAccountsByNumber()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
CreateAccount("7200", "IT-udgifter", "expense", "I25"),
CreateAccount("5000", "Vareforbrug", "cogs", "I25"),
CreateAccount("7100", "Kontorudgifter", "expense", "I25"),
CreateAccount("6000", "Løn", "personnel", null)
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
var accountNumbers = result.Accounts.Select(a => a.AccountNumber).ToList();
accountNumbers.Should().BeInAscendingOrder();
accountNumbers.Should().Equal(["5000", "6000", "7100", "7200"]);
}
[Fact]
public async Task GetChartOfAccountsAsync_ReturnsCorrectCompanyId()
{
// Arrange
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync("company-42", It.IsAny<CancellationToken>()))
.ReturnsAsync([]);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-42");
// Assert
result.CompanyId.Should().Be("company-42");
}
[Fact]
public async Task GetChartOfAccountsAsync_MapsAllAccountProperties()
{
// Arrange
var accounts = new List<AccountReadModelDto>
{
new()
{
Id = "acc-1",
CompanyId = "company-1",
AccountNumber = "7320",
Name = "Software abonnementer",
AccountType = "expense",
VatCodeId = "I25",
StandardAccountNumber = "11400",
IsActive = true,
IsSystemAccount = false,
CreatedAt = DateTime.UtcNow,
UpdatedAt = DateTime.UtcNow
}
};
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(accounts);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
result.Accounts.Should().HaveCount(1);
var account = result.Accounts[0];
account.AccountNumber.Should().Be("7320");
account.Name.Should().Be("Software abonnementer");
account.AccountType.Should().Be("expense");
account.VatCodeId.Should().Be("I25");
account.StandardAccountNumber.Should().Be("11400");
}
[Fact]
public async Task GetChartOfAccountsAsync_HandlesEmptyAccountList()
{
// Arrange
_accountRepository.Setup(r => r.GetActiveByCompanyIdAsync(It.IsAny<string>(), It.IsAny<CancellationToken>()))
.ReturnsAsync([]);
// Act
var result = await _sut.GetChartOfAccountsAsync("company-1");
// Assert
result.Accounts.Should().BeEmpty();
result.CompanyId.Should().Be("company-1");
}
private static AccountReadModelDto CreateAccount(string number, string name, string type, string? vatCode)
{
return new AccountReadModelDto
{
Id = $"account-{number}",
CompanyId = "company-1",
AccountNumber = number,
Name = name,
AccountType = type,
VatCodeId = vatCode,
IsActive = true,
IsSystemAccount = false,
CreatedAt = DateTime.UtcNow,
UpdatedAt = DateTime.UtcNow
};
}
}