PluralKit/PluralKit.Bot/Commands/Api.cs

146 lines
6.1 KiB
C#
Raw Normal View History

using System.Text.RegularExpressions;
2019-06-20 19:15:57 +00:00
using System.Threading.Tasks;
2020-12-25 12:19:35 +00:00
using Myriad.Extensions;
using Myriad.Rest.Exceptions;
2020-12-25 12:19:35 +00:00
using Myriad.Rest.Types.Requests;
using Myriad.Types;
2019-10-05 05:41:00 +00:00
using PluralKit.Core;
2019-06-20 19:15:57 +00:00
namespace PluralKit.Bot
2019-06-20 19:15:57 +00:00
{
public class Api
2019-06-20 19:15:57 +00:00
{
2020-08-29 11:46:27 +00:00
private readonly ModelRepository _repo;
2021-11-03 06:01:35 +00:00
private readonly DispatchService _dispatch;
private static readonly Regex _webhookRegex = new("https://(?:\\w+.)?discord(?:app)?.com/api(?:/v.*)?/webhooks/(.*)");
2021-11-03 06:01:35 +00:00
public Api(ModelRepository repo, DispatchService dispatch)
2019-10-05 05:41:00 +00:00
{
2020-08-29 11:46:27 +00:00
_repo = repo;
2021-11-03 06:01:35 +00:00
_dispatch = dispatch;
2019-10-05 05:41:00 +00:00
}
public async Task GetToken(Context ctx)
2019-06-20 19:15:57 +00:00
{
2019-10-05 05:41:00 +00:00
ctx.CheckSystem();
2019-06-20 19:15:57 +00:00
// Get or make a token
2019-10-05 05:41:00 +00:00
var token = ctx.System.Token ?? await MakeAndSetNewToken(ctx.System);
2021-08-27 15:03:47 +00:00
try
2019-06-20 19:15:57 +00:00
{
// DM the user a security disclaimer, and then the token in a separate message (for easy copying on mobile)
2021-01-31 15:16:52 +00:00
var dm = await ctx.Cache.GetOrCreateDmChannel(ctx.Rest, ctx.Author.Id);
await ctx.Rest.CreateMessage(dm.Id, new MessageRequest
2020-12-25 12:19:35 +00:00
{
Content = $"{Emojis.Warn} Please note that this grants access to modify (and delete!) all your system data, so keep it safe and secure. If it leaks or you need a new one, you can invalidate this one with `pk;token refresh`.\n\nYour token is below:"
});
2021-08-27 15:03:47 +00:00
await ctx.Rest.CreateMessage(dm.Id, new MessageRequest { Content = token });
2019-06-20 19:15:57 +00:00
// If we're not already in a DM, reply with a reminder to check
2021-01-31 15:16:52 +00:00
if (ctx.Channel.Type != Channel.ChannelType.Dm)
2020-12-25 12:19:35 +00:00
await ctx.Reply($"{Emojis.Success} Check your DMs!");
}
2021-03-18 10:38:28 +00:00
catch (ForbiddenException)
{
// Can't check for permission errors beforehand, so have to handle here :/
2021-01-31 15:16:52 +00:00
if (ctx.Channel.Type != Channel.ChannelType.Dm)
2020-12-25 12:19:35 +00:00
await ctx.Reply($"{Emojis.Error} Could not send token in DMs. Are your DMs closed?");
}
2019-06-20 19:15:57 +00:00
}
2019-10-05 05:41:00 +00:00
private async Task<string> MakeAndSetNewToken(PKSystem system)
2019-06-20 19:15:57 +00:00
{
system = await _repo.UpdateSystem(system.Id, new() { Token = StringUtils.GenerateToken() });
2019-10-05 05:41:00 +00:00
return system.Token;
2019-06-20 19:15:57 +00:00
}
2021-08-27 15:03:47 +00:00
2019-10-05 05:41:00 +00:00
public async Task RefreshToken(Context ctx)
2019-06-20 19:15:57 +00:00
{
2019-10-05 05:41:00 +00:00
ctx.CheckSystem();
2021-08-27 15:03:47 +00:00
2019-10-05 05:41:00 +00:00
if (ctx.System.Token == null)
2019-06-20 19:15:57 +00:00
{
// If we don't have a token, call the other method instead
// This does pretty much the same thing, except words the messages more appropriately for that :)
2019-10-05 05:41:00 +00:00
await GetToken(ctx);
2019-06-20 19:15:57 +00:00
return;
}
2021-08-27 15:03:47 +00:00
try
{
// DM the user an invalidation disclaimer, and then the token in a separate message (for easy copying on mobile)
2021-01-31 15:16:52 +00:00
var dm = await ctx.Cache.GetOrCreateDmChannel(ctx.Rest, ctx.Author.Id);
await ctx.Rest.CreateMessage(dm.Id, new MessageRequest
2020-12-25 12:19:35 +00:00
{
Content = $"{Emojis.Warn} Your previous API token has been invalidated. You will need to change it anywhere it's currently used.\n\nYour token is below:"
});
2021-08-27 15:03:47 +00:00
// Make the new token after sending the first DM; this ensures if we can't DM, we also don't end up
// breaking their existing token as a side effect :)
var token = await MakeAndSetNewToken(ctx.System);
2021-01-31 15:16:52 +00:00
await ctx.Rest.CreateMessage(dm.Id, new MessageRequest { Content = token });
2021-08-27 15:03:47 +00:00
// If we're not already in a DM, reply with a reminder to check
2021-01-31 15:16:52 +00:00
if (ctx.Channel.Type != Channel.ChannelType.Dm)
2020-12-25 12:19:35 +00:00
await ctx.Reply($"{Emojis.Success} Check your DMs!");
}
2021-03-18 10:38:28 +00:00
catch (ForbiddenException)
2019-06-20 19:15:57 +00:00
{
// Can't check for permission errors beforehand, so have to handle here :/
2021-01-31 15:16:52 +00:00
if (ctx.Channel.Type != Channel.ChannelType.Dm)
2020-12-25 12:19:35 +00:00
await ctx.Reply($"{Emojis.Error} Could not send token in DMs. Are your DMs closed?");
2019-06-20 19:15:57 +00:00
}
}
2021-11-03 06:01:35 +00:00
public async Task SystemWebhook(Context ctx)
{
ctx.CheckSystem().CheckDMContext();
2021-11-03 06:01:35 +00:00
if (!ctx.HasNext(false))
{
if (ctx.System.WebhookUrl == null)
await ctx.Reply("Your system does not have a webhook URL set. Set one with `pk;system webhook <url>`!");
else
await ctx.Reply($"Your system's webhook URL is <{ctx.System.WebhookUrl}>.");
return;
}
if (await ctx.MatchClear("your system's webhook URL"))
{
await _repo.UpdateSystem(ctx.System.Id, new()
{
WebhookUrl = null,
WebhookToken = null,
2021-11-03 06:01:35 +00:00
});
await ctx.Reply($"{Emojis.Success} System webhook URL removed.");
return;
}
var newUrl = ctx.RemainderOrNull();
if (!await DispatchExt.ValidateUri(newUrl))
throw new PKError($"The URL {newUrl.AsCode()} is invalid or I cannot access it. Are you sure this is a valid, publicly accessible URL?");
2021-11-03 06:01:35 +00:00
if (_webhookRegex.IsMatch(newUrl))
throw new PKError("PluralKit does not currently support setting a Discord webhook URL as your system's webhook URL.");
2021-11-03 06:01:35 +00:00
var newToken = StringUtils.GenerateToken();
await _repo.UpdateSystem(ctx.System.Id, new()
{
WebhookUrl = newUrl,
WebhookToken = newToken,
2021-11-03 06:01:35 +00:00
});
await ctx.Reply($"{Emojis.Success} Successfully the new webhook URL for your system."
+ $"\n\n{Emojis.Warn} The following token is used to authenticate requests from PluralKit to you."
+ " If it leaks, you should clear and re-set the webhook URL to get a new token."
+ "\ntodo: add link to docs or something"
);
await ctx.Reply(newToken);
}
2019-06-20 19:15:57 +00:00
}
}