add fp1 sample
This commit is contained in:
@@ -0,0 +1,37 @@
|
||||
using CsharpFp1.Domain;
|
||||
using CsharpFp1.Infrastructure;
|
||||
|
||||
namespace CsharpFp1.Application;
|
||||
|
||||
// delegate type is optional, but nice to illustrate
|
||||
public delegate void WithdrawMoney(Guid accountId, decimal amount);
|
||||
|
||||
public static class AccountApplication
|
||||
{
|
||||
public static WithdrawMoney CreateWithdrawMoney(
|
||||
LoadAccount loadAccount,
|
||||
SaveAccount saveAccount
|
||||
)
|
||||
{
|
||||
return (accountId, amount) =>
|
||||
{
|
||||
var account =
|
||||
loadAccount(accountId) ?? throw new InvalidOperationException("Account not found.");
|
||||
|
||||
Console.WriteLine(
|
||||
$"[App] Account loaded. Current balance: {account.Balance.Amount:0.00}"
|
||||
);
|
||||
Console.WriteLine($"[App] Executing withdrawal of {amount:0.00}...");
|
||||
|
||||
var modifiedAccount = AccountDomain.Withdraw(account, new Money(amount));
|
||||
|
||||
Console.WriteLine(
|
||||
$"[App] Withdrawal applied. New balance: {modifiedAccount.Balance.Amount:0.00}"
|
||||
);
|
||||
|
||||
saveAccount(modifiedAccount);
|
||||
|
||||
Console.WriteLine("[App] Account persisted.");
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,32 @@
|
||||
namespace CsharpFp1.Domain;
|
||||
|
||||
public sealed record Account(Guid Id, Money Balance);
|
||||
|
||||
public static class AccountDomain
|
||||
{
|
||||
// Choosing a "clean" FP approach here of instantiating the Account
|
||||
// Depending on needs, code to prevent the Account type from
|
||||
// being instantiated without this helper needs to be added
|
||||
// to the record type.
|
||||
public static Account Open(Guid id, Money openingBalance)
|
||||
{
|
||||
if (openingBalance.Amount < 0)
|
||||
throw new ArgumentOutOfRangeException(nameof(openingBalance));
|
||||
|
||||
return new Account(id, openingBalance);
|
||||
}
|
||||
|
||||
public static Account Withdraw(Account account, Money amount)
|
||||
{
|
||||
if (amount.Amount <= 0)
|
||||
throw new InvalidOperationException("Withdrawal amount must be positive.");
|
||||
|
||||
if (account.Balance.Amount < amount.Amount)
|
||||
throw new InsufficientBalanceException(account.Balance, amount);
|
||||
|
||||
return account with
|
||||
{
|
||||
Balance = account.Balance.Subtract(amount),
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
namespace CsharpFp1.Domain;
|
||||
|
||||
/// Custom domain exception thrown when a withdrawal would cause the balance to go below zero
|
||||
public sealed class InsufficientBalanceException : InvalidOperationException
|
||||
{
|
||||
public Money CurrentBalance { get; }
|
||||
public Money RequestedAmount { get; }
|
||||
|
||||
public InsufficientBalanceException(Money currentBalance, Money requestedAmount)
|
||||
: base(
|
||||
$"Insufficient balance. Current: {currentBalance.Amount:0.00}, Requested: {requestedAmount.Amount:0.00}"
|
||||
)
|
||||
{
|
||||
CurrentBalance = currentBalance;
|
||||
RequestedAmount = requestedAmount;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,33 @@
|
||||
namespace CsharpFp1.Domain;
|
||||
|
||||
/// Value object used to represent money and enforce simple invariants.
|
||||
/// Note that this implementation uses immutable patterns for the data
|
||||
/// by returning a new instance for each modification. This is an early
|
||||
/// recommendation for DDD with OO, but not necessarily the common practice
|
||||
/// in many real-world implementations.
|
||||
public sealed class Money
|
||||
{
|
||||
// Potentially with a setter - see note above
|
||||
public decimal Amount { get; }
|
||||
|
||||
public Money(decimal amount)
|
||||
{
|
||||
Amount = amount;
|
||||
}
|
||||
|
||||
// In many existing DDD/OO codebases you may actually see the use
|
||||
// of mutable value types.
|
||||
//
|
||||
// public void Add(Money other)
|
||||
// {
|
||||
// this.Amount += other.Amount;
|
||||
// }
|
||||
|
||||
// On the other hand, sometimes these helpers may be left out
|
||||
// and operations encoded directly "from the outside":
|
||||
// newBalance = new Money(oldBalance.Amount - charge.Amount)
|
||||
//
|
||||
public Money Add(Money other) => new(Amount + other.Amount);
|
||||
|
||||
public Money Subtract(Money other) => new(Amount - other.Amount);
|
||||
}
|
||||
@@ -0,0 +1,40 @@
|
||||
using CsharpFp1.Domain;
|
||||
|
||||
namespace CsharpFp1.Infrastructure;
|
||||
|
||||
// If we don't like working with generic delegates directly, we can
|
||||
// create custom named delegate types.
|
||||
public delegate Account? LoadAccount(Guid id);
|
||||
|
||||
public delegate void SaveAccount(Account accunt);
|
||||
|
||||
// If we don't want to use tuples or really miss the interface idea, we can create a named container
|
||||
// public sealed record AccountPersistence(LoadAccount Load, SaveAccount Save);
|
||||
|
||||
public static class InMemoryAccount
|
||||
{
|
||||
public static (LoadAccount, SaveAccount) Create()
|
||||
{
|
||||
Dictionary<Guid, Account> store = new();
|
||||
|
||||
Account? GetById(Guid id)
|
||||
{
|
||||
var found = store.TryGetValue(id, out var account);
|
||||
Console.WriteLine(
|
||||
found ? $"[Repo] Loaded account {id}" : $"[Repo] Account {id} not found"
|
||||
);
|
||||
|
||||
return found ? account : null;
|
||||
}
|
||||
|
||||
void Save(Account account)
|
||||
{
|
||||
store[account.Id] = account;
|
||||
Console.WriteLine(
|
||||
$"[Repo] Saved account {account.Id} with balance {account.Balance.Amount:0.00}"
|
||||
);
|
||||
}
|
||||
|
||||
return (GetById, Save);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,28 @@
|
||||
using CsharpFp1.Application;
|
||||
using CsharpFp1.Domain;
|
||||
using CsharpFp1.Infrastructure;
|
||||
|
||||
namespace CsharpFp1;
|
||||
|
||||
public class Program
|
||||
{
|
||||
public static void Main()
|
||||
{
|
||||
Console.WriteLine("[csharp-fp1] Starting withdraw money demo...");
|
||||
|
||||
var (loadAccount, saveAccount) = InMemoryAccount.Create();
|
||||
var withdrawMoney = AccountApplication.CreateWithdrawMoney(loadAccount, saveAccount);
|
||||
|
||||
var accountId = Guid.NewGuid();
|
||||
Console.WriteLine($"[csharp-fp1] Seeding account {accountId} with opening balance 200.00");
|
||||
saveAccount(new Account(accountId, new Money(200m)));
|
||||
|
||||
decimal amount = 100m;
|
||||
Console.WriteLine(
|
||||
$"[csharp-fp1] Executing withdrawal {amount:0.00} from account {accountId}"
|
||||
);
|
||||
withdrawMoney(accountId, amount);
|
||||
|
||||
Console.WriteLine("[csharp-fp1] Demo completed.");
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<OutputType>Exe</OutputType>
|
||||
<TargetFramework>net10.0</TargetFramework>
|
||||
<RootNamespace>CsharpFp1</RootNamespace>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
</PropertyGroup>
|
||||
</Project>
|
||||
Reference in New Issue
Block a user