Generate token with random prefix
Added getitem function for specific item - that decrypt is not invoked for previous run of the server - and that server restart always triggers a client update
This commit is contained in:
parent
0b9cb03bae
commit
cd006fb268
@ -54,6 +54,8 @@ local function getResponse(parsed)
|
|||||||
return textutils.serializeJSON(getPeripheral("rsBridge").listFluids())
|
return textutils.serializeJSON(getPeripheral("rsBridge").listFluids())
|
||||||
elseif parsed.method == "craft" then
|
elseif parsed.method == "craft" then
|
||||||
return tostring(getPeripheral("rsBridge").craftItem(parsed.params))
|
return tostring(getPeripheral("rsBridge").craftItem(parsed.params))
|
||||||
|
elseif parsed.method == "getitem" then
|
||||||
|
return textutils.serializeJSON(getPeripheral("rsBridge").getItem(parsed.params))
|
||||||
end
|
end
|
||||||
|
|
||||||
error("No message handler for method: "..parsed.method.."!")
|
error("No message handler for method: "..parsed.method.."!")
|
||||||
|
@ -4,7 +4,6 @@ using Fleck;
|
|||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Runtime.Serialization;
|
using System.Runtime.Serialization;
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace MinecraftDiscordBot;
|
namespace MinecraftDiscordBot;
|
||||||
|
|
||||||
@ -18,124 +17,6 @@ public sealed class CommandHandlerAttribute : Attribute {
|
|||||||
public string? HelpText { get; init; }
|
public string? HelpText { get; init; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class RefinedStorageService : CommandRouter {
|
|
||||||
private readonly ITaskWaitSource _taskSource;
|
|
||||||
public override string HelpTextPrefix => "!rs ";
|
|
||||||
public RefinedStorageService(ITaskWaitSource taskSource) : base() => _taskSource = taskSource;
|
|
||||||
public override Task<ResponseType> FallbackHandler(SocketUserMessage message, string method, string[] parameters, CancellationToken ct)
|
|
||||||
=> Task.FromResult(ResponseType.AsString($"The RS system has no command '{method}'!"));
|
|
||||||
public override Task<ResponseType> RootAnswer(SocketUserMessage message, CancellationToken ct)
|
|
||||||
=> Task.FromResult(ResponseType.AsString("The RS system is online!"));
|
|
||||||
|
|
||||||
private async Task<T> Method<T>(string methodName, Func<string, T> parser, CancellationToken ct, Dictionary<string, object>? parameters = null) {
|
|
||||||
var waiter = _taskSource.GetWaiter(parser, ct);
|
|
||||||
await _taskSource.Send(new RequestMessage(waiter.ID, methodName, parameters));
|
|
||||||
return await waiter.Task;
|
|
||||||
}
|
|
||||||
|
|
||||||
private const string CmdEnergyUsage = "energyusage";
|
|
||||||
private const string CmdEnergyStorage = "energystorage";
|
|
||||||
private const string CmdListItems = "listitems";
|
|
||||||
private const string CmdItemName = "itemname";
|
|
||||||
private const string CmdListFluids = "listfluids";
|
|
||||||
private const string CmdCraftItem = "craft";
|
|
||||||
|
|
||||||
public async Task<int> GetEnergyUsageAsync(CancellationToken ct) => await Method(CmdEnergyUsage, int.Parse, ct);
|
|
||||||
public async Task<int> GetEnergyStorageAsync(CancellationToken ct) => await Method(CmdEnergyStorage, int.Parse, ct);
|
|
||||||
public async Task<IEnumerable<Item>> ListItemsAsync(CancellationToken ct) => await Method(CmdListItems, ConnectedComputer.Deserialize<IEnumerable<Item>>(), ct);
|
|
||||||
public async Task<IEnumerable<Fluid>> ListFluidsAsync(CancellationToken ct) => await Method(CmdListFluids, ConnectedComputer.Deserialize<IEnumerable<Fluid>>(), ct);
|
|
||||||
public async Task<bool> CraftItem(string itemid, int amount, CancellationToken ct) => await Method(CmdCraftItem, ConnectedComputer.Deserialize<bool>(), ct, new() {
|
|
||||||
["name"] = itemid,
|
|
||||||
["count"] = amount
|
|
||||||
});
|
|
||||||
|
|
||||||
private Task<IEnumerable<Item>> FilterItems(SocketUserMessage message, IEnumerable<string> filters, CancellationToken ct)
|
|
||||||
=> FilterItems(message, filters.Select(ItemFilter.Parse), ct);
|
|
||||||
|
|
||||||
private async Task<IEnumerable<Item>> FilterItems(SocketUserMessage message, IEnumerable<ItemFilter> filters, CancellationToken ct) {
|
|
||||||
var items = Items?.ToList().AsEnumerable();
|
|
||||||
if (items is null) items = (await RefreshItemList(ct)).ToList();
|
|
||||||
foreach (var filter in filters)
|
|
||||||
items = items.Where(filter.MatchItem);
|
|
||||||
return items.ToList();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<List<Item>> RefreshItemList(CancellationToken ct) {
|
|
||||||
var response = await ListItemsAsync(ct);
|
|
||||||
lock (_itemLock) {
|
|
||||||
Items = response.OrderByDescending(i => i.Amount).ToList();
|
|
||||||
return Items;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private List<Item>? Items;
|
|
||||||
private readonly object _itemLock = new();
|
|
||||||
|
|
||||||
[CommandHandler(CmdEnergyStorage, HelpText = "Get the amount of energy stored in the RS system.")]
|
|
||||||
public async Task<ResponseType> HandleEnergyStorage(SocketUserMessage message, string[] parameters, CancellationToken ct)
|
|
||||||
=> ResponseType.AsString($"Refined Storage system stores {await GetEnergyStorageAsync(ct)} RF/t");
|
|
||||||
[CommandHandler(CmdEnergyUsage, HelpText = "Get the amount of energy used by the RS system.")]
|
|
||||||
public async Task<ResponseType> HandleEnergyUsage(SocketUserMessage message, string[] parameters, CancellationToken ct)
|
|
||||||
=> ResponseType.AsString($"Refined Storage system currently uses {await GetEnergyUsageAsync(ct)} RF/t");
|
|
||||||
[CommandHandler(CmdCraftItem, HelpText = "Craft a specific item given an item ID and optionally an amount.")]
|
|
||||||
public async Task<ResponseType> HandleCraftItem(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
|
||||||
var amount = 1;
|
|
||||||
string itemid;
|
|
||||||
if (parameters.Length is 1 or 2) {
|
|
||||||
itemid = parameters[0];
|
|
||||||
if (parameters.Length is 2)
|
|
||||||
if (int.TryParse(parameters[1], out var value)) amount = value;
|
|
||||||
else return ResponseType.AsString($"I expected an amount to craft, not '{parameters[1]}'!");
|
|
||||||
} else return parameters.Length is < 1
|
|
||||||
? ResponseType.AsString("You have to give me at least an item name!")
|
|
||||||
: parameters.Length is > 2
|
|
||||||
? ResponseType.AsString("Yo, those are way too many arguments! I want only item name and maybe an amount!")
|
|
||||||
: throw new InvalidOperationException($"Forgot to match parameter length {parameters.Length}!");
|
|
||||||
return await CraftItem(itemid, amount, ct)
|
|
||||||
? ResponseType.AsString($"Alright, I'm starting to craft {amount} {itemid}.")
|
|
||||||
: ResponseType.AsString($"Nope, that somehow doesn't work!");
|
|
||||||
}
|
|
||||||
[CommandHandler(CmdItemName, HelpText = "Filter items by name.")]
|
|
||||||
public async Task<ResponseType> HandleItemName(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
|
||||||
if (parameters.Length < 2) return ResponseType.AsString($"Usage: {CmdItemName} filters...");
|
|
||||||
else {
|
|
||||||
var items = await FilterItems(message, parameters[1..], ct);
|
|
||||||
var sb = new StringBuilder();
|
|
||||||
sb.AppendLine("Did you mean:");
|
|
||||||
sb.AppendJoin("\n", items.Select(i => i.ToString()));
|
|
||||||
return ResponseType.AsString(sb.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandHandler(CmdListFluids, HelpText = "Gets a list of fluids that are currently stored in the RS system.")]
|
|
||||||
public async Task<ResponseType> HandleFluidListing(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
|
||||||
var sb = new StringBuilder();
|
|
||||||
sb.Append("The Refined Storage system stores those fluids:");
|
|
||||||
var fluids = await ListFluidsAsync(ct);
|
|
||||||
foreach (var fluid in fluids.OrderByDescending(i => i.Amount))
|
|
||||||
if (fluid.Amount > 10000) sb.AppendFormat("\n{0:n2} B of {1}", fluid.Amount / 1000.0f, fluid.DisplayName);
|
|
||||||
else sb.AppendFormat("\n{0:n0} mB of {1}", fluid.Amount, fluid.DisplayName);
|
|
||||||
return ResponseType.AsString(sb.ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandHandler(CmdListItems, HelpText = "Gets a list of items that are currently stored in the RS system.")]
|
|
||||||
public async Task<ResponseType> HandleItemListing(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
|
||||||
var sb = new StringBuilder();
|
|
||||||
sb.Append("The Refined Storage system currently stores these items:");
|
|
||||||
var items = await RefreshItemList(ct);
|
|
||||||
lock (_itemLock) {
|
|
||||||
int taken = 0;
|
|
||||||
foreach (var item in items) {
|
|
||||||
if (sb.Length > 500) break;
|
|
||||||
sb.AppendFormat("\n{0:n0}x {1}", item.Amount, item.DisplayName);
|
|
||||||
taken++;
|
|
||||||
}
|
|
||||||
if (items.Count > taken) sb.AppendFormat("\nand {0} more items.", items.Skip(taken).Sum(i => i.Amount));
|
|
||||||
}
|
|
||||||
return ResponseType.AsString(sb.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class ConnectedComputer : CommandRouter, ITaskWaitSource {
|
public class ConnectedComputer : CommandRouter, ITaskWaitSource {
|
||||||
protected readonly IWebSocketConnection _socket;
|
protected readonly IWebSocketConnection _socket;
|
||||||
public override string HelpTextPrefix => "!";
|
public override string HelpTextPrefix => "!";
|
||||||
|
@ -25,7 +25,6 @@
|
|||||||
<PackageReference Include="Fleck" Version="1.2.0" />
|
<PackageReference Include="Fleck" Version="1.2.0" />
|
||||||
<PackageReference Include="Microsoft.VisualStudio.Azure.Containers.Tools.Targets" Version="1.14.0" />
|
<PackageReference Include="Microsoft.VisualStudio.Azure.Containers.Tools.Targets" Version="1.14.0" />
|
||||||
<PackageReference Include="Newtonsoft.Json" Version="13.0.1" />
|
<PackageReference Include="Newtonsoft.Json" Version="13.0.1" />
|
||||||
<PackageReference Include="OneOf" Version="3.0.205" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
@ -4,8 +4,6 @@ using Discord.Commands;
|
|||||||
using Discord.Rest;
|
using Discord.Rest;
|
||||||
using Discord.WebSocket;
|
using Discord.WebSocket;
|
||||||
using Fleck;
|
using Fleck;
|
||||||
using Newtonsoft.Json;
|
|
||||||
using OneOf;
|
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
@ -31,7 +29,8 @@ public class Program : IDisposable, ICommandHandler<ResponseType> {
|
|||||||
private bool disposedValue;
|
private bool disposedValue;
|
||||||
public static bool OnlineNotifications => false;
|
public static bool OnlineNotifications => false;
|
||||||
public static readonly string ClientScript = GetClientScript();
|
public static readonly string ClientScript = GetClientScript();
|
||||||
private readonly ITokenProvider _tokenProvider = new TimeoutTokenProvider(10);
|
private readonly ITokenProvider _tokenProvider = new TimeoutTokenProvider(InstanceId, 10);
|
||||||
|
private static readonly int InstanceId = new Random().Next();
|
||||||
|
|
||||||
private string GetVerifiedClientScript() => ClientScript.Replace("$TOKEN", _tokenProvider.GenerateToken());
|
private string GetVerifiedClientScript() => ClientScript.Replace("$TOKEN", _tokenProvider.GenerateToken());
|
||||||
|
|
||||||
@ -86,11 +85,11 @@ public class Program : IDisposable, ICommandHandler<ResponseType> {
|
|||||||
private static Task<int> RunWithConfig(IBotConfigurator arg) => new Program(arg.Config).RunAsync();
|
private static Task<int> RunWithConfig(IBotConfigurator arg) => new Program(arg.Config).RunAsync();
|
||||||
|
|
||||||
public async Task<int> RunAsync() {
|
public async Task<int> RunAsync() {
|
||||||
|
StartWebSocketServer();
|
||||||
await _client.LoginAsync(TokenType.Bot, _config.Token);
|
await _client.LoginAsync(TokenType.Bot, _config.Token);
|
||||||
await _client.StartAsync();
|
await _client.StartAsync();
|
||||||
if (!await HasValidChannels())
|
if (!await HasValidChannels())
|
||||||
return 1;
|
return 1;
|
||||||
StartWebSocketServer();
|
|
||||||
|
|
||||||
// Block this task until the program is closed.
|
// Block this task until the program is closed.
|
||||||
await Task.Delay(-1);
|
await Task.Delay(-1);
|
||||||
|
143
MinecraftDiscordBot/RefinedStorageService.cs
Normal file
143
MinecraftDiscordBot/RefinedStorageService.cs
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
using Discord.WebSocket;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
namespace MinecraftDiscordBot;
|
||||||
|
|
||||||
|
public class RefinedStorageService : CommandRouter {
|
||||||
|
private readonly ITaskWaitSource _taskSource;
|
||||||
|
public override string HelpTextPrefix => "!rs ";
|
||||||
|
public RefinedStorageService(ITaskWaitSource taskSource) : base() => _taskSource = taskSource;
|
||||||
|
public override Task<ResponseType> FallbackHandler(SocketUserMessage message, string method, string[] parameters, CancellationToken ct)
|
||||||
|
=> Task.FromResult(ResponseType.AsString($"The RS system has no command '{method}'!"));
|
||||||
|
public override Task<ResponseType> RootAnswer(SocketUserMessage message, CancellationToken ct)
|
||||||
|
=> Task.FromResult(ResponseType.AsString("The RS system is online!"));
|
||||||
|
|
||||||
|
private async Task<T> Method<T>(string methodName, Func<string, T> parser, CancellationToken ct, Dictionary<string, object>? parameters = null) {
|
||||||
|
var waiter = _taskSource.GetWaiter(parser, ct);
|
||||||
|
await _taskSource.Send(new RequestMessage(waiter.ID, methodName, parameters));
|
||||||
|
return await waiter.Task;
|
||||||
|
}
|
||||||
|
|
||||||
|
private const string CmdEnergyUsage = "energyusage";
|
||||||
|
private const string CmdEnergyStorage = "energystorage";
|
||||||
|
private const string CmdListItems = "listitems";
|
||||||
|
private const string CmdItemName = "itemname";
|
||||||
|
private const string CmdListFluids = "listfluids";
|
||||||
|
private const string CmdCraftItem = "craft";
|
||||||
|
private const string CmdGetItem = "getitem";
|
||||||
|
|
||||||
|
public async Task<int> GetEnergyUsageAsync(CancellationToken ct) => await Method(CmdEnergyUsage, int.Parse, ct);
|
||||||
|
public async Task<int> GetEnergyStorageAsync(CancellationToken ct) => await Method(CmdEnergyStorage, int.Parse, ct);
|
||||||
|
public async Task<IEnumerable<Item>> ListItemsAsync(CancellationToken ct) => await Method(CmdListItems, ConnectedComputer.Deserialize<IEnumerable<Item>>(), ct);
|
||||||
|
public async Task<IEnumerable<Fluid>> ListFluidsAsync(CancellationToken ct) => await Method(CmdListFluids, ConnectedComputer.Deserialize<IEnumerable<Fluid>>(), ct);
|
||||||
|
public async Task<Item> GetItemData(string itemid, CancellationToken ct) => await Method(CmdGetItem, ConnectedComputer.Deserialize<Item>(), ct, new() {
|
||||||
|
["name"] = itemid
|
||||||
|
});
|
||||||
|
public async Task<bool> CraftItem(string itemid, int amount, CancellationToken ct) => await Method(CmdCraftItem, ConnectedComputer.Deserialize<bool>(), ct, new() {
|
||||||
|
["name"] = itemid,
|
||||||
|
["count"] = amount
|
||||||
|
});
|
||||||
|
|
||||||
|
private Task<IEnumerable<Item>> FilterItems(SocketUserMessage message, IEnumerable<string> filters, CancellationToken ct)
|
||||||
|
=> FilterItems(message, filters.Select(ItemFilter.Parse), ct);
|
||||||
|
|
||||||
|
private async Task<IEnumerable<Item>> FilterItems(SocketUserMessage message, IEnumerable<ItemFilter> filters, CancellationToken ct) {
|
||||||
|
var items = Items?.ToList().AsEnumerable();
|
||||||
|
if (items is null) items = (await RefreshItemList(ct)).ToList();
|
||||||
|
foreach (var filter in filters)
|
||||||
|
items = items.Where(filter.MatchItem);
|
||||||
|
return items.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<Item>> RefreshItemList(CancellationToken ct) {
|
||||||
|
var response = await ListItemsAsync(ct);
|
||||||
|
lock (_itemLock) {
|
||||||
|
Items = response.OrderByDescending(i => i.Amount).ToList();
|
||||||
|
return Items;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<Item>? Items;
|
||||||
|
private readonly object _itemLock = new();
|
||||||
|
|
||||||
|
[CommandHandler(CmdEnergyStorage, HelpText = "Get the amount of energy stored in the RS system.")]
|
||||||
|
public async Task<ResponseType> HandleEnergyStorage(SocketUserMessage message, string[] parameters, CancellationToken ct)
|
||||||
|
=> ResponseType.AsString($"Refined Storage system stores {await GetEnergyStorageAsync(ct)} RF/t");
|
||||||
|
[CommandHandler(CmdEnergyUsage, HelpText = "Get the amount of energy used by the RS system.")]
|
||||||
|
public async Task<ResponseType> HandleEnergyUsage(SocketUserMessage message, string[] parameters, CancellationToken ct)
|
||||||
|
=> ResponseType.AsString($"Refined Storage system currently uses {await GetEnergyUsageAsync(ct)} RF/t");
|
||||||
|
[CommandHandler(CmdCraftItem, HelpText = "Craft a specific item given an item ID and optionally an amount.")]
|
||||||
|
public async Task<ResponseType> HandleCraftItem(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
||||||
|
var amount = 1;
|
||||||
|
string itemid;
|
||||||
|
if (parameters.Length is 1 or 2) {
|
||||||
|
itemid = parameters[0];
|
||||||
|
if (parameters.Length is 2)
|
||||||
|
if (int.TryParse(parameters[1], out var value)) amount = value;
|
||||||
|
else return ResponseType.AsString($"I expected an amount to craft, not '{parameters[1]}'!");
|
||||||
|
} else return parameters.Length is < 1
|
||||||
|
? ResponseType.AsString("You have to give me at least an item name!")
|
||||||
|
: parameters.Length is > 2
|
||||||
|
? ResponseType.AsString("Yo, those are way too many arguments! I want only item name and maybe an amount!")
|
||||||
|
: throw new InvalidOperationException($"Forgot to match parameter length {parameters.Length}!");
|
||||||
|
return await CraftItem(itemid, amount, ct)
|
||||||
|
? ResponseType.AsString($"Alright, I'm starting to craft {amount} {itemid}.")
|
||||||
|
: ResponseType.AsString($"Nope, that somehow doesn't work!");
|
||||||
|
}
|
||||||
|
[CommandHandler(CmdGetItem, HelpText = "Get information about a specific item.")]
|
||||||
|
public async Task<ResponseType> HandleGetItemData(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
||||||
|
var amount = 1;
|
||||||
|
string itemid;
|
||||||
|
if (parameters.Length is 1 or 2) {
|
||||||
|
itemid = parameters[0];
|
||||||
|
if (parameters.Length is 2)
|
||||||
|
if (int.TryParse(parameters[1], out var value)) amount = value;
|
||||||
|
else return ResponseType.AsString($"I expected an amount to craft, not '{parameters[1]}'!");
|
||||||
|
} else return parameters.Length is < 1
|
||||||
|
? ResponseType.AsString("You have to give me at least an item name!")
|
||||||
|
: parameters.Length is > 2
|
||||||
|
? ResponseType.AsString("Yo, those are way too many arguments! I want only item name and maybe an amount!")
|
||||||
|
: throw new InvalidOperationException($"Forgot to match parameter length {parameters.Length}!");
|
||||||
|
var data = await GetItemData(itemid, ct);
|
||||||
|
return ResponseType.AsString(data.ToString());
|
||||||
|
}
|
||||||
|
[CommandHandler(CmdItemName, HelpText = "Filter items by name.")]
|
||||||
|
public async Task<ResponseType> HandleItemName(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
||||||
|
if (parameters.Length < 2) return ResponseType.AsString($"Usage: {CmdItemName} filters...");
|
||||||
|
else {
|
||||||
|
var items = await FilterItems(message, parameters[1..], ct);
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
sb.AppendLine("Did you mean:");
|
||||||
|
sb.AppendJoin("\n", items.Select(i => i.ToString()));
|
||||||
|
return ResponseType.AsString(sb.ToString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[CommandHandler(CmdListFluids, HelpText = "Gets a list of fluids that are currently stored in the RS system.")]
|
||||||
|
public async Task<ResponseType> HandleFluidListing(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
sb.Append("The Refined Storage system stores those fluids:");
|
||||||
|
var fluids = await ListFluidsAsync(ct);
|
||||||
|
foreach (var fluid in fluids.OrderByDescending(i => i.Amount))
|
||||||
|
if (fluid.Amount > 10000) sb.AppendFormat("\n{0:n2} B of {1}", fluid.Amount / 1000.0f, fluid.DisplayName);
|
||||||
|
else sb.AppendFormat("\n{0:n0} mB of {1}", fluid.Amount, fluid.DisplayName);
|
||||||
|
return ResponseType.AsString(sb.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
[CommandHandler(CmdListItems, HelpText = "Gets a list of items that are currently stored in the RS system.")]
|
||||||
|
public async Task<ResponseType> HandleItemListing(SocketUserMessage message, string[] parameters, CancellationToken ct) {
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
sb.Append("The Refined Storage system currently stores these items:");
|
||||||
|
var items = await RefreshItemList(ct);
|
||||||
|
lock (_itemLock) {
|
||||||
|
int taken = 0;
|
||||||
|
foreach (var item in items) {
|
||||||
|
if (sb.Length > 500) break;
|
||||||
|
sb.AppendFormat("\n{0:n0}x {1}", item.Amount, item.DisplayName);
|
||||||
|
taken++;
|
||||||
|
}
|
||||||
|
if (items.Count > taken) sb.AppendFormat("\nand {0} more items.", items.Skip(taken).Sum(i => i.Amount));
|
||||||
|
}
|
||||||
|
return ResponseType.AsString(sb.ToString());
|
||||||
|
}
|
||||||
|
}
|
@ -1,13 +1,17 @@
|
|||||||
namespace MinecraftDiscordBot;
|
namespace MinecraftDiscordBot;
|
||||||
|
|
||||||
public class TimeoutTokenProvider : ITokenProvider {
|
public class TimeoutTokenProvider : ITokenProvider {
|
||||||
public TimeoutTokenProvider(int timeoutSeconds, ICipher? cipher = null) {
|
public TimeoutTokenProvider(int instanceId, int timeoutSeconds, ICipher? cipher = null) {
|
||||||
|
InstancePrefix = Convert.ToHexString(BitConverter.GetBytes(instanceId));
|
||||||
_timeout = timeoutSeconds;
|
_timeout = timeoutSeconds;
|
||||||
_cipher = cipher ?? new AesCipher();
|
_cipher = cipher ?? new AesCipher();
|
||||||
}
|
}
|
||||||
private readonly ICipher _cipher;
|
private readonly ICipher _cipher;
|
||||||
private readonly int _timeout;
|
private readonly int _timeout;
|
||||||
|
public string InstancePrefix { get; }
|
||||||
public bool VerifyToken(string token) {
|
public bool VerifyToken(string token) {
|
||||||
|
if (!token.StartsWith(InstancePrefix)) return false;
|
||||||
|
token = token[InstancePrefix.Length..];
|
||||||
byte[] data;
|
byte[] data;
|
||||||
try {
|
try {
|
||||||
data = _cipher.Decrypt(Convert.FromHexString(token));
|
data = _cipher.Decrypt(Convert.FromHexString(token));
|
||||||
@ -21,7 +25,7 @@ public class TimeoutTokenProvider : ITokenProvider {
|
|||||||
public string GenerateToken() {
|
public string GenerateToken() {
|
||||||
var time = BitConverter.GetBytes(DateTime.UtcNow.ToBinary());
|
var time = BitConverter.GetBytes(DateTime.UtcNow.ToBinary());
|
||||||
var key = Guid.NewGuid().ToByteArray();
|
var key = Guid.NewGuid().ToByteArray();
|
||||||
var token = Convert.ToHexString(_cipher.Encrypt(time.Concat(key).ToArray()));
|
var token = InstancePrefix + Convert.ToHexString(_cipher.Encrypt(time.Concat(key).ToArray()));
|
||||||
return token;
|
return token;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user