75 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			75 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
using System.ComponentModel.DataAnnotations;
 | 
						|
using System.ComponentModel.DataAnnotations.Schema;
 | 
						|
using System.Text.Json;
 | 
						|
using System.Text.Json.Serialization;
 | 
						|
using Microsoft.EntityFrameworkCore;
 | 
						|
using NodaTime;
 | 
						|
using NodaTime.Serialization.Protobuf;
 | 
						|
 | 
						|
namespace DysonNetwork.Shared.Models;
 | 
						|
 | 
						|
/// The permission node model provides the infrastructure of permission control in Dyson Network.
 | 
						|
/// It based on the ABAC permission model.
 | 
						|
/// 
 | 
						|
/// The value can be any type, boolean and number for most cases and stored in jsonb.
 | 
						|
/// 
 | 
						|
/// The area represents the region this permission affects. For example, the pub:<publisherId>
 | 
						|
/// indicates it's a permission node for the publishers managing.
 | 
						|
///
 | 
						|
/// And the actor shows who owns the permission, in most cases, the user:<userId>
 | 
						|
/// and when the permission node has a GroupId, the actor will be set to the group, but it won't work on checking
 | 
						|
/// expect the member of that permission group inherent the permission from the group.
 | 
						|
[Index(nameof(Key), nameof(Area), nameof(Actor))]
 | 
						|
public class SnPermissionNode : ModelBase, IDisposable
 | 
						|
{
 | 
						|
    public Guid Id { get; set; } = Guid.NewGuid();
 | 
						|
    [MaxLength(1024)] public string Actor { get; set; } = null!;
 | 
						|
    [MaxLength(1024)] public string Area { get; set; } = null!;
 | 
						|
    [MaxLength(1024)] public string Key { get; set; } = null!;
 | 
						|
    [Column(TypeName = "jsonb")] public JsonDocument Value { get; set; } = null!;
 | 
						|
    public Instant? ExpiredAt { get; set; } = null;
 | 
						|
    public Instant? AffectedAt { get; set; } = null;
 | 
						|
 | 
						|
    public Guid? GroupId { get; set; } = null;
 | 
						|
    [JsonIgnore] public SnPermissionGroup? Group { get; set; } = null;
 | 
						|
 | 
						|
    public Proto.PermissionNode ToProtoValue()
 | 
						|
    {
 | 
						|
        return new Proto.PermissionNode
 | 
						|
        {
 | 
						|
            Id = Id.ToString(),
 | 
						|
            Actor = Actor,
 | 
						|
            Area = Area,
 | 
						|
            Key = Key,
 | 
						|
            Value = Google.Protobuf.WellKnownTypes.Value.Parser.ParseJson(Value.RootElement.GetRawText()),
 | 
						|
            ExpiredAt = ExpiredAt?.ToTimestamp(),
 | 
						|
            AffectedAt = AffectedAt?.ToTimestamp(),
 | 
						|
            GroupId = GroupId?.ToString() ?? string.Empty
 | 
						|
        };
 | 
						|
    }
 | 
						|
 | 
						|
    public void Dispose()
 | 
						|
    {
 | 
						|
        Value.Dispose();
 | 
						|
        GC.SuppressFinalize(this);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
public class SnPermissionGroup : ModelBase
 | 
						|
{
 | 
						|
    public Guid Id { get; set; } = Guid.NewGuid();
 | 
						|
    [MaxLength(1024)] public string Key { get; set; } = null!;
 | 
						|
 | 
						|
    public ICollection<SnPermissionNode> Nodes { get; set; } = [];
 | 
						|
    [JsonIgnore] public ICollection<SnPermissionGroupMember> Members { get; set; } = [];
 | 
						|
}
 | 
						|
 | 
						|
public class SnPermissionGroupMember : ModelBase
 | 
						|
{
 | 
						|
    public Guid GroupId { get; set; }
 | 
						|
    public SnPermissionGroup Group { get; set; } = null!;
 | 
						|
    [MaxLength(1024)] public string Actor { get; set; } = null!;
 | 
						|
 | 
						|
    public Instant? ExpiredAt { get; set; }
 | 
						|
    public Instant? AffectedAt { get; set; }
 | 
						|
} |