Added IP banning support
This commit is contained in:
parent
ecc4b6d999
commit
7ae8d6908a
@ -1,8 +1,7 @@
|
|||||||
package com.redstoner.bungeeBans;
|
package com.redstoner.bungeeBans;
|
||||||
|
|
||||||
import com.redstoner.bungeeBans.commands.BanCommand;
|
import com.redstoner.bungeeBans.commands.*;
|
||||||
import com.redstoner.bungeeBans.commands.GetBanCommand;
|
import com.redstoner.bungeeBans.json.IPBan;
|
||||||
import com.redstoner.bungeeBans.commands.UnbanCommand;
|
|
||||||
import com.redstoner.bungeeBans.json.PlayerBan;
|
import com.redstoner.bungeeBans.json.PlayerBan;
|
||||||
import com.redstoner.bungeeBans.listeners.BanJoinListener;
|
import com.redstoner.bungeeBans.listeners.BanJoinListener;
|
||||||
import com.redstoner.bungeeBans.listeners.DisableJoinListener;
|
import com.redstoner.bungeeBans.listeners.DisableJoinListener;
|
||||||
@ -16,15 +15,18 @@ import java.nio.file.NoSuchFileException;
|
|||||||
|
|
||||||
public class Main extends Plugin {
|
public class Main extends Plugin {
|
||||||
private File playerBanFile = new File("banned-players.json");
|
private File playerBanFile = new File("banned-players.json");
|
||||||
|
private File ipBanFile = new File("banned-ips.json");
|
||||||
|
|
||||||
private BanManager<PlayerBan> playerBanManager = new BanManager<>(playerBanFile, PlayerBan.class);
|
private BanManager<PlayerBan> playerBanManager = new BanManager<>(playerBanFile, PlayerBan.class);
|
||||||
|
private BanManager<IPBan> ipBanManager = new BanManager<>(ipBanFile, IPBan.class);
|
||||||
|
|
||||||
private boolean shouldSave = true;
|
private boolean shouldSave = true;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onEnable() {
|
public void onEnable() {
|
||||||
if (
|
if (
|
||||||
loadBans("player", playerBanManager, playerBanFile)
|
loadBans("player", playerBanManager, playerBanFile) ||
|
||||||
|
loadBans("IP", ipBanManager, ipBanFile)
|
||||||
) {
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -35,19 +37,19 @@ public class Main extends Plugin {
|
|||||||
pm.registerCommand(this, new UnbanCommand(playerBanManager));
|
pm.registerCommand(this, new UnbanCommand(playerBanManager));
|
||||||
pm.registerCommand(this, new GetBanCommand(playerBanManager));
|
pm.registerCommand(this, new GetBanCommand(playerBanManager));
|
||||||
|
|
||||||
pm.registerListener(this, new BanJoinListener<>(playerBanManager));
|
pm.registerCommand(this, new BanIPCommand(ipBanManager, this));
|
||||||
|
pm.registerCommand(this, new UnbanIPCommand(ipBanManager));
|
||||||
|
pm.registerCommand(this, new GetIPBanCommand(ipBanManager));
|
||||||
|
|
||||||
|
pm.registerListener(this, new BanJoinListener<>("player", playerBanManager));
|
||||||
|
pm.registerListener(this, new BanJoinListener<>("IP", ipBanManager));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onDisable() {
|
public void onDisable() {
|
||||||
if (shouldSave) {
|
if (shouldSave) {
|
||||||
try {
|
saveBans("player", playerBanManager);
|
||||||
playerBanManager.saveBans();
|
saveBans("IP", ipBanManager);
|
||||||
getLogger().info("Saved bans to file!");
|
|
||||||
} catch (IOException e) {
|
|
||||||
getLogger().severe("Failed to save bans: " + e.getMessage());
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,4 +93,14 @@ public class Main extends Plugin {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void saveBans(String name, BanManager banManager) {
|
||||||
|
try {
|
||||||
|
playerBanManager.saveBans();
|
||||||
|
getLogger().info("Saved " + name + " bans to file!");
|
||||||
|
} catch (IOException e) {
|
||||||
|
getLogger().severe("Failed to save " + name + " bans: " + e.getMessage());
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,9 +5,12 @@ import com.mojang.api.profiles.Profile;
|
|||||||
|
|
||||||
import java.time.ZonedDateTime;
|
import java.time.ZonedDateTime;
|
||||||
import java.time.format.DateTimeFormatter;
|
import java.time.format.DateTimeFormatter;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
public class Util {
|
public class Util {
|
||||||
private static final HttpProfileRepository profileRepo = new HttpProfileRepository("minecraft");
|
private static final HttpProfileRepository profileRepo = new HttpProfileRepository("minecraft");
|
||||||
|
private static final Pattern ipValidity = Pattern.compile(
|
||||||
|
"^([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])$");
|
||||||
|
|
||||||
public static String dashUUID(String uuid) {
|
public static String dashUUID(String uuid) {
|
||||||
return uuid.replaceFirst("([0-9a-fA-F]{8})([0-9a-fA-F]{4})([0-9a-fA-F]{4})([0-9a-fA-F]{4})([0-9a-fA-F]+)", "$1-$2-$3-$4-$5");
|
return uuid.replaceFirst("([0-9a-fA-F]{8})([0-9a-fA-F]{4})([0-9a-fA-F]{4})([0-9a-fA-F]{4})([0-9a-fA-F]+)", "$1-$2-$3-$4-$5");
|
||||||
@ -20,4 +23,8 @@ public class Util {
|
|||||||
public static Profile[] findProfilesByNames(String... names) {
|
public static Profile[] findProfilesByNames(String... names) {
|
||||||
return profileRepo.findProfilesByNames(names);
|
return profileRepo.findProfilesByNames(names);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static boolean validateIP(String ip) {
|
||||||
|
return ipValidity.matcher(ip).matches();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -33,7 +33,7 @@ public class BanCommand extends Command {
|
|||||||
switch (args.length) {
|
switch (args.length) {
|
||||||
case 0:
|
case 0:
|
||||||
sender.sendMessage(
|
sender.sendMessage(
|
||||||
new ComponentBuilder("Usage: ")
|
new ComponentBuilder(ChatColor.RED + "Usage: ")
|
||||||
.append(ChatColor.AQUA + "/ban ")
|
.append(ChatColor.AQUA + "/ban ")
|
||||||
.append(ChatColor.GOLD + "<username> ")
|
.append(ChatColor.GOLD + "<username> ")
|
||||||
.append(ChatColor.YELLOW + "[reason]")
|
.append(ChatColor.YELLOW + "[reason]")
|
||||||
@ -67,9 +67,8 @@ public class BanCommand extends Command {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
bm.saveBans();
|
bm.saveBans();
|
||||||
sender.sendMessage(new TextComponent(ChatColor.GREEN + "Saved bans to file!"));
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save bans to file!"));
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save player bans to file! (nothing was changed)"));
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
|
|
||||||
bm.removeBan(ban);
|
bm.removeBan(ban);
|
||||||
|
@ -0,0 +1,95 @@
|
|||||||
|
package com.redstoner.bungeeBans.commands;
|
||||||
|
|
||||||
|
import com.redstoner.bungeeBans.BanManager;
|
||||||
|
import com.redstoner.bungeeBans.Main;
|
||||||
|
import com.redstoner.bungeeBans.Util;
|
||||||
|
import com.redstoner.bungeeBans.json.IPBan;
|
||||||
|
import net.md_5.bungee.api.ChatColor;
|
||||||
|
import net.md_5.bungee.api.CommandSender;
|
||||||
|
import net.md_5.bungee.api.chat.ComponentBuilder;
|
||||||
|
import net.md_5.bungee.api.chat.TextComponent;
|
||||||
|
import net.md_5.bungee.api.connection.ProxiedPlayer;
|
||||||
|
import net.md_5.bungee.api.plugin.Command;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
public class BanIPCommand extends Command {
|
||||||
|
private BanManager<IPBan> bm;
|
||||||
|
private Main plugin;
|
||||||
|
|
||||||
|
public BanIPCommand(BanManager<IPBan> bm, Main plugin) {
|
||||||
|
super("banip", "bungeeBans.banip");
|
||||||
|
|
||||||
|
this.bm = bm;
|
||||||
|
this.plugin = plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute(CommandSender sender, String[] args) {
|
||||||
|
String reason = "Banned by an operator.";
|
||||||
|
|
||||||
|
switch (args.length) {
|
||||||
|
case 0:
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.RED + "Usage: ")
|
||||||
|
.append(ChatColor.AQUA + "/banip ")
|
||||||
|
.append(ChatColor.GOLD + "<ip> ")
|
||||||
|
.append(ChatColor.YELLOW + "[reason]")
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
case 1:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
reason = String.join(" ", Arrays.copyOfRange(args, 1, args.length));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
String ip = args[0];
|
||||||
|
String expires = "forever"; //TODO: expiry option
|
||||||
|
|
||||||
|
if (!Util.validateIP(ip)) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Invalid IP!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bm.getBan(ip) != null) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "That IP is already banned!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
IPBan ban = new IPBan(ip, Util.getNow(), sender.getName(), expires, reason);
|
||||||
|
bm.addBan(ban);
|
||||||
|
|
||||||
|
try {
|
||||||
|
bm.saveBans();
|
||||||
|
} catch (IOException e) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save IP bans to file! (nothing was changed)"));
|
||||||
|
e.printStackTrace();
|
||||||
|
|
||||||
|
bm.removeBan(ban);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ProxiedPlayer player : plugin.getProxy().getPlayers()) {
|
||||||
|
if (player.getAddress().getAddress().toString().equals(ip)) {
|
||||||
|
player.disconnect(
|
||||||
|
new ComponentBuilder(ChatColor.RED + "Your IP was banned by ")
|
||||||
|
.append(ChatColor.AQUA + sender.getName())
|
||||||
|
.append(ChatColor.RED + " for ")
|
||||||
|
.append(ChatColor.YELLOW + reason)
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.GREEN + "Banned IP ")
|
||||||
|
.append(ChatColor.AQUA + ip)
|
||||||
|
.append(ChatColor.GREEN + " for ")
|
||||||
|
.append(ChatColor.AQUA + reason)
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,59 @@
|
|||||||
|
package com.redstoner.bungeeBans.commands;
|
||||||
|
|
||||||
|
import com.redstoner.bungeeBans.BanManager;
|
||||||
|
import com.redstoner.bungeeBans.Util;
|
||||||
|
import com.redstoner.bungeeBans.json.IPBan;
|
||||||
|
import net.md_5.bungee.api.ChatColor;
|
||||||
|
import net.md_5.bungee.api.CommandSender;
|
||||||
|
import net.md_5.bungee.api.chat.ComponentBuilder;
|
||||||
|
import net.md_5.bungee.api.chat.TextComponent;
|
||||||
|
import net.md_5.bungee.api.plugin.Command;
|
||||||
|
|
||||||
|
public class GetIPBanCommand extends Command {
|
||||||
|
private BanManager<IPBan> bm;
|
||||||
|
|
||||||
|
public GetIPBanCommand(BanManager<IPBan> bm) {
|
||||||
|
super("getipban", "bungeebans.getipban");
|
||||||
|
|
||||||
|
this.bm = bm;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute(CommandSender sender, String[] args) {
|
||||||
|
if (args.length != 1) {
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.RED + "Usage: ")
|
||||||
|
.append(ChatColor.AQUA + "/getipban ")
|
||||||
|
.append(ChatColor.GOLD + "<ip> ")
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String ip = args[0];
|
||||||
|
|
||||||
|
if (!Util.validateIP(ip)) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Invalid IP!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
IPBan ban = bm.getBan(ip);
|
||||||
|
|
||||||
|
if (ban == null) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "That IP is not banned!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.GREEN + "IP ")
|
||||||
|
.append(ChatColor.AQUA + ip)
|
||||||
|
.append(ChatColor.GREEN + " is banned for ")
|
||||||
|
.append(ChatColor.AQUA + ban.getReason())
|
||||||
|
.append(ChatColor.GREEN + " until ")
|
||||||
|
.append(ChatColor.AQUA + ban.getExpires())
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -1,6 +1,5 @@
|
|||||||
package com.redstoner.bungeeBans.commands;
|
package com.redstoner.bungeeBans.commands;
|
||||||
|
|
||||||
import com.mojang.api.profiles.HttpProfileRepository;
|
|
||||||
import com.mojang.api.profiles.Profile;
|
import com.mojang.api.profiles.Profile;
|
||||||
import com.redstoner.bungeeBans.BanManager;
|
import com.redstoner.bungeeBans.BanManager;
|
||||||
import com.redstoner.bungeeBans.Util;
|
import com.redstoner.bungeeBans.Util;
|
||||||
@ -14,8 +13,6 @@ import net.md_5.bungee.api.plugin.Command;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
public class UnbanCommand extends Command {
|
public class UnbanCommand extends Command {
|
||||||
HttpProfileRepository profileRepo = new HttpProfileRepository("minecraft");
|
|
||||||
|
|
||||||
private BanManager<PlayerBan> bm;
|
private BanManager<PlayerBan> bm;
|
||||||
|
|
||||||
public UnbanCommand(BanManager<PlayerBan> bm) {
|
public UnbanCommand(BanManager<PlayerBan> bm) {
|
||||||
@ -28,8 +25,7 @@ public class UnbanCommand extends Command {
|
|||||||
public void execute(CommandSender sender, String[] args) {
|
public void execute(CommandSender sender, String[] args) {
|
||||||
if (args.length != 1) {
|
if (args.length != 1) {
|
||||||
sender.sendMessage(
|
sender.sendMessage(
|
||||||
new ComponentBuilder(ChatColor.RED + "Invalid command! ")
|
new ComponentBuilder(ChatColor.RED + "Usage: ")
|
||||||
.append("Usage: ")
|
|
||||||
.append(ChatColor.AQUA + "/unban ")
|
.append(ChatColor.AQUA + "/unban ")
|
||||||
.append(ChatColor.GOLD + "<username> ")
|
.append(ChatColor.GOLD + "<username> ")
|
||||||
.create()
|
.create()
|
||||||
@ -38,7 +34,7 @@ public class UnbanCommand extends Command {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Profile[] profiles = profileRepo.findProfilesByNames(args[0]);
|
Profile[] profiles = Util.findProfilesByNames(args[0]);
|
||||||
|
|
||||||
if (profiles.length != 1) {
|
if (profiles.length != 1) {
|
||||||
sender.sendMessage(new TextComponent(ChatColor.RED + "Invalid name!"));
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Invalid name!"));
|
||||||
@ -59,9 +55,8 @@ public class UnbanCommand extends Command {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
bm.saveBans();
|
bm.saveBans();
|
||||||
sender.sendMessage(new TextComponent(ChatColor.GREEN + "Saved bans to file!"));
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save bans to file!"));
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save player bans to file! (nothing was changed)"));
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
|
|
||||||
bm.addBan(ban);
|
bm.addBan(ban);
|
||||||
|
@ -0,0 +1,68 @@
|
|||||||
|
package com.redstoner.bungeeBans.commands;
|
||||||
|
|
||||||
|
import com.redstoner.bungeeBans.BanManager;
|
||||||
|
import com.redstoner.bungeeBans.Util;
|
||||||
|
import com.redstoner.bungeeBans.json.IPBan;
|
||||||
|
import net.md_5.bungee.api.ChatColor;
|
||||||
|
import net.md_5.bungee.api.CommandSender;
|
||||||
|
import net.md_5.bungee.api.chat.ComponentBuilder;
|
||||||
|
import net.md_5.bungee.api.chat.TextComponent;
|
||||||
|
import net.md_5.bungee.api.plugin.Command;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class UnbanIPCommand extends Command {
|
||||||
|
private BanManager<IPBan> bm;
|
||||||
|
|
||||||
|
public UnbanIPCommand(BanManager<IPBan> bm) {
|
||||||
|
super("unbanip", "bungeebans.unbanip", "pardonip");
|
||||||
|
|
||||||
|
this.bm = bm;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute(CommandSender sender, String[] args) {
|
||||||
|
if (args.length != 1) {
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.RED + "Usage: ")
|
||||||
|
.append(ChatColor.AQUA + "/unbanip ")
|
||||||
|
.append(ChatColor.GOLD + "<ip> ")
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String ip = args[0];
|
||||||
|
|
||||||
|
if (!Util.validateIP(ip)) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Invalid IP!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
IPBan ban = bm.getBan(ip);
|
||||||
|
|
||||||
|
if (ban == null) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "That IP is not banned!"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bm.removeBan(ban);
|
||||||
|
|
||||||
|
try {
|
||||||
|
bm.saveBans();
|
||||||
|
} catch (IOException e) {
|
||||||
|
sender.sendMessage(new TextComponent(ChatColor.RED + "Failed to save IP bans to file! (nothing was changed)"));
|
||||||
|
e.printStackTrace();
|
||||||
|
|
||||||
|
bm.addBan(ban);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sender.sendMessage(
|
||||||
|
new ComponentBuilder(ChatColor.GREEN + "Unbanned IP ")
|
||||||
|
.append(ChatColor.AQUA + ip)
|
||||||
|
.create()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -25,7 +25,12 @@ public class BanJoinListener<T extends Ban> implements Listener {
|
|||||||
public void onJoin(PreLoginEvent event) {
|
public void onJoin(PreLoginEvent event) {
|
||||||
event.setCancelled(true);
|
event.setCancelled(true);
|
||||||
|
|
||||||
|
T ban;
|
||||||
|
|
||||||
PendingConnection conn = event.getConnection();
|
PendingConnection conn = event.getConnection();
|
||||||
|
|
||||||
|
switch (this.name) {
|
||||||
|
case "player":
|
||||||
String name = conn.getName();
|
String name = conn.getName();
|
||||||
|
|
||||||
Profile[] profiles = Util.findProfilesByNames(name);
|
Profile[] profiles = Util.findProfilesByNames(name);
|
||||||
@ -40,11 +45,20 @@ public class BanJoinListener<T extends Ban> implements Listener {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
T ban = bm.getBan(Util.dashUUID(profiles[0].getId()));
|
ban = bm.getBan(Util.dashUUID(profiles[0].getId()));
|
||||||
|
break;
|
||||||
|
case "IP":
|
||||||
|
String address = conn.getAddress().getAddress().getHostAddress();
|
||||||
|
System.out.println("IP connection: " + address);
|
||||||
|
ban = bm.getBan(address);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
if (ban != null) {
|
if (ban != null) {
|
||||||
event.setCancelReason(
|
event.setCancelReason(
|
||||||
new ComponentBuilder(ChatColor.RED + "You were " + name + " banned by ")
|
new ComponentBuilder(ChatColor.RED + "You were " + this.name + " banned by ")
|
||||||
.append(ChatColor.AQUA + ban.getSource())
|
.append(ChatColor.AQUA + ban.getSource())
|
||||||
.append(ChatColor.RED + " for ")
|
.append(ChatColor.RED + " for ")
|
||||||
.append(ChatColor.AQUA + ban.getReason())
|
.append(ChatColor.AQUA + ban.getReason())
|
||||||
|
Reference in New Issue
Block a user