Implement remote banning
- Use IPID for IP bans instead of remote IP. - Remove 2 extraneous DB queries by altering isIPBanned() - Allow banning unconnected clients
This commit is contained in:
parent
4c32cf86cc
commit
555b4a0cbf
@ -56,13 +56,15 @@ public:
|
||||
~DBManager();
|
||||
|
||||
/**
|
||||
* @brief Checks if there is a record in the Bans table with the given IP address.
|
||||
* @brief Checks if there is a record in the Bans table with the given IPID.
|
||||
*
|
||||
* @param ip The IP address to check if it is banned.
|
||||
* @param ipid The IPID to check if it is banned.
|
||||
*
|
||||
* @return True if the query could return at least one such record.
|
||||
* @return A pair of values:
|
||||
* * First, a `bool` that is true if the query could return at least one such record.
|
||||
* * Then, a `QString` that is the reason for the ban.
|
||||
*/
|
||||
bool isIPBanned(QHostAddress ip);
|
||||
std::pair<bool, QString> isIPBanned(QString ipid);
|
||||
|
||||
/**
|
||||
* @brief Checks if there is a record in the Bans table with the given hardware ID.
|
||||
|
@ -82,8 +82,12 @@ void AOClient::cmdBan(int argc, QStringList argv)
|
||||
|
||||
if (kick_counter > 1)
|
||||
sendServerMessage("Kicked " + QString::number(kick_counter) + " clients with matching ipids.");
|
||||
if (!ban_logged)
|
||||
sendServerMessage("User with ipid not found!");
|
||||
|
||||
// We're banning someone not connected.
|
||||
if (!ban_logged) {
|
||||
server->db_manager->addBan(ban);
|
||||
sendServerMessage("Banned " + ban.ipid + " for reason: " + ban.reason);
|
||||
}
|
||||
}
|
||||
|
||||
void AOClient::cmdKick(int argc, QStringList argv)
|
||||
|
@ -31,23 +31,24 @@ DBManager::DBManager() :
|
||||
updateDB(db_version);
|
||||
}
|
||||
|
||||
bool DBManager::isIPBanned(QHostAddress ip)
|
||||
std::pair<bool, QString> DBManager::isIPBanned(QString ipid)
|
||||
{
|
||||
QSqlQuery query;
|
||||
query.prepare("SELECT TIME FROM BANS WHERE IP = ? ORDER BY TIME DESC");
|
||||
query.addBindValue(ip.toString());
|
||||
query.prepare("SELECT * FROM BANS WHERE IPID = ? ORDER BY TIME DESC");
|
||||
query.addBindValue(ipid);
|
||||
query.exec();
|
||||
if (query.first()) {
|
||||
long long duration = getBanDuration(ip);
|
||||
long long ban_time = query.value(0).toLongLong();
|
||||
long long ban_time = query.value(4).toLongLong();
|
||||
QString reason = query.value(5).toString();
|
||||
long long duration = query.value(6).toLongLong();
|
||||
if (duration == -2)
|
||||
return true;
|
||||
return {true, reason};
|
||||
long long current_time = QDateTime::currentDateTime().toSecsSinceEpoch();
|
||||
if (ban_time + duration > current_time)
|
||||
return true;
|
||||
else return false;
|
||||
return {true, reason};
|
||||
else return {false, nullptr};
|
||||
}
|
||||
else return false;
|
||||
else return {false, nullptr};
|
||||
}
|
||||
|
||||
bool DBManager::isHDIDBanned(QString hdid)
|
||||
|
@ -315,9 +315,13 @@ void AOClient::pktWebSocketIp(AreaData* area, int argc, QStringList argv, AOPack
|
||||
{
|
||||
// Special packet to set remote IP from the webao proxy
|
||||
// Only valid if from a local ip
|
||||
calculateIpid();
|
||||
if (remote_ip.isLoopback()) {
|
||||
if(server->db_manager->isIPBanned(QHostAddress(argv[0]))) {
|
||||
sendPacket("BD", {server->db_manager->getBanReason(QHostAddress(argv[0]))});
|
||||
auto ban = server->db_manager->isIPBanned(ipid);
|
||||
bool is_banned = ban.first;
|
||||
if(is_banned) {
|
||||
QString reason = ban.second;
|
||||
sendPacket("BD", {reason});
|
||||
socket->close();
|
||||
return;
|
||||
}
|
||||
@ -325,7 +329,6 @@ void AOClient::pktWebSocketIp(AreaData* area, int argc, QStringList argv, AOPack
|
||||
qDebug() << "ws ip set to" << argv[0];
|
||||
#endif
|
||||
remote_ip = QHostAddress(argv[0]);
|
||||
calculateIpid();
|
||||
|
||||
int multiclient_count = 0;
|
||||
for (AOClient* joined_client : server->clients) {
|
||||
|
@ -125,7 +125,9 @@ void Server::clientConnected()
|
||||
|
||||
int multiclient_count = 1;
|
||||
bool is_at_multiclient_limit = false;
|
||||
bool is_banned = db_manager->isIPBanned(socket->peerAddress());
|
||||
client->calculateIpid();
|
||||
auto ban = db_manager->isIPBanned(client->getIpid());
|
||||
bool is_banned = ban.first;
|
||||
for (AOClient* joined_client : clients) {
|
||||
if (client->remote_ip.isEqual(joined_client->remote_ip))
|
||||
multiclient_count++;
|
||||
@ -135,7 +137,8 @@ void Server::clientConnected()
|
||||
is_at_multiclient_limit = true;
|
||||
|
||||
if (is_banned) {
|
||||
AOPacket ban_reason("BD", {db_manager->getBanReason(socket->peerAddress())});
|
||||
QString reason = ban.second;
|
||||
AOPacket ban_reason("BD", {reason});
|
||||
socket->write(ban_reason.toUtf8());
|
||||
}
|
||||
if (is_banned || is_at_multiclient_limit) {
|
||||
@ -158,7 +161,6 @@ void Server::clientConnected()
|
||||
// tsuserver4. It should disable fantacrypt
|
||||
// completely in any client 2.4.3 or newer
|
||||
client->sendPacket(decryptor);
|
||||
client->calculateIpid();
|
||||
#ifdef NET_DEBUG
|
||||
qDebug() << client->remote_ip.toString() << "connected";
|
||||
#endif
|
||||
|
Loading…
Reference in New Issue
Block a user