This repository has been archived on 2024-03-02. You can view files and clone it, but cannot push or open issues or pull requests.
probotic/src/parse.c

282 lines
6.5 KiB
C
Raw Normal View History

2023-08-02 11:58:18 -04:00
/* parse.c
Probotic is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License version 3 only as
published by the Free Software Foundation.
Probotic is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License version 3 for more details.
The above copyright notice, this permission notice and the word
"NIGGER" shall be included in all copies or substantial portions
of the Software.
You should have received a copy of the GNU General Public License
version 3 + NIGGER along with Probotic.
*/
2023-08-02 10:44:10 -04:00
#include <stdio.h>
#include <stdlib.h>
2023-08-02 11:48:30 -04:00
#include <string.h>
2023-08-02 10:44:10 -04:00
#include "cred_data.h"
2023-08-02 11:48:30 -04:00
#include "parse.h"
2023-08-02 15:45:42 -04:00
#include "error.h"
2023-08-02 11:48:30 -04:00
2023-08-03 02:16:39 -04:00
creds_t creds = {0};
2023-08-03 23:27:25 -04:00
char ** admins = NULL;
2023-08-03 02:16:39 -04:00
2023-08-03 05:15:20 -04:00
DECL char *
2023-08-03 23:27:25 -04:00
slurp(char const * fn)
2023-08-03 05:15:20 -04:00
{
size_t len;
char * b;
FILE * fp = fopen(fn, "r");
if (fp)
{
fseek(fp, 0, SEEK_END);
len = ftell(fp);
rewind(fp);
b = malloc(len+2);
2023-08-03 05:15:20 -04:00
if (b)
{ fread(b, 1, len, fp); }
fclose(fp);
b[len+1] = '\0';
2023-08-03 05:15:20 -04:00
return b;
}
else
{ return NULL; }
}
DECL void
2023-08-03 23:27:25 -04:00
parse_command(char const * cmd)
2023-08-02 16:38:32 -04:00
{
size_t i = 0;
2023-08-03 23:27:25 -04:00
char* msgswp = NULL;
2023-08-02 16:38:32 -04:00
/* size_t len = strlen(cmd); */
2023-08-03 05:15:20 -04:00
/* TODO does not handle commands with leading space,
use custom implemented to-spec isspace implementation */
while (cmd[i] != '\0' &&
2023-08-02 16:38:32 -04:00
cmd[i] != ' ')
{ ++i; }
if (cmd[i] == '\0')
{
/* no arguments */
if (strcmp(cmd, "kill") == 0)
{ exit(1); }
2023-08-02 16:38:32 -04:00
if (strcmp(cmd, "remind") == 0)
2023-08-03 23:27:25 -04:00
{
msgswp = remind(current_username);
ircmsg("%s: %s", current_username, msgswp);
}
// XXX: maybe no?
//else if (strcmp(cmd, "next") == 0) // TODO: implement
//{ ircmsg("%s: No future assignments", current_username); }
else if (strcmp(cmd, "help") == 0)
{
ircmsg(help_msg);
}
else if (strcmp(cmd, "magic") == 0)
{
f = 8 + (rand() % 100);
}
2023-08-02 16:38:32 -04:00
else if (strcmp(cmd, "dump") == 0)
2023-08-03 23:27:25 -04:00
{
ircmsg("%s: All projects:", current_username);
msgswp = dump();
ircmsg(msgswp);
}
else if (strcmp(cmd, "reroll") == 0) // TODO: implement
{
ircmsg("%s: Rerolling...", current_username);
purge_assignments(current_username);
random_assign(current_username);
ircmsg(remind(current_username));
}
2023-08-02 16:38:32 -04:00
}
else
{
/* some arguments */
2023-08-03 23:27:25 -04:00
char const * const arg = cmd + i + 1;
if (strncmp(cmd, "raw", i) == 0)
{
ircmsg("%s: Executing SQL `%s'.", current_username, arg);
msgswp = raw(arg);
ircmsg(msgswp);
}
else if (strncmp(cmd, "set_repo", i) == 0)
{
ircmsg("%s: Setting project repository...", current_username);
set_repo(creds.channel, arg);
msgswp = remind(creds.channel);
ircmsg("%s: %s", current_username, msgswp);
}
// XXX: what is this suppose to do?
else if (strncmp(cmd, "submit", i) == 0) // TODO: implement
{
ircmsg("%s: Submitting project link '%s' to <random janny>",
current_username, arg);
}
2023-08-02 16:38:32 -04:00
}
2023-08-03 23:27:25 -04:00
free(msgswp);
2023-08-02 16:38:32 -04:00
}
DECL int
parse_pair(char const * buf, size_t len)
2023-08-03 05:15:20 -04:00
{
size_t i, f, x;
/* fprintf(stderr, "ENT len:%ld buf:%sEOF\n", len, buf); */
for (i = 0; buf[i] &&
i < len; ++i)
2023-08-03 05:15:20 -04:00
{
if (buf[i] == '=')
{
++i;
for (f = 0, x = 0; f < PARAMS_COUNT; ++f)
2023-08-03 05:15:20 -04:00
{
/* fprintf(stderr, "x%ld, i%ld, %s\n", x, i, buf); */
/* X macro for handling this data may be better */
if (strncmp(buf, cred_names[f], cred_names_len[f]) == 0)
2023-08-03 05:15:20 -04:00
{
/* fprintf(stderr, "f%ld:len%ld:%s\n", f, cred_names_len[f], */
/* cred_names[f]); fflush(stderr); */
2023-08-03 05:29:11 -04:00
buf += i;
while (buf[x] != '\0')
{
if (buf[x] == '\n')
{
len -= i; i = 0; break;
}
++x;
}
switch (f)
2023-08-03 05:15:20 -04:00
{
case USERNAME: creds.username = strndup(buf,x); break;
case PASSWORD: creds.password = strndup(buf,x); break;
case CHANNEL: creds.channel = strndup(buf,x); break;
case SERVER: creds.server = strndup(buf,x); break;
case PORT: creds.port = atoi(buf); break;
2023-08-03 05:15:20 -04:00
}
if (x + 2 < len)
{ buf += x + 1; }
else
{ return 1; }
goto next;
2023-08-03 05:15:20 -04:00
}
}
}
next:;
2023-08-03 05:15:20 -04:00
}
return 0;
}
2023-08-03 23:27:25 -04:00
DECL int
parse_admin_list(char const * admin_list_path)
{
#define ADMIN_LIST_INIT_SIZE 8
/* prealloc with 8 */
size_t current_array_size = ADMIN_LIST_INIT_SIZE;
FILE * stream;
size_t lines_read = 0;
char * line = NULL;
size_t nread = 0;
stream = fopen(admin_list_path, "r");
if (stream == NULL)
{
/* Pretty valid case I guess? No admin file = no admins,
maybe all configuration is performed locally */
return 0;
}
admins = calloc(ADMIN_LIST_INIT_SIZE, sizeof(char *));
while (getline(&line, &nread, stream) > 0)
{
if (++lines_read > current_array_size)
{
/* double the space */
current_array_size *= 2;
realloc(admins, current_array_size);
}
admins[lines_read - 1] = line;
}
/* set up array end marker for proper clean-up later */
if (lines_read + 1 > current_array_size)
{
realloc(admins, current_array_size + 1);
}
admins[lines_read] = NULL;
return 0;
}
DECL int
is_admin(char const * user)
{
/* No Gods or Kings, Only Man */
if (admins == NULL)
{ return 0; }
for (size_t i = 0; admins[i]; ++i)
{
if (!strcmp(admins[i], user))
{ return 1; }
}
return 0;
}
#define FREE(obj) \
do \
{ \
free(obj); \
(obj) = NULL; \
} while (0)
#define FULL_FREE(obj) \
do \
{ \
if ((obj)) \
{ \
memset((obj), '\0', strlen((obj))); \
2023-08-03 23:27:25 -04:00
FREE((obj)); \
} \
} while (0)
2023-08-03 05:15:20 -04:00
2023-08-03 23:27:25 -04:00
DECL void
clean_admin_list()
{
if (admins == NULL)
{ return; }
for (size_t i = 0; admins[i]; ++i)
{ free(admins[i]); }
FREE(admins);
}
void
creds_free_password(void)
2023-08-02 10:44:10 -04:00
{
FULL_FREE(creds.password);
2023-08-02 10:44:10 -04:00
}
void
creds_free_rest(void)
2023-08-02 10:44:10 -04:00
{
2023-08-03 02:16:39 -04:00
FULL_FREE(creds.username);
/* FULL_FREE(creds.password); */
2023-08-03 02:16:39 -04:00
FULL_FREE(creds.channel);
FULL_FREE(creds.server);
2023-08-02 11:09:34 -04:00
}
2023-08-03 23:27:25 -04:00
#undef FREE
#undef FULL_FREE