Files
budget/src/Budget.Api/Controllers/IncomesController.cs
T
Spencer Twaddle 087fbdd176 Split into Budget.Core / Budget.Infrastructure / Budget.Api projects
Budget.Core: entities, DTOs, enums, FrequencyCalculator (no EF/ASP.NET deps)
Budget.Infrastructure: AppDbContext, migrations, BudgetAuthorizationService
Budget.Api: controllers, middleware, Program.cs — references both projects

EF and Npgsql packages moved to Infrastructure; Api retains only JwtBearer,
HealthChecks, and EF.Design (needed for dotnet ef CLI). Dockerfile updated
to copy all three project directories before publishing. Migration namespaces
updated from Budget.Api.Data.* to Budget.Infrastructure.Data.* and model
type strings updated to Budget.Core.Models.* in the snapshot.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-02 16:30:31 -05:00

111 lines
4.0 KiB
C#

using Budget.Api.Services;
using Budget.Core.DTOs;
using Budget.Core.Models;
using Budget.Core.Services;
using Budget.Infrastructure.Data;
using Budget.Infrastructure.Services;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.RateLimiting;
using Microsoft.EntityFrameworkCore;
namespace Budget.Api.Controllers;
[ApiController]
[Route("api/budgets/{budgetId:guid}/incomes")]
[Authorize]
public class IncomesController(AppDbContext db, BudgetAuthorizationService authz) : ControllerBase
{
private IActionResult? TryGetUserId(out string userId)
{
var id = User.GetUserId();
if (id is null) { userId = string.Empty; return Unauthorized(); }
userId = id;
return null;
}
private static IncomeDto ToDto(Income i) => new(
i.Id, i.BudgetId, i.Name, i.Frequency, i.Amount,
FrequencyCalculator.ToMonthly(i.Amount, i.Frequency),
FrequencyCalculator.ToAnnually(i.Amount, i.Frequency),
i.SortOrder);
[HttpGet]
public async Task<IActionResult> List(Guid budgetId)
{
if (TryGetUserId(out var userId) is { } err) return err;
if (!await authz.CanReadAsync(budgetId, userId)) return Forbid();
var incomes = await db.Incomes
.Where(i => i.BudgetId == budgetId)
.OrderBy(i => i.SortOrder)
.ToListAsync();
return Ok(incomes.Select(ToDto));
}
[HttpPost]
[EnableRateLimiting("writes")]
public async Task<IActionResult> Create(Guid budgetId, [FromBody] CreateIncomeRequest req)
{
if (TryGetUserId(out var userId) is { } err) return err;
if (!await authz.CanWriteAsync(budgetId, userId)) return Forbid();
var maxOrder = await db.Incomes.Where(i => i.BudgetId == budgetId).MaxAsync(i => (int?)i.SortOrder) ?? -1;
var income = new Income
{
Id = Guid.NewGuid(),
BudgetId = budgetId,
Name = req.Name,
Frequency = req.Frequency,
Amount = req.Amount,
SortOrder = maxOrder + 1,
};
db.Incomes.Add(income);
await db.SaveChangesAsync();
return Ok(ToDto(income));
}
[HttpPut("{incomeId:guid}")]
[EnableRateLimiting("writes")]
public async Task<IActionResult> Update(Guid budgetId, Guid incomeId, [FromBody] UpdateIncomeRequest req)
{
if (TryGetUserId(out var userId) is { } err) return err;
if (!await authz.CanWriteAsync(budgetId, userId)) return Forbid();
var income = await db.Incomes.FirstOrDefaultAsync(i => i.Id == incomeId && i.BudgetId == budgetId);
if (income is null) return NotFound();
income.Name = req.Name;
income.Frequency = req.Frequency;
income.Amount = req.Amount;
await db.SaveChangesAsync();
return Ok(ToDto(income));
}
[HttpDelete("{incomeId:guid}")]
[EnableRateLimiting("writes")]
public async Task<IActionResult> Delete(Guid budgetId, Guid incomeId)
{
if (TryGetUserId(out var userId) is { } err) return err;
if (!await authz.CanWriteAsync(budgetId, userId)) return Forbid();
var income = await db.Incomes.FirstOrDefaultAsync(i => i.Id == incomeId && i.BudgetId == budgetId);
if (income is null) return NotFound();
db.Incomes.Remove(income);
await db.SaveChangesAsync();
return NoContent();
}
[HttpPut("order")]
[EnableRateLimiting("writes")]
public async Task<IActionResult> Reorder(Guid budgetId, [FromBody] ReorderIncomesRequest req)
{
if (TryGetUserId(out var userId) is { } err) return err;
if (!await authz.CanWriteAsync(budgetId, userId)) return Forbid();
var incomes = await db.Incomes.Where(i => i.BudgetId == budgetId).ToListAsync();
var lookup = incomes.ToDictionary(i => i.Id);
for (int idx = 0; idx < req.OrderedIds.Count; idx++)
{
if (lookup.TryGetValue(req.OrderedIds[idx], out var income))
income.SortOrder = idx;
}
await db.SaveChangesAsync();
return NoContent();
}
}