8b86f17645
PlansController had a class-level [Authorize] that gated the public plans list, contradicting the gateway's optionalAuth on /plans. Mark List/GetById [AllowAnonymous] and resolve tenant optionally so anonymous callers receive global plans (purchase/current-plan stay authenticated). Frontend container stayed "unhealthy" because busybox wget resolves localhost to IPv6 [::1] while the Next.js standalone server binds IPv4 only. Use 127.0.0.1 in the healthcheck. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
51 lines
1.9 KiB
C#
51 lines
1.9 KiB
C#
using FlatRender.IdentitySvc.Application.Services.Interfaces;
|
|
using FlatRender.IdentitySvc.Models.Requests;
|
|
using FlatRender.IdentitySvc.Models.Responses;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
namespace FlatRender.IdentitySvc.Controllers;
|
|
|
|
[ApiController]
|
|
[Route("v1")]
|
|
[Authorize]
|
|
public class PlansController(IPlanService planService) : ControllerBase
|
|
{
|
|
[AllowAnonymous]
|
|
[HttpGet("plans")]
|
|
[ProducesResponseType(typeof(object), 200)]
|
|
public async Task<IActionResult> List([FromQuery] string? scope)
|
|
{
|
|
var plans = await planService.ListAsync(GetTenantIdOrNull(), scope);
|
|
return Ok(new { data = plans });
|
|
}
|
|
|
|
[AllowAnonymous]
|
|
[HttpGet("plans/{planId:guid}")]
|
|
[ProducesResponseType(typeof(PlanResponse), 200)]
|
|
public async Task<IActionResult> GetById(Guid planId)
|
|
=> Ok(await planService.GetByIdAsync(planId));
|
|
|
|
[HttpGet("users/me/plan")]
|
|
[ProducesResponseType(typeof(UserPlanResponse), 200)]
|
|
public async Task<IActionResult> GetCurrentPlan()
|
|
=> Ok(await planService.GetCurrentPlanAsync(GetUserId()));
|
|
|
|
[HttpPost("users/me/plan/purchase")]
|
|
[ProducesResponseType(typeof(PurchasePlanResponse), 200)]
|
|
public async Task<IActionResult> Purchase([FromBody] PurchasePlanRequest request)
|
|
{
|
|
var result = await planService.PurchasePlanAsync(GetUserId(), GetTenantId(), request);
|
|
return Ok(result);
|
|
}
|
|
|
|
private Guid GetUserId() => Guid.Parse(User.FindFirst(System.Security.Claims.ClaimTypes.NameIdentifier)?.Value
|
|
?? User.FindFirst("sub")?.Value ?? throw new UnauthorizedAccessException());
|
|
|
|
private Guid GetTenantId() => Guid.Parse(User.FindFirst("tenant_id")?.Value
|
|
?? throw new UnauthorizedAccessException());
|
|
|
|
private Guid? GetTenantIdOrNull() => User.FindFirst("tenant_id")?.Value is { } t && Guid.TryParse(t, out var id)
|
|
? id : null;
|
|
}
|