♻️ Extract the Storage service to DysonNetwork.Drive microservice

This commit is contained in:
2025-07-06 17:29:26 +08:00
parent 6a3d04af3d
commit 14b79f16f4
71 changed files with 2629 additions and 346 deletions

View File

@ -0,0 +1,68 @@
using System.Security.Claims;
using DysonNetwork.Drive.Data;
using DysonNetwork.Drive.Models;
using Microsoft.AspNetCore.Authentication.JwtBearer;
using Microsoft.AspNetCore.Http;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Configuration;
using Microsoft.IdentityModel.Tokens;
using System.IdentityModel.Tokens.Jwt;
using System.Text;
namespace DysonNetwork.Drive.Auth;
public interface IAuthService
{
Task<string> GenerateJwtToken(Account account);
Task<Account?> GetAuthenticatedAccountAsync(ClaimsPrincipal user);
Task<Account?> GetAuthenticatedAccountAsync(HttpContext context);
}
public class AuthService : IAuthService
{
private readonly IConfiguration _configuration;
private readonly AppDatabase _db;
public AuthService(IConfiguration configuration, AppDatabase db)
{
_configuration = configuration;
_db = db;
}
public Task<string> GenerateJwtToken(Account account)
{
var tokenHandler = new JwtSecurityTokenHandler();
var key = Encoding.ASCII.GetBytes(_configuration["Jwt:Secret"] ?? throw new InvalidOperationException("JWT Secret not configured"));
var tokenDescriptor = new SecurityTokenDescriptor
{
Subject = new ClaimsIdentity(new[]
{
new Claim(ClaimTypes.NameIdentifier, account.Id.ToString()),
new Claim(ClaimTypes.Name, account.Username),
new Claim(ClaimTypes.Email, account.Email)
}),
Expires = DateTime.UtcNow.AddDays(7),
SigningCredentials = new SigningCredentials(
new SymmetricSecurityKey(key),
SecurityAlgorithms.HmacSha256Signature)
};
var token = tokenHandler.CreateToken(tokenDescriptor);
return Task.FromResult(tokenHandler.WriteToken(token));
}
public async Task<Account?> GetAuthenticatedAccountAsync(ClaimsPrincipal user)
{
var userIdClaim = user.FindFirst(ClaimTypes.NameIdentifier)?.Value;
if (string.IsNullOrEmpty(userIdClaim) || !Guid.TryParse(userIdClaim, out var userId))
return null;
return await _db.Set<Account>().FindAsync(userId);
}
public async Task<Account?> GetAuthenticatedAccountAsync(HttpContext context)
{
return await GetAuthenticatedAccountAsync(context.User);
}
}

View File

@ -0,0 +1,42 @@
using System;
using System.ComponentModel.DataAnnotations;
using DysonNetwork.Drive.Models;
namespace DysonNetwork.Drive.Auth;
public class Session
{
[Key]
public Guid Id { get; set; } = Guid.NewGuid();
[Required]
public Guid AccountId { get; set; }
public virtual Account? Account { get; set; }
[Required]
[MaxLength(64)]
public string Token { get; set; } = null!;
public string? UserAgent { get; set; }
public string? IpAddress { get; set; }
public DateTimeOffset CreatedAt { get; set; } = DateTimeOffset.UtcNow;
public DateTimeOffset? ExpiresAt { get; set; }
public DateTimeOffset LastActiveAt { get; set; } = DateTimeOffset.UtcNow;
public bool IsActive { get; set; } = true;
// Additional metadata
public string? DeviceInfo { get; set; }
public string? LocationInfo { get; set; }
public void UpdateLastActive()
{
LastActiveAt = DateTimeOffset.UtcNow;
}
public bool IsExpired()
{
return ExpiresAt.HasValue && DateTimeOffset.UtcNow >= ExpiresAt.Value;
}
}