Files
meezi/src/Meezi.API/Middleware/PlanLimitMiddleware.cs
T
soroush.asadi ef15fd6247 feat(api): .NET 10 multi-tenant REST API
Full backend implementation:
- Multi-tenant cafe/restaurant management (menus, orders, tables, staff)
- POS order flow with ZarinPal and Snappfood payment integration
- OTP authentication via Kavenegar SMS
- QR digital menu with public discover/finder endpoints
- Customer loyalty, coupons, CRM
- PostgreSQL via EF Core, Redis for caching/sessions
- Background jobs, webhook handlers
- Full migration history

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-05-27 21:33:48 +03:30

65 lines
1.8 KiB
C#

using System.Text.Json;
using Meezi.API.Services;
using Meezi.Core.Interfaces;
using Meezi.Shared;
namespace Meezi.API.Middleware;
public class PlanLimitMiddleware
{
private static readonly string[] SkipPrefixes =
[
"/api/auth",
"/api/customers/me",
"/api/admin",
"/hubs/guest-order",
"/api/public",
"/api/q/",
"/api/webhooks",
"/api/billing/verify",
"/health",
"/swagger",
"/hangfire"
];
private readonly RequestDelegate _next;
public PlanLimitMiddleware(RequestDelegate next)
{
_next = next;
}
public async Task InvokeAsync(HttpContext context, ITenantContext tenant, IPlanLimitChecker planLimitChecker)
{
if (ShouldSkip(context.Request.Path))
{
await _next(context);
return;
}
if (context.User.Identity?.IsAuthenticated == true)
{
var (allowed, code, message) = await planLimitChecker.CheckAsync(context, tenant, context.RequestAborted);
if (!allowed)
{
context.Response.StatusCode = StatusCodes.Status403Forbidden;
context.Response.ContentType = "application/json";
var payload = new ApiResponse<object>(false, null, new ApiError(code!, message!));
await context.Response.WriteAsync(JsonSerializer.Serialize(payload, new JsonSerializerOptions
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase
}));
return;
}
}
await _next(context);
}
private static bool ShouldSkip(PathString path)
{
var value = path.Value ?? string.Empty;
return SkipPrefixes.Any(p => value.StartsWith(p, StringComparison.OrdinalIgnoreCase));
}
}