1136 lines
35 KiB
C#
1136 lines
35 KiB
C#
using DotNext;
|
||
using LinqToDB;
|
||
using LinqToDB.Data;
|
||
using LinqToDB.Mapping;
|
||
|
||
namespace Database;
|
||
|
||
/// <summary>
|
||
/// 用户类,表示用户信息
|
||
/// </summary>
|
||
public class User
|
||
{
|
||
/// <summary>
|
||
/// 用户的唯一标识符
|
||
/// </summary>
|
||
[PrimaryKey]
|
||
public Guid ID { get; set; } = Guid.NewGuid();
|
||
|
||
/// <summary>
|
||
/// 用户的名称
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string Name { get; set; }
|
||
|
||
/// <summary>
|
||
/// 用户的电子邮箱
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string EMail { get; set; }
|
||
|
||
/// <summary>
|
||
/// 用户的密码(应该进行哈希处理)
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string Password { get; set; }
|
||
|
||
/// <summary>
|
||
/// 用户权限等级
|
||
/// </summary>
|
||
[NotNull]
|
||
public required UserPermission Permission { get; set; }
|
||
|
||
/// <summary>
|
||
/// 绑定的实验板ID,如果未绑定则为空
|
||
/// </summary>
|
||
[Nullable]
|
||
public Guid BoardID { get; set; }
|
||
|
||
/// <summary>
|
||
/// 用户绑定板子的过期时间
|
||
/// </summary>
|
||
[Nullable]
|
||
public DateTime? BoardExpireTime { get; set; }
|
||
|
||
/// <summary>
|
||
/// 用户权限枚举
|
||
/// </summary>
|
||
public enum UserPermission
|
||
{
|
||
/// <summary>
|
||
/// 管理员权限,可以管理用户和实验板
|
||
/// </summary>
|
||
Admin,
|
||
|
||
/// <summary>
|
||
/// 普通用户权限,只能使用实验板
|
||
/// </summary>
|
||
Normal,
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// FPGA 板子类,表示板子信息
|
||
/// </summary>
|
||
public class Board
|
||
{
|
||
/// <summary>
|
||
/// FPGA 板子的唯一标识符
|
||
/// </summary>
|
||
[PrimaryKey]
|
||
public Guid ID { get; set; } = Guid.NewGuid();
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的名称
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string BoardName { get; set; }
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的IP地址
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string IpAddr { get; set; }
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的MAC地址
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string MacAddr { get; set; }
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的通信端口
|
||
/// </summary>
|
||
[NotNull]
|
||
public int Port { get; set; } = 1234;
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的当前状态
|
||
/// </summary>
|
||
[NotNull]
|
||
public required BoardStatus Status { get; set; }
|
||
|
||
/// <summary>
|
||
/// 占用该板子的用户的唯一标识符
|
||
/// </summary>
|
||
[Nullable]
|
||
public Guid OccupiedUserID { get; set; }
|
||
|
||
/// <summary>
|
||
/// 占用该板子的用户的用户名
|
||
/// </summary>
|
||
[Nullable]
|
||
public string? OccupiedUserName { get; set; }
|
||
|
||
/// <summary>
|
||
/// FPGA 板子的固件版本号
|
||
/// </summary>
|
||
[NotNull]
|
||
public string FirmVersion { get; set; } = "1.0.0";
|
||
|
||
/// <summary>
|
||
/// FPGA 板子状态枚举
|
||
/// </summary>
|
||
public enum BoardStatus
|
||
{
|
||
/// <summary>
|
||
/// 未启用状态,无法被使用
|
||
/// </summary>
|
||
Disabled,
|
||
|
||
/// <summary>
|
||
/// 繁忙状态,正在被用户使用
|
||
/// </summary>
|
||
Busy,
|
||
|
||
/// <summary>
|
||
/// 可用状态,可以被分配给用户
|
||
/// </summary>
|
||
Available,
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 实验类,表示实验信息
|
||
/// </summary>
|
||
public class Exam
|
||
{
|
||
/// <summary>
|
||
/// 实验的唯一标识符
|
||
/// </summary>
|
||
[PrimaryKey]
|
||
public required string ID { get; set; }
|
||
|
||
/// <summary>
|
||
/// 实验名称
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string Name { get; set; }
|
||
|
||
/// <summary>
|
||
/// 实验描述
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string Description { get; set; }
|
||
|
||
/// <summary>
|
||
/// 实验创建时间
|
||
/// </summary>
|
||
[NotNull]
|
||
public DateTime CreatedTime { get; set; } = DateTime.Now;
|
||
|
||
/// <summary>
|
||
/// 实验最后更新时间
|
||
/// </summary>
|
||
[NotNull]
|
||
public DateTime UpdatedTime { get; set; } = DateTime.Now;
|
||
|
||
/// <summary>
|
||
/// 实验标签(以逗号分隔的字符串)
|
||
/// </summary>
|
||
[NotNull]
|
||
public string Tags { get; set; } = "";
|
||
|
||
/// <summary>
|
||
/// 实验难度(1-5,1为最简单)
|
||
/// </summary>
|
||
[NotNull]
|
||
public int Difficulty { get; set; } = 1;
|
||
|
||
/// <summary>
|
||
/// 普通用户是否可见
|
||
/// </summary>
|
||
[NotNull]
|
||
public bool IsVisibleToUsers { get; set; } = true;
|
||
|
||
/// <summary>
|
||
/// 获取标签列表
|
||
/// </summary>
|
||
/// <returns>标签数组</returns>
|
||
public string[] GetTagsList()
|
||
{
|
||
if (string.IsNullOrWhiteSpace(Tags))
|
||
return Array.Empty<string>();
|
||
|
||
return Tags.Split(',', StringSplitOptions.RemoveEmptyEntries)
|
||
.Select(tag => tag.Trim())
|
||
.Where(tag => !string.IsNullOrEmpty(tag))
|
||
.ToArray();
|
||
}
|
||
|
||
/// <summary>
|
||
/// 设置标签列表
|
||
/// </summary>
|
||
/// <param name="tags">标签数组</param>
|
||
public void SetTagsList(string[] tags)
|
||
{
|
||
Tags = string.Join(",", tags.Where(tag => !string.IsNullOrWhiteSpace(tag)).Select(tag => tag.Trim()));
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 实验资源表(图片等)
|
||
/// </summary>
|
||
public class ExamResource
|
||
{
|
||
/// <summary>
|
||
/// 资源的唯一标识符
|
||
/// </summary>
|
||
[PrimaryKey, Identity]
|
||
public int ID { get; set; }
|
||
|
||
/// <summary>
|
||
/// 所属实验ID
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string ExamID { get; set; }
|
||
|
||
/// <summary>
|
||
/// 资源类型(images, markdown, bitstream, diagram, project)
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string ResourceType { get; set; }
|
||
|
||
/// <summary>
|
||
/// 资源名称(包含文件扩展名)
|
||
/// </summary>
|
||
[NotNull]
|
||
public required string ResourceName { get; set; }
|
||
|
||
/// <summary>
|
||
/// 资源的二进制数据
|
||
/// </summary>
|
||
[NotNull]
|
||
public required byte[] Data { get; set; }
|
||
|
||
/// <summary>
|
||
/// 资源创建时间
|
||
/// </summary>
|
||
[NotNull]
|
||
public DateTime CreatedTime { get; set; } = DateTime.Now;
|
||
|
||
/// <summary>
|
||
/// 资源的MIME类型
|
||
/// </summary>
|
||
[NotNull]
|
||
public string MimeType { get; set; } = "application/octet-stream";
|
||
|
||
/// <summary>
|
||
/// 资源类型枚举
|
||
/// </summary>
|
||
public static class ResourceTypes
|
||
{
|
||
/// <summary>
|
||
/// 图片资源类型
|
||
/// </summary>
|
||
public const string Images = "images";
|
||
|
||
/// <summary>
|
||
/// Markdown文档资源类型
|
||
/// </summary>
|
||
public const string Markdown = "markdown";
|
||
|
||
/// <summary>
|
||
/// 比特流文件资源类型
|
||
/// </summary>
|
||
public const string Bitstream = "bitstream";
|
||
|
||
/// <summary>
|
||
/// 原理图资源类型
|
||
/// </summary>
|
||
public const string Diagram = "diagram";
|
||
|
||
/// <summary>
|
||
/// 项目文件资源类型
|
||
/// </summary>
|
||
public const string Project = "project";
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 应用程序数据连接类,用于与数据库交互
|
||
/// </summary>
|
||
public class AppDataConnection : DataConnection
|
||
{
|
||
private static NLog.Logger logger = NLog.LogManager.GetCurrentClassLogger();
|
||
|
||
static readonly string DATABASE_FILEPATH = $"{Environment.CurrentDirectory}/Database.sqlite";
|
||
|
||
static readonly LinqToDB.DataOptions options =
|
||
new LinqToDB.DataOptions().UseSQLite($"Data Source={DATABASE_FILEPATH}");
|
||
|
||
/// <summary>
|
||
/// 初始化应用程序数据连接
|
||
/// </summary>
|
||
public AppDataConnection() : base(options)
|
||
{
|
||
if (!Path.Exists(DATABASE_FILEPATH))
|
||
{
|
||
logger.Info($"数据库文件不存在,正在创建新数据库: {DATABASE_FILEPATH}");
|
||
LinqToDB.DataProvider.SQLite.SQLiteTools.CreateDatabase(DATABASE_FILEPATH);
|
||
this.CreateAllTables();
|
||
var user = new User()
|
||
{
|
||
Name = "Admin",
|
||
EMail = "selfconfusion@gmail.com",
|
||
Password = "12345678",
|
||
Permission = Database.User.UserPermission.Admin,
|
||
};
|
||
this.Insert(user);
|
||
logger.Info("默认管理员用户已创建");
|
||
}
|
||
else
|
||
{
|
||
logger.Info($"数据库连接已建立: {DATABASE_FILEPATH}");
|
||
}
|
||
}
|
||
|
||
|
||
/// <summary>
|
||
/// 创建所有数据库表
|
||
/// </summary>
|
||
public void CreateAllTables()
|
||
{
|
||
logger.Info("正在创建数据库表...");
|
||
this.CreateTable<User>();
|
||
this.CreateTable<Board>();
|
||
this.CreateTable<Exam>();
|
||
this.CreateTable<ExamResource>();
|
||
logger.Info("数据库表创建完成");
|
||
}
|
||
|
||
/// <summary>
|
||
/// 删除所有数据库表
|
||
/// </summary>
|
||
public void DropAllTables()
|
||
{
|
||
logger.Warn("正在删除所有数据库表...");
|
||
this.DropTable<User>();
|
||
this.DropTable<Board>();
|
||
this.DropTable<Exam>();
|
||
this.DropTable<ExamResource>();
|
||
logger.Warn("所有数据库表已删除");
|
||
}
|
||
|
||
/// <summary>
|
||
/// 添加一个新的用户到数据库
|
||
/// </summary>
|
||
/// <param name="name">用户的名称</param>
|
||
/// <param name="email">用户的电子邮箱地址</param>
|
||
/// <param name="password">用户的密码</param>
|
||
/// <returns>插入的记录数</returns>
|
||
public int AddUser(string name, string email, string password)
|
||
{
|
||
var user = new User()
|
||
{
|
||
Name = name,
|
||
EMail = email,
|
||
Password = password,
|
||
Permission = Database.User.UserPermission.Normal,
|
||
};
|
||
var result = this.Insert(user);
|
||
logger.Info($"新用户已添加: {name} ({email})");
|
||
return result;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据用户名获取用户信息
|
||
/// </summary>
|
||
/// <param name="name">用户名</param>
|
||
/// <returns>包含用户信息的结果,如果未找到或出错则返回相应状态</returns>
|
||
public Result<Optional<User>> GetUserByName(string name)
|
||
{
|
||
var user = this.UserTable.Where((user) => user.Name == name).ToArray();
|
||
|
||
if (user.Length > 1)
|
||
{
|
||
logger.Error($"数据库中存在多个同名用户: {name}");
|
||
return new(new Exception($"数据库中存在多个同名用户: {name}"));
|
||
}
|
||
|
||
if (user.Length == 0)
|
||
{
|
||
logger.Info($"未找到用户: {name}");
|
||
return new(Optional<User>.None);
|
||
}
|
||
|
||
logger.Debug($"成功获取用户信息: {name}");
|
||
return new(user[0]);
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据电子邮箱获取用户信息
|
||
/// </summary>
|
||
/// <param name="email">用户的电子邮箱地址</param>
|
||
/// <returns>包含用户信息的结果,如果未找到或出错则返回相应状态</returns>
|
||
public Result<Optional<User>> GetUserByEMail(string email)
|
||
{
|
||
var user = this.UserTable.Where((user) => user.EMail == email).ToArray();
|
||
|
||
if (user.Length > 1)
|
||
{
|
||
logger.Error($"数据库中存在多个相同邮箱的用户: {email}");
|
||
return new(new Exception($"数据库中存在多个相同邮箱的用户: {email}"));
|
||
}
|
||
|
||
if (user.Length == 0)
|
||
{
|
||
logger.Info($"未找到邮箱对应的用户: {email}");
|
||
return new(Optional<User>.None);
|
||
}
|
||
|
||
logger.Debug($"成功获取用户信息: {email}");
|
||
return new(user[0]);
|
||
}
|
||
|
||
/// <summary>
|
||
/// 验证用户密码
|
||
/// </summary>
|
||
/// <param name="name">用户名</param>
|
||
/// <param name="password">用户密码</param>
|
||
/// <returns>如果密码正确返回用户信息,否则返回空</returns>
|
||
public Result<Optional<User>> CheckUserPassword(string name, string password)
|
||
{
|
||
var ret = this.GetUserByName(name);
|
||
if (!ret.IsSuccessful)
|
||
return new(ret.Error);
|
||
|
||
if (!ret.Value.HasValue)
|
||
return new(Optional<User>.None);
|
||
|
||
var user = ret.Value.Value;
|
||
|
||
if (user.Password == password)
|
||
{
|
||
logger.Info($"用户 {name} 密码验证成功");
|
||
return new(user);
|
||
}
|
||
else
|
||
{
|
||
logger.Warn($"用户 {name} 密码验证失败");
|
||
return new(Optional<User>.None);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 绑定用户与实验板
|
||
/// </summary>
|
||
/// <param name="userId">用户的唯一标识符</param>
|
||
/// <param name="boardId">实验板的唯一标识符</param>
|
||
/// <param name="expireTime">绑定过期时间</param>
|
||
/// <returns>更新的记录数</returns>
|
||
public int BindUserToBoard(Guid userId, Guid boardId, DateTime expireTime)
|
||
{
|
||
// 获取用户信息
|
||
var user = this.UserTable.Where(u => u.ID == userId).FirstOrDefault();
|
||
if (user == null)
|
||
{
|
||
logger.Error($"未找到用户: {userId}");
|
||
return 0;
|
||
}
|
||
|
||
// 更新用户的板子绑定信息
|
||
var userResult = this.UserTable
|
||
.Where(u => u.ID == userId)
|
||
.Set(u => u.BoardID, boardId)
|
||
.Set(u => u.BoardExpireTime, expireTime)
|
||
.Update();
|
||
|
||
// 更新板子的用户绑定信息
|
||
var boardResult = this.BoardTable
|
||
.Where(b => b.ID == boardId)
|
||
.Set(b => b.Status, Board.BoardStatus.Busy)
|
||
.Set(b => b.OccupiedUserID, userId)
|
||
.Set(b => b.OccupiedUserName, user.Name)
|
||
.Update();
|
||
|
||
logger.Info($"用户 {userId} ({user.Name}) 已绑定到实验板 {boardId},过期时间: {expireTime}");
|
||
return userResult + boardResult;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 解除用户与实验板的绑定
|
||
/// </summary>
|
||
/// <param name="userId">用户的唯一标识符</param>
|
||
/// <returns>更新的记录数</returns>
|
||
public int UnbindUserFromBoard(Guid userId)
|
||
{
|
||
// 获取用户当前绑定的板子ID
|
||
var user = this.UserTable.Where(u => u.ID == userId).FirstOrDefault();
|
||
Guid boardId = user?.BoardID ?? Guid.Empty;
|
||
|
||
// 清空用户的板子绑定信息
|
||
var userResult = this.UserTable
|
||
.Where(u => u.ID == userId)
|
||
.Set(u => u.BoardID, Guid.Empty)
|
||
.Set(u => u.BoardExpireTime, (DateTime?)null)
|
||
.Update();
|
||
|
||
// 如果用户原本绑定了板子,则清空板子的用户绑定信息
|
||
int boardResult = 0;
|
||
if (boardId != Guid.Empty)
|
||
{
|
||
boardResult = this.BoardTable
|
||
.Where(b => b.ID == boardId)
|
||
.Set(b => b.Status, Board.BoardStatus.Available)
|
||
.Set(b => b.OccupiedUserID, Guid.Empty)
|
||
.Set(b => b.OccupiedUserName, (string?)null)
|
||
.Update();
|
||
logger.Info($"实验板 {boardId} 状态已设置为空闲,用户绑定信息已清空");
|
||
}
|
||
|
||
logger.Info($"用户 {userId} 已解除实验板绑定");
|
||
return userResult + boardResult;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 自动分配一个未被占用的IP地址
|
||
/// </summary>
|
||
/// <returns>分配的IP地址字符串</returns>
|
||
public string AllocateIpAddr()
|
||
{
|
||
var usedIps = this.BoardTable.Select(b => b.IpAddr).ToArray();
|
||
for (int i = 1; i <= 254; i++)
|
||
{
|
||
string ip = $"169.254.109.{i}";
|
||
if (!usedIps.Contains(ip))
|
||
return ip;
|
||
}
|
||
throw new Exception("没有可用的IP地址");
|
||
}
|
||
|
||
/// <summary>
|
||
/// 自动分配一个未被占用的MAC地址
|
||
/// </summary>
|
||
/// <returns>分配的MAC地址字符串</returns>
|
||
public string AllocateMacAddr()
|
||
{
|
||
var usedMacs = this.BoardTable.Select(b => b.MacAddr).ToArray();
|
||
// 以 02-00-00-xx-xx-xx 格式分配,02 表示本地管理地址
|
||
for (int i = 1; i <= 0xFFFFFF; i++)
|
||
{
|
||
string mac = $"02-00-00-{(i >> 16) & 0xFF:X2}-{(i >> 8) & 0xFF:X2}-{i & 0xFF:X2}";
|
||
if (!usedMacs.Contains(mac))
|
||
return mac;
|
||
}
|
||
throw new Exception("没有可用的MAC地址");
|
||
}
|
||
|
||
/// <summary>
|
||
/// 添加一块新的 FPGA 板子到数据库
|
||
/// </summary>
|
||
/// <param name="name">FPGA 板子的名称</param>
|
||
/// <returns>插入的记录数</returns>
|
||
public Guid AddBoard(string name)
|
||
{
|
||
if (string.IsNullOrWhiteSpace(name) || name.Contains('\'') || name.Contains(';'))
|
||
{
|
||
logger.Error("实验板名称非法,包含不允许的字符");
|
||
throw new ArgumentException("实验板名称非法");
|
||
}
|
||
var board = new Board()
|
||
{
|
||
BoardName = name,
|
||
IpAddr = AllocateIpAddr(),
|
||
MacAddr = AllocateMacAddr(),
|
||
Status = Database.Board.BoardStatus.Disabled,
|
||
};
|
||
var result = this.Insert(board);
|
||
logger.Info($"新实验板已添加: {name} ({board.IpAddr}:{board.MacAddr})");
|
||
return board.ID;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据名称删除实验板
|
||
/// </summary>
|
||
/// <param name="name">实验板的名称</param>
|
||
/// <returns>删除的记录数</returns>
|
||
public int DeleteBoardByName(string name)
|
||
{
|
||
// 先获取要删除的板子信息
|
||
var board = this.BoardTable.Where(b => b.BoardName == name).FirstOrDefault();
|
||
if (board == null)
|
||
{
|
||
logger.Warn($"未找到名称为 {name} 的实验板");
|
||
return 0;
|
||
}
|
||
|
||
// 如果板子被占用,先解除绑定
|
||
if (board.OccupiedUserID != Guid.Empty)
|
||
{
|
||
this.UserTable
|
||
.Where(u => u.ID == board.OccupiedUserID)
|
||
.Set(u => u.BoardID, Guid.Empty)
|
||
.Set(u => u.BoardExpireTime, (DateTime?)null)
|
||
.Update();
|
||
logger.Info($"已解除用户 {board.OccupiedUserID} 与实验板 {name} 的绑定");
|
||
}
|
||
|
||
var result = this.BoardTable.Where(b => b.BoardName == name).Delete();
|
||
logger.Info($"实验板已删除: {name},删除记录数: {result}");
|
||
return result;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据ID删除实验板
|
||
/// </summary>
|
||
/// <param name="id">实验板的唯一标识符</param>
|
||
/// <returns>删除的记录数</returns>
|
||
public int DeleteBoardByID(Guid id)
|
||
{
|
||
// 先获取要删除的板子信息
|
||
var board = this.BoardTable.Where(b => b.ID == id).FirstOrDefault();
|
||
if (board == null)
|
||
{
|
||
logger.Warn($"未找到ID为 {id} 的实验板");
|
||
return 0;
|
||
}
|
||
|
||
// 如果板子被占用,先解除绑定
|
||
if (board.OccupiedUserID != Guid.Empty)
|
||
{
|
||
this.UserTable
|
||
.Where(u => u.ID == board.OccupiedUserID)
|
||
.Set(u => u.BoardID, Guid.Empty)
|
||
.Set(u => u.BoardExpireTime, (DateTime?)null)
|
||
.Update();
|
||
logger.Info($"已解除用户 {board.OccupiedUserID} 与实验板 {id} 的绑定");
|
||
}
|
||
|
||
var result = this.BoardTable.Where(b => b.ID == id).Delete();
|
||
logger.Info($"实验板已删除: {id},删除记录数: {result}");
|
||
return result;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据实验板ID获取实验板信息
|
||
/// </summary>
|
||
/// <param name="id">实验板的唯一标识符</param>
|
||
/// <returns>包含实验板信息的结果,如果未找到则返回空</returns>
|
||
public Result<Optional<Board>> GetBoardByID(Guid id)
|
||
{
|
||
var boards = this.BoardTable.Where(board => board.ID == id).ToArray();
|
||
|
||
if (boards.Length > 1)
|
||
{
|
||
logger.Error($"数据库中存在多个相同ID的实验板: {id}");
|
||
return new(new Exception($"数据库中存在多个相同ID的实验板: {id}"));
|
||
}
|
||
|
||
if (boards.Length == 0)
|
||
{
|
||
logger.Info($"未找到ID对应的实验板: {id}");
|
||
return new(Optional<Board>.None);
|
||
}
|
||
|
||
logger.Debug($"成功获取实验板信息: {id}");
|
||
return new(boards[0]);
|
||
}
|
||
|
||
/// <summary>
|
||
/// 获取所有实验板信息
|
||
/// </summary>
|
||
/// <returns>所有实验板的数组</returns>
|
||
public Board[] GetAllBoard()
|
||
{
|
||
var boards = this.BoardTable.ToArray();
|
||
logger.Debug($"获取所有实验板,共 {boards.Length} 块");
|
||
return boards;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 获取一块可用的实验板并将其状态设置为繁忙
|
||
/// </summary>
|
||
/// <param name="userId">要分配板子的用户ID</param>
|
||
/// <param name="expireTime">绑定过期时间</param>
|
||
/// <returns>可用的实验板,如果没有可用的板子则返回空</returns>
|
||
public Optional<Board> GetAvailableBoard(Guid userId, DateTime expireTime)
|
||
{
|
||
var boards = this.BoardTable.Where(
|
||
(board) => board.Status == Database.Board.BoardStatus.Available
|
||
).ToArray();
|
||
|
||
if (boards.Length == 0)
|
||
{
|
||
logger.Warn("没有可用的实验板");
|
||
return new(null);
|
||
}
|
||
else
|
||
{
|
||
var board = boards[0];
|
||
var user = this.UserTable.Where(u => u.ID == userId).FirstOrDefault();
|
||
|
||
if (user == null)
|
||
{
|
||
logger.Error($"未找到用户: {userId}");
|
||
return new(null);
|
||
}
|
||
|
||
// 更新板子状态和用户绑定信息
|
||
this.BoardTable
|
||
.Where(target => target.ID == board.ID)
|
||
.Set(target => target.Status, Board.BoardStatus.Busy)
|
||
.Set(target => target.OccupiedUserID, userId)
|
||
.Set(target => target.OccupiedUserName, user.Name)
|
||
.Update();
|
||
|
||
// 更新用户的板子绑定信息
|
||
this.UserTable
|
||
.Where(u => u.ID == userId)
|
||
.Set(u => u.BoardID, board.ID)
|
||
.Set(u => u.BoardExpireTime, expireTime)
|
||
.Update();
|
||
|
||
board.Status = Database.Board.BoardStatus.Busy;
|
||
board.OccupiedUserID = userId;
|
||
board.OccupiedUserName = user.Name;
|
||
|
||
logger.Info($"实验板 {board.BoardName} ({board.ID}) 已分配给用户 {user.Name} ({userId}),过期时间: {expireTime}");
|
||
return new(board);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// [TODO:description]
|
||
/// </summary>
|
||
/// <param name="boardId">[TODO:parameter]</param>
|
||
/// <param name="newName">[TODO:parameter]</param>
|
||
/// <returns>[TODO:return]</returns>
|
||
public int UpdateBoardName(Guid boardId, string newName)
|
||
{
|
||
if (string.IsNullOrWhiteSpace(newName) || newName.Contains('\'') || newName.Contains(';'))
|
||
{
|
||
logger.Error("实验板名称非法,包含不允许的字符");
|
||
return 0;
|
||
}
|
||
var result = this.BoardTable
|
||
.Where(b => b.ID == boardId)
|
||
.Set(b => b.BoardName, newName)
|
||
.Update();
|
||
logger.Info($"实验板名称已更新: {boardId} -> {newName}");
|
||
return result;
|
||
}
|
||
|
||
/// <summary>
|
||
/// [TODO:description]
|
||
/// </summary>
|
||
/// <param name="boardId">[TODO:parameter]</param>
|
||
/// <param name="newStatus">[TODO:parameter]</param>
|
||
/// <returns>[TODO:return]</returns>
|
||
public int UpdateBoardStatus(Guid boardId, Board.BoardStatus newStatus)
|
||
{
|
||
var result = this.BoardTable
|
||
.Where(b => b.ID == boardId)
|
||
.Set(b => b.Status, newStatus)
|
||
.Update();
|
||
logger.Info($"TODO");
|
||
return result;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 用户表
|
||
/// </summary>
|
||
public ITable<User> UserTable => this.GetTable<User>();
|
||
|
||
/// <summary>
|
||
/// FPGA 板子表
|
||
/// </summary>
|
||
public ITable<Board> BoardTable => this.GetTable<Board>();
|
||
|
||
/// <summary>
|
||
/// 实验表
|
||
/// </summary>
|
||
public ITable<Exam> ExamTable => this.GetTable<Exam>();
|
||
|
||
/// <summary>
|
||
/// 实验资源表
|
||
/// </summary>
|
||
public ITable<ExamResource> ExamResourceTable => this.GetTable<ExamResource>();
|
||
|
||
/// <summary>
|
||
/// 创建新实验
|
||
/// </summary>
|
||
/// <param name="id">实验ID</param>
|
||
/// <param name="name">实验名称</param>
|
||
/// <param name="description">实验描述</param>
|
||
/// <param name="tags">实验标签</param>
|
||
/// <param name="difficulty">实验难度</param>
|
||
/// <param name="isVisibleToUsers">普通用户是否可见</param>
|
||
/// <returns>创建的实验</returns>
|
||
public Result<Exam> CreateExam(string id, string name, string description, string[]? tags = null, int difficulty = 1, bool isVisibleToUsers = true)
|
||
{
|
||
try
|
||
{
|
||
// 检查实验ID是否已存在
|
||
var existingExam = this.ExamTable.Where(e => e.ID == id).FirstOrDefault();
|
||
if (existingExam != null)
|
||
{
|
||
logger.Error($"实验ID已存在: {id}");
|
||
return new(new Exception($"实验ID已存在: {id}"));
|
||
}
|
||
|
||
var exam = new Exam
|
||
{
|
||
ID = id,
|
||
Name = name,
|
||
Description = description,
|
||
Difficulty = Math.Max(1, Math.Min(5, difficulty)),
|
||
IsVisibleToUsers = isVisibleToUsers,
|
||
CreatedTime = DateTime.Now,
|
||
UpdatedTime = DateTime.Now
|
||
};
|
||
|
||
if (tags != null)
|
||
{
|
||
exam.SetTagsList(tags);
|
||
}
|
||
|
||
this.Insert(exam);
|
||
logger.Info($"新实验已创建: {id} ({name})");
|
||
return new(exam);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"创建实验时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 更新实验信息
|
||
/// </summary>
|
||
/// <param name="id">实验ID</param>
|
||
/// <param name="name">实验名称</param>
|
||
/// <param name="description">实验描述</param>
|
||
/// <param name="tags">实验标签</param>
|
||
/// <param name="difficulty">实验难度</param>
|
||
/// <param name="isVisibleToUsers">普通用户是否可见</param>
|
||
/// <returns>更新的记录数</returns>
|
||
public Result<int> UpdateExam(string id, string? name = null, string? description = null, string[]? tags = null, int? difficulty = null, bool? isVisibleToUsers = null)
|
||
{
|
||
try
|
||
{
|
||
int result = 0;
|
||
|
||
if (name != null)
|
||
{
|
||
result += this.ExamTable.Where(e => e.ID == id).Set(e => e.Name, name).Update();
|
||
}
|
||
if (description != null)
|
||
{
|
||
result += this.ExamTable.Where(e => e.ID == id).Set(e => e.Description, description).Update();
|
||
}
|
||
if (tags != null)
|
||
{
|
||
var tagsString = string.Join(",", tags.Where(tag => !string.IsNullOrWhiteSpace(tag)).Select(tag => tag.Trim()));
|
||
result += this.ExamTable.Where(e => e.ID == id).Set(e => e.Tags, tagsString).Update();
|
||
}
|
||
if (difficulty.HasValue)
|
||
{
|
||
result += this.ExamTable.Where(e => e.ID == id).Set(e => e.Difficulty, Math.Max(1, Math.Min(5, difficulty.Value))).Update();
|
||
}
|
||
if (isVisibleToUsers.HasValue)
|
||
{
|
||
result += this.ExamTable.Where(e => e.ID == id).Set(e => e.IsVisibleToUsers, isVisibleToUsers.Value).Update();
|
||
}
|
||
|
||
// 更新时间
|
||
this.ExamTable.Where(e => e.ID == id).Set(e => e.UpdatedTime, DateTime.Now).Update();
|
||
|
||
logger.Info($"实验已更新: {id},更新记录数: {result}");
|
||
return new(result);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"更新实验时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 添加实验资源
|
||
/// </summary>
|
||
/// <param name="examId">所属实验ID</param>
|
||
/// <param name="resourceType">资源类型</param>
|
||
/// <param name="resourceName">资源名称</param>
|
||
/// <param name="data">资源二进制数据</param>
|
||
/// <param name="mimeType">MIME类型(可选,将根据文件扩展名自动确定)</param>
|
||
/// <returns>创建的资源</returns>
|
||
public Result<ExamResource> AddExamResource(string examId, string resourceType, string resourceName, byte[] data, string? mimeType = null)
|
||
{
|
||
try
|
||
{
|
||
// 验证实验是否存在
|
||
var exam = this.ExamTable.Where(e => e.ID == examId).FirstOrDefault();
|
||
if (exam == null)
|
||
{
|
||
logger.Error($"实验不存在: {examId}");
|
||
return new(new Exception($"实验不存在: {examId}"));
|
||
}
|
||
|
||
// 检查资源是否已存在
|
||
var existingResource = this.ExamResourceTable
|
||
.Where(r => r.ExamID == examId && r.ResourceType == resourceType && r.ResourceName == resourceName)
|
||
.FirstOrDefault();
|
||
if (existingResource != null)
|
||
{
|
||
logger.Error($"资源已存在: {examId}/{resourceType}/{resourceName}");
|
||
return new(new Exception($"资源已存在: {examId}/{resourceType}/{resourceName}"));
|
||
}
|
||
|
||
// 如果未指定MIME类型,根据文件扩展名自动确定
|
||
if (string.IsNullOrEmpty(mimeType))
|
||
{
|
||
var extension = Path.GetExtension(resourceName).ToLowerInvariant();
|
||
mimeType = GetMimeTypeFromExtension(extension, resourceName);
|
||
}
|
||
|
||
var resource = new ExamResource
|
||
{
|
||
ExamID = examId,
|
||
ResourceType = resourceType,
|
||
ResourceName = resourceName,
|
||
Data = data,
|
||
MimeType = mimeType,
|
||
CreatedTime = DateTime.Now
|
||
};
|
||
|
||
this.Insert(resource);
|
||
logger.Info($"新资源已添加: {examId}/{resourceType}/{resourceName} ({data.Length} bytes)");
|
||
return new(resource);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"添加实验资源时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 获取指定实验ID的指定资源类型的所有资源的ID和名称
|
||
/// </summary>
|
||
/// <param name="examId">实验ID</param>
|
||
/// <param name="resourceType">资源类型</param>
|
||
/// <returns>资源信息列表</returns>
|
||
public Result<(int ID, string Name)[]> GetExamResourceList(string examId, string resourceType)
|
||
{
|
||
try
|
||
{
|
||
var resources = this.ExamResourceTable
|
||
.Where(r => r.ExamID == examId && r.ResourceType == resourceType)
|
||
.Select(r => new { r.ID, r.ResourceName })
|
||
.ToArray();
|
||
|
||
var result = resources.Select(r => (r.ID, r.ResourceName)).ToArray();
|
||
logger.Info($"获取实验资源列表: {examId}/{resourceType},共 {result.Length} 个资源");
|
||
return new(result);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"获取实验资源列表时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据资源ID获取资源
|
||
/// </summary>
|
||
/// <param name="resourceId">资源ID</param>
|
||
/// <returns>资源数据</returns>
|
||
public Result<Optional<ExamResource>> GetExamResourceById(int resourceId)
|
||
{
|
||
try
|
||
{
|
||
var resource = this.ExamResourceTable.Where(r => r.ID == resourceId).FirstOrDefault();
|
||
|
||
if (resource == null)
|
||
{
|
||
logger.Info($"未找到资源: {resourceId}");
|
||
return new(Optional<ExamResource>.None);
|
||
}
|
||
|
||
logger.Info($"成功获取资源: {resourceId} ({resource.ResourceName})");
|
||
return new(resource);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"获取资源时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 删除实验资源
|
||
/// </summary>
|
||
/// <param name="resourceId">资源ID</param>
|
||
/// <returns>删除的记录数</returns>
|
||
public Result<int> DeleteExamResource(int resourceId)
|
||
{
|
||
try
|
||
{
|
||
var result = this.ExamResourceTable.Where(r => r.ID == resourceId).Delete();
|
||
logger.Info($"资源已删除: {resourceId},删除记录数: {result}");
|
||
return new(result);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
logger.Error($"删除资源时出错: {ex.Message}");
|
||
return new(ex);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据文件扩展名获取MIME类型
|
||
/// </summary>
|
||
/// <param name="extension">文件扩展名</param>
|
||
/// <param name="fileName">文件名(可选,用于特殊文件判断)</param>
|
||
/// <returns>MIME类型</returns>
|
||
private string GetMimeTypeFromExtension(string extension, string fileName = "")
|
||
{
|
||
// 特殊文件名处理
|
||
if (!string.IsNullOrEmpty(fileName) && fileName.Equals("diagram.json", StringComparison.OrdinalIgnoreCase))
|
||
{
|
||
return "application/json";
|
||
}
|
||
|
||
return extension.ToLowerInvariant() switch
|
||
{
|
||
".png" => "image/png",
|
||
".jpg" or ".jpeg" => "image/jpeg",
|
||
".gif" => "image/gif",
|
||
".bmp" => "image/bmp",
|
||
".svg" => "image/svg+xml",
|
||
".sbit" => "application/octet-stream",
|
||
".bit" => "application/octet-stream",
|
||
".bin" => "application/octet-stream",
|
||
".json" => "application/json",
|
||
".zip" => "application/zip",
|
||
".md" => "text/markdown",
|
||
_ => "application/octet-stream"
|
||
};
|
||
}
|
||
|
||
/// <summary>
|
||
/// 获取所有实验信息
|
||
/// </summary>
|
||
/// <returns>所有实验的数组</returns>
|
||
public Exam[] GetAllExams()
|
||
{
|
||
var exams = this.ExamTable.OrderBy(e => e.ID).ToArray();
|
||
logger.Debug($"获取所有实验,共 {exams.Length} 个");
|
||
return exams;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 根据实验ID获取实验信息
|
||
/// </summary>
|
||
/// <param name="examId">实验ID</param>
|
||
/// <returns>包含实验信息的结果,如果未找到则返回空</returns>
|
||
public Result<Optional<Exam>> GetExamByID(string examId)
|
||
{
|
||
var exams = this.ExamTable.Where(exam => exam.ID == examId).ToArray();
|
||
|
||
if (exams.Length > 1)
|
||
{
|
||
logger.Error($"数据库中存在多个相同ID的实验: {examId}");
|
||
return new(new Exception($"数据库中存在多个相同ID的实验: {examId}"));
|
||
}
|
||
|
||
if (exams.Length == 0)
|
||
{
|
||
logger.Info($"未找到ID对应的实验: {examId}");
|
||
return new(Optional<Exam>.None);
|
||
}
|
||
|
||
logger.Debug($"成功获取实验信息: {examId}");
|
||
return new(exams[0]);
|
||
}
|
||
|
||
/// <summary>
|
||
/// 删除所有实验
|
||
/// </summary>
|
||
/// <returns>删除的实验数量</returns>
|
||
public int DeleteAllExams()
|
||
{
|
||
// 先删除所有实验资源
|
||
var resourceDeleteCount = this.DeleteAllExamResources();
|
||
logger.Info($"已删除所有实验资源,共删除 {resourceDeleteCount} 个资源");
|
||
|
||
// 再删除所有实验
|
||
var examDeleteCount = this.ExamTable.Delete();
|
||
logger.Info($"已删除所有实验,共删除 {examDeleteCount} 个实验");
|
||
return examDeleteCount;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 删除所有实验资源
|
||
/// </summary>
|
||
/// <returns>删除的资源数量</returns>
|
||
public int DeleteAllExamResources()
|
||
{
|
||
var deleteCount = this.ExamResourceTable.Delete();
|
||
logger.Info($"已删除所有实验资源,共删除 {deleteCount} 个资源");
|
||
return deleteCount;
|
||
}
|
||
}
|