Work on permission system, along with initial work on database(postgres) system
This commit is contained in:
parent
3a85a9f18f
commit
25d691c4e9
|
@ -0,0 +1,28 @@
|
||||||
|
using System;
|
||||||
|
using DeukBot4.MessageHandlers.Permissions;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace DeukBot4.Database
|
||||||
|
{
|
||||||
|
public class DatabaseConnection : IDisposable
|
||||||
|
{
|
||||||
|
public static string ConnectionString { private get; set; }
|
||||||
|
private readonly NpgsqlConnection _connection;
|
||||||
|
|
||||||
|
public DatabaseConnection()
|
||||||
|
{
|
||||||
|
_connection = new NpgsqlConnection(ConnectionString);
|
||||||
|
_connection.Open();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
_connection.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator NpgsqlConnection(DatabaseConnection conn)
|
||||||
|
{
|
||||||
|
return conn._connection;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,25 @@
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace DeukBot4.Database
|
||||||
|
{
|
||||||
|
public static class DatabaseInitializer
|
||||||
|
{
|
||||||
|
public static void Initialize()
|
||||||
|
{
|
||||||
|
using (var conn = new DatabaseConnection())
|
||||||
|
{
|
||||||
|
using (var cmd = new NpgsqlCommand())
|
||||||
|
{
|
||||||
|
cmd.Connection = conn;
|
||||||
|
cmd.CommandText = "CREATE TABLE IF NOT EXISTS permission_roles (" +
|
||||||
|
"server_id bigint NOT NULL," +
|
||||||
|
"role_id bigint NOT NULL," +
|
||||||
|
"permission_level smallint NOT NULL," +
|
||||||
|
"PRIMARY KEY(server_id, role_id)" +
|
||||||
|
")";
|
||||||
|
cmd.ExecuteNonQuery();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,51 @@
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using DeukBot4.MessageHandlers.Permissions;
|
||||||
|
using DeukBot4.Utilities;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
namespace DeukBot4.Database
|
||||||
|
{
|
||||||
|
public static class DatabaseRolePermissions
|
||||||
|
{
|
||||||
|
public static async Task SetRolePermission(ulong serverId, ulong roleId, PermissionLevel permissionLevel)
|
||||||
|
{
|
||||||
|
using (var conn = new DatabaseConnection())
|
||||||
|
{
|
||||||
|
using (var cmd = new NpgsqlCommand())
|
||||||
|
{
|
||||||
|
cmd.Connection = conn;
|
||||||
|
cmd.CommandText = "INSERT INTO permission_roles (server_id, role_id, permission_level)" +
|
||||||
|
"VALUES (@server_id, @role_id, @permission_level) " +
|
||||||
|
"ON CONFLICT (server_id, role_id) DO UPDATE SET permission_level = @permission_level";
|
||||||
|
cmd.Parameters.AddWithValue("server_id", serverId.ToLong());
|
||||||
|
cmd.Parameters.AddWithValue("role_id", roleId.ToLong());
|
||||||
|
cmd.Parameters.AddWithValue("permission_level", (sbyte)permissionLevel);
|
||||||
|
await cmd.ExecuteNonQueryAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task<PermissionLevel> GetRolePermission(ulong serverId, ulong roleId)
|
||||||
|
{
|
||||||
|
using (var conn = new DatabaseConnection())
|
||||||
|
{
|
||||||
|
using (var cmd = new NpgsqlCommand())
|
||||||
|
{
|
||||||
|
cmd.Connection = conn;
|
||||||
|
cmd.CommandText =
|
||||||
|
"SELECT permission_level FROM permission_roles WHERE server_id = @server_id AND role_id = @role_id";
|
||||||
|
cmd.Parameters.AddWithValue("server_id", serverId.ToLong());
|
||||||
|
cmd.Parameters.AddWithValue("role_id", roleId.ToLong());
|
||||||
|
|
||||||
|
var reader = cmd.ExecuteReader();
|
||||||
|
while (await reader.ReadAsync())
|
||||||
|
{
|
||||||
|
return (PermissionLevel)(sbyte)reader.GetInt16(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (PermissionLevel)sbyte.MinValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,5 +7,6 @@
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Discord.Net" Version="2.0.0-beta" />
|
<PackageReference Include="Discord.Net" Version="2.0.0-beta" />
|
||||||
<PackageReference Include="Newtonsoft.Json" Version="11.0.2" />
|
<PackageReference Include="Newtonsoft.Json" Version="11.0.2" />
|
||||||
|
<PackageReference Include="Npgsql" Version="4.0.0-preview1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
</Project>
|
|
@ -39,7 +39,7 @@ namespace DeukBot4.MessageHandlers.CommandHandler
|
||||||
public static async Task HandleMessage(SocketMessage message)
|
public static async Task HandleMessage(SocketMessage message)
|
||||||
{
|
{
|
||||||
if (message.Content[0] != CommandTrigger) return;
|
if (message.Content[0] != CommandTrigger) return;
|
||||||
var req = CommandRequest.Create(message);
|
var req = await CommandRequest.Create(message);
|
||||||
var resultCode = req.Item2;
|
var resultCode = req.Item2;
|
||||||
if (resultCode == CommandRequest.RequestCode.Invalid)
|
if (resultCode == CommandRequest.RequestCode.Invalid)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading.Tasks;
|
||||||
using DeukBot4.MessageHandlers.Permissions;
|
using DeukBot4.MessageHandlers.Permissions;
|
||||||
using Discord.WebSocket;
|
using Discord.WebSocket;
|
||||||
|
|
||||||
|
@ -30,7 +31,7 @@ namespace DeukBot4.MessageHandlers.CommandHandler.RequestStructure
|
||||||
OK, Invalid, Forbidden
|
OK, Invalid, Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
public static (CommandRequest, RequestCode) Create(SocketMessage message)
|
public static async Task<(CommandRequest, RequestCode)> Create(SocketMessage message)
|
||||||
{
|
{
|
||||||
var originalMessage = message;
|
var originalMessage = message;
|
||||||
var content = message.Content;
|
var content = message.Content;
|
||||||
|
@ -45,7 +46,16 @@ namespace DeukBot4.MessageHandlers.CommandHandler.RequestStructure
|
||||||
return (null, RequestCode.Invalid);
|
return (null, RequestCode.Invalid);
|
||||||
}
|
}
|
||||||
|
|
||||||
var permission = PermissionValidator.GetUserPermissionLevel(message);
|
PermissionLevel permission;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
permission = await PermissionValidator.GetUserPermissionLevel(message);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
await Logger.LogError(e.Message);
|
||||||
|
return (null, RequestCode.Forbidden);
|
||||||
|
}
|
||||||
if (permission < command.Permission)
|
if (permission < command.Permission)
|
||||||
{
|
{
|
||||||
return (null, RequestCode.Forbidden);
|
return (null, RequestCode.Forbidden);
|
||||||
|
|
|
@ -0,0 +1,100 @@
|
||||||
|
using System;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using DeukBot4.Database;
|
||||||
|
using DeukBot4.MessageHandlers.CommandHandler.RequestStructure;
|
||||||
|
using DeukBot4.MessageHandlers.Permissions;
|
||||||
|
using Discord;
|
||||||
|
|
||||||
|
namespace DeukBot4.MessageHandlers.CommandHandler
|
||||||
|
{
|
||||||
|
public class RolePermissionCommands : CommandContainerBase
|
||||||
|
{
|
||||||
|
public override string Name => "Role Related";
|
||||||
|
|
||||||
|
[Command("roles", PermissionLevel.Admin)]
|
||||||
|
[CommandHelp("Lists all roles on the server along with their IDs", "Lists all roles on the server along with their IDs")]
|
||||||
|
public async Task ListServerRoles(CommandRequest request)
|
||||||
|
{
|
||||||
|
var channel = request.OriginalMessage.Channel;
|
||||||
|
if (!(channel is IGuildChannel serverChannel))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var roles = serverChannel.Guild.Roles;
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
foreach (var role in roles)
|
||||||
|
{
|
||||||
|
sb.Append($"``{role.Name}`` --> {role.Id}\n");
|
||||||
|
}
|
||||||
|
await channel.SendMessageAsync(sb.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Command("adminrole", PermissionLevel.ServerOwner)]
|
||||||
|
[CommandParameters(new []{ParameterMatcher.ParameterType.Number})]
|
||||||
|
public async Task SetAdminRole(CommandRequest request)
|
||||||
|
{
|
||||||
|
await SetRolePermission(request, PermissionLevel.Admin);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Command("moderatorrole", PermissionLevel.ServerOwner)]
|
||||||
|
[CommandParameters(new []{ParameterMatcher.ParameterType.Number})]
|
||||||
|
public async Task SetModRole(CommandRequest request)
|
||||||
|
{
|
||||||
|
await SetRolePermission(request, PermissionLevel.Moderator);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Command("helperrole", PermissionLevel.ServerOwner)]
|
||||||
|
[CommandParameters(new []{ParameterMatcher.ParameterType.Number})]
|
||||||
|
public async Task SetHelperRole(CommandRequest request)
|
||||||
|
{
|
||||||
|
await SetRolePermission(request, PermissionLevel.Helper);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Command("bannedrole", PermissionLevel.ServerOwner)]
|
||||||
|
[CommandParameters(new []{ParameterMatcher.ParameterType.Number})]
|
||||||
|
public async Task SetBannedRole(CommandRequest request)
|
||||||
|
{
|
||||||
|
await SetRolePermission(request, PermissionLevel.Banned);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static async Task SetRolePermission(CommandRequest request, PermissionLevel permissionLevel)
|
||||||
|
{
|
||||||
|
var channel = request.OriginalMessage.Channel;
|
||||||
|
if (!(channel is IGuildChannel serverChannel))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (request.Parameters.Length == 0)
|
||||||
|
{
|
||||||
|
await request.OriginalMessage.Channel.SendMessageAsync(
|
||||||
|
$"You did not give a valid role ID. Use ``!roles`` to list all current server roles, along with their ids");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ulong.TryParse(request.Parameters[0].AsString(), out var roleId))
|
||||||
|
{
|
||||||
|
await request.OriginalMessage.Channel.SendMessageAsync(
|
||||||
|
$"You did not give a valid role ID. Use ``!roles`` to list all current server roles, along with their ids");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var role = serverChannel.Guild.GetRole(roleId);
|
||||||
|
if (role == null)
|
||||||
|
{
|
||||||
|
await request.OriginalMessage.Channel.SendMessageAsync("No role with that id exists on this server");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await DatabaseRolePermissions.SetRolePermission(serverChannel.GuildId, roleId, permissionLevel);
|
||||||
|
PermissionValidator.UpdateCache(serverChannel.GuildId, roleId, permissionLevel);
|
||||||
|
}
|
||||||
|
catch(Exception e)
|
||||||
|
{
|
||||||
|
await Logger.LogError(e.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
using System.Threading.Tasks;
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
using Discord.WebSocket;
|
using Discord.WebSocket;
|
||||||
|
|
||||||
namespace DeukBot4.MessageHandlers
|
namespace DeukBot4.MessageHandlers
|
||||||
|
@ -7,7 +8,14 @@ namespace DeukBot4.MessageHandlers
|
||||||
{
|
{
|
||||||
public static async Task HandleMessage(SocketMessage message)
|
public static async Task HandleMessage(SocketMessage message)
|
||||||
{
|
{
|
||||||
await CommandHandler.CommandHandler.HandleMessage(message);
|
try
|
||||||
|
{
|
||||||
|
await CommandHandler.CommandHandler.HandleMessage(message);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
await Logger.LogError(e.ToString());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -8,6 +8,7 @@
|
||||||
Helper = 20,
|
Helper = 20,
|
||||||
Moderator = 40,
|
Moderator = 40,
|
||||||
Admin = 60,
|
Admin = 60,
|
||||||
Owner = 100
|
ServerOwner = 80,
|
||||||
|
BotCreator = 100
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,27 +1,104 @@
|
||||||
using DeukBot4.MessageHandlers.CommandHandler.RequestStructure;
|
using System;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using DeukBot4.Database;
|
||||||
|
using DeukBot4.MessageHandlers.CommandHandler.RequestStructure;
|
||||||
|
using Discord;
|
||||||
using Discord.WebSocket;
|
using Discord.WebSocket;
|
||||||
|
|
||||||
namespace DeukBot4.MessageHandlers.Permissions
|
namespace DeukBot4.MessageHandlers.Permissions
|
||||||
{
|
{
|
||||||
public static class PermissionValidator
|
public static class PermissionValidator
|
||||||
{
|
{
|
||||||
public static PermissionLevel GetUserPermissionLevel(SocketMessage message)
|
private static readonly ConcurrentDictionary<ulong, ConcurrentDictionary<ulong, PermissionLevel>> PermissionCache =
|
||||||
|
new ConcurrentDictionary<ulong, ConcurrentDictionary<ulong, PermissionLevel>>();
|
||||||
|
|
||||||
|
private static async Task<PermissionLevel> GetDatabasePermissionLevelForRole(ulong serverId, ulong roleId)
|
||||||
|
{
|
||||||
|
if (PermissionCache.TryGetValue(serverId, out var serverPermissions))
|
||||||
|
{
|
||||||
|
if (serverPermissions.TryGetValue(roleId, out var permissions))
|
||||||
|
{
|
||||||
|
return permissions;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
PermissionCache.TryAdd(serverId, new ConcurrentDictionary<ulong, PermissionLevel>());
|
||||||
|
}
|
||||||
|
|
||||||
|
serverPermissions = PermissionCache[serverId];
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var permission = await DatabaseRolePermissions.GetRolePermission(serverId, roleId);
|
||||||
|
serverPermissions.TryAdd(serverId, permission);
|
||||||
|
return permission;
|
||||||
|
}
|
||||||
|
catch(Exception e)
|
||||||
|
{
|
||||||
|
await Logger.LogError(e.ToString());
|
||||||
|
return PermissionLevel.Everyone;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void UpdateCache(ulong serverId, ulong roleId, PermissionLevel level)
|
||||||
|
{
|
||||||
|
if (PermissionCache.TryGetValue(serverId, out var serverPermissions))
|
||||||
|
{
|
||||||
|
if (serverPermissions.ContainsKey(roleId))
|
||||||
|
{
|
||||||
|
serverPermissions[roleId] = level;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
serverPermissions.TryAdd(roleId, level);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<PermissionLevel> GetDatabasePermissionLevel(ulong serverId, ulong[] roles)
|
||||||
|
{
|
||||||
|
var highestRole = (PermissionLevel)sbyte.MinValue;
|
||||||
|
foreach (var role in roles)
|
||||||
|
{
|
||||||
|
var perms = await GetDatabasePermissionLevelForRole(serverId, role);
|
||||||
|
if (perms > highestRole)
|
||||||
|
{
|
||||||
|
highestRole = perms;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (highestRole == (PermissionLevel) sbyte.MinValue)
|
||||||
|
{
|
||||||
|
return PermissionLevel.Everyone;
|
||||||
|
}
|
||||||
|
|
||||||
|
return highestRole;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task<PermissionLevel> GetUserPermissionLevel(SocketMessage message)
|
||||||
{
|
{
|
||||||
if (message.Author.Id == Program.Settings.OwnerId)
|
if (message.Author.Id == Program.Settings.OwnerId)
|
||||||
{
|
{
|
||||||
return PermissionLevel.Owner;
|
return PermissionLevel.BotCreator;
|
||||||
}
|
}
|
||||||
if (message.Author.IsBot)
|
if (message.Author.IsBot)
|
||||||
{
|
{
|
||||||
return PermissionLevel.Bot;
|
return PermissionLevel.Bot;
|
||||||
}
|
}
|
||||||
return PermissionLevel.Everyone;
|
if (!(message.Channel is IGuildChannel serverChannel))
|
||||||
}
|
return PermissionLevel.Everyone;
|
||||||
|
|
||||||
public static bool CanUse(this CommandRequest req)
|
if (serverChannel.Guild.OwnerId == message.Author.Id)
|
||||||
{
|
return PermissionLevel.ServerOwner;
|
||||||
var level = GetUserPermissionLevel(req.OriginalMessage);
|
|
||||||
return level >= req.Command.Permission;
|
if (!(message.Author is IGuildUser user))
|
||||||
|
return PermissionLevel.Everyone;
|
||||||
|
|
||||||
|
var perms = await GetDatabasePermissionLevel(serverChannel.GuildId, user.RoleIds.ToArray());
|
||||||
|
return perms;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,5 +1,6 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using DeukBot4.Database;
|
||||||
using DeukBot4.MessageHandlers;
|
using DeukBot4.MessageHandlers;
|
||||||
using DeukBot4.MessageHandlers.CommandHandler;
|
using DeukBot4.MessageHandlers.CommandHandler;
|
||||||
using Discord;
|
using Discord;
|
||||||
|
@ -21,6 +22,9 @@ namespace DeukBot4
|
||||||
private static async Task MainAsync()
|
private static async Task MainAsync()
|
||||||
{
|
{
|
||||||
Settings = Settings.FromJsonFile("settings.json");
|
Settings = Settings.FromJsonFile("settings.json");
|
||||||
|
DatabaseConnection.ConnectionString = Settings.DatabaseConnectionString;
|
||||||
|
|
||||||
|
DatabaseInitializer.Initialize();
|
||||||
CommandHandler.Build();
|
CommandHandler.Build();
|
||||||
Client = new DiscordSocketClient();
|
Client = new DiscordSocketClient();
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,8 @@ namespace DeukBot4
|
||||||
public string Username { get; private set; }
|
public string Username { get; private set; }
|
||||||
[JsonProperty]
|
[JsonProperty]
|
||||||
public ulong OwnerId { get; private set; }
|
public ulong OwnerId { get; private set; }
|
||||||
|
[JsonProperty]
|
||||||
|
public string DatabaseConnectionString { get; private set; }
|
||||||
|
|
||||||
|
|
||||||
public static Settings FromJsonFile(string filepath)
|
public static Settings FromJsonFile(string filepath)
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
namespace DeukBot4.Utilities
|
||||||
|
{
|
||||||
|
public static class LongExtensions
|
||||||
|
{
|
||||||
|
public static ulong ToUlong(this long l)
|
||||||
|
{
|
||||||
|
return unchecked((ulong)(l - long.MinValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static long ToLong(this ulong l)
|
||||||
|
{
|
||||||
|
return unchecked((long)l + long.MinValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue