using Microsoft.AspNetCore.Mvc; using Newtonsoft.Json.Linq; using PluralKit.Core; namespace PluralKit.API; [ApiController] [ApiVersion("2.0")] [Route("v{version:apiVersion}")] public class MemberControllerV2: PKControllerBase { public MemberControllerV2(IServiceProvider svc) : base(svc) { } [HttpGet("systems/{systemRef}/members")] public async Task GetSystemMembers(string systemRef) { var system = await ResolveSystem(systemRef); if (system == null) throw Errors.SystemNotFound; var ctx = ContextFor(system); if (!system.MemberListPrivacy.CanAccess(ContextFor(system))) throw Errors.UnauthorizedMemberList; var members = _repo.GetSystemMembers(system.Id); return Ok(await members .Where(m => m.MemberVisibility.CanAccess(ctx)) .Select(m => m.ToJson(ctx, v: APIVersion.V2)) .ToListAsync()); } [HttpPost("members")] public async Task MemberCreate([FromBody] JObject data) { var system = await ResolveSystem("@me"); var memberCount = await _repo.GetSystemMemberCount(system.Id); var memberLimit = system.MemberLimitOverride ?? Limits.MaxMemberCount; if (memberCount >= memberLimit) throw Errors.MemberLimitReached; var patch = MemberPatch.FromJSON(data); patch.AssertIsValid(); if (!patch.Name.IsPresent) patch.Errors.Add(new ValidationError("name", "Key 'name' is required when creating new member.")); if (patch.Errors.Count > 0) throw new ModelParseError(patch.Errors); using var conn = await _db.Obtain(); using var tx = await conn.BeginTransactionAsync(); var newMember = await _repo.CreateMember(system.Id, patch.Name.Value, conn); newMember = await _repo.UpdateMember(newMember.Id, patch, conn); _ = _dispatch.Dispatch(newMember.Id, new() { Event = DispatchEvent.CREATE_MEMBER, EventData = patch.ToJson(), }); await tx.CommitAsync(); return Ok(newMember.ToJson(LookupContext.ByOwner, v: APIVersion.V2)); } [HttpGet("members/{memberRef}")] public async Task MemberGet(string memberRef) { var member = await ResolveMember(memberRef); if (member == null) throw Errors.MemberNotFound; var system = await _repo.GetSystem(member.System); return Ok(member.ToJson(ContextFor(member), systemStr: system.Hid, v: APIVersion.V2)); } [HttpPatch("members/{memberRef}")] public async Task DoMemberPatch(string memberRef, [FromBody] JObject data) { var system = await ResolveSystem("@me"); var member = await ResolveMember(memberRef); if (member == null) throw Errors.MemberNotFound; if (member.System != system.Id) throw Errors.NotOwnMemberError; var patch = MemberPatch.FromJSON(data, APIVersion.V2); patch.AssertIsValid(); if (patch.Errors.Count > 0) throw new ModelParseError(patch.Errors); var newMember = await _repo.UpdateMember(member.Id, patch); return Ok(newMember.ToJson(LookupContext.ByOwner, v: APIVersion.V2)); } [HttpDelete("members/{memberRef}")] public async Task MemberDelete(string memberRef) { var member = await ResolveMember(memberRef); if (member == null) throw Errors.MemberNotFound; var system = await ResolveSystem("@me"); if (system.Id != member.System) throw Errors.NotOwnMemberError; await _repo.DeleteMember(member.Id); return NoContent(); } }