PluralKit/PluralKit.API/Controllers/PrivateController.cs

139 lines
4.5 KiB
C#
Raw Normal View History

using Microsoft.AspNetCore.Mvc;
2022-04-07 03:49:06 -04:00
using SqlKata;
2022-03-15 00:29:34 -04:00
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using PluralKit.Core;
namespace PluralKit.API;
// Internal API definitions
// I would prefer if you do not use any of these APIs in your own integrations.
// It is unstable and subject to change at any time (which is why it's not versioned)
// If for some reason you do need access to something defined here,
// let us know in #api-support on the support server (https://discord.com/invite/PczBt78) and I'll see if it can be made public
[ApiController]
[Route("private")]
public class PrivateController: PKControllerBase
{
public PrivateController(IServiceProvider svc) : base(svc) { }
2022-04-07 03:49:06 -04:00
[HttpPost("bulk_privacy/member")]
public async Task<IActionResult> BulkMemberPrivacy([FromBody] JObject inner)
{
HttpContext.Items.TryGetValue("SystemId", out var systemId);
if (systemId == null)
throw Errors.GenericAuthError;
var data = new JObject();
data.Add("privacy", inner);
2022-04-20 12:20:03 -04:00
var patch = MemberPatch.FromJSON(data);
2022-04-07 03:49:06 -04:00
patch.AssertIsValid();
if (patch.Errors.Count > 0)
throw new ModelParseError(patch.Errors);
await _db.ExecuteQuery(patch.Apply(new Query("members").Where("system", systemId)));
return NoContent();
}
[HttpPost("bulk_privacy/group")]
public async Task<IActionResult> BulkGroupPrivacy([FromBody] JObject inner)
{
HttpContext.Items.TryGetValue("SystemId", out var systemId);
if (systemId == null)
throw Errors.GenericAuthError;
var data = new JObject();
data.Add("privacy", inner);
var patch = GroupPatch.FromJson(data);
patch.AssertIsValid();
if (patch.Errors.Count > 0)
throw new ModelParseError(patch.Errors);
await _db.ExecuteQuery(patch.Apply(new Query("groups").Where("system", systemId)));
return NoContent();
}
2022-04-07 09:26:12 -04:00
[HttpPost("discord/callback")]
public async Task<IActionResult> DiscordLogin([FromBody] JObject data)
{
if (_config.ClientId == null) return NotFound();
using var client = new HttpClient();
var res = await client.PostAsync("https://discord.com/api/v10/oauth2/token", new FormUrlEncodedContent(
new Dictionary<string, string>{
{ "client_id", _config.ClientId },
{ "client_secret", _config.ClientSecret },
{ "grant_type", "authorization_code" },
{ "redirect_uri", data.Value<string>("redirect_domain") + "/login/discord" },
{ "code", data.Value<string>("code") },
}));
var h = await res.Content.ReadAsStringAsync();
var c = JsonConvert.DeserializeObject<OAuth2TokenResponse>(h);
if (c.access_token == null)
return BadRequest(PrivateJsonExt.ObjectWithError(c.error_description));
client.DefaultRequestHeaders.Add("Authorization", $"Bearer {c.access_token}");
var resp = await client.GetAsync("https://discord.com/api/v10/users/@me");
var user = JsonConvert.DeserializeObject<JObject>(await resp.Content.ReadAsStringAsync());
var userId = user.Value<String>("id");
var system = await ResolveSystem(userId);
if (system == null)
return BadRequest(PrivateJsonExt.ObjectWithError("User does not have a system registered!"));
var config = await _repo.GetSystemConfig(system.Id);
2022-04-07 09:26:12 -04:00
// TODO
// resp = await client.GetAsync("https://discord.com/api/v10/users/@me/guilds");
// var guilds = JsonConvert.DeserializeObject<JArray>(await resp.Content.ReadAsStringAsync());
// await _redis.Connection.GetDatabase().HashSetAsync(
// $"user_guilds::{userId}",
// guilds.Select(g => new HashEntry(g.Value<string>("id"), true)).ToArray()
// );
if (system.Token == null)
system = await _repo.UpdateSystem(system.Id, new SystemPatch { Token = StringUtils.GenerateToken() });
2022-04-07 09:26:12 -04:00
var o = new JObject();
2022-04-20 12:20:03 -04:00
o.Add("system", system.ToJson(LookupContext.ByOwner));
o.Add("config", config.ToJson());
2022-04-07 09:26:12 -04:00
o.Add("user", user);
o.Add("token", system.Token);
return Ok(o);
}
}
public record OAuth2TokenResponse
{
public string access_token;
public string? error;
public string? error_description;
2022-01-22 03:52:52 -05:00
}
public static class PrivateJsonExt
{
2022-04-07 09:26:12 -04:00
public static JObject ObjectWithError(string error)
{
var o = new JObject();
o.Add("error", error);
return o;
}
}