Add --exclude-domains flag. Fix #100

This commit is contained in:
Vadim Vetrov 2024-09-03 21:22:17 +03:00
parent 3d50c00e4f
commit 5f2e423dfa
No known key found for this signature in database
GPG Key ID: E8A308689D7A73A5
4 changed files with 57 additions and 3 deletions

View File

@ -137,7 +137,9 @@ Put flags to the **BINARY**, not an init script. If you are on OpenWRT you shoul
Available flags: Available flags:
- `--sni-domains=<comma separated domain list>|all` List of domains you want to be handled by SNI. Use this string if you want to change default domain list. Defaults to `googlevideo.com,ggpht.com,ytimg.com,youtube.com,play.google.com,youtu.be,googleapis.com,googleusercontent.com,gstatic.com,l.google.com`. You can pass **all** if you want for every *ClientHello* to be handled. - `--sni-domains=<comma separated domain list>|all` List of domains you want to be handled by SNI. Use this string if you want to change default domain list. Defaults to `googlevideo.com,ggpht.com,ytimg.com,youtube.com,play.google.com,youtu.be,googleapis.com,googleusercontent.com,gstatic.com,l.google.com`. You can pass **all** if you want for every *ClientHello* to be handled. You can exclude some domains from _all_ with `--exclude-domains` flag.
- `--exclude-domains=<comma separated domain list>` List of domains to be excluded from targetting. Useful if you use `--sni-domains=all` and want for some websites to not be targetted by youtubeUnblock. Also the use case is subdomains (for example if you unblock l.google.com, dl.google.com will be also targetted. You can pass it to this flag and it will be ignored).
- `--queue-num=<number of netfilter queue>` The number of netfilter queue **youtubeUnblock** will be linked to. Defaults to **537**. - `--queue-num=<number of netfilter queue>` The number of netfilter queue **youtubeUnblock** will be linked to. Defaults to **537**.

12
args.c
View File

@ -49,12 +49,16 @@ struct config_t config = {
.domains_str = defaul_snistr, .domains_str = defaul_snistr,
.domains_strlen = sizeof(defaul_snistr), .domains_strlen = sizeof(defaul_snistr),
.exclude_domains_str = "",
.exclude_domains_strlen = 0,
.queue_start_num = DEFAULT_QUEUE_NUM, .queue_start_num = DEFAULT_QUEUE_NUM,
.fake_sni_pkt = fake_sni_old, .fake_sni_pkt = fake_sni_old,
.fake_sni_pkt_sz = sizeof(fake_sni_old) - 1, // - 1 for null-terminator .fake_sni_pkt_sz = sizeof(fake_sni_old) - 1, // - 1 for null-terminator
}; };
#define OPT_SNI_DOMAINS 1 #define OPT_SNI_DOMAINS 1
#define OPT_EXCLUDE_DOMAINS 25
#define OPT_FAKE_SNI 2 #define OPT_FAKE_SNI 2
#define OPT_FAKING_TTL 3 #define OPT_FAKING_TTL 3
#define OPT_FAKING_STRATEGY 10 #define OPT_FAKING_STRATEGY 10
@ -79,12 +83,13 @@ struct config_t config = {
#define OPT_NO_GSO 8 #define OPT_NO_GSO 8
#define OPT_QUEUE_NUM 9 #define OPT_QUEUE_NUM 9
#define OPT_MAX OPT_SYNFAKE_LEN #define OPT_MAX OPT_SNI_DOMAINS
static struct option long_opt[] = { static struct option long_opt[] = {
{"help", 0, 0, 'h'}, {"help", 0, 0, 'h'},
{"version", 0, 0, 'v'}, {"version", 0, 0, 'v'},
{"sni-domains", 1, 0, OPT_SNI_DOMAINS}, {"sni-domains", 1, 0, OPT_SNI_DOMAINS},
{"exclude-domains", 1, 0, OPT_EXCLUDE_DOMAINS},
{"fake-sni", 1, 0, OPT_FAKE_SNI}, {"fake-sni", 1, 0, OPT_FAKE_SNI},
{"synfake", 1, 0, OPT_SYNFAKE}, {"synfake", 1, 0, OPT_SYNFAKE},
{"synfake-len", 1, 0, OPT_SYNFAKE_LEN}, {"synfake-len", 1, 0, OPT_SYNFAKE_LEN},
@ -142,6 +147,7 @@ void print_usage(const char *argv0) {
printf("Options:\n"); printf("Options:\n");
printf("\t--queue-num=<number of netfilter queue>\n"); printf("\t--queue-num=<number of netfilter queue>\n");
printf("\t--sni-domains=<comma separated domain list>|all\n"); printf("\t--sni-domains=<comma separated domain list>|all\n");
printf("\t--exclude-domains=<comma separated domain list>\n");
printf("\t--fake-sni={1|0}\n"); printf("\t--fake-sni={1|0}\n");
printf("\t--fake-sni-seq-len=<length>\n"); printf("\t--fake-sni-seq-len=<length>\n");
printf("\t--fake-seq-offset=<offset>\n"); printf("\t--fake-seq-offset=<offset>\n");
@ -203,6 +209,10 @@ int parse_args(int argc, char *argv[]) {
config.domains_str = optarg; config.domains_str = optarg;
config.domains_strlen = strlen(config.domains_str); config.domains_strlen = strlen(config.domains_str);
break; break;
case OPT_EXCLUDE_DOMAINS:
config.exclude_domains_str = optarg;
config.exclude_domains_strlen = strlen(config.exclude_domains_str);
break;
case OPT_SNI_DETECTION: case OPT_SNI_DETECTION:
if (strcmp(optarg, "parse") == 0) { if (strcmp(optarg, "parse") == 0) {
config.sni_detection = SNI_DETECTION_PARSE; config.sni_detection = SNI_DETECTION_PARSE;

View File

@ -40,6 +40,8 @@ struct config_t {
unsigned int seg2_delay; unsigned int seg2_delay;
const char *domains_str; const char *domains_str;
unsigned int domains_strlen; unsigned int domains_strlen;
const char *exclude_domains_str;
unsigned int exclude_domains_strlen;
unsigned int all_domains; unsigned int all_domains;
const char *fake_sni_pkt; const char *fake_sni_pkt;
unsigned int fake_sni_pkt_sz; unsigned int fake_sni_pkt_sz;

View File

@ -704,7 +704,7 @@ struct tls_verdict analyze_tls_data(
if (config.all_domains) { if (config.all_domains) {
vrd.target_sni = 1; vrd.target_sni = 1;
goto out; goto check_domain;
} }
@ -726,12 +726,48 @@ struct tls_verdict analyze_tls_data(
domain_startp, domain_startp,
domain_len)) { domain_len)) {
vrd.target_sni = 1; vrd.target_sni = 1;
goto check_domain;
} }
j = i + 1; j = i + 1;
} }
} }
continue;
check_domain:
if (vrd.target_sni == 1 && config.exclude_domains_strlen != 0) {
unsigned int j = 0;
for (unsigned int i = 0; i <= config.exclude_domains_strlen; i++) {
if ( i > j &&
(i == config.exclude_domains_strlen ||
config.exclude_domains_str[i] == '\0' ||
config.exclude_domains_str[i] == ',' ||
config.exclude_domains_str[i] == '\n' )) {
unsigned int domain_len = (i - j);
const char *sni_startp = sni_name + sni_len - domain_len;
const char *domain_startp = config.exclude_domains_str + j;
if (sni_len >= domain_len &&
sni_len < 128 &&
!strncmp(sni_startp,
domain_startp,
domain_len)) {
vrd.target_sni = 0;
lgdebugmsg("Excluded SNI: %.*s",
vrd.sni_len, data + vrd.sni_offset);
goto out;
}
j = i + 1;
}
}
}
goto out;
nextExtension: nextExtension:
extensionsPtr += 2 + 2 + extensionLen; extensionsPtr += 2 + 2 + extensionLen;
} }
@ -739,9 +775,13 @@ nextMessage:
i += 5 + message_length; i += 5 + message_length;
} }
goto out;
out: out:
return vrd; return vrd;
brute: brute:
if (config.all_domains) { if (config.all_domains) {
vrd.target_sni = 1; vrd.target_sni = 1;