| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103 |
- const Protocols = {
- Freedom: "freedom",
- Blackhole: "blackhole",
- DNS: "dns",
- VMess: "vmess",
- VLESS: "vless",
- Trojan: "trojan",
- Shadowsocks: "shadowsocks",
- Wireguard: "wireguard",
- Hysteria: "hysteria",
- Socks: "socks",
- HTTP: "http",
- };
- const SSMethods = {
- AES_256_GCM: 'aes-256-gcm',
- AES_128_GCM: 'aes-128-gcm',
- CHACHA20_POLY1305: 'chacha20-poly1305',
- CHACHA20_IETF_POLY1305: 'chacha20-ietf-poly1305',
- XCHACHA20_POLY1305: 'xchacha20-poly1305',
- XCHACHA20_IETF_POLY1305: 'xchacha20-ietf-poly1305',
- BLAKE3_AES_128_GCM: '2022-blake3-aes-128-gcm',
- BLAKE3_AES_256_GCM: '2022-blake3-aes-256-gcm',
- BLAKE3_CHACHA20_POLY1305: '2022-blake3-chacha20-poly1305',
- };
- const TLS_FLOW_CONTROL = {
- VISION: "xtls-rprx-vision",
- VISION_UDP443: "xtls-rprx-vision-udp443",
- };
- const UTLS_FINGERPRINT = {
- UTLS_CHROME: "chrome",
- UTLS_FIREFOX: "firefox",
- UTLS_SAFARI: "safari",
- UTLS_IOS: "ios",
- UTLS_android: "android",
- UTLS_EDGE: "edge",
- UTLS_360: "360",
- UTLS_QQ: "qq",
- UTLS_RANDOM: "random",
- UTLS_RANDOMIZED: "randomized",
- UTLS_RONDOMIZEDNOALPN: "randomizednoalpn",
- UTLS_UNSAFE: "unsafe",
- };
- const ALPN_OPTION = {
- H3: "h3",
- H2: "h2",
- HTTP1: "http/1.1",
- };
- const SNIFFING_OPTION = {
- HTTP: "http",
- TLS: "tls",
- QUIC: "quic",
- FAKEDNS: "fakedns"
- };
- const OutboundDomainStrategies = [
- "AsIs",
- "UseIP",
- "UseIPv4",
- "UseIPv6",
- "UseIPv6v4",
- "UseIPv4v6",
- "ForceIP",
- "ForceIPv6v4",
- "ForceIPv6",
- "ForceIPv4v6",
- "ForceIPv4"
- ];
- const WireguardDomainStrategy = [
- "ForceIP",
- "ForceIPv4",
- "ForceIPv4v6",
- "ForceIPv6",
- "ForceIPv6v4"
- ];
- const USERS_SECURITY = {
- AES_128_GCM: "aes-128-gcm",
- CHACHA20_POLY1305: "chacha20-poly1305",
- AUTO: "auto",
- NONE: "none",
- ZERO: "zero",
- };
- const MODE_OPTION = {
- AUTO: "auto",
- PACKET_UP: "packet-up",
- STREAM_UP: "stream-up",
- STREAM_ONE: "stream-one",
- };
- const Address_Port_Strategy = {
- NONE: "none",
- SrvPortOnly: "srvportonly",
- SrvAddressOnly: "srvaddressonly",
- SrvPortAndAddress: "srvportandaddress",
- TxtPortOnly: "txtportonly",
- TxtAddressOnly: "txtaddressonly",
- TxtPortAndAddress: "txtportandaddress"
- };
- const DNSRuleActions = ['direct', 'drop', 'reject', 'hijack'];
- function normalizeDNSRuleField(value) {
- if (value === null || value === undefined) {
- return '';
- }
- if (Array.isArray(value)) {
- return value.map(item => item.toString().trim()).filter(item => item.length > 0).join(',');
- }
- return value.toString().trim();
- }
- function normalizeDNSRuleAction(action) {
- action = ObjectUtil.isEmpty(action) ? 'direct' : action.toString().toLowerCase().trim();
- return DNSRuleActions.includes(action) ? action : 'direct';
- }
- function parseLegacyDNSBlockTypes(blockTypes) {
- if (blockTypes === null || blockTypes === undefined || blockTypes === '') {
- return [];
- }
- if (Array.isArray(blockTypes)) {
- return blockTypes
- .map(item => Number(item))
- .filter(item => Number.isInteger(item) && item >= 0 && item <= 65535);
- }
- if (typeof blockTypes === 'number') {
- return Number.isInteger(blockTypes) && blockTypes >= 0 && blockTypes <= 65535 ? [blockTypes] : [];
- }
- return blockTypes
- .toString()
- .split(',')
- .map(item => item.trim())
- .filter(item => /^\d+$/.test(item))
- .map(item => Number(item))
- .filter(item => item >= 0 && item <= 65535);
- }
- function buildLegacyDNSRules(nonIPQuery, blockTypes) {
- const mode = ['reject', 'drop', 'skip'].includes(nonIPQuery) ? nonIPQuery : 'reject';
- const rules = [];
- const parsedBlockTypes = parseLegacyDNSBlockTypes(blockTypes);
- if (parsedBlockTypes.length > 0) {
- rules.push(new Outbound.DNSRule(mode === 'reject' ? 'reject' : 'drop', parsedBlockTypes.join(',')));
- }
- rules.push(new Outbound.DNSRule('hijack', '1,28'));
- rules.push(new Outbound.DNSRule(mode === 'skip' ? 'direct' : mode));
- return rules;
- }
- function getDNSRulesFromJson(json = {}) {
- if (Array.isArray(json.rules) && json.rules.length > 0) {
- return json.rules.map(rule => Outbound.DNSRule.fromJson(rule));
- }
- if (json.nonIPQuery !== undefined || json.blockTypes !== undefined) {
- return buildLegacyDNSRules(json.nonIPQuery, json.blockTypes);
- }
- return [];
- }
- Object.freeze(Protocols);
- Object.freeze(SSMethods);
- Object.freeze(TLS_FLOW_CONTROL);
- Object.freeze(UTLS_FINGERPRINT);
- Object.freeze(ALPN_OPTION);
- Object.freeze(SNIFFING_OPTION);
- Object.freeze(OutboundDomainStrategies);
- Object.freeze(WireguardDomainStrategy);
- Object.freeze(USERS_SECURITY);
- Object.freeze(MODE_OPTION);
- Object.freeze(Address_Port_Strategy);
- Object.freeze(DNSRuleActions);
- class CommonClass {
- static toJsonArray(arr) {
- return arr.map(obj => obj.toJson());
- }
- static fromJson() {
- return new CommonClass();
- }
- toJson() {
- return this;
- }
- toString(format = true) {
- return format ? JSON.stringify(this.toJson(), null, 2) : JSON.stringify(this.toJson());
- }
- }
- class ReverseSniffing extends CommonClass {
- constructor(
- enabled = false,
- destOverride = ['http', 'tls', 'quic', 'fakedns'],
- metadataOnly = false,
- routeOnly = false,
- ipsExcluded = [],
- domainsExcluded = [],
- ) {
- super();
- this.enabled = enabled;
- this.destOverride = Array.isArray(destOverride) && destOverride.length > 0 ? destOverride : ['http', 'tls', 'quic', 'fakedns'];
- this.metadataOnly = metadataOnly;
- this.routeOnly = routeOnly;
- this.ipsExcluded = Array.isArray(ipsExcluded) ? ipsExcluded : [];
- this.domainsExcluded = Array.isArray(domainsExcluded) ? domainsExcluded : [];
- }
- static fromJson(json = {}) {
- if (!json || Object.keys(json).length === 0) {
- return new ReverseSniffing();
- }
- return new ReverseSniffing(
- !!json.enabled,
- json.destOverride,
- json.metadataOnly,
- json.routeOnly,
- json.ipsExcluded || [],
- json.domainsExcluded || [],
- );
- }
- toJson() {
- return {
- enabled: this.enabled,
- destOverride: this.destOverride,
- metadataOnly: this.metadataOnly,
- routeOnly: this.routeOnly,
- ipsExcluded: this.ipsExcluded.length > 0 ? this.ipsExcluded : undefined,
- domainsExcluded: this.domainsExcluded.length > 0 ? this.domainsExcluded : undefined,
- };
- }
- }
- class TcpStreamSettings extends CommonClass {
- constructor(type = 'none', host, path) {
- super();
- this.type = type;
- this.host = host;
- this.path = path;
- }
- static fromJson(json = {}) {
- let header = json.header;
- if (!header) return new TcpStreamSettings();
- if (header.type == 'http' && header.request) {
- return new TcpStreamSettings(
- header.type,
- header.request.headers.Host.join(','),
- header.request.path.join(','),
- );
- }
- return new TcpStreamSettings(header.type, '', '');
- }
- toJson() {
- return {
- header: {
- type: this.type,
- request: this.type === 'http' ? {
- headers: {
- Host: ObjectUtil.isEmpty(this.host) ? [] : this.host.split(',')
- },
- path: ObjectUtil.isEmpty(this.path) ? ["/"] : this.path.split(',')
- } : undefined,
- }
- };
- }
- }
- class KcpStreamSettings extends CommonClass {
- constructor(
- mtu = 1350,
- tti = 20,
- uplinkCapacity = 5,
- downlinkCapacity = 20,
- cwndMultiplier = 1,
- maxSendingWindow = 1350,
- ) {
- super();
- this.mtu = mtu;
- this.tti = tti;
- this.upCap = uplinkCapacity;
- this.downCap = downlinkCapacity;
- this.cwndMultiplier = cwndMultiplier;
- this.maxSendingWindow = maxSendingWindow;
- }
- static fromJson(json = {}) {
- return new KcpStreamSettings(
- json.mtu,
- json.tti,
- json.uplinkCapacity,
- json.downlinkCapacity,
- json.cwndMultiplier,
- json.maxSendingWindow,
- );
- }
- toJson() {
- return {
- mtu: this.mtu,
- tti: this.tti,
- uplinkCapacity: this.upCap,
- downlinkCapacity: this.downCap,
- cwndMultiplier: this.cwndMultiplier,
- maxSendingWindow: this.maxSendingWindow,
- };
- }
- }
- class WsStreamSettings extends CommonClass {
- constructor(
- path = '/',
- host = '',
- heartbeatPeriod = 0,
- ) {
- super();
- this.path = path;
- this.host = host;
- this.heartbeatPeriod = heartbeatPeriod;
- }
- static fromJson(json = {}) {
- return new WsStreamSettings(
- json.path,
- json.host,
- json.heartbeatPeriod,
- );
- }
- toJson() {
- return {
- path: this.path,
- host: this.host,
- heartbeatPeriod: this.heartbeatPeriod
- };
- }
- }
- class GrpcStreamSettings extends CommonClass {
- constructor(
- serviceName = "",
- authority = "",
- multiMode = false
- ) {
- super();
- this.serviceName = serviceName;
- this.authority = authority;
- this.multiMode = multiMode;
- }
- static fromJson(json = {}) {
- return new GrpcStreamSettings(json.serviceName, json.authority, json.multiMode);
- }
- toJson() {
- return {
- serviceName: this.serviceName,
- authority: this.authority,
- multiMode: this.multiMode
- }
- }
- }
- class HttpUpgradeStreamSettings extends CommonClass {
- constructor(path = '/', host = '') {
- super();
- this.path = path;
- this.host = host;
- }
- static fromJson(json = {}) {
- return new HttpUpgradeStreamSettings(
- json.path,
- json.host,
- );
- }
- toJson() {
- return {
- path: this.path,
- host: this.host,
- };
- }
- }
- class xHTTPStreamSettings extends CommonClass {
- constructor(
- path = '/',
- host = '',
- mode = '',
- noGRPCHeader = false,
- scMinPostsIntervalMs = "30",
- xmux = {
- maxConcurrency: "16-32",
- maxConnections: 0,
- cMaxReuseTimes: 0,
- hMaxRequestTimes: "600-900",
- hMaxReusableSecs: "1800-3000",
- hKeepAlivePeriod: 0,
- },
- ) {
- super();
- this.path = path;
- this.host = host;
- this.mode = mode;
- this.noGRPCHeader = noGRPCHeader;
- this.scMinPostsIntervalMs = scMinPostsIntervalMs;
- this.xmux = xmux;
- }
- static fromJson(json = {}) {
- return new xHTTPStreamSettings(
- json.path,
- json.host,
- json.mode,
- json.noGRPCHeader,
- json.scMinPostsIntervalMs,
- json.xmux
- );
- }
- toJson() {
- return {
- path: this.path,
- host: this.host,
- mode: this.mode,
- noGRPCHeader: this.noGRPCHeader,
- scMinPostsIntervalMs: this.scMinPostsIntervalMs,
- xmux: {
- maxConcurrency: this.xmux.maxConcurrency,
- maxConnections: this.xmux.maxConnections,
- cMaxReuseTimes: this.xmux.cMaxReuseTimes,
- hMaxRequestTimes: this.xmux.hMaxRequestTimes,
- hMaxReusableSecs: this.xmux.hMaxReusableSecs,
- hKeepAlivePeriod: this.xmux.hKeepAlivePeriod,
- },
- };
- }
- }
- class TlsStreamSettings extends CommonClass {
- constructor(
- serverName = '',
- alpn = [],
- fingerprint = '',
- echConfigList = '',
- verifyPeerCertByName = '',
- pinnedPeerCertSha256 = '',
- ) {
- super();
- this.serverName = serverName;
- this.alpn = alpn;
- this.fingerprint = fingerprint;
- this.echConfigList = echConfigList;
- this.verifyPeerCertByName = verifyPeerCertByName;
- this.pinnedPeerCertSha256 = pinnedPeerCertSha256;
- }
- static fromJson(json = {}) {
- return new TlsStreamSettings(
- json.serverName,
- json.alpn,
- json.fingerprint,
- json.echConfigList,
- json.verifyPeerCertByName,
- json.pinnedPeerCertSha256,
- );
- }
- toJson() {
- return {
- serverName: this.serverName,
- alpn: this.alpn,
- fingerprint: this.fingerprint,
- echConfigList: this.echConfigList,
- verifyPeerCertByName: this.verifyPeerCertByName,
- pinnedPeerCertSha256: this.pinnedPeerCertSha256
- };
- }
- }
- class RealityStreamSettings extends CommonClass {
- constructor(
- publicKey = '',
- fingerprint = '',
- serverName = '',
- shortId = '',
- spiderX = '',
- mldsa65Verify = ''
- ) {
- super();
- this.publicKey = publicKey;
- this.fingerprint = fingerprint;
- this.serverName = serverName;
- this.shortId = shortId
- this.spiderX = spiderX;
- this.mldsa65Verify = mldsa65Verify;
- }
- static fromJson(json = {}) {
- return new RealityStreamSettings(
- json.publicKey,
- json.fingerprint,
- json.serverName,
- json.shortId,
- json.spiderX,
- json.mldsa65Verify
- );
- }
- toJson() {
- return {
- publicKey: this.publicKey,
- fingerprint: this.fingerprint,
- serverName: this.serverName,
- shortId: this.shortId,
- spiderX: this.spiderX,
- mldsa65Verify: this.mldsa65Verify
- };
- }
- };
- class HysteriaStreamSettings extends CommonClass {
- constructor(
- version = 2,
- auth = '',
- congestion = '',
- up = '0',
- down = '0',
- udphopPort = '',
- udphopIntervalMin = 30,
- udphopIntervalMax = 30,
- initStreamReceiveWindow = 8388608,
- maxStreamReceiveWindow = 8388608,
- initConnectionReceiveWindow = 20971520,
- maxConnectionReceiveWindow = 20971520,
- maxIdleTimeout = 30,
- keepAlivePeriod = 2,
- disablePathMTUDiscovery = false
- ) {
- super();
- this.version = version;
- this.auth = auth;
- this.congestion = congestion;
- this.up = up;
- this.down = down;
- this.udphopPort = udphopPort;
- this.udphopIntervalMin = udphopIntervalMin;
- this.udphopIntervalMax = udphopIntervalMax;
- this.initStreamReceiveWindow = initStreamReceiveWindow;
- this.maxStreamReceiveWindow = maxStreamReceiveWindow;
- this.initConnectionReceiveWindow = initConnectionReceiveWindow;
- this.maxConnectionReceiveWindow = maxConnectionReceiveWindow;
- this.maxIdleTimeout = maxIdleTimeout;
- this.keepAlivePeriod = keepAlivePeriod;
- this.disablePathMTUDiscovery = disablePathMTUDiscovery;
- }
- static fromJson(json = {}) {
- let udphopPort = '';
- let udphopIntervalMin = 30;
- let udphopIntervalMax = 30;
- if (json.udphop) {
- udphopPort = json.udphop.port || '';
- // Backward compatibility: if old 'interval' exists, use it for both min/max
- if (json.udphop.interval !== undefined) {
- udphopIntervalMin = json.udphop.interval;
- udphopIntervalMax = json.udphop.interval;
- } else {
- udphopIntervalMin = json.udphop.intervalMin || 30;
- udphopIntervalMax = json.udphop.intervalMax || 30;
- }
- }
- return new HysteriaStreamSettings(
- json.version,
- json.auth,
- json.congestion,
- json.up,
- json.down,
- udphopPort,
- udphopIntervalMin,
- udphopIntervalMax,
- json.initStreamReceiveWindow,
- json.maxStreamReceiveWindow,
- json.initConnectionReceiveWindow,
- json.maxConnectionReceiveWindow,
- json.maxIdleTimeout,
- json.keepAlivePeriod,
- json.disablePathMTUDiscovery
- );
- }
- toJson() {
- const result = {
- version: this.version,
- auth: this.auth,
- congestion: this.congestion,
- up: this.up,
- down: this.down,
- initStreamReceiveWindow: this.initStreamReceiveWindow,
- maxStreamReceiveWindow: this.maxStreamReceiveWindow,
- initConnectionReceiveWindow: this.initConnectionReceiveWindow,
- maxConnectionReceiveWindow: this.maxConnectionReceiveWindow,
- maxIdleTimeout: this.maxIdleTimeout,
- keepAlivePeriod: this.keepAlivePeriod,
- disablePathMTUDiscovery: this.disablePathMTUDiscovery
- };
- if (this.udphopPort) {
- result.udphop = {
- port: this.udphopPort,
- intervalMin: this.udphopIntervalMin,
- intervalMax: this.udphopIntervalMax
- };
- }
- return result;
- }
- };
- class SockoptStreamSettings extends CommonClass {
- constructor(
- dialerProxy = "",
- tcpFastOpen = false,
- tcpKeepAliveInterval = 0,
- tcpMptcp = false,
- penetrate = false,
- addressPortStrategy = Address_Port_Strategy.NONE,
- trustedXForwardedFor = [],
- ) {
- super();
- this.dialerProxy = dialerProxy;
- this.tcpFastOpen = tcpFastOpen;
- this.tcpKeepAliveInterval = tcpKeepAliveInterval;
- this.tcpMptcp = tcpMptcp;
- this.penetrate = penetrate;
- this.addressPortStrategy = addressPortStrategy;
- this.trustedXForwardedFor = trustedXForwardedFor;
- }
- static fromJson(json = {}) {
- if (Object.keys(json).length === 0) return undefined;
- return new SockoptStreamSettings(
- json.dialerProxy,
- json.tcpFastOpen,
- json.tcpKeepAliveInterval,
- json.tcpMptcp,
- json.penetrate,
- json.addressPortStrategy,
- json.trustedXForwardedFor || []
- );
- }
- toJson() {
- const result = {
- dialerProxy: this.dialerProxy,
- tcpFastOpen: this.tcpFastOpen,
- tcpKeepAliveInterval: this.tcpKeepAliveInterval,
- tcpMptcp: this.tcpMptcp,
- penetrate: this.penetrate,
- addressPortStrategy: this.addressPortStrategy
- };
- if (this.trustedXForwardedFor && this.trustedXForwardedFor.length > 0) {
- result.trustedXForwardedFor = this.trustedXForwardedFor;
- }
- return result;
- }
- }
- class UdpMask extends CommonClass {
- constructor(type = 'salamander', settings = {}) {
- super();
- this.type = type;
- this.settings = this._getDefaultSettings(type, settings);
- }
- _getDefaultSettings(type, settings = {}) {
- switch (type) {
- case 'salamander':
- case 'mkcp-aes128gcm':
- return { password: settings.password || '' };
- case 'header-dns':
- return { domain: settings.domain || '' };
- case 'xdns':
- return { resolvers: Array.isArray(settings.resolvers) ? settings.resolvers : [] };
- case 'xicmp':
- return { ip: settings.ip || '', id: settings.id ?? 0 };
- case 'mkcp-original':
- case 'header-dtls':
- case 'header-srtp':
- case 'header-utp':
- case 'header-wechat':
- case 'header-wireguard':
- return {}; // No settings needed
- case 'header-custom':
- return {
- client: Array.isArray(settings.client) ? settings.client : [],
- server: Array.isArray(settings.server) ? settings.server : [],
- };
- case 'noise':
- return {
- reset: settings.reset ?? 0,
- noise: Array.isArray(settings.noise) ? settings.noise : [],
- };
- case 'sudoku':
- return {
- ascii: settings.ascii || '',
- customTable: settings.customTable || '',
- customTables: Array.isArray(settings.customTables) ? settings.customTables : [],
- paddingMin: settings.paddingMin ?? 0,
- paddingMax: settings.paddingMax ?? 0
- };
- default:
- return settings;
- }
- }
- static fromJson(json = {}) {
- return new UdpMask(
- json.type || 'salamander',
- json.settings || {}
- );
- }
- toJson() {
- const cleanItem = item => {
- const out = { ...item };
- if (out.type === 'array') {
- delete out.packet;
- } else {
- delete out.rand;
- delete out.randRange;
- }
- return out;
- };
- let settings = this.settings;
- if (this.type === 'noise' && settings && Array.isArray(settings.noise)) {
- settings = { ...settings, noise: settings.noise.map(cleanItem) };
- } else if (this.type === 'header-custom' && settings) {
- settings = {
- ...settings,
- client: Array.isArray(settings.client) ? settings.client.map(cleanItem) : settings.client,
- server: Array.isArray(settings.server) ? settings.server.map(cleanItem) : settings.server,
- };
- }
- return {
- type: this.type,
- settings: (settings && Object.keys(settings).length > 0) ? settings : undefined
- };
- }
- }
- class TcpMask extends CommonClass {
- constructor(type = 'fragment', settings = {}) {
- super();
- this.type = type;
- this.settings = this._getDefaultSettings(type, settings);
- }
- _getDefaultSettings(type, settings = {}) {
- switch (type) {
- case 'fragment':
- return {
- packets: settings.packets ?? 'tlshello',
- length: settings.length ?? '',
- delay: settings.delay ?? '',
- maxSplit: settings.maxSplit ?? '',
- };
- case 'sudoku':
- return {
- password: settings.password ?? '',
- ascii: settings.ascii ?? '',
- customTable: settings.customTable ?? '',
- customTables: Array.isArray(settings.customTables) ? settings.customTables : [],
- paddingMin: settings.paddingMin ?? 0,
- paddingMax: settings.paddingMax ?? 0,
- };
- case 'header-custom':
- return {
- clients: Array.isArray(settings.clients) ? settings.clients : [],
- servers: Array.isArray(settings.servers) ? settings.servers : [],
- };
- default:
- return settings;
- }
- }
- static fromJson(json = {}) {
- return new TcpMask(
- json.type || 'fragment',
- json.settings || {}
- );
- }
- toJson() {
- const cleanItem = item => {
- const out = { ...item };
- if (out.type === 'array') {
- delete out.packet;
- } else {
- delete out.rand;
- delete out.randRange;
- }
- return out;
- };
- let settings = this.settings;
- if (this.type === 'header-custom' && settings) {
- const cleanGroup = group => Array.isArray(group) ? group.map(cleanItem) : group;
- settings = {
- ...settings,
- clients: Array.isArray(settings.clients) ? settings.clients.map(cleanGroup) : settings.clients,
- servers: Array.isArray(settings.servers) ? settings.servers.map(cleanGroup) : settings.servers,
- };
- }
- return {
- type: this.type,
- settings: (settings && Object.keys(settings).length > 0) ? settings : undefined
- };
- }
- }
- class QuicParams extends CommonClass {
- constructor(
- congestion = 'bbr',
- debug = false,
- brutalUp = 65537,
- brutalDown = 65537,
- udpHop = undefined,
- initStreamReceiveWindow = 8388608,
- maxStreamReceiveWindow = 8388608,
- initConnectionReceiveWindow = 20971520,
- maxConnectionReceiveWindow = 20971520,
- maxIdleTimeout = 30,
- keepAlivePeriod = 5,
- disablePathMTUDiscovery = false,
- maxIncomingStreams = 1024,
- ) {
- super();
- this.congestion = congestion;
- this.debug = debug;
- this.brutalUp = brutalUp;
- this.brutalDown = brutalDown;
- this.udpHop = udpHop;
- this.initStreamReceiveWindow = initStreamReceiveWindow;
- this.maxStreamReceiveWindow = maxStreamReceiveWindow;
- this.initConnectionReceiveWindow = initConnectionReceiveWindow;
- this.maxConnectionReceiveWindow = maxConnectionReceiveWindow;
- this.maxIdleTimeout = maxIdleTimeout;
- this.keepAlivePeriod = keepAlivePeriod;
- this.disablePathMTUDiscovery = disablePathMTUDiscovery;
- this.maxIncomingStreams = maxIncomingStreams;
- }
- get hasUdpHop() {
- return this.udpHop != null;
- }
- set hasUdpHop(value) {
- this.udpHop = value ? (this.udpHop || { ports: '20000-50000', interval: '5-10' }) : undefined;
- }
- static fromJson(json = {}) {
- if (!json || Object.keys(json).length === 0) return undefined;
- return new QuicParams(
- json.congestion,
- json.debug,
- json.brutalUp,
- json.brutalDown,
- json.udpHop ? { ports: json.udpHop.ports, interval: json.udpHop.interval } : undefined,
- json.initStreamReceiveWindow,
- json.maxStreamReceiveWindow,
- json.initConnectionReceiveWindow,
- json.maxConnectionReceiveWindow,
- json.maxIdleTimeout,
- json.keepAlivePeriod,
- json.disablePathMTUDiscovery,
- json.maxIncomingStreams,
- );
- }
- toJson() {
- const result = { congestion: this.congestion };
- if (this.debug) result.debug = this.debug;
- if (['brutal', 'force-brutal'].includes(this.congestion)) {
- if (this.brutalUp) result.brutalUp = this.brutalUp;
- if (this.brutalDown) result.brutalDown = this.brutalDown;
- }
- if (this.udpHop) result.udpHop = { ports: this.udpHop.ports, interval: this.udpHop.interval };
- if (this.initStreamReceiveWindow > 0) result.initStreamReceiveWindow = this.initStreamReceiveWindow;
- if (this.maxStreamReceiveWindow > 0) result.maxStreamReceiveWindow = this.maxStreamReceiveWindow;
- if (this.initConnectionReceiveWindow > 0) result.initConnectionReceiveWindow = this.initConnectionReceiveWindow;
- if (this.maxConnectionReceiveWindow > 0) result.maxConnectionReceiveWindow = this.maxConnectionReceiveWindow;
- if (this.maxIdleTimeout !== 30 && this.maxIdleTimeout > 0) result.maxIdleTimeout = this.maxIdleTimeout;
- if (this.keepAlivePeriod > 0) result.keepAlivePeriod = this.keepAlivePeriod;
- if (this.disablePathMTUDiscovery) result.disablePathMTUDiscovery = this.disablePathMTUDiscovery;
- if (this.maxIncomingStreams > 0) result.maxIncomingStreams = this.maxIncomingStreams;
- return result;
- }
- }
- class FinalMaskStreamSettings extends CommonClass {
- constructor(tcp = [], udp = [], quicParams = undefined) {
- super();
- this.tcp = Array.isArray(tcp) ? tcp.map(t => t instanceof TcpMask ? t : new TcpMask(t.type, t.settings)) : [];
- this.udp = Array.isArray(udp) ? udp.map(u => new UdpMask(u.type, u.settings)) : [new UdpMask(udp.type, udp.settings)];
- this.quicParams = quicParams instanceof QuicParams ? quicParams : (quicParams ? QuicParams.fromJson(quicParams) : undefined);
- }
- get enableQuicParams() {
- return this.quicParams != null;
- }
- set enableQuicParams(value) {
- this.quicParams = value ? (this.quicParams || new QuicParams()) : undefined;
- }
- static fromJson(json = {}) {
- return new FinalMaskStreamSettings(
- json.tcp || [],
- json.udp || [],
- json.quicParams ? QuicParams.fromJson(json.quicParams) : undefined,
- );
- }
- toJson() {
- const result = {};
- if (this.tcp && this.tcp.length > 0) {
- result.tcp = this.tcp.map(t => t.toJson());
- }
- if (this.udp && this.udp.length > 0) {
- result.udp = this.udp.map(udp => udp.toJson());
- }
- if (this.quicParams) {
- result.quicParams = this.quicParams.toJson();
- }
- return result;
- }
- }
- class StreamSettings extends CommonClass {
- constructor(
- network = 'tcp',
- security = 'none',
- tlsSettings = new TlsStreamSettings(),
- realitySettings = new RealityStreamSettings(),
- tcpSettings = new TcpStreamSettings(),
- kcpSettings = new KcpStreamSettings(),
- wsSettings = new WsStreamSettings(),
- grpcSettings = new GrpcStreamSettings(),
- httpupgradeSettings = new HttpUpgradeStreamSettings(),
- xhttpSettings = new xHTTPStreamSettings(),
- hysteriaSettings = new HysteriaStreamSettings(),
- finalmask = new FinalMaskStreamSettings(),
- sockopt = undefined,
- ) {
- super();
- this.network = network;
- this.security = security;
- this.tls = tlsSettings;
- this.reality = realitySettings;
- this.tcp = tcpSettings;
- this.kcp = kcpSettings;
- this.ws = wsSettings;
- this.grpc = grpcSettings;
- this.httpupgrade = httpupgradeSettings;
- this.xhttp = xhttpSettings;
- this.hysteria = hysteriaSettings;
- this.finalmask = finalmask;
- this.sockopt = sockopt;
- }
- addTcpMask(type = 'fragment') {
- this.finalmask.tcp.push(new TcpMask(type));
- }
- delTcpMask(index) {
- if (this.finalmask.tcp) {
- this.finalmask.tcp.splice(index, 1);
- }
- }
- addUdpMask(type = 'salamander') {
- this.finalmask.udp.push(new UdpMask(type));
- }
- delUdpMask(index) {
- if (this.finalmask.udp) {
- this.finalmask.udp.splice(index, 1);
- }
- }
- get hasFinalMask() {
- const hasTcp = this.finalmask.tcp && this.finalmask.tcp.length > 0;
- const hasUdp = this.finalmask.udp && this.finalmask.udp.length > 0;
- const hasQuicParams = this.finalmask.quicParams != null;
- return hasTcp || hasUdp || hasQuicParams;
- }
- get isTls() {
- return this.security === 'tls';
- }
- get isReality() {
- return this.security === "reality";
- }
- get sockoptSwitch() {
- return this.sockopt != undefined;
- }
- set sockoptSwitch(value) {
- this.sockopt = value ? new SockoptStreamSettings() : undefined;
- }
- static fromJson(json = {}) {
- return new StreamSettings(
- json.network,
- json.security,
- TlsStreamSettings.fromJson(json.tlsSettings),
- RealityStreamSettings.fromJson(json.realitySettings),
- TcpStreamSettings.fromJson(json.tcpSettings),
- KcpStreamSettings.fromJson(json.kcpSettings),
- WsStreamSettings.fromJson(json.wsSettings),
- GrpcStreamSettings.fromJson(json.grpcSettings),
- HttpUpgradeStreamSettings.fromJson(json.httpupgradeSettings),
- xHTTPStreamSettings.fromJson(json.xhttpSettings),
- HysteriaStreamSettings.fromJson(json.hysteriaSettings),
- FinalMaskStreamSettings.fromJson(json.finalmask),
- SockoptStreamSettings.fromJson(json.sockopt),
- );
- }
- toJson() {
- const network = this.network;
- return {
- network: network,
- security: this.security,
- tlsSettings: this.security == 'tls' ? this.tls.toJson() : undefined,
- realitySettings: this.security == 'reality' ? this.reality.toJson() : undefined,
- tcpSettings: network === 'tcp' ? this.tcp.toJson() : undefined,
- kcpSettings: network === 'kcp' ? this.kcp.toJson() : undefined,
- wsSettings: network === 'ws' ? this.ws.toJson() : undefined,
- grpcSettings: network === 'grpc' ? this.grpc.toJson() : undefined,
- httpupgradeSettings: network === 'httpupgrade' ? this.httpupgrade.toJson() : undefined,
- xhttpSettings: network === 'xhttp' ? this.xhttp.toJson() : undefined,
- hysteriaSettings: network === 'hysteria' ? this.hysteria.toJson() : undefined,
- finalmask: this.hasFinalMask ? this.finalmask.toJson() : undefined,
- sockopt: this.sockopt != undefined ? this.sockopt.toJson() : undefined,
- };
- }
- }
- class Mux extends CommonClass {
- constructor(enabled = false, concurrency = 8, xudpConcurrency = 16, xudpProxyUDP443 = "reject") {
- super();
- this.enabled = enabled;
- this.concurrency = concurrency;
- this.xudpConcurrency = xudpConcurrency;
- this.xudpProxyUDP443 = xudpProxyUDP443;
- }
- static fromJson(json = {}) {
- if (Object.keys(json).length === 0) return undefined;
- return new Mux(
- json.enabled,
- json.concurrency,
- json.xudpConcurrency,
- json.xudpProxyUDP443,
- );
- }
- toJson() {
- return {
- enabled: this.enabled,
- concurrency: this.concurrency,
- xudpConcurrency: this.xudpConcurrency,
- xudpProxyUDP443: this.xudpProxyUDP443,
- };
- }
- }
- class Outbound extends CommonClass {
- constructor(
- tag = '',
- protocol = Protocols.VLESS,
- settings = null,
- streamSettings = new StreamSettings(),
- sendThrough,
- mux = new Mux(),
- ) {
- super();
- this.tag = tag;
- this._protocol = protocol;
- this.settings = settings == null ? Outbound.Settings.getSettings(protocol) : settings;
- this.stream = streamSettings;
- this.sendThrough = sendThrough;
- this.mux = mux;
- }
- get protocol() {
- return this._protocol;
- }
- set protocol(protocol) {
- this._protocol = protocol;
- this.settings = Outbound.Settings.getSettings(protocol);
- this.stream = new StreamSettings();
- }
- canEnableTls() {
- if (this.protocol === Protocols.Hysteria) return true;
- if (![Protocols.VMess, Protocols.VLESS, Protocols.Trojan, Protocols.Shadowsocks].includes(this.protocol)) return false;
- return ["tcp", "ws", "http", "grpc", "httpupgrade", "xhttp"].includes(this.stream.network);
- }
- //this is used for xtls-rprx-vision
- canEnableTlsFlow() {
- if ((this.stream.security != 'none') && (this.stream.network === "tcp")) {
- return this.protocol === Protocols.VLESS;
- }
- return false;
- }
- // Vision seed applies only when vision flow is selected
- canEnableVisionSeed() {
- if (!this.canEnableTlsFlow()) return false;
- const flow = this.settings?.flow;
- return flow === TLS_FLOW_CONTROL.VISION || flow === TLS_FLOW_CONTROL.VISION_UDP443;
- }
- canEnableReality() {
- if (![Protocols.VLESS, Protocols.Trojan].includes(this.protocol)) return false;
- return ["tcp", "http", "grpc", "xhttp"].includes(this.stream.network);
- }
- canEnableStream() {
- return [Protocols.VMess, Protocols.VLESS, Protocols.Trojan, Protocols.Shadowsocks, Protocols.Hysteria].includes(this.protocol);
- }
- canEnableMux() {
- // Disable Mux if flow is set
- if (this.settings.flow && this.settings.flow !== '') {
- this.mux.enabled = false;
- return false;
- }
- // Disable Mux if network is xhttp
- if (this.stream.network === 'xhttp') {
- this.mux.enabled = false;
- return false;
- }
- // Allow Mux only for these protocols
- return [
- Protocols.VMess,
- Protocols.VLESS,
- Protocols.Trojan,
- Protocols.Shadowsocks,
- Protocols.HTTP,
- Protocols.Socks
- ].includes(this.protocol);
- }
- hasServers() {
- return [Protocols.Trojan, Protocols.Shadowsocks, Protocols.Socks, Protocols.HTTP].includes(this.protocol);
- }
- hasAddressPort() {
- return [
- Protocols.DNS,
- Protocols.VMess,
- Protocols.VLESS,
- Protocols.Trojan,
- Protocols.Shadowsocks,
- Protocols.Socks,
- Protocols.HTTP,
- Protocols.Hysteria
- ].includes(this.protocol);
- }
- hasUsername() {
- return [Protocols.Socks, Protocols.HTTP].includes(this.protocol);
- }
- static fromJson(json = {}) {
- return new Outbound(
- json.tag,
- json.protocol,
- Outbound.Settings.fromJson(json.protocol, json.settings),
- StreamSettings.fromJson(json.streamSettings),
- json.sendThrough,
- Mux.fromJson(json.mux),
- )
- }
- toJson() {
- var stream;
- if (this.canEnableStream()) {
- stream = this.stream.toJson();
- } else {
- if (this.stream?.sockopt)
- stream = { sockopt: this.stream.sockopt.toJson() };
- }
- let settingsOut = this.settings instanceof CommonClass ? this.settings.toJson() : this.settings;
- return {
- protocol: this.protocol,
- settings: settingsOut,
- // Only include tag, streamSettings, sendThrough, mux if present and not empty
- ...(this.tag ? { tag: this.tag } : {}),
- ...(stream ? { streamSettings: stream } : {}),
- ...(this.sendThrough ? { sendThrough: this.sendThrough } : {}),
- ...(this.mux?.enabled ? { mux: this.mux } : {}),
- };
- }
- static fromLink(link) {
- data = link.split('://');
- if (data.length != 2) return null;
- switch (data[0].toLowerCase()) {
- case Protocols.VMess:
- return this.fromVmessLink(JSON.parse(Base64.decode(data[1])));
- case Protocols.VLESS:
- case Protocols.Trojan:
- case 'ss':
- return this.fromParamLink(link);
- case 'hysteria2':
- case Protocols.Hysteria:
- return this.fromHysteriaLink(link);
- default:
- return null;
- }
- }
- static fromVmessLink(json = {}) {
- let stream = new StreamSettings(json.net, json.tls);
- let network = json.net;
- if (network === 'tcp') {
- stream.tcp = new TcpStreamSettings(
- json.type,
- json.host ?? '',
- json.path ?? '');
- } else if (network === 'kcp') {
- stream.kcp = new KcpStreamSettings();
- stream.type = json.type;
- stream.seed = json.path;
- const mtu = Number(json.mtu);
- if (Number.isFinite(mtu) && mtu > 0) stream.kcp.mtu = mtu;
- const tti = Number(json.tti);
- if (Number.isFinite(tti) && tti > 0) stream.kcp.tti = tti;
- } else if (network === 'ws') {
- stream.ws = new WsStreamSettings(json.path, json.host);
- } else if (network === 'grpc') {
- stream.grpc = new GrpcStreamSettings(json.path, json.authority, json.type == 'multi');
- } else if (network === 'httpupgrade') {
- stream.httpupgrade = new HttpUpgradeStreamSettings(json.path, json.host);
- } else if (network === 'xhttp') {
- // xHTTPStreamSettings positional args are (path, host, headers, ..., mode);
- // passing `json.mode` as the 3rd argument used to land in the `headers`
- // slot, dropping the mode on the floor. Build the object and set mode
- // explicitly to avoid that.
- const xh = new xHTTPStreamSettings(json.path, json.host);
- if (json.mode) xh.mode = json.mode;
- stream.xhttp = xh;
- }
- if (json.tls && json.tls == 'tls') {
- stream.tls = new TlsStreamSettings(
- json.sni,
- json.alpn ? json.alpn.split(',') : [],
- json.fp);
- }
- const port = json.port * 1;
- return new Outbound(json.ps, Protocols.VMess, new Outbound.VmessSettings(json.add, port, json.id, json.scy), stream);
- }
- static fromParamLink(link) {
- const url = new URL(link);
- let type = url.searchParams.get('type') ?? 'tcp';
- let security = url.searchParams.get('security') ?? 'none';
- let stream = new StreamSettings(type, security);
- let headerType = url.searchParams.get('headerType') ?? undefined;
- let host = url.searchParams.get('host') ?? undefined;
- let path = url.searchParams.get('path') ?? undefined;
- let seed = url.searchParams.get('seed') ?? path ?? undefined;
- let mode = url.searchParams.get('mode') ?? undefined;
- if (type === 'tcp' || type === 'none') {
- stream.tcp = new TcpStreamSettings(headerType ?? 'none', host, path);
- } else if (type === 'kcp') {
- stream.kcp = new KcpStreamSettings();
- stream.kcp.type = headerType ?? 'none';
- stream.kcp.seed = seed;
- const mtu = Number(url.searchParams.get('mtu'));
- if (Number.isFinite(mtu) && mtu > 0) stream.kcp.mtu = mtu;
- const tti = Number(url.searchParams.get('tti'));
- if (Number.isFinite(tti) && tti > 0) stream.kcp.tti = tti;
- } else if (type === 'ws') {
- stream.ws = new WsStreamSettings(path, host);
- } else if (type === 'grpc') {
- stream.grpc = new GrpcStreamSettings(
- url.searchParams.get('serviceName') ?? '',
- url.searchParams.get('authority') ?? '',
- url.searchParams.get('mode') == 'multi');
- } else if (type === 'httpupgrade') {
- stream.httpupgrade = new HttpUpgradeStreamSettings(path, host);
- } else if (type === 'xhttp') {
- // Same positional bug as in the VMess-JSON branch above:
- // passing `mode` as the 3rd positional arg put it into the
- // `headers` slot. Build explicitly instead.
- const xh = new xHTTPStreamSettings(path, host);
- if (mode) xh.mode = mode;
- const xpb = url.searchParams.get('x_padding_bytes');
- if (xpb) xh.xPaddingBytes = xpb;
- const extraRaw = url.searchParams.get('extra');
- if (extraRaw) {
- try {
- const extra = JSON.parse(extraRaw);
- if (typeof extra.xPaddingBytes === 'string' && extra.xPaddingBytes) xh.xPaddingBytes = extra.xPaddingBytes;
- if (extra.xPaddingObfsMode === true) xh.xPaddingObfsMode = true;
- ["xPaddingKey", "xPaddingHeader", "xPaddingPlacement", "xPaddingMethod"].forEach(k => {
- if (typeof extra[k] === 'string' && extra[k]) xh[k] = extra[k];
- });
- } catch (_) { /* ignore malformed extra */ }
- }
- stream.xhttp = xh;
- }
- if (security == 'tls') {
- let fp = url.searchParams.get('fp') ?? 'none';
- let alpn = url.searchParams.get('alpn');
- let sni = url.searchParams.get('sni') ?? '';
- let ech = url.searchParams.get('ech') ?? '';
- stream.tls = new TlsStreamSettings(sni, alpn ? alpn.split(',') : [], fp, ech);
- }
- if (security == 'reality') {
- let pbk = url.searchParams.get('pbk');
- let fp = url.searchParams.get('fp');
- let sni = url.searchParams.get('sni') ?? '';
- let sid = url.searchParams.get('sid') ?? '';
- let spx = url.searchParams.get('spx') ?? '';
- let pqv = url.searchParams.get('pqv') ?? '';
- stream.reality = new RealityStreamSettings(pbk, fp, sni, sid, spx, pqv);
- }
- const regex = /([^@]+):\/\/([^@]+)@(.+):(\d+)(.*)$/;
- const match = link.match(regex);
- if (!match) return null;
- let [, protocol, userData, address, port,] = match;
- port *= 1;
- if (protocol == 'ss') {
- protocol = 'shadowsocks';
- userData = atob(userData).split(':');
- }
- var settings;
- switch (protocol) {
- case Protocols.VLESS:
- settings = new Outbound.VLESSSettings(address, port, userData, url.searchParams.get('flow') ?? '', url.searchParams.get('encryption') ?? 'none');
- break;
- case Protocols.Trojan:
- settings = new Outbound.TrojanSettings(address, port, userData);
- break;
- case Protocols.Shadowsocks:
- let method = userData.splice(0, 1)[0];
- settings = new Outbound.ShadowsocksSettings(address, port, userData.join(":"), method, true);
- break;
- default:
- return null;
- }
- let remark = decodeURIComponent(url.hash);
- // Remove '#' from url.hash
- remark = remark.length > 0 ? remark.substring(1) : 'out-' + protocol + '-' + port;
- return new Outbound(remark, protocol, settings, stream);
- }
- static fromHysteriaLink(link) {
- // Parse hysteria2://password@address:port[?param1=value1¶m2=value2...][#remarks]
- const regex = /^hysteria2?:\/\/([^@]+)@([^:?#]+):(\d+)([^#]*)(#.*)?$/;
- const match = link.match(regex);
- if (!match) return null;
- let [, password, address, port, params, hash] = match;
- port = parseInt(port);
- // Parse URL parameters if present
- let urlParams = new URLSearchParams(params);
- // Create stream settings with hysteria network
- let stream = new StreamSettings('hysteria', 'none');
- // Set hysteria stream settings
- stream.hysteria.auth = password;
- stream.hysteria.congestion = urlParams.get('congestion') ?? '';
- stream.hysteria.up = urlParams.get('up') ?? '0';
- stream.hysteria.down = urlParams.get('down') ?? '0';
- stream.hysteria.udphopPort = urlParams.get('udphopPort') ?? '';
- // Support both old single interval and new min/max range
- if (urlParams.has('udphopInterval')) {
- const interval = parseInt(urlParams.get('udphopInterval'));
- stream.hysteria.udphopIntervalMin = interval;
- stream.hysteria.udphopIntervalMax = interval;
- } else {
- stream.hysteria.udphopIntervalMin = parseInt(urlParams.get('udphopIntervalMin') ?? '30');
- stream.hysteria.udphopIntervalMax = parseInt(urlParams.get('udphopIntervalMax') ?? '30');
- }
- // Optional QUIC parameters
- if (urlParams.has('initStreamReceiveWindow')) {
- stream.hysteria.initStreamReceiveWindow = parseInt(urlParams.get('initStreamReceiveWindow'));
- }
- if (urlParams.has('maxStreamReceiveWindow')) {
- stream.hysteria.maxStreamReceiveWindow = parseInt(urlParams.get('maxStreamReceiveWindow'));
- }
- if (urlParams.has('initConnectionReceiveWindow')) {
- stream.hysteria.initConnectionReceiveWindow = parseInt(urlParams.get('initConnectionReceiveWindow'));
- }
- if (urlParams.has('maxConnectionReceiveWindow')) {
- stream.hysteria.maxConnectionReceiveWindow = parseInt(urlParams.get('maxConnectionReceiveWindow'));
- }
- if (urlParams.has('maxIdleTimeout')) {
- stream.hysteria.maxIdleTimeout = parseInt(urlParams.get('maxIdleTimeout'));
- }
- if (urlParams.has('keepAlivePeriod')) {
- stream.hysteria.keepAlivePeriod = parseInt(urlParams.get('keepAlivePeriod'));
- }
- if (urlParams.has('disablePathMTUDiscovery')) {
- stream.hysteria.disablePathMTUDiscovery = urlParams.get('disablePathMTUDiscovery') === 'true';
- }
- // Create settings
- let settings = new Outbound.HysteriaSettings(address, port, 2);
- // Extract remark from hash
- let remark = hash ? decodeURIComponent(hash.substring(1)) : `out-hysteria-${port}`;
- return new Outbound(remark, Protocols.Hysteria, settings, stream);
- }
- }
- Outbound.Settings = class extends CommonClass {
- constructor(protocol) {
- super();
- this.protocol = protocol;
- }
- static getSettings(protocol) {
- switch (protocol) {
- case Protocols.Freedom: return new Outbound.FreedomSettings();
- case Protocols.Blackhole: return new Outbound.BlackholeSettings();
- case Protocols.DNS: return new Outbound.DNSSettings();
- case Protocols.VMess: return new Outbound.VmessSettings();
- case Protocols.VLESS: return new Outbound.VLESSSettings();
- case Protocols.Trojan: return new Outbound.TrojanSettings();
- case Protocols.Shadowsocks: return new Outbound.ShadowsocksSettings();
- case Protocols.Socks: return new Outbound.SocksSettings();
- case Protocols.HTTP: return new Outbound.HttpSettings();
- case Protocols.Wireguard: return new Outbound.WireguardSettings();
- case Protocols.Hysteria: return new Outbound.HysteriaSettings();
- default: return null;
- }
- }
- static fromJson(protocol, json) {
- switch (protocol) {
- case Protocols.Freedom: return Outbound.FreedomSettings.fromJson(json);
- case Protocols.Blackhole: return Outbound.BlackholeSettings.fromJson(json);
- case Protocols.DNS: return Outbound.DNSSettings.fromJson(json);
- case Protocols.VMess: return Outbound.VmessSettings.fromJson(json);
- case Protocols.VLESS: return Outbound.VLESSSettings.fromJson(json);
- case Protocols.Trojan: return Outbound.TrojanSettings.fromJson(json);
- case Protocols.Shadowsocks: return Outbound.ShadowsocksSettings.fromJson(json);
- case Protocols.Socks: return Outbound.SocksSettings.fromJson(json);
- case Protocols.HTTP: return Outbound.HttpSettings.fromJson(json);
- case Protocols.Wireguard: return Outbound.WireguardSettings.fromJson(json);
- case Protocols.Hysteria: return Outbound.HysteriaSettings.fromJson(json);
- default: return null;
- }
- }
- toJson() {
- return {};
- }
- };
- Outbound.FreedomSettings = class extends CommonClass {
- constructor(
- domainStrategy = '',
- redirect = '',
- fragment = {},
- noises = [],
- finalRules = [],
- ) {
- super();
- this.domainStrategy = domainStrategy;
- this.redirect = redirect;
- this.fragment = fragment || {};
- this.noises = Array.isArray(noises) ? noises : [];
- this.finalRules = Array.isArray(finalRules)
- ? finalRules.map(rule => rule instanceof Outbound.FreedomSettings.FinalRule ? rule : Outbound.FreedomSettings.FinalRule.fromJson(rule))
- : [];
- }
- addNoise() {
- this.noises.push(new Outbound.FreedomSettings.Noise());
- }
- delNoise(index) {
- this.noises.splice(index, 1);
- }
- addFinalRule(action = 'block') {
- this.finalRules.push(new Outbound.FreedomSettings.FinalRule(action));
- }
- delFinalRule(index) {
- this.finalRules.splice(index, 1);
- }
- static fromJson(json = {}) {
- const finalRules = Array.isArray(json.finalRules)
- ? json.finalRules.map(rule => Outbound.FreedomSettings.FinalRule.fromJson(rule))
- : [];
- // Backward compatibility: map legacy ipsBlocked entries to blocking finalRules.
- if (finalRules.length === 0 && Array.isArray(json.ipsBlocked) && json.ipsBlocked.length > 0) {
- finalRules.push(new Outbound.FreedomSettings.FinalRule('block', '', '', json.ipsBlocked, ''));
- }
- return new Outbound.FreedomSettings(
- json.domainStrategy,
- json.redirect,
- json.fragment ? Outbound.FreedomSettings.Fragment.fromJson(json.fragment) : {},
- json.noises ? json.noises.map(noise => Outbound.FreedomSettings.Noise.fromJson(noise)) : [],
- finalRules,
- );
- }
- toJson() {
- return {
- domainStrategy: ObjectUtil.isEmpty(this.domainStrategy) ? undefined : this.domainStrategy,
- redirect: ObjectUtil.isEmpty(this.redirect) ? undefined : this.redirect,
- fragment: Object.keys(this.fragment).length === 0 ? undefined : this.fragment,
- noises: this.noises.length === 0 ? undefined : Outbound.FreedomSettings.Noise.toJsonArray(this.noises),
- finalRules: this.finalRules.length === 0 ? undefined : Outbound.FreedomSettings.FinalRule.toJsonArray(this.finalRules),
- };
- }
- };
- Outbound.FreedomSettings.Fragment = class extends CommonClass {
- constructor(
- packets = '1-3',
- length = '',
- interval = '',
- maxSplit = ''
- ) {
- super();
- this.packets = packets;
- this.length = length;
- this.interval = interval;
- this.maxSplit = maxSplit;
- }
- static fromJson(json = {}) {
- return new Outbound.FreedomSettings.Fragment(
- json.packets,
- json.length,
- json.interval,
- json.maxSplit
- );
- }
- };
- Outbound.FreedomSettings.Noise = class extends CommonClass {
- constructor(
- type = 'rand',
- packet = '10-20',
- delay = '10-16',
- applyTo = 'ip'
- ) {
- super();
- this.type = type;
- this.packet = packet;
- this.delay = delay;
- this.applyTo = applyTo;
- }
- static fromJson(json = {}) {
- return new Outbound.FreedomSettings.Noise(
- json.type,
- json.packet,
- json.delay,
- json.applyTo
- );
- }
- toJson() {
- return {
- type: this.type,
- packet: this.packet,
- delay: this.delay,
- applyTo: this.applyTo
- };
- }
- };
- Outbound.FreedomSettings.FinalRule = class extends CommonClass {
- constructor(action = 'block', network = '', port = '', ip = [], blockDelay = '') {
- super();
- this.action = action;
- this.network = network;
- this.port = port;
- this.ip = Array.isArray(ip) ? ip : [];
- this.blockDelay = blockDelay;
- }
- static fromJson(json = {}) {
- return new Outbound.FreedomSettings.FinalRule(
- json.action,
- Array.isArray(json.network) ? json.network.join(',') : json.network,
- json.port,
- json.ip || [],
- json.blockDelay,
- );
- }
- toJson() {
- return {
- action: ['allow', 'block'].includes(this.action) ? this.action : 'block',
- network: ObjectUtil.isEmpty(this.network) ? undefined : this.network,
- port: ObjectUtil.isEmpty(this.port) ? undefined : this.port,
- ip: this.ip.length === 0 ? undefined : this.ip,
- blockDelay: this.action === 'block' && !ObjectUtil.isEmpty(this.blockDelay) ? this.blockDelay : undefined,
- };
- }
- };
- Outbound.BlackholeSettings = class extends CommonClass {
- constructor(type) {
- super();
- this.type = type;
- }
- static fromJson(json = {}) {
- return new Outbound.BlackholeSettings(
- json.response ? json.response.type : undefined,
- );
- }
- toJson() {
- return {
- response: ObjectUtil.isEmpty(this.type) ? undefined : { type: this.type },
- };
- }
- };
- Outbound.DNSRule = class extends CommonClass {
- constructor(action = 'direct', qtype = '', domain = '') {
- super();
- this.action = action;
- this.qtype = qtype;
- this.domain = domain;
- }
- static fromJson(json = {}) {
- return new Outbound.DNSRule(
- json.action,
- normalizeDNSRuleField(json.qtype),
- normalizeDNSRuleField(json.domain),
- );
- }
- toJson() {
- const rule = {
- action: normalizeDNSRuleAction(this.action),
- };
- const qtype = normalizeDNSRuleField(this.qtype);
- if (!ObjectUtil.isEmpty(qtype)) {
- if (/^\d+$/.test(qtype)) {
- rule.qtype = Number(qtype);
- } else {
- rule.qtype = qtype;
- }
- }
- const domains = normalizeDNSRuleField(this.domain)
- .split(',')
- .map(d => d.trim())
- .filter(d => d.length > 0);
- if (domains.length > 0) {
- rule.domain = domains;
- }
- return rule;
- }
- };
- Outbound.DNSSettings = class extends CommonClass {
- constructor(
- network = 'udp',
- address = '',
- port = 53,
- rules = []
- ) {
- super();
- this.network = network;
- this.address = address;
- this.port = port;
- this.rules = Array.isArray(rules) ? rules.map(rule => rule instanceof Outbound.DNSRule ? rule : Outbound.DNSRule.fromJson(rule)) : [];
- }
- addRule(action = 'direct') {
- this.rules.push(new Outbound.DNSRule(action));
- }
- delRule(index) {
- this.rules.splice(index, 1);
- }
- static fromJson(json = {}) {
- return new Outbound.DNSSettings(
- json.network,
- json.address,
- json.port,
- getDNSRulesFromJson(json),
- );
- }
- toJson() {
- const json = {
- network: this.network,
- address: this.address,
- port: this.port,
- };
- if (this.rules.length > 0) {
- json.rules = Outbound.DNSRule.toJsonArray(this.rules);
- }
- return json;
- }
- };
- Outbound.VmessSettings = class extends CommonClass {
- constructor(address, port, id, security) {
- super();
- this.address = address;
- this.port = port;
- this.id = id;
- this.security = security;
- }
- static fromJson(json = {}) {
- if (!ObjectUtil.isArrEmpty(json.vnext)) {
- const v = json.vnext[0] || {};
- const u = ObjectUtil.isArrEmpty(v.users) ? {} : v.users[0];
- return new Outbound.VmessSettings(
- v.address,
- v.port,
- u.id,
- u.security,
- );
- }
- }
- toJson() {
- return {
- vnext: [{
- address: this.address,
- port: this.port,
- users: [{
- id: this.id,
- security: this.security
- }]
- }]
- };
- }
- };
- Outbound.VLESSSettings = class extends CommonClass {
- constructor(address, port, id, flow, encryption, reverseTag = '', reverseSniffing = new ReverseSniffing(), testpre = 0, testseed = [900, 500, 900, 256]) {
- super();
- this.address = address;
- this.port = port;
- this.id = id;
- this.flow = flow;
- this.encryption = encryption;
- this.reverseTag = reverseTag;
- this.reverseSniffing = reverseSniffing;
- this.testpre = testpre;
- this.testseed = testseed;
- }
- static fromJson(json = {}) {
- if (ObjectUtil.isEmpty(json.address) || ObjectUtil.isEmpty(json.port)) return new Outbound.VLESSSettings();
- return new Outbound.VLESSSettings(
- json.address,
- json.port,
- json.id,
- json.flow,
- json.encryption,
- json.reverse?.tag || '',
- ReverseSniffing.fromJson(json.reverse?.sniffing || {}),
- json.testpre || 0,
- json.testseed && json.testseed.length >= 4 ? json.testseed : [900, 500, 900, 256]
- );
- }
- toJson() {
- const result = {
- address: this.address,
- port: this.port,
- id: this.id,
- flow: this.flow,
- encryption: this.encryption,
- };
- if (!ObjectUtil.isEmpty(this.reverseTag)) {
- const reverseSniffing = this.reverseSniffing ? this.reverseSniffing.toJson() : {};
- const defaultReverseSniffing = new ReverseSniffing().toJson();
- result.reverse = {
- tag: this.reverseTag,
- sniffing: JSON.stringify(reverseSniffing) === JSON.stringify(defaultReverseSniffing) ? {} : reverseSniffing,
- };
- }
- // Only include Vision settings when flow is set
- if (this.flow && this.flow !== '') {
- if (this.testpre > 0) {
- result.testpre = this.testpre;
- }
- if (this.testseed && this.testseed.length >= 4) {
- result.testseed = this.testseed;
- }
- }
- return result;
- }
- };
- Outbound.TrojanSettings = class extends CommonClass {
- constructor(address, port, password) {
- super();
- this.address = address;
- this.port = port;
- this.password = password;
- }
- static fromJson(json = {}) {
- if (ObjectUtil.isArrEmpty(json.servers)) return new Outbound.TrojanSettings();
- return new Outbound.TrojanSettings(
- json.servers[0].address,
- json.servers[0].port,
- json.servers[0].password,
- );
- }
- toJson() {
- return {
- servers: [{
- address: this.address,
- port: this.port,
- password: this.password,
- }],
- };
- }
- };
- Outbound.ShadowsocksSettings = class extends CommonClass {
- constructor(address, port, password, method, uot, UoTVersion) {
- super();
- this.address = address;
- this.port = port;
- this.password = password;
- this.method = method;
- this.uot = uot;
- this.UoTVersion = UoTVersion;
- }
- static fromJson(json = {}) {
- let servers = json.servers;
- if (ObjectUtil.isArrEmpty(servers)) servers = [{}];
- return new Outbound.ShadowsocksSettings(
- servers[0].address,
- servers[0].port,
- servers[0].password,
- servers[0].method,
- servers[0].uot,
- servers[0].UoTVersion,
- );
- }
- toJson() {
- return {
- servers: [{
- address: this.address,
- port: this.port,
- password: this.password,
- method: this.method,
- uot: this.uot,
- UoTVersion: this.UoTVersion,
- }],
- };
- }
- };
- Outbound.SocksSettings = class extends CommonClass {
- constructor(address, port, user, pass) {
- super();
- this.address = address;
- this.port = port;
- this.user = user;
- this.pass = pass;
- }
- static fromJson(json = {}) {
- let servers = json.servers;
- if (ObjectUtil.isArrEmpty(servers)) servers = [{ users: [{}] }];
- return new Outbound.SocksSettings(
- servers[0].address,
- servers[0].port,
- ObjectUtil.isArrEmpty(servers[0].users) ? '' : servers[0].users[0].user,
- ObjectUtil.isArrEmpty(servers[0].users) ? '' : servers[0].users[0].pass,
- );
- }
- toJson() {
- return {
- servers: [{
- address: this.address,
- port: this.port,
- users: ObjectUtil.isEmpty(this.user) ? [] : [{ user: this.user, pass: this.pass }],
- }],
- };
- }
- };
- Outbound.HttpSettings = class extends CommonClass {
- constructor(address, port, user, pass) {
- super();
- this.address = address;
- this.port = port;
- this.user = user;
- this.pass = pass;
- }
- static fromJson(json = {}) {
- let servers = json.servers;
- if (ObjectUtil.isArrEmpty(servers)) servers = [{ users: [{}] }];
- return new Outbound.HttpSettings(
- servers[0].address,
- servers[0].port,
- ObjectUtil.isArrEmpty(servers[0].users) ? '' : servers[0].users[0].user,
- ObjectUtil.isArrEmpty(servers[0].users) ? '' : servers[0].users[0].pass,
- );
- }
- toJson() {
- return {
- servers: [{
- address: this.address,
- port: this.port,
- users: ObjectUtil.isEmpty(this.user) ? [] : [{ user: this.user, pass: this.pass }],
- }],
- };
- }
- };
- Outbound.WireguardSettings = class extends CommonClass {
- constructor(
- mtu = 1420,
- secretKey = '',
- address = [''],
- workers = 2,
- domainStrategy = '',
- reserved = '',
- peers = [new Outbound.WireguardSettings.Peer()],
- noKernelTun = false,
- ) {
- super();
- this.mtu = mtu;
- this.secretKey = secretKey;
- this.pubKey = secretKey.length > 0 ? Wireguard.generateKeypair(secretKey).publicKey : '';
- this.address = Array.isArray(address) ? address.join(',') : address;
- this.workers = workers;
- this.domainStrategy = domainStrategy;
- this.reserved = Array.isArray(reserved) ? reserved.join(',') : reserved;
- this.peers = peers;
- this.noKernelTun = noKernelTun;
- }
- addPeer() {
- this.peers.push(new Outbound.WireguardSettings.Peer());
- }
- delPeer(index) {
- this.peers.splice(index, 1);
- }
- static fromJson(json = {}) {
- return new Outbound.WireguardSettings(
- json.mtu,
- json.secretKey,
- json.address,
- json.workers,
- json.domainStrategy,
- json.reserved,
- json.peers.map(peer => Outbound.WireguardSettings.Peer.fromJson(peer)),
- json.noKernelTun,
- );
- }
- toJson() {
- return {
- mtu: this.mtu ?? undefined,
- secretKey: this.secretKey,
- address: this.address ? this.address.split(",") : [],
- workers: this.workers ?? undefined,
- domainStrategy: WireguardDomainStrategy.includes(this.domainStrategy) ? this.domainStrategy : undefined,
- reserved: this.reserved ? this.reserved.split(",").map(Number) : undefined,
- peers: Outbound.WireguardSettings.Peer.toJsonArray(this.peers),
- noKernelTun: this.noKernelTun,
- };
- }
- };
- Outbound.WireguardSettings.Peer = class extends CommonClass {
- constructor(
- publicKey = '',
- psk = '',
- allowedIPs = ['0.0.0.0/0', '::/0'],
- endpoint = '',
- keepAlive = 0
- ) {
- super();
- this.publicKey = publicKey;
- this.psk = psk;
- this.allowedIPs = allowedIPs;
- this.endpoint = endpoint;
- this.keepAlive = keepAlive;
- }
- static fromJson(json = {}) {
- return new Outbound.WireguardSettings.Peer(
- json.publicKey,
- json.preSharedKey,
- json.allowedIPs,
- json.endpoint,
- json.keepAlive
- );
- }
- toJson() {
- return {
- publicKey: this.publicKey,
- preSharedKey: this.psk.length > 0 ? this.psk : undefined,
- allowedIPs: this.allowedIPs ? this.allowedIPs : undefined,
- endpoint: this.endpoint,
- keepAlive: this.keepAlive ?? undefined,
- };
- }
- };
- Outbound.HysteriaSettings = class extends CommonClass {
- constructor(address = '', port = 443, version = 2) {
- super();
- this.address = address;
- this.port = port;
- this.version = version;
- }
- static fromJson(json = {}) {
- if (Object.keys(json).length === 0) return new Outbound.HysteriaSettings();
- return new Outbound.HysteriaSettings(
- json.address,
- json.port,
- json.version
- );
- }
- toJson() {
- return {
- address: this.address,
- port: this.port,
- version: this.version
- };
- }
- };
|