mirror of
https://github.com/bol-van/zapret.git
synced 2025-01-12 19:20:35 +05:00
266 lines
6.4 KiB
C
266 lines
6.4 KiB
C
#include "params.h"
|
|
|
|
#include <stdarg.h>
|
|
#include <syslog.h>
|
|
#include <errno.h>
|
|
|
|
#include "pools.h"
|
|
#include "desync.h"
|
|
|
|
#ifdef BSD
|
|
const char *progname = "dvtws";
|
|
#elif defined(__CYGWIN__)
|
|
const char *progname = "winws";
|
|
#elif defined(__linux__)
|
|
const char *progname = "nfqws";
|
|
#else
|
|
#error UNKNOWN_SYSTEM_TIME
|
|
#endif
|
|
|
|
|
|
int DLOG_FILE(FILE *F, const char *format, va_list args)
|
|
{
|
|
return vfprintf(F, format, args);
|
|
}
|
|
int DLOG_CON(const char *format, int syslog_priority, va_list args)
|
|
{
|
|
return DLOG_FILE(syslog_priority==LOG_ERR ? stderr : stdout, format, args);
|
|
}
|
|
int DLOG_FILENAME(const char *filename, const char *format, va_list args)
|
|
{
|
|
int r;
|
|
FILE *F = fopen(filename,"at");
|
|
if (F)
|
|
{
|
|
r = DLOG_FILE(F, format, args);
|
|
fclose(F);
|
|
}
|
|
else
|
|
r=-1;
|
|
return r;
|
|
}
|
|
|
|
static char syslog_buf[1024];
|
|
static size_t syslog_buf_sz=0;
|
|
static void syslog_buffered(int priority, const char *format, va_list args)
|
|
{
|
|
if (vsnprintf(syslog_buf+syslog_buf_sz,sizeof(syslog_buf)-syslog_buf_sz,format,args)>0)
|
|
{
|
|
syslog_buf_sz=strlen(syslog_buf);
|
|
// log when buffer is full or buffer ends with \n
|
|
if (syslog_buf_sz>=(sizeof(syslog_buf)-1) || (syslog_buf_sz && syslog_buf[syslog_buf_sz-1]=='\n'))
|
|
{
|
|
syslog(priority,"%s",syslog_buf);
|
|
syslog_buf_sz = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int DLOG_VA(const char *format, int syslog_priority, bool condup, va_list args)
|
|
{
|
|
int r=0;
|
|
va_list args2;
|
|
|
|
if (condup && !(params.debug && params.debug_target==LOG_TARGET_CONSOLE))
|
|
{
|
|
va_copy(args2,args);
|
|
DLOG_CON(format,syslog_priority,args2);
|
|
}
|
|
if (params.debug)
|
|
{
|
|
switch(params.debug_target)
|
|
{
|
|
case LOG_TARGET_CONSOLE:
|
|
r = DLOG_CON(format,syslog_priority,args);
|
|
break;
|
|
case LOG_TARGET_FILE:
|
|
r = DLOG_FILENAME(params.debug_logfile,format,args);
|
|
break;
|
|
case LOG_TARGET_SYSLOG:
|
|
// skip newlines
|
|
syslog_buffered(syslog_priority,format,args);
|
|
r = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
int DLOG(const char *format, ...)
|
|
{
|
|
int r;
|
|
va_list args;
|
|
va_start(args, format);
|
|
r = DLOG_VA(format, LOG_DEBUG, false, args);
|
|
va_end(args);
|
|
return r;
|
|
}
|
|
int DLOG_CONDUP(const char *format, ...)
|
|
{
|
|
int r;
|
|
va_list args;
|
|
va_start(args, format);
|
|
r = DLOG_VA(format, LOG_DEBUG, true, args);
|
|
va_end(args);
|
|
return r;
|
|
}
|
|
int DLOG_ERR(const char *format, ...)
|
|
{
|
|
int r;
|
|
va_list args;
|
|
va_start(args, format);
|
|
r = DLOG_VA(format, LOG_ERR, true, args);
|
|
va_end(args);
|
|
return r;
|
|
}
|
|
int DLOG_PERROR(const char *s)
|
|
{
|
|
return DLOG_ERR("%s: %s\n", s, strerror(errno));
|
|
}
|
|
|
|
|
|
int LOG_APPEND(const char *filename, const char *format, va_list args)
|
|
{
|
|
int r;
|
|
FILE *F = fopen(filename,"at");
|
|
if (F)
|
|
{
|
|
fprint_localtime(F);
|
|
fprintf(F, " : ");
|
|
r = vfprintf(F, format, args);
|
|
fprintf(F, "\n");
|
|
fclose(F);
|
|
}
|
|
else
|
|
r=-1;
|
|
return r;
|
|
}
|
|
|
|
int HOSTLIST_DEBUGLOG_APPEND(const char *format, ...)
|
|
{
|
|
if (*params.hostlist_auto_debuglog)
|
|
{
|
|
int r;
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
r = LOG_APPEND(params.hostlist_auto_debuglog, format, args);
|
|
va_end(args);
|
|
return r;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
void hexdump_limited_dlog(const uint8_t *data, size_t size, size_t limit)
|
|
{
|
|
size_t k;
|
|
bool bcut = false;
|
|
if (size > limit)
|
|
{
|
|
size = limit;
|
|
bcut = true;
|
|
}
|
|
if (!size) return;
|
|
for (k = 0; k < size; k++) DLOG("%02X ", data[k]);
|
|
DLOG(bcut ? "... : " : ": ");
|
|
for (k = 0; k < size; k++) DLOG("%c", data[k] >= 0x20 && data[k] <= 0x7F ? (char)data[k] : '.');
|
|
if (bcut) DLOG(" ...");
|
|
}
|
|
|
|
void dp_init(struct desync_profile *dp)
|
|
{
|
|
LIST_INIT(&dp->hl_collection);
|
|
LIST_INIT(&dp->hl_collection_exclude);
|
|
LIST_INIT(&dp->ips_collection);
|
|
LIST_INIT(&dp->ips_collection_exclude);
|
|
LIST_INIT(&dp->pf_tcp);
|
|
LIST_INIT(&dp->pf_udp);
|
|
|
|
memcpy(dp->hostspell, "host", 4); // default hostspell
|
|
dp->desync_skip_nosni = true;
|
|
dp->desync_ipfrag_pos_udp = IPFRAG_UDP_DEFAULT;
|
|
dp->desync_ipfrag_pos_tcp = IPFRAG_TCP_DEFAULT;
|
|
dp->desync_repeats = 1;
|
|
dp->fake_tls_size = sizeof(fake_tls_clienthello_default);
|
|
memcpy(dp->fake_tls,fake_tls_clienthello_default,dp->fake_tls_size);
|
|
randomize_default_tls_payload(dp->fake_tls);
|
|
dp->fake_http_size = strlen(fake_http_request_default);
|
|
memcpy(dp->fake_http,fake_http_request_default,dp->fake_http_size);
|
|
dp->fake_quic_size = 620; // must be 601+ for TSPU hack
|
|
dp->fake_quic[0] = 0x40; // russian TSPU QUIC short header fake
|
|
dp->fake_wg_size = 64;
|
|
dp->fake_dht_size = 64;
|
|
dp->fake_unknown_size = 256;
|
|
dp->fake_syndata_size = 16;
|
|
dp->fake_unknown_udp_size = 64;
|
|
dp->wscale=-1; // default - dont change scale factor (client)
|
|
dp->desync_ttl6 = 0xFF; // unused
|
|
dp->desync_badseq_increment = BADSEQ_INCREMENT_DEFAULT;
|
|
dp->desync_badseq_ack_increment = BADSEQ_ACK_INCREMENT_DEFAULT;
|
|
dp->wssize_cutoff_mode = dp->desync_start_mode = dp->desync_cutoff_mode = 'n'; // packet number by default
|
|
dp->udplen_increment = UDPLEN_INCREMENT_DEFAULT;
|
|
dp->hostlist_auto_fail_threshold = HOSTLIST_AUTO_FAIL_THRESHOLD_DEFAULT;
|
|
dp->hostlist_auto_fail_time = HOSTLIST_AUTO_FAIL_TIME_DEFAULT;
|
|
dp->hostlist_auto_retrans_threshold = HOSTLIST_AUTO_RETRANS_THRESHOLD_DEFAULT;
|
|
dp->filter_ipv4 = dp->filter_ipv6 = true;
|
|
}
|
|
struct desync_profile_list *dp_list_add(struct desync_profile_list_head *head)
|
|
{
|
|
struct desync_profile_list *entry = calloc(1,sizeof(struct desync_profile_list));
|
|
if (!entry) return NULL;
|
|
|
|
dp_init(&entry->dp);
|
|
|
|
// add to the tail
|
|
struct desync_profile_list *dpn,*dpl=LIST_FIRST(¶ms.desync_profiles);
|
|
if (dpl)
|
|
{
|
|
while ((dpn=LIST_NEXT(dpl,next))) dpl = dpn;
|
|
LIST_INSERT_AFTER(dpl, entry, next);
|
|
}
|
|
else
|
|
LIST_INSERT_HEAD(¶ms.desync_profiles, entry, next);
|
|
|
|
return entry;
|
|
}
|
|
static void dp_clear_dynamic(struct desync_profile *dp)
|
|
{
|
|
hostlist_collection_destroy(&dp->hl_collection);
|
|
hostlist_collection_destroy(&dp->hl_collection_exclude);
|
|
ipset_collection_destroy(&dp->ips_collection);
|
|
ipset_collection_destroy(&dp->ips_collection_exclude);
|
|
port_filters_destroy(&dp->pf_tcp);
|
|
port_filters_destroy(&dp->pf_udp);
|
|
HostFailPoolDestroy(&dp->hostlist_auto_fail_counters);
|
|
}
|
|
void dp_clear(struct desync_profile *dp)
|
|
{
|
|
dp_clear_dynamic(dp);
|
|
memset(dp,0,sizeof(*dp));
|
|
}
|
|
void dp_entry_destroy(struct desync_profile_list *entry)
|
|
{
|
|
dp_clear_dynamic(&entry->dp);
|
|
free(entry);
|
|
}
|
|
void dp_list_destroy(struct desync_profile_list_head *head)
|
|
{
|
|
struct desync_profile_list *entry;
|
|
while ((entry = LIST_FIRST(head)))
|
|
{
|
|
LIST_REMOVE(entry, next);
|
|
dp_entry_destroy(entry);
|
|
}
|
|
}
|
|
bool dp_list_have_autohostlist(struct desync_profile_list_head *head)
|
|
{
|
|
struct desync_profile_list *dpl;
|
|
LIST_FOREACH(dpl, head, next)
|
|
if (dpl->dp.hostlist_auto)
|
|
return true;
|
|
return false;
|
|
}
|