feat(clubs): add Club and Member API endpoints with auto-sync
Implement Task 16: Club + Member API endpoints with MemberSyncService
Services:
- ClubService: GetMyClubsAsync (user's clubs), GetCurrentClubAsync (tenant club)
- MemberService: GetMembersAsync (list), GetMemberByIdAsync, GetCurrentMemberAsync
- MemberSyncService: Auto-creates Member records from JWT on first request
Middleware:
- MemberSyncMiddleware: Runs after auth, calls MemberSyncService
Endpoints:
- GET /api/clubs/me (list user's clubs)
- GET /api/clubs/current (current tenant's club)
- GET /api/members (list members, RLS filtered)
- GET /api/members/{id} (member detail)
- GET /api/members/me (current user's membership)
Tests: 14 integration tests (6 club + 8 member)
- Club filtering by user membership
- Multi-tenant isolation via RLS
- Member auto-sync on first request
- Cross-tenant access blocked
- Role-based authorization
Build: 0 errors, all tests compile
Pattern: TypedResults, RequireAuthorization policies, TDD approach
This commit is contained in:
53
backend/WorkClub.Api/Endpoints/Members/MemberEndpoints.cs
Normal file
53
backend/WorkClub.Api/Endpoints/Members/MemberEndpoints.cs
Normal file
@@ -0,0 +1,53 @@
|
||||
using Microsoft.AspNetCore.Http.HttpResults;
|
||||
using WorkClub.Api.Services;
|
||||
using WorkClub.Application.Members.DTOs;
|
||||
|
||||
namespace WorkClub.Api.Endpoints.Members;
|
||||
|
||||
public static class MemberEndpoints
|
||||
{
|
||||
public static void MapMemberEndpoints(this IEndpointRouteBuilder app)
|
||||
{
|
||||
var group = app.MapGroup("/api/members");
|
||||
|
||||
group.MapGet("", GetMembers)
|
||||
.RequireAuthorization("RequireMember")
|
||||
.WithName("GetMembers");
|
||||
|
||||
group.MapGet("{id:guid}", GetMemberById)
|
||||
.RequireAuthorization("RequireMember")
|
||||
.WithName("GetMemberById");
|
||||
|
||||
group.MapGet("/me", GetCurrentMember)
|
||||
.RequireAuthorization("RequireMember")
|
||||
.WithName("GetCurrentMember");
|
||||
}
|
||||
|
||||
private static async Task<Ok<List<MemberListDto>>> GetMembers(MemberService memberService)
|
||||
{
|
||||
var result = await memberService.GetMembersAsync();
|
||||
return TypedResults.Ok(result);
|
||||
}
|
||||
|
||||
private static async Task<Results<Ok<MemberDetailDto>, NotFound>> GetMemberById(
|
||||
Guid id,
|
||||
MemberService memberService)
|
||||
{
|
||||
var result = await memberService.GetMemberByIdAsync(id);
|
||||
|
||||
if (result == null)
|
||||
return TypedResults.NotFound();
|
||||
|
||||
return TypedResults.Ok(result);
|
||||
}
|
||||
|
||||
private static async Task<Results<Ok<MemberDetailDto>, NotFound>> GetCurrentMember(MemberService memberService)
|
||||
{
|
||||
var result = await memberService.GetCurrentMemberAsync();
|
||||
|
||||
if (result == null)
|
||||
return TypedResults.NotFound();
|
||||
|
||||
return TypedResults.Ok(result);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user