You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
230 lines
4.9 KiB
230 lines
4.9 KiB
/* Copyright 2016 Alex 'AdUser' Z (ad_user@runbox.com) |
|
* |
|
* This program is free software; you can redistribute it and/or modify |
|
* it under the terms of the GNU General Public License version 2 as |
|
* published by the Free Software Foundation. |
|
*/ |
|
#include <assert.h> |
|
#include <errno.h> |
|
#include <stdio.h> |
|
#include <string.h> |
|
#include <stdbool.h> |
|
#include <stdint.h> |
|
#include <stdlib.h> |
|
#include <sys/types.h> |
|
#include <unistd.h> |
|
#include <netinet/in.h> |
|
#include <arpa/inet.h> |
|
#include <net/if.h> |
|
#include <netdb.h> |
|
|
|
#include "../strlcpy.h" |
|
#include "../commands.h" |
|
#include "../cmsg.h" |
|
#include "../csocket.h" |
|
|
|
#include "backend.h" |
|
#include "shared.c" |
|
|
|
#define DEFAULT_MCAST_ADDR "239.255.186.1" |
|
#define DEFAULT_MCAST_PORT "3370" |
|
#define DEFAULT_PING_NUM 5 |
|
|
|
struct _config { |
|
char name[ID_MAX + 1]; |
|
char error[256]; |
|
bool shared; |
|
uint8_t ping_num; /*< current number of ping() call */ |
|
uint8_t ping_max; /*< max ping() calls before actually send CMD_PING packet */ |
|
char maddr[INET_ADDRSTRLEN]; /**< multicast address */ |
|
char mport[6]; /**< multicast port */ |
|
char iface[IF_NAMESIZE]; /**< bind interface */ |
|
int sock; |
|
struct sockaddr_storage sa; |
|
socklen_t sa_len; |
|
}; |
|
|
|
cfg_t * |
|
create(const char *id) { |
|
cfg_t *cfg = NULL; |
|
|
|
assert(id != NULL); |
|
|
|
if ((cfg = calloc(1, sizeof(cfg_t))) == NULL) |
|
return NULL; |
|
cfg->ping_max = DEFAULT_PING_NUM; |
|
strlcpy(cfg->name, id, sizeof(cfg->name)); |
|
strlcpy(cfg->maddr, DEFAULT_MCAST_ADDR, sizeof(cfg->maddr)); |
|
strlcpy(cfg->mport, DEFAULT_MCAST_PORT, sizeof(cfg->mport)); |
|
|
|
return cfg; |
|
} |
|
|
|
bool |
|
config(cfg_t *cfg, const char *key, const char *value) { |
|
assert(cfg != NULL); |
|
assert(key != NULL); |
|
assert(value != NULL); |
|
|
|
if (strcmp(key, "group") == 0) { |
|
if (strncmp(value, "239.255.", 8) != 0) { |
|
strlcpy(cfg->error, "mcast group address should be inside 239.255.0.0/16 block", sizeof(cfg->error)); |
|
return false; |
|
} |
|
strlcpy(cfg->maddr, value, sizeof(cfg->maddr)); |
|
return true; |
|
} |
|
if (strcmp(key, "port") == 0) { |
|
strlcpy(cfg->mport, value, sizeof(cfg->mport)); |
|
return true; |
|
} |
|
if (strcmp(key, "iface") == 0) { |
|
strlcpy(cfg->iface, value, sizeof(cfg->iface)); |
|
return true; |
|
} |
|
if (strcmp(key, "ping") == 0) { |
|
cfg->ping_max = atoi(value); |
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
bool |
|
ready(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
if (cfg->maddr[0] && cfg->mport[0]) |
|
return true; |
|
|
|
return false; |
|
} |
|
|
|
char * |
|
error(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
return cfg->error; |
|
} |
|
|
|
bool |
|
start(cfg_t *cfg) { |
|
struct addrinfo hints; |
|
struct addrinfo *result; |
|
assert(cfg != NULL); |
|
int ret, sock = -1; |
|
|
|
if (cfg->shared && usage_inc(cfg->name) > 1) |
|
return true; |
|
|
|
memset(&hints, 0, sizeof(struct addrinfo)); |
|
hints.ai_family = AF_INET; |
|
hints.ai_socktype = SOCK_DGRAM; |
|
hints.ai_protocol = 0; |
|
|
|
if ((ret = getaddrinfo(cfg->maddr, cfg->mport, &hints, &result)) < 0) { |
|
snprintf(cfg->error, sizeof(cfg->error), "can't create socket: %s", gai_strerror(ret)); |
|
return false; |
|
} |
|
|
|
for (struct addrinfo *rp = result; rp != NULL; rp = rp->ai_next) { |
|
if (sock >= 0) { |
|
close(sock); /* from prev iteration */ |
|
sock = -1; |
|
} |
|
if ((sock = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol)) < 0) { |
|
snprintf(cfg->error, sizeof(cfg->error), "can't create socket: %s", strerror(errno)); |
|
continue; |
|
} |
|
memcpy(&cfg->sa, rp->ai_addr, rp->ai_addrlen); |
|
cfg->sa_len = rp->ai_addrlen; |
|
break; |
|
} |
|
freeaddrinfo(result); |
|
|
|
if (sock < 0) |
|
return false; |
|
|
|
cfg->sock = sock; |
|
return true; |
|
|
|
return true; |
|
} |
|
|
|
bool |
|
stop(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
if (cfg->shared && usage_dec(cfg->name) > 0) |
|
return true; |
|
|
|
close(cfg->sock); |
|
cfg->sock = -1; |
|
|
|
return true; |
|
} |
|
|
|
bool |
|
ban(cfg_t *cfg, const char *ip) { |
|
f2b_cmsg_t cmsg; |
|
int ret; |
|
|
|
assert(cfg != NULL); |
|
|
|
memset(&cmsg, 0x0, sizeof(cmsg)); |
|
|
|
strncpy(cmsg.magic, "F2B", sizeof(cmsg.magic)); |
|
cmsg.version = F2B_PROTO_VER; |
|
cmsg.type = CMD_JAIL_IP_BAN; |
|
f2b_cmd_append_arg(cmsg.data, sizeof(cmsg.data), cfg->name); |
|
f2b_cmd_append_arg(cmsg.data, sizeof(cmsg.data), ip); |
|
cmsg.size = strlen(cmsg.data); |
|
cmsg.data[cmsg.size] = '\0'; |
|
f2b_cmsg_convert_args(&cmsg); |
|
|
|
ret = f2b_csocket_send(cfg->sock, &cmsg, &cfg->sa, &cfg->sa_len); |
|
if (ret <= 0) { |
|
strlcpy(cfg->error, f2b_csocket_error(ret), sizeof(cfg->error)); |
|
return false; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
bool |
|
unban(cfg_t *cfg, const char *ip) { |
|
assert(cfg != NULL); |
|
|
|
(void)(ip); /* suppress warning for unused variable 'ip' */ |
|
return true; |
|
} |
|
|
|
bool |
|
check(cfg_t *cfg, const char *ip) { |
|
assert(cfg != NULL); |
|
|
|
(void)(ip); /* suppress warning for unused variable 'ip' */ |
|
return false; |
|
} |
|
|
|
bool |
|
ping(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
cfg->ping_num++; |
|
if (cfg->ping_num < cfg->ping_max) |
|
return true; /* skip this try */ |
|
|
|
/* max empty calls reached, make real ping */ |
|
cfg->ping_num = 0; |
|
/* TODO */ |
|
|
|
return false; |
|
} |
|
|
|
void |
|
destroy(cfg_t *cfg) { |
|
assert(cfg != NULL); |
|
|
|
free(cfg); |
|
}
|
|
|