| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503 |
- import { describe, expect, it } from 'vitest';
- import {
- parseOutboundLink,
- parseShadowsocksLink,
- parseTrojanLink,
- parseVlessLink,
- parseVmessLink,
- parseHysteria2Link,
- parseWireguardLink,
- } from '@/lib/xray/outbound-link-parser';
- import { Base64 } from '@/utils';
- // Focused acceptance tests for the share-link parsers — one happy-path
- // case per protocol family, plus a few common edge cases. The parsers
- // produce wire-shape outbound rows; the modal hands them to
- // rawOutboundToFormValues to seed Form.useForm.
- describe('parseVmessLink', () => {
- it('parses a vmess:// link with ws + tls', () => {
- const json = {
- v: '2', ps: 'imported-vmess', add: '1.2.3.4', port: 8443,
- id: '11111111-2222-4333-8444-555555555555', aid: 0, scy: 'auto',
- net: 'ws', host: 'example.com', path: '/ws',
- tls: 'tls', sni: 'example.com', fp: 'chrome', alpn: 'h2,http/1.1',
- };
- const link = `vmess://${Base64.encode(JSON.stringify(json))}`;
- const out = parseVmessLink(link);
- expect(out).not.toBeNull();
- expect(out?.protocol).toBe('vmess');
- expect(out?.tag).toBe('imported-vmess');
- const settings = out?.settings as { vnext: Array<{ address: string; port: number; users: Array<{ id: string; security: string }> }> };
- expect(settings.vnext[0].address).toBe('1.2.3.4');
- expect(settings.vnext[0].port).toBe(8443);
- expect(settings.vnext[0].users[0].id).toBe('11111111-2222-4333-8444-555555555555');
- const stream = out?.streamSettings as Record<string, unknown>;
- expect(stream.network).toBe('ws');
- expect(stream.security).toBe('tls');
- expect((stream.wsSettings as Record<string, unknown>).path).toBe('/ws');
- expect((stream.tlsSettings as Record<string, unknown>).serverName).toBe('example.com');
- expect((stream.tlsSettings as Record<string, unknown>).alpn).toEqual(['h2', 'http/1.1']);
- });
- it('returns null for non-vmess links', () => {
- expect(parseVmessLink('vless://x@y:1')).toBeNull();
- });
- it('returns null for malformed base64', () => {
- expect(parseVmessLink('vmess://!!!not-base64!!!')).toBeNull();
- });
- });
- describe('parseVmessLink — XHTTP advanced fields', () => {
- it('round-trips xhttp knobs from the vmess JSON', () => {
- const json = {
- v: '2', ps: 'imported-xhttp', add: '1.2.3.4', port: 443,
- id: '11111111-2222-4333-8444-555555555555', aid: 0, scy: 'auto',
- net: 'xhttp', host: 'edge.example', path: '/sp', mode: 'stream-up',
- xPaddingBytes: '500-1500',
- scMaxEachPostBytes: '2000000',
- scMinPostsIntervalMs: '60',
- uplinkChunkSize: 8192,
- noGRPCHeader: true,
- tls: 'tls', sni: 'edge.example',
- };
- const link = `vmess://${Base64.encode(JSON.stringify(json))}`;
- const out = parseVmessLink(link);
- const stream = out?.streamSettings as Record<string, unknown>;
- const xhttp = stream.xhttpSettings as Record<string, unknown>;
- expect(xhttp.host).toBe('edge.example');
- expect(xhttp.path).toBe('/sp');
- expect(xhttp.mode).toBe('stream-up');
- expect(xhttp.xPaddingBytes).toBe('500-1500');
- expect(xhttp.scMaxEachPostBytes).toBe('2000000');
- expect(xhttp.scMinPostsIntervalMs).toBe('60');
- expect(xhttp.uplinkChunkSize).toBe(8192);
- expect(xhttp.noGRPCHeader).toBe(true);
- });
- it('round-trips xhttp padding-obfs knobs from the vmess JSON', () => {
- const json = {
- v: '2', ps: 'imported-pad', add: '1.2.3.4', port: 443,
- id: '11111111-2222-4333-8444-555555555555', aid: 0, scy: 'auto',
- net: 'xhttp', host: 'edge.example', path: '/sp',
- xPaddingObfsMode: true,
- xPaddingKey: 'secret-key',
- xPaddingHeader: 'X-Pad',
- xPaddingPlacement: 'header',
- xPaddingMethod: 'random',
- sessionKey: 'X-Session',
- seqKey: 'X-Seq',
- noSSEHeader: true,
- scMaxBufferedPosts: 50,
- tls: 'tls',
- };
- const link = `vmess://${Base64.encode(JSON.stringify(json))}`;
- const out = parseVmessLink(link);
- const xhttp = (out?.streamSettings as Record<string, unknown>).xhttpSettings as Record<string, unknown>;
- expect(xhttp.xPaddingObfsMode).toBe(true);
- expect(xhttp.xPaddingKey).toBe('secret-key');
- expect(xhttp.xPaddingHeader).toBe('X-Pad');
- expect(xhttp.xPaddingPlacement).toBe('header');
- expect(xhttp.xPaddingMethod).toBe('random');
- expect(xhttp.sessionKey).toBe('X-Session');
- expect(xhttp.seqKey).toBe('X-Seq');
- expect(xhttp.noSSEHeader).toBe(true);
- expect(xhttp.scMaxBufferedPosts).toBe(50);
- });
- });
- describe('parseVlessLink — XHTTP advanced fields', () => {
- it('round-trips xhttp knobs from URL query params', () => {
- const link
- = 'vless://[email protected]:443'
- + '?type=xhttp&security=tls&host=edge.example&path=%2Fsp&mode=stream-up'
- + '&xPaddingBytes=500-1500&scMaxEachPostBytes=2000000'
- + '&scMinPostsIntervalMs=60&uplinkChunkSize=8192&noGRPCHeader=true'
- + '#imported-xhttp';
- const out = parseVlessLink(link);
- const stream = out?.streamSettings as Record<string, unknown>;
- const xhttp = stream.xhttpSettings as Record<string, unknown>;
- expect(xhttp.host).toBe('edge.example');
- expect(xhttp.path).toBe('/sp');
- expect(xhttp.mode).toBe('stream-up');
- expect(xhttp.xPaddingBytes).toBe('500-1500');
- expect(xhttp.scMaxEachPostBytes).toBe('2000000');
- expect(xhttp.scMinPostsIntervalMs).toBe('60');
- expect(xhttp.uplinkChunkSize).toBe(8192);
- expect(xhttp.noGRPCHeader).toBe(true);
- });
- it('round-trips xhttp padding-obfs knobs from URL query params', () => {
- const link
- = 'vless://[email protected]:443'
- + '?type=xhttp&security=tls&host=edge.example&path=%2Fsp'
- + '&xPaddingObfsMode=true&xPaddingKey=secret-key&xPaddingHeader=X-Pad'
- + '&xPaddingPlacement=header&xPaddingMethod=random'
- + '&sessionKey=X-Session&seqKey=X-Seq&noSSEHeader=true'
- + '&scMaxBufferedPosts=50'
- + '#imported-pad';
- const out = parseVlessLink(link);
- const xhttp = (out?.streamSettings as Record<string, unknown>).xhttpSettings as Record<string, unknown>;
- expect(xhttp.xPaddingObfsMode).toBe(true);
- expect(xhttp.xPaddingKey).toBe('secret-key');
- expect(xhttp.xPaddingHeader).toBe('X-Pad');
- expect(xhttp.xPaddingPlacement).toBe('header');
- expect(xhttp.xPaddingMethod).toBe('random');
- expect(xhttp.sessionKey).toBe('X-Session');
- expect(xhttp.seqKey).toBe('X-Seq');
- expect(xhttp.noSSEHeader).toBe(true);
- expect(xhttp.scMaxBufferedPosts).toBe(50);
- });
- });
- describe('parseVlessLink', () => {
- it('parses a vless:// link with reality', () => {
- const link
- = 'vless://[email protected]:443'
- + '?type=tcp&security=reality&pbk=pubkey&sid=abcd&fp=chrome&sni=cloudflare.com&flow=xtls-rprx-vision'
- + '#imported-vless';
- const out = parseVlessLink(link);
- expect(out?.protocol).toBe('vless');
- expect(out?.tag).toBe('imported-vless');
- const settings = out?.settings as { id: string; flow: string; address: string; port: number };
- expect(settings.id).toBe('11111111-2222-4333-8444-555555555555');
- expect(settings.address).toBe('srv.example');
- expect(settings.port).toBe(443);
- expect(settings.flow).toBe('xtls-rprx-vision');
- const stream = out?.streamSettings as Record<string, unknown>;
- expect(stream.security).toBe('reality');
- const reality = stream.realitySettings as Record<string, unknown>;
- expect(reality.publicKey).toBe('pubkey');
- expect(reality.shortId).toBe('abcd');
- expect(reality.serverName).toBe('cloudflare.com');
- });
- it('parses encryption + pqv (post-quantum) into settings and mldsa65Verify', () => {
- const enc = 'mlkem768x25519plus.native.0rtt.G3cdPSd1-NnlpTbWNSM5vHsT5VNzWfFzYSKwbUMnV1Y';
- const pqv = 'GIsemxbGPjDRH1ONfmoGlVkJ4etNuLmYDvzpjmFFreDLd8WjoJxJ4Fmt_NQJaC6';
- const link
- = 'vless://9406c224-8ac6-4675-ae0b-f93785959418@localhost:1121'
- + `?encryption=${enc}&pqv=${pqv}`
- + '&security=reality&sid=29cf418813d5bac7&sni=aws.amazon.com'
- + '&pbk=aQaGBOT2hMfXWebYtjADoOVUrP8qZRdwXVap7nrId0I&fp=chrome&spx=%2FOUTjB7xHRiP4zBP&type=tcp'
- + '#giqssbgmo9';
- const out = parseVlessLink(link);
- const settings = out?.settings as { encryption: string };
- expect(settings.encryption).toBe(enc);
- const reality = (out?.streamSettings as Record<string, unknown>).realitySettings as Record<string, unknown>;
- expect(reality.mldsa65Verify).toBe(pqv);
- expect(reality.publicKey).toBe('aQaGBOT2hMfXWebYtjADoOVUrP8qZRdwXVap7nrId0I');
- });
- });
- describe('parseTrojanLink', () => {
- it('parses a trojan:// link with ws + tls', () => {
- const link = 'trojan://[email protected]:8443?type=ws&security=tls&host=example.com&path=/tj&sni=example.com#imported-trojan';
- const out = parseTrojanLink(link);
- expect(out?.protocol).toBe('trojan');
- const settings = out?.settings as { servers: Array<{ address: string; port: number; password: string }> };
- expect(settings.servers[0].address).toBe('srv.example');
- expect(settings.servers[0].port).toBe(8443);
- expect(settings.servers[0].password).toBe('secret-pw');
- const stream = out?.streamSettings as Record<string, unknown>;
- expect(stream.network).toBe('ws');
- expect((stream.wsSettings as Record<string, unknown>).path).toBe('/tj');
- });
- });
- describe('parseShadowsocksLink', () => {
- it('parses the modern userinfo@host:port form', () => {
- // ss://base64(method:password)@host:port#remark
- const userinfo = Base64.encode('2022-blake3-aes-128-gcm:supersecret');
- const link = `ss://${userinfo}@1.2.3.4:8388#imported-ss`;
- const out = parseShadowsocksLink(link);
- expect(out?.protocol).toBe('shadowsocks');
- expect(out?.tag).toBe('imported-ss');
- const settings = out?.settings as { servers: Array<{ address: string; port: number; method: string; password: string }> };
- expect(settings.servers[0].address).toBe('1.2.3.4');
- expect(settings.servers[0].port).toBe(8388);
- expect(settings.servers[0].method).toBe('2022-blake3-aes-128-gcm');
- expect(settings.servers[0].password).toBe('supersecret');
- });
- it('keeps the port when the link carries a query string (2022 two-key password)', () => {
- const link = 'ss://MjAyMi1ibGFrZTMtYWVzLTI1Ni1nY206LzhsdFZKaU90azE2QmhKZG9WZVRmSkNNUEJlRGhjcmkycTN0dzU1OUZvYz06YUhuTTB6ZnpFaTdRejc5dzlxNWFFWWVQVnpDU0wxaHV4RnZXZFB6OFZHST0@localhost:30757?type=tcp#pahf4urt53';
- const out = parseShadowsocksLink(link);
- expect(out?.protocol).toBe('shadowsocks');
- expect(out?.tag).toBe('pahf4urt53');
- const settings = out?.settings as { servers: Array<{ address: string; port: number; method: string; password: string }> };
- expect(settings.servers[0].address).toBe('localhost');
- expect(settings.servers[0].port).toBe(30757);
- expect(settings.servers[0].method).toBe('2022-blake3-aes-256-gcm');
- expect(settings.servers[0].password).toBe('/8ltVJiOtk16BhJdoVeTfJCMPBeDhcri2q3tw559Foc=:aHnM0zfzEi7Qz79w9q5aEYePVzCSL1huxFvWdPz8VGI=');
- });
- it('parses the legacy base64-of-whole form', () => {
- // ss://base64(method:password@host:port)#remark
- const inner = Base64.encode('aes-256-gcm:[email protected]:1080');
- const link = `ss://${inner}#imported-legacy`;
- const out = parseShadowsocksLink(link);
- const settings = out?.settings as { servers: Array<{ address: string; port: number; method: string; password: string }> };
- expect(settings.servers[0].address).toBe('10.0.0.1');
- expect(settings.servers[0].port).toBe(1080);
- expect(settings.servers[0].method).toBe('aes-256-gcm');
- expect(settings.servers[0].password).toBe('legacypw');
- });
- });
- describe('parseHysteria2Link', () => {
- it('parses a hysteria2:// link with sni', () => {
- const link = 'hysteria2://[email protected]:443?sni=example.com#imported-hy2';
- const out = parseHysteria2Link(link);
- expect(out?.protocol).toBe('hysteria');
- expect(out?.tag).toBe('imported-hy2');
- const settings = out?.settings as { address: string; port: number; version: number };
- expect(settings.address).toBe('srv.example');
- expect(settings.port).toBe(443);
- expect(settings.version).toBe(2);
- const stream = out?.streamSettings as Record<string, unknown>;
- const hys = stream.hysteriaSettings as Record<string, unknown>;
- expect(hys.auth).toBe('auth-secret');
- expect((stream.tlsSettings as Record<string, unknown>).serverName).toBe('example.com');
- });
- it('also accepts hy2:// alias', () => {
- const out = parseHysteria2Link('hy2://auth@srv:443?sni=example.com');
- expect(out?.protocol).toBe('hysteria');
- });
- it('parses alpn, fingerprint and the salamander UDP mask (fm) — #4760', () => {
- const link = 'hysteria2://[email protected]:8443?'
- + 'alpn=h2%2Chttp%2F1.1&'
- + 'fm=%7B%22udp%22%3A%5B%7B%22settings%22%3A%7B%22password%22%3A%22ftwfgb9655hh2mgo%22%7D%2C%22type%22%3A%22salamander%22%7D%5D%7D&'
- + 'fp=chrome&obfs=salamander&obfs-password=655hh2mgo&security=tls&sni=news.domain.org'
- + '#hy2-ej596ty350qs';
- const out = parseHysteria2Link(link);
- expect(out).not.toBeNull();
- const stream = out!.streamSettings as Record<string, unknown>;
- const tls = stream.tlsSettings as Record<string, unknown>;
- expect(tls.alpn).toEqual(['h2', 'http/1.1']);
- expect(tls.fingerprint).toBe('chrome');
- expect(tls.serverName).toBe('news.domain.org');
- const finalmask = stream.finalmask as Record<string, unknown>;
- expect(finalmask).toBeDefined();
- const udp = finalmask.udp as Array<Record<string, unknown>>;
- expect(udp[0].type).toBe('salamander');
- expect((udp[0].settings as Record<string, unknown>).password).toBe('ftwfgb9655hh2mgo');
- });
- it('round-trips the salamander packetSize (Gecko) under fm', () => {
- const fm = encodeURIComponent(JSON.stringify({
- udp: [{ type: 'salamander', settings: { password: 'ftwfgb9655hh2mgo', packetSize: '100-200' } }],
- }));
- const link = `hysteria2://[email protected]:8443?security=tls&sni=news.domain.org&fm=${fm}#hy2-gecko`;
- const out = parseHysteria2Link(link);
- expect(out).not.toBeNull();
- const finalmask = (out!.streamSettings as Record<string, unknown>).finalmask as Record<string, unknown>;
- const udp = finalmask.udp as Array<Record<string, unknown>>;
- const settings = udp[0].settings as Record<string, unknown>;
- expect(udp[0].type).toBe('salamander');
- expect(settings.password).toBe('ftwfgb9655hh2mgo');
- expect(settings.packetSize).toBe('100-200');
- });
- it('round-trips the realm tlsConfig under fm', () => {
- const fm = encodeURIComponent(JSON.stringify({
- udp: [{
- type: 'realm',
- settings: {
- url: 'realm://[email protected]/my-realm',
- stunServers: ['stun.l.google.com:19302'],
- tlsConfig: { serverName: 'example.com', alpn: ['h3'], fingerprint: 'chrome', allowInsecure: false },
- },
- }],
- }));
- const link = `hysteria2://auth@srv:443?security=tls&sni=srv&fm=${fm}#hy2-realm`;
- const out = parseHysteria2Link(link);
- expect(out).not.toBeNull();
- const finalmask = (out!.streamSettings as Record<string, unknown>).finalmask as Record<string, unknown>;
- const udp = finalmask.udp as Array<Record<string, unknown>>;
- const settings = udp[0].settings as Record<string, unknown>;
- expect(udp[0].type).toBe('realm');
- expect(settings.url).toBe('realm://[email protected]/my-realm');
- const tlsConfig = settings.tlsConfig as Record<string, unknown>;
- expect(tlsConfig.serverName).toBe('example.com');
- expect(tlsConfig.alpn).toEqual(['h3']);
- expect(tlsConfig.fingerprint).toBe('chrome');
- });
- it('defaults alpn to h3 when the link omits it', () => {
- const out = parseHysteria2Link('hysteria2://auth@srv:443?sni=example.com');
- const tls = (out!.streamSettings as Record<string, unknown>).tlsSettings as Record<string, unknown>;
- expect(tls.alpn).toEqual(['h3']);
- });
- });
- describe('parseVlessLink — extra / fm / x_padding_bytes (B20)', () => {
- it('round-trips a real inbound-generated link with extra+fm+reality+xhttp', () => {
- // Real user-reported link — bundled xhttp knobs via `extra` JSON,
- // full finalmask via `fm` JSON, reality auth, snake_case
- // x_padding_bytes alias. All three parse-paths must combine.
- const link = 'vless://b622ac2f-f155-47db-a3b2-b64e8d7f6342@localhost:37723?'
- + 'encryption=none&'
- + 'extra=%7B%22scMaxEachPostBytes%22%3A%221000000%22%2C%22scMinPostsIntervalMs%22%3A%2230%22%2C%22xPaddingBytes%22%3A%22100-1000%22%7D&'
- + 'fm=%7B%22quicParams%22%3A%7B%22congestion%22%3A%22bbr%22%2C%22maxIdleTimeout%22%3A30%2C%22udpHop%22%3A%7B%22interval%22%3A%225-10%22%2C%22ports%22%3A%2220000-50000%22%7D%7D%7D&'
- + 'fp=chrome&host=&mode=auto&path=%2F&'
- + 'pbk=nJw4k4CPf5jf64V8nnDwWa8iClDnUvQ1lCI4iKzfJ0o&'
- + 'security=reality&sid=14ebccc4d3&sni=aws.amazon.com&'
- + 'spx=%2F97L2FjycXEwrE67&type=xhttp&x_padding_bytes=100-1000'
- + '#sda-8ud3us6rt';
- const parsed = parseVlessLink(link);
- expect(parsed).not.toBeNull();
- expect(parsed!.tag).toBe('sda-8ud3us6rt');
- const stream = parsed!.streamSettings as Record<string, unknown>;
- expect(stream.network).toBe('xhttp');
- expect(stream.security).toBe('reality');
- const xhttp = stream.xhttpSettings as Record<string, unknown>;
- expect(xhttp.xPaddingBytes).toBe('100-1000');
- expect(xhttp.scMaxEachPostBytes).toBe('1000000');
- expect(xhttp.scMinPostsIntervalMs).toBe('30');
- const reality = stream.realitySettings as Record<string, unknown>;
- expect(reality.publicKey).toBe('nJw4k4CPf5jf64V8nnDwWa8iClDnUvQ1lCI4iKzfJ0o');
- expect(reality.shortId).toBe('14ebccc4d3');
- expect(reality.spiderX).toBe('/97L2FjycXEwrE67');
- expect(reality.serverName).toBe('aws.amazon.com');
- const finalmask = stream.finalmask as Record<string, unknown>;
- expect(finalmask).toBeDefined();
- const quicParams = finalmask.quicParams as Record<string, unknown>;
- expect(quicParams.congestion).toBe('bbr');
- expect(quicParams.maxIdleTimeout).toBe(30);
- expect((quicParams.udpHop as Record<string, unknown>).interval).toBe('5-10');
- expect((quicParams.udpHop as Record<string, unknown>).ports).toBe('20000-50000');
- });
- it('falls back to x_padding_bytes when extra has no xPaddingBytes', () => {
- const link = 'vless://u@h:1?type=xhttp&security=none&path=%2F&host=&mode=auto&x_padding_bytes=200-2000#t';
- const parsed = parseVlessLink(link);
- const xhttp = (parsed!.streamSettings as Record<string, unknown>).xhttpSettings as Record<string, unknown>;
- expect(xhttp.xPaddingBytes).toBe('200-2000');
- });
- it('extra takes precedence — camelCase wins over snake_case alias', () => {
- const link = 'vless://u@h:1?type=xhttp&security=none&path=%2F&host=&mode=auto'
- + '&xPaddingBytes=900-9000&x_padding_bytes=100-1000#t';
- const parsed = parseVlessLink(link);
- const xhttp = (parsed!.streamSettings as Record<string, unknown>).xhttpSettings as Record<string, unknown>;
- expect(xhttp.xPaddingBytes).toBe('900-9000');
- });
- it('extracts the nested xmux object from the extra JSON blob', () => {
- // The inbound link bundles xmux into `extra` as a nested object
- // (sub/service.go). It must survive import so the outbound form's
- // XMUX sub-form populates rather than silently dropping it (#5353).
- const extra = encodeURIComponent(JSON.stringify({
- xmux: { maxConcurrency: '8-16', hMaxRequestTimes: '700-1000' },
- }));
- const link = 'vless://u@h:1?type=xhttp&security=none&path=%2F&host=&mode=auto'
- + '&extra=' + extra + '#t';
- const parsed = parseVlessLink(link);
- const xhttp = (parsed!.streamSettings as Record<string, unknown>).xhttpSettings as Record<string, unknown>;
- const xmux = xhttp.xmux as Record<string, unknown>;
- expect(xmux).toBeDefined();
- expect(xmux.maxConcurrency).toBe('8-16');
- expect(xmux.hMaxRequestTimes).toBe('700-1000');
- });
- it('ignores malformed extra JSON without breaking the rest of the link', () => {
- const link = 'vless://u@h:1?type=xhttp&security=none&path=%2F&host=&mode=auto'
- + '&extra=not-json&fp=chrome#t';
- const parsed = parseVlessLink(link);
- expect(parsed).not.toBeNull();
- const stream = parsed!.streamSettings as Record<string, unknown>;
- expect((stream.xhttpSettings as Record<string, unknown>).mode).toBe('auto');
- });
- it('round-trips ech and pcs from a TLS vless link', () => {
- const ech = 'AFb+DQBSAAAgACAL7gYwrvaSFCIEs34G3SkfpuIbjMuYQxAiJsPK1oO7cwAkAAEAAQABAAIAAQADAAIAAQACAAIAAgADAAMAAQADAAIAAwADAAMxMjMAAA==';
- const pcs = '6fbc15ba46dfed152ad6c8d2129dd774707dd667a9ab4965476fa0f79ba82670';
- const link = 'vless://e3d307ae-c074-4aa3-af08-4f9e0f1d298b@localhost:15282?'
- + 'alpn=h3&ech=' + encodeURIComponent(ech) + '&encryption=none&fp=firefox&host=&'
- + 'mode=packet-up&path=%2F&pcs=' + pcs + '&security=tls&sni=123&type=xhttp#i5sboxj07w';
- const parsed = parseVlessLink(link);
- expect(parsed).not.toBeNull();
- const tls = (parsed!.streamSettings as Record<string, unknown>).tlsSettings as Record<string, unknown>;
- expect(tls.echConfigList).toBe(ech);
- expect(tls.pinnedPeerCertSha256).toBe(pcs);
- expect(tls.serverName).toBe('123');
- expect(tls.fingerprint).toBe('firefox');
- });
- });
- describe('parseWireguardLink', () => {
- it('parses a wireguard:// link with percent-encoded secret and publickey', () => {
- const link = 'wireguard://IKeuy2+BNspvMffiC47z16seLIGxGtbDIYiZcbh9C1U%3D@localhost:22824'
- + '?publickey=3CnNsCy74TOlupjaii%2BRFp%2FgDMk5vvUuFD0SNZ%2FGl2s%3D'
- + '&address=10.0.0.2%2F32&mtu=1420#-1';
- const out = parseWireguardLink(link);
- expect(out?.protocol).toBe('wireguard');
- expect(out?.tag).toBe('-1');
- const settings = out?.settings as {
- secretKey: string; address: string[]; mtu: number;
- peers: Array<{ publicKey: string; endpoint: string; allowedIPs: string[] }>;
- };
- expect(settings.secretKey).toBe('IKeuy2+BNspvMffiC47z16seLIGxGtbDIYiZcbh9C1U=');
- expect(settings.address).toEqual(['10.0.0.2/32']);
- expect(settings.mtu).toBe(1420);
- expect(settings.peers[0].publicKey).toBe('3CnNsCy74TOlupjaii+RFp/gDMk5vvUuFD0SNZ/Gl2s=');
- expect(settings.peers[0].endpoint).toBe('localhost:22824');
- expect(settings.peers[0].allowedIPs).toEqual(['0.0.0.0/0', '::/0']);
- });
- it('parses reserved, presharedkey and keepalive aliases', () => {
- const link = 'wireguard://[email protected]:51820'
- + '?publickey=peerpub&address=10.0.0.2/32,fd00::2/128'
- + '&reserved=1,2,3&presharedkey=psk-secret&persistentkeepalive=25'
- + '&allowedips=0.0.0.0/0#wg-peer';
- const out = parseWireguardLink(link);
- const settings = out?.settings as {
- reserved: number[];
- peers: Array<{ preSharedKey: string; keepAlive: number; allowedIPs: string[] }>;
- address: string[];
- };
- expect(settings.address).toEqual(['10.0.0.2/32', 'fd00::2/128']);
- expect(settings.reserved).toEqual([1, 2, 3]);
- expect(settings.peers[0].preSharedKey).toBe('psk-secret');
- expect(settings.peers[0].keepAlive).toBe(25);
- expect(settings.peers[0].allowedIPs).toEqual(['0.0.0.0/0']);
- });
- it('returns null for non-wireguard links', () => {
- expect(parseWireguardLink('vless://x@y:1')).toBeNull();
- });
- });
- describe('parseOutboundLink dispatcher', () => {
- it('dispatches vmess via base64 JSON', () => {
- const json = { v: '2', ps: 'x', add: '1.1.1.1', port: 443, id: '11111111-2222-4333-8444-555555555555', net: 'tcp', tls: 'none' };
- const link = `vmess://${Base64.encode(JSON.stringify(json))}`;
- expect(parseOutboundLink(link)?.protocol).toBe('vmess');
- });
- it('dispatches vless via URL', () => {
- expect(parseOutboundLink('vless://uuid@host:443?type=tcp&security=none')?.protocol).toBe('vless');
- });
- it('dispatches wireguard via URL', () => {
- expect(parseOutboundLink('wireguard://pk@host:22824?publickey=pub&address=10.0.0.2/32')?.protocol).toBe('wireguard');
- });
- it('returns null for an unknown scheme', () => {
- expect(parseOutboundLink('socks5://user:pass@host:1080')).toBeNull();
- });
- it('returns null for empty input', () => {
- expect(parseOutboundLink('')).toBeNull();
- expect(parseOutboundLink(' ')).toBeNull();
- });
- });
|