#include "config.h" #include "raw_replacements.h" #include #include #include #include #include #include #include struct config_t config = { .threads = THREADS_NUM, .frag_sni_reverse = 0, .frag_sni_faked = 0, .fragmentation_strategy = FRAGMENTATION_STRATEGY, .faking_strategy = FAKING_STRATEGY, .faking_ttl = FAKE_TTL, .fake_sni = 1, .fake_sni_seq_len = 1, #ifdef SEG2_DELAY .seg2_delay = SEG2_DELAY, #else .seg2_delay = 0, #endif #ifdef USE_GSO .use_gso = true, #else .use_gso = false, #endif #ifdef DEBUG .verbose = true, #else .verbose = false, #endif .domains_str = defaul_snistr, .domains_strlen = sizeof(defaul_snistr), .queue_start_num = DEFAULT_QUEUE_NUM, .fake_sni_pkt = fake_sni, .fake_sni_pkt_sz = sizeof(fake_sni) - 1, // - 1 for null-terminator }; #define OPT_SNI_DOMAINS 1 #define OPT_FAKE_SNI 2 #define OPT_FAKING_TTL 3 #define OPT_FAKING_STRATEGY 10 #define OPT_FAKE_SNI_SEQ_LEN 11 #define OPT_FRAG 4 #define OPT_FRAG_SNI_REVERSE 12 #define OPT_FRAG_SNI_FAKED 13 #define OPT_SEG2DELAY 5 #define OPT_THREADS 6 #define OPT_SILENT 7 #define OPT_NO_GSO 8 #define OPT_QUEUE_NUM 9 #define OPT_MAX OPT_FRAG_SNI_FAKED static struct option long_opt[] = { {"help", 0, 0, 'h'}, {"version", 0, 0, 'v'}, {"sni-domains", 1, 0, OPT_SNI_DOMAINS}, {"fake-sni", 1, 0, OPT_FAKE_SNI}, {"fake-sni-seq-len", 1, 0, OPT_FAKE_SNI_SEQ_LEN}, {"faking-strategy", 1, 0, OPT_FAKING_STRATEGY}, {"faking-ttl", 1, 0, OPT_FAKING_TTL}, {"frag", 1, 0, OPT_FRAG}, {"frag-sni-reverse", 1, 0, OPT_FRAG_SNI_REVERSE}, {"frag-sni-faked", 1, 0, OPT_FRAG_SNI_FAKED}, {"seg2delay", 1, 0, OPT_SEG2DELAY}, {"threads", 1, 0, OPT_THREADS}, {"silent", 0, 0, OPT_SILENT}, {"no-gso", 0, 0, OPT_NO_GSO}, {"queue-num", 1, 0, OPT_QUEUE_NUM}, {0,0,0,0} }; static long parse_numeric_option(const char* value) { errno = 0; if (*value == '\0') { errno = EINVAL; return 0; } char* end; long result = strtol(value, &end, 10); if (*end != '\0') { errno = EINVAL; return 0; } return result; } void print_version() { printf("youtubeUnblock\n"); printf("Bypasses youtube detection systems that relies on SNI\n"); } void print_usage(const char *argv0) { print_version(); printf("Usage: %s [ OPTIONS ] \n", argv0); printf("Options:\n"); printf("\t--queue-num=\n"); printf("\t--sni-domains=|all\n"); printf("\t--fake-sni={1|0}\n"); printf("\t--fake-sni-seq-len=\n"); printf("\t--faking-ttl=\n"); printf("\t--faking-strategy={ack,ttl}\n"); printf("\t--frag={tcp,ip,none}\n"); printf("\t--frag-sni-reverse={0|1}\n"); printf("\t--frag-sni-faked={0|1}\n"); printf("\t--seg2delay=\n"); printf("\t--threads=\n"); printf("\t--silent\n"); printf("\t--no-gso\n"); printf("\n"); } int parse_args(int argc, char *argv[]) { int opt; int optIdx; long num; while ((opt = getopt_long(argc, argv, "hv", long_opt, &optIdx)) != -1) { switch (opt) { case 'h': print_usage(argv[0]); goto out; case 'v': print_version(); goto out; case OPT_SILENT: config.verbose = 0; break; case OPT_NO_GSO: config.use_gso = 0; break; case OPT_SNI_DOMAINS: if (strcmp(optarg, "all")) { config.all_domains = 1; } config.domains_str = optarg; config.domains_strlen = strlen(config.domains_str); break; case OPT_FRAG: if (strcmp(optarg, "tcp") == 0) { config.fragmentation_strategy = FRAG_STRAT_TCP; } else if (strcmp(optarg, "ip") == 0) { config.fragmentation_strategy = FRAG_STRAT_IP; } else if (strcmp(optarg, "none") == 0) { config.fragmentation_strategy = FRAG_STRAT_NONE; } else { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } break; case OPT_FRAG_SNI_FAKED: if (strcmp(optarg, "1") == 0) { config.frag_sni_faked = 1; } else if (strcmp(optarg, "0") == 0) { config.frag_sni_faked = 0; } else { errno = EINVAL; printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } break; case OPT_FRAG_SNI_REVERSE: if (strcmp(optarg, "1") == 0) { config.frag_sni_reverse = 1; } else if (strcmp(optarg, "0") == 0) { config.frag_sni_reverse = 0; } else { errno = EINVAL; printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } break; case OPT_FAKING_STRATEGY: if (strcmp(optarg, "ack") == 0) { config.faking_strategy = FAKE_STRAT_ACK_SEQ; } else if (strcmp(optarg, "ttl") == 0) { config.faking_strategy = FAKE_STRAT_TTL; } else { errno = EINVAL; printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } break; case OPT_FAKING_TTL: num = parse_numeric_option(optarg); if (errno != 0 || num < 0 || num > 255) { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } config.faking_ttl = num; break; case OPT_FAKE_SNI: if (strcmp(optarg, "1") == 0) { config.fake_sni = 1; } else if (strcmp(optarg, "0") == 0) { config.fake_sni = 0; } else { errno = EINVAL; printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } break; case OPT_FAKE_SNI_SEQ_LEN: num = parse_numeric_option(optarg); if (errno != 0 || num < 0 || num > 255) { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } config.fake_sni_seq_len = num; break; case OPT_SEG2DELAY: num = parse_numeric_option(optarg); if (errno != 0 || num < 0) { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } config.seg2_delay = num; break; case OPT_THREADS: num = parse_numeric_option(optarg); if (errno != 0 || num < 0 || num > MAX_THREADS) { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } config.threads = num; break; case OPT_QUEUE_NUM: num = parse_numeric_option(optarg); if (errno != 0 || num < 0) { printf("Invalid option %s\n", long_opt[optIdx].name); goto error; } config.queue_start_num = num; break; default: goto error; } } errno = 0; return 0; out: errno = 0; return 1; error: print_usage(argv[0]); errno = EINVAL; return -1; }