2026-05-22 14:29:22 +08:00
|
|
|
|
using FileShare_Common.Core;
|
|
|
|
|
|
using FileShare_Services.Core;
|
|
|
|
|
|
using FileShare_Services.Services.FileLibrary;
|
|
|
|
|
|
using FileShare_Services.Services.QrCode;
|
2026-05-21 15:52:36 +08:00
|
|
|
|
|
2026-05-22 14:29:22 +08:00
|
|
|
|
namespace FileShare_Services.Endpoints
|
2026-05-21 15:52:36 +08:00
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 统一端点配置 —— 所有业务端点在此定义一次。
|
2026-05-22 14:29:22 +08:00
|
|
|
|
/// 这是 FileShare-API 和 FileShare-PC 的唯一入口。
|
2026-05-21 15:52:36 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
public static class AppEndpoints
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 配置所有业务端点。调用方传入 builder,按需叠加鉴权、过滤器等。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="builder">端点构建器</param>
|
|
|
|
|
|
/// <param name="includeDetails">是否在错误响应中包含异常详情(开发环境 true)</param>
|
|
|
|
|
|
public static ServiceEndpointBuilder Configure(ServiceEndpointBuilder builder, bool includeDetails = false)
|
|
|
|
|
|
{
|
|
|
|
|
|
// ---- 全局异常拦截(自动捕获所有端点中未处理的异常) ----
|
|
|
|
|
|
builder.Endpoints.AddGlobalFilter(new GlobalExceptionFilter(includeDetails));
|
|
|
|
|
|
|
|
|
|
|
|
builder.ConfigureEndpoints(endpoints =>
|
|
|
|
|
|
{
|
|
|
|
|
|
// ---- 全局日志过滤器(记录每个请求) ----
|
|
|
|
|
|
endpoints.AddGlobalFilter(async (ctx, next) =>
|
|
|
|
|
|
{
|
2026-05-22 15:17:59 +08:00
|
|
|
|
Serilog.Log.Information("→ {Method} {Path}", ctx.Method, ctx.Path);
|
2026-05-21 15:52:36 +08:00
|
|
|
|
await next(ctx);
|
2026-05-22 15:17:59 +08:00
|
|
|
|
Serilog.Log.Information("← {Method} {Path} | {StatusCode}", ctx.Method, ctx.Path, ctx.StatusCode);
|
2026-05-21 15:52:36 +08:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// ---- 业务端点注册 ----
|
|
|
|
|
|
|
2026-05-21 16:45:56 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService>("api/library/drives", (service, ctx) => service.GetDrivesAsync(ctx))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "查询服务器磁盘。")
|
|
|
|
|
|
.WithName("GetLibraryDrives");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, DirectoryQueryRequest>("api/library/directories", (service, request, _) => service.GetDirectoriesAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "查询服务器目录。")
|
|
|
|
|
|
.WithName("GetLibraryDirectories");
|
|
|
|
|
|
|
|
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService>("api/library/roots", (service, ctx) => service.GetRootsAsync(ctx))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "查询文件库目录。")
|
|
|
|
|
|
.WithName("GetLibraryRoots");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, AddLibraryRootRequest>("api/library/roots", (service, request, _) => service.AddRootAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "添加文件库目录。")
|
|
|
|
|
|
.WithName("AddLibraryRoot");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, UpdateLibraryRootRequest>("api/library/roots/enabled", (service, request, _) => service.SetRootEnabledAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "启用或禁用文件库目录。")
|
|
|
|
|
|
.WithName("SetLibraryRootEnabled");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, DeleteLibraryRootRequest>("api/library/roots/delete", (service, request, _) => service.DeleteRootAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "删除文件库目录。")
|
|
|
|
|
|
.WithName("DeleteLibraryRoot");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, ScanLibraryRootRequest>("api/library/roots/scan", (service, request, _) => service.ScanRootAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "立即扫描文件库目录。")
|
|
|
|
|
|
.WithName("ScanLibraryRoot");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, SearchFilesRequest>("api/files", (service, request, _) => service.SearchFilesAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "分页查询已扫描文件。")
|
|
|
|
|
|
.WithName("SearchFiles");
|
|
|
|
|
|
|
2026-05-22 11:59:45 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, BrowseDirectoryRequest>("api/files/browse", (service, request, _) => service.BrowseDirectoryAsync(request))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "浏览文件库目录结构。")
|
|
|
|
|
|
.WithName("BrowseDirectory");
|
|
|
|
|
|
|
2026-05-22 17:01:49 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, RecentFilesRequest>("api/files/recent", (service, request, _) => service.GetRecentFilesAsync(request))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "获取最近添加或最近播放的文件。")
|
|
|
|
|
|
.WithName("GetRecentFiles");
|
|
|
|
|
|
|
|
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, MarkFilePlayedRequest>("api/files/played", (service, request, _) => service.MarkFilePlayedAsync(request))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "标记文件已播放。")
|
|
|
|
|
|
.WithName("MarkFilePlayed");
|
|
|
|
|
|
|
2026-05-22 17:44:04 +08:00
|
|
|
|
endpoints.MapPost<IFileLibraryEndpointService, SaveFileProgressRequest>("api/files/progress", (service, request, _) => service.SaveFileProgressAsync(request))
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "保存文件播放进度。")
|
|
|
|
|
|
.WithName("SaveFileProgress");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, FileQueryRequest>("api/files/detail", (service, request, _) => service.GetFileAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "查询文件详情。")
|
|
|
|
|
|
.WithName("GetFileDetail");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapGet<IFileLibraryEndpointService, FileQueryRequest>("api/files/text", (service, request, _) => service.GetTextPreviewAsync(request))
|
2026-05-21 16:45:56 +08:00
|
|
|
|
.WithOpenApi("FileLibrary", "预览文本文件。")
|
|
|
|
|
|
.WithName("GetTextPreview");
|
|
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
endpoints.MapGet("api/files/stream", GetFileStreamAsync)
|
|
|
|
|
|
.WithOpenApi("FileLibrary", "流式传输文件(支持 Range 请求)。")
|
|
|
|
|
|
.WithName("StreamManagedFile");
|
|
|
|
|
|
|
|
|
|
|
|
endpoints.MapGet<IQrCodeService>("api/qrcode", (service, ctx) => service.GenerateQrCodeAsync(ctx))
|
|
|
|
|
|
.WithOpenApi("Utility", "生成局域网访问二维码。")
|
|
|
|
|
|
.WithName("GetQrCode");
|
|
|
|
|
|
|
2026-05-21 15:52:36 +08:00
|
|
|
|
// ---- 需要鉴权的端点示例 ----
|
|
|
|
|
|
// endpoints.MapGet("api/admin/dashboard", AdminDashboardAsync)
|
|
|
|
|
|
// .WithName("AdminDashboard")
|
|
|
|
|
|
// .RequireAuthorization = true
|
|
|
|
|
|
// .Policy = "AdminOnly";
|
|
|
|
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
return builder;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#region 业务处理方法
|
|
|
|
|
|
|
2026-05-22 14:45:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 从 <see cref="ServiceEndpointContext.Items"/> 中解析 <see cref="IFileStreamService"/>,
|
|
|
|
|
|
/// 读取查询参数中的文件 ID,返回文件流响应。
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="ctx">端点上下文。</param>
|
|
|
|
|
|
/// <returns>文件流响应对象,服务不可用或 ID 无效时返回 null。</returns>
|
2026-05-22 11:18:47 +08:00
|
|
|
|
private static async Task<object?> GetFileStreamAsync(ServiceEndpointContext ctx)
|
2026-05-21 15:52:36 +08:00
|
|
|
|
{
|
|
|
|
|
|
var sp = ctx.Items["ServiceProvider"] as IServiceProvider;
|
2026-05-22 11:18:47 +08:00
|
|
|
|
var service = sp?.GetService(typeof(IFileStreamService)) as IFileStreamService;
|
|
|
|
|
|
if (service is null) return null;
|
2026-05-21 15:52:36 +08:00
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
if (!int.TryParse(ctx.Query.GetValueOrDefault("id"), out var id) || id <= 0)
|
|
|
|
|
|
return null;
|
2026-05-21 15:52:36 +08:00
|
|
|
|
|
2026-05-22 11:18:47 +08:00
|
|
|
|
return await service.GetFileStreamAsync(id);
|
2026-05-21 15:52:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|