- Attach ban reasons to banned_t

- Properly call D_SaveBan after remote bans. Bans are no longer saved in the ban command and instead wait for the actual kick to process, since before they were split between the two, which is what caused the discrepancy.

# Conflicts:
#	src/d_clisrv.c
#	src/i_tcp.c
This commit is contained in:
toaster 2022-06-14 16:58:07 +01:00
parent 2f0e1521d0
commit 24181ae738
4 changed files with 77 additions and 106 deletions

View file

@ -2637,21 +2637,10 @@ static void CL_ConnectToServer(void)
} }
#ifndef NONET #ifndef NONET
typedef struct banreason_s
{
char *reason;
struct banreason_s *prev; //-1
struct banreason_s *next; //+1
} banreason_t;
static banreason_t *reasontail = NULL; //last entry, use prev
static banreason_t *reasonhead = NULL; //1st entry, use next
static void Command_ShowBan(void) //Print out ban list static void Command_ShowBan(void) //Print out ban list
{ {
size_t i; size_t i;
const char *address, *mask; const char *address, *mask, *reason;
banreason_t *reasonlist = reasonhead;
if (I_GetBanAddress) if (I_GetBanAddress)
CONS_Printf(M_GetText("Ban List:\n")); CONS_Printf(M_GetText("Ban List:\n"));
@ -2665,12 +2654,10 @@ static void Command_ShowBan(void) //Print out ban list
else else
CONS_Printf("%s: %s/%s ", sizeu1(i+1), address, mask); CONS_Printf("%s: %s/%s ", sizeu1(i+1), address, mask);
if (reasonlist && reasonlist->reason) if (I_GetBanReason && (reason = I_GetBanReason(i)) != NULL)
CONS_Printf("(%s)\n", reasonlist->reason); CONS_Printf("(%s)\n", reason);
else else
CONS_Printf("\n"); CONS_Printf("\n");
if (reasonlist) reasonlist = reasonlist->next;
} }
if (i == 0 && !address) if (i == 0 && !address)
@ -2681,13 +2668,10 @@ void D_SaveBan(void)
{ {
FILE *f; FILE *f;
size_t i; size_t i;
banreason_t *reasonlist = reasonhead; const char *address, *mask, *reason;
const char *address, *mask; const char *path = va("%s"PATHSEP"%s", srb2home, "ban.txt");
if (!reasonhead) f = fopen(path, "w");
return;
f = fopen(va("%s"PATHSEP"%s", srb2home, "ban.txt"), "w");
if (!f) if (!f)
{ {
@ -2702,59 +2686,29 @@ void D_SaveBan(void)
else else
fprintf(f, "%s %s", address, mask); fprintf(f, "%s %s", address, mask);
if (reasonlist && reasonlist->reason) if (I_GetBanReason && (reason = I_GetBanReason(i)) != NULL)
fprintf(f, " %s\n", reasonlist->reason); fprintf(f, " %s\n", reason);
else else
fprintf(f, " %s\n", "NA"); fprintf(f, " %s\n", "No reason given");
if (reasonlist) reasonlist = reasonlist->next;
} }
fclose(f); fclose(f);
} }
static void Ban_Add(const char *reason)
{
banreason_t *reasonlist = malloc(sizeof(*reasonlist));
if (!reasonlist)
return;
if (!reason)
reason = "NA";
reasonlist->next = NULL;
reasonlist->reason = Z_StrDup(reason);
if ((reasonlist->prev = reasontail) == NULL)
reasonhead = reasonlist;
else
reasontail->next = reasonlist;
reasontail = reasonlist;
}
static void Command_ClearBans(void) static void Command_ClearBans(void)
{ {
banreason_t *temp;
if (!I_ClearBans) if (!I_ClearBans)
return; return;
I_ClearBans(); I_ClearBans();
D_SaveBan(); D_SaveBan();
reasontail = NULL;
while (reasonhead)
{
temp = reasonhead->next;
Z_Free(reasonhead->reason);
free(reasonhead);
reasonhead = temp;
}
} }
static void Ban_Load_File(boolean warning) static void Ban_Load_File(boolean warning)
{ {
FILE *f; FILE *f;
size_t i; size_t i;
const char *address, *mask; const char *address, *mask, *reason;
char buffer[MAX_WADPATH]; char buffer[MAX_WADPATH];
f = fopen(va("%s"PATHSEP"%s", srb2home, "ban.txt"), "r"); f = fopen(va("%s"PATHSEP"%s", srb2home, "ban.txt"), "r");
@ -2766,22 +2720,17 @@ static void Ban_Load_File(boolean warning)
return; return;
} }
if (I_ClearBans) I_ClearBans();
Command_ClearBans();
else
{
fclose(f);
return;
}
for (i=0; fgets(buffer, (int)sizeof(buffer), f); i++) for (i=0; fgets(buffer, (int)sizeof(buffer), f); i++)
{ {
address = strtok(buffer, " \t\r\n"); address = strtok(buffer, " \t\r\n");
mask = strtok(NULL, " \t\r\n"); mask = strtok(NULL, " \t\r\n");
reason = strtok(NULL, "\r\n");
I_SetBanAddress(address, mask); I_SetBanAddress(address, mask);
if (I_SetBanReason)
Ban_Add(strtok(NULL, "\r\n")); I_SetBanReason(reason);
} }
fclose(f); fclose(f);
@ -3140,27 +3089,12 @@ static void Command_Ban(void)
XBOXSTATIC UINT8 buf[3 + MAX_REASONLENGTH]; XBOXSTATIC UINT8 buf[3 + MAX_REASONLENGTH];
UINT8 *p = buf; UINT8 *p = buf;
const SINT8 pn = nametonum(COM_Argv(1)); const SINT8 pn = nametonum(COM_Argv(1));
const INT32 node = playernode[(INT32)pn];
if (pn == -1 || pn == 0) if (pn == -1 || pn == 0)
return; return;
WRITEUINT8(p, pn); WRITEUINT8(p, pn);
if (server && I_Ban && !I_Ban(node)) // only the server is allowed to do this right now
{
CONS_Alert(CONS_WARNING, M_GetText("Ban failed. Invalid node?\n"));
WRITEUINT8(p, KICK_MSG_GO_AWAY);
SendNetXCmd(XD_KICK, &buf, 2);
}
else
{
if (server) // only the server is allowed to do this right now
{
Ban_Add(COM_Argv(2));
D_SaveBan(); // save the ban list
}
if (COM_Argc() == 2) if (COM_Argc() == 2)
{ {
WRITEUINT8(p, KICK_MSG_BANNED); WRITEUINT8(p, KICK_MSG_BANNED);
@ -3184,10 +3118,8 @@ static void Command_Ban(void)
SendNetXCmd(XD_KICK, &buf, p - buf); SendNetXCmd(XD_KICK, &buf, p - buf);
} }
} }
}
else else
CONS_Printf(M_GetText("Only the server or a remote admin can use this.\n")); CONS_Printf(M_GetText("Only the server or a remote admin can use this.\n"));
} }
static void Command_BanIP(void) static void Command_BanIP(void)
@ -3208,7 +3140,6 @@ static void Command_BanIP(void)
else else
reason = COM_Argv(2); reason = COM_Argv(2);
if (I_SetBanAddress && I_SetBanAddress(address, NULL)) if (I_SetBanAddress && I_SetBanAddress(address, NULL))
{ {
if (reason) if (reason)
@ -3216,7 +3147,8 @@ static void Command_BanIP(void)
else else
CONS_Printf("Banned IP address %s\n", address); CONS_Printf("Banned IP address %s\n", address);
Ban_Add(reason); if (I_SetBanReason)
I_SetBanReason(reason);
D_SaveBan(); D_SaveBan();
} }
else else
@ -3357,16 +3289,21 @@ static void Got_KickCmd(UINT8 **p, INT32 playernum)
//CONS_Printf("\x82%s ", player_names[pnum]); //CONS_Printf("\x82%s ", player_names[pnum]);
// If a verified admin banned someone, the server needs to know about it. // Save bans here. Used to be split between here and the actual command, depending on
// If the playernum isn't zero (the server) then the server needs to record the ban. // whenever the server did it or a remote admin did it, but it's simply more convenient
if (server && playernum && (msg == KICK_MSG_BANNED || msg == KICK_MSG_CUSTOM_BAN)) // to keep it all in one place.
if (server && (msg == KICK_MSG_BANNED || msg == KICK_MSG_CUSTOM_BAN))
{ {
if (I_Ban && !I_Ban(playernode[(INT32)pnum])) if (I_Ban && !I_Ban(playernode[(INT32)pnum]))
CONS_Alert(CONS_WARNING, M_GetText("Too many bans! Geez, that's a lot of people you're excluding...\n")); {
#ifndef NONET CONS_Alert(CONS_WARNING, M_GetText("Ban failed. Invalid node?\n"));
}
else else
Ban_Add(reason); {
#endif if (I_SetBanReason)
I_SetBanReason(reason);
D_SaveBan();
}
} }
if (msg == KICK_MSG_PLAYER_QUIT) if (msg == KICK_MSG_PLAYER_QUIT)

View file

@ -79,7 +79,9 @@ void (*I_ClearBans)(void) = NULL;
const char *(*I_GetNodeAddress) (INT32 node) = NULL; const char *(*I_GetNodeAddress) (INT32 node) = NULL;
const char *(*I_GetBanAddress) (size_t ban) = NULL; const char *(*I_GetBanAddress) (size_t ban) = NULL;
const char *(*I_GetBanMask) (size_t ban) = NULL; const char *(*I_GetBanMask) (size_t ban) = NULL;
const char *(*I_GetBanReason) (size_t ban) = NULL;
boolean (*I_SetBanAddress) (const char *address, const char *mask) = NULL; boolean (*I_SetBanAddress) (const char *address, const char *mask) = NULL;
boolean (*I_SetBanReason) (const char *reason) = NULL;
boolean *bannednode = NULL; boolean *bannednode = NULL;

View file

@ -145,7 +145,9 @@ extern void (*I_ClearBans)(void);
extern const char *(*I_GetNodeAddress) (INT32 node); extern const char *(*I_GetNodeAddress) (INT32 node);
extern const char *(*I_GetBanAddress) (size_t ban); extern const char *(*I_GetBanAddress) (size_t ban);
extern const char *(*I_GetBanMask) (size_t ban); extern const char *(*I_GetBanMask) (size_t ban);
extern const char *(*I_GetBanReason) (size_t ban);
extern boolean (*I_SetBanAddress) (const char *address,const char *mask); extern boolean (*I_SetBanAddress) (const char *address,const char *mask);
extern boolean (*I_SetBanReason) (const char *reason);
extern boolean *bannednode; extern boolean *bannednode;
/// \brief Called by D_SRB2Main to be defined by extern network driver /// \brief Called by D_SRB2Main to be defined by extern network driver

View file

@ -235,6 +235,7 @@ typedef struct
{ {
mysockaddr_t address; mysockaddr_t address;
UINT8 mask; UINT8 mask;
char *reason;
// TODO: timestamp, for tempbans! // TODO: timestamp, for tempbans!
} banned_t; } banned_t;
@ -498,6 +499,16 @@ static const char *SOCK_GetBanMask(size_t ban)
return NULL; return NULL;
} }
static const char *SOCK_GetBanReason(size_t ban)
{
#ifdef NONET
(void)ban;
return NULL;
#else
return banned[ban].reason;
#endif
}
#ifndef NONET #ifndef NONET
static boolean SOCK_cmpaddr(mysockaddr_t *a, mysockaddr_t *b, UINT8 mask) static boolean SOCK_cmpaddr(mysockaddr_t *a, mysockaddr_t *b, UINT8 mask)
{ {
@ -1521,6 +1532,23 @@ static boolean SOCK_SetBanAddress(const char *address, const char *mask)
#endif #endif
} }
static boolean SOCK_SetBanReason(const char *reason)
{
#ifdef NONET
(void)reason;
return false;
#else
if (!reason)
{
reason = "No reason given";
}
banned[numbans - 1].reason = Z_StrDup(reason);
return true;
#endif
}
static void SOCK_ClearBans(void) static void SOCK_ClearBans(void)
{ {
numbans = 0; numbans = 0;
@ -1617,7 +1645,9 @@ boolean I_InitTcpNetwork(void)
I_GetNodeAddress = SOCK_GetNodeAddress; I_GetNodeAddress = SOCK_GetNodeAddress;
I_GetBanAddress = SOCK_GetBanAddress; I_GetBanAddress = SOCK_GetBanAddress;
I_GetBanMask = SOCK_GetBanMask; I_GetBanMask = SOCK_GetBanMask;
I_GetBanReason = SOCK_GetBanReason;
I_SetBanAddress = SOCK_SetBanAddress; I_SetBanAddress = SOCK_SetBanAddress;
I_SetBanReason = SOCK_SetBanReason;
bannednode = SOCK_bannednode; bannednode = SOCK_bannednode;
return ret; return ret;