后端: - 新增 ManagedLibraryRoot / ManagedFileRecord 数据模型及 SQLite 迁移 - 新增文件库服务、端点服务及定时扫描后台任务 - 新增 REST API: drives、directories、roots CRUD、files 分页搜索、文本预览 - 新增文件流端点支持视频/音频流式传输 - 数据库切换为 SQLite,Kestrel 绑定 0.0.0.0 支持局域网访问 前端: - 管理端:磁盘浏览、目录选择、根目录添加/启用/删除/扫描 - 客户端:根目录选择、文件搜索/筛选/分页、音视频播放、文本预览 - 全新响应式 UI(桌面+移动端),CSS 变量设计系统 - HTTP 客户端支持 Vite 开发代理与生产同源自动切换 - 移除 HTTPS 强制重定向以提升移动端视频流兼容性
85 lines
3.9 KiB
C#
85 lines
3.9 KiB
C#
using Avalonia_EFCore.Models;
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace Avalonia_EFCore.Database
|
|
{
|
|
/// <summary>
|
|
/// 应用数据库上下文 —— 继承自 Avalonia-EFCore 的 AppDbContext。
|
|
/// 所有业务实体在此注册 DbSet。
|
|
/// 这是 Avalonia-API 和 Avalonia-PC 共用的具体数据上下文。
|
|
/// </summary>
|
|
public class AppDataContext(DatabaseConfiguration dbConfig) : AppDbContext(dbConfig)
|
|
{
|
|
/// <summary>天气预报数据</summary>
|
|
public DbSet<WeatherForecastEntity> WeatherForecasts => Set<WeatherForecastEntity>();
|
|
|
|
/// <summary>用户数据</summary>
|
|
public DbSet<UserEntity> Users => Set<UserEntity>();
|
|
|
|
/// <summary>API refresh token 数据</summary>
|
|
public DbSet<ApiRefreshTokenEntity> ApiRefreshTokens => Set<ApiRefreshTokenEntity>();
|
|
|
|
/// <summary>文件库根目录数据</summary>
|
|
public DbSet<ManagedLibraryRoot> ManagedLibraryRoots => Set<ManagedLibraryRoot>();
|
|
|
|
/// <summary>文件库文件记录数据</summary>
|
|
public DbSet<ManagedFileRecord> ManagedFileRecords => Set<ManagedFileRecord>();
|
|
|
|
/// <summary>
|
|
/// 配置实体映射,包括主键、索引和属性约束。
|
|
/// </summary>
|
|
/// <param name="modelBuilder">模型构建器。</param>
|
|
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
|
{
|
|
base.OnModelCreating(modelBuilder);
|
|
|
|
modelBuilder.Entity<WeatherForecastEntity>(entity =>
|
|
{
|
|
entity.HasKey(e => e.Id).HasName("pk-weather-forecast");
|
|
entity.Property(e => e.Summary).HasMaxLength(200);
|
|
});
|
|
|
|
modelBuilder.Entity<UserEntity>(entity =>
|
|
{
|
|
entity.HasKey(e => e.Id).HasName("pk-user");
|
|
entity.Property(e => e.Email).HasMaxLength(200);
|
|
});
|
|
|
|
modelBuilder.Entity<ApiRefreshTokenEntity>(entity =>
|
|
{
|
|
entity.HasKey(e => e.Id).HasName("pk-api-refresh-token");
|
|
entity.HasIndex(e => e.TokenHash).IsUnique().HasDatabaseName("idx-api-refresh-token-hash");
|
|
entity.HasIndex(e => e.UserId).HasDatabaseName("idx-api-refresh-token-user-id");
|
|
});
|
|
|
|
modelBuilder.Entity<ManagedLibraryRoot>(entity =>
|
|
{
|
|
entity.HasKey(e => e.Id).HasName("pk-managed-library-root");
|
|
entity.HasIndex(e => e.Path).IsUnique().HasDatabaseName("idx-managed-library-root-path");
|
|
entity.Property(e => e.Path).HasMaxLength(1024);
|
|
entity.Property(e => e.DisplayName).HasMaxLength(200);
|
|
entity.Property(e => e.LastScanError).HasMaxLength(2000);
|
|
entity.Property(e => e.IsAvailable).HasDefaultValue(true);
|
|
});
|
|
|
|
modelBuilder.Entity<ManagedFileRecord>(entity =>
|
|
{
|
|
entity.HasKey(e => e.Id).HasName("pk-managed-file-record");
|
|
entity.HasIndex(e => e.LibraryRootId).HasDatabaseName("idx-managed-file-record-root-id");
|
|
entity.HasIndex(e => e.AbsolutePath).IsUnique().HasDatabaseName("idx-managed-file-record-absolute-path");
|
|
entity.HasIndex(e => new { e.MediaType, e.Exists }).HasDatabaseName("idx-managed-file-record-media-type-exists");
|
|
entity.Property(e => e.FileName).HasMaxLength(260);
|
|
entity.Property(e => e.RelativePath).HasMaxLength(1024);
|
|
entity.Property(e => e.AbsolutePath).HasMaxLength(2048);
|
|
entity.Property(e => e.Extension).HasMaxLength(32);
|
|
entity.Property(e => e.MediaType).HasMaxLength(20);
|
|
entity.Property(e => e.ContentType).HasMaxLength(100);
|
|
entity.HasOne(e => e.LibraryRoot)
|
|
.WithMany(e => e.Files)
|
|
.HasForeignKey(e => e.LibraryRootId)
|
|
.OnDelete(DeleteBehavior.Cascade);
|
|
});
|
|
}
|
|
}
|
|
}
|