From 0aca2d3b3d7e54f0408670db5f7075eaee190066 Mon Sep 17 00:00:00 2001 From: MHSanaei Date: Sun, 26 Apr 2026 23:04:47 +0200 Subject: sub: kcp finalmask --- web/assets/js/model/inbound.js | 54 ++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 54 insertions(+) (limited to 'web/assets/js/model/inbound.js') diff --git a/web/assets/js/model/inbound.js b/web/assets/js/model/inbound.js index a091e1ef..ef9ad19c 100644 --- a/web/assets/js/model/inbound.js +++ b/web/assets/js/model/inbound.js @@ -1370,6 +1370,50 @@ class Inbound extends XrayCommonClass { } } + static hasShareableFinalMaskValue(value) { + if (value == null) { + return false; + } + if (Array.isArray(value)) { + return value.some(item => Inbound.hasShareableFinalMaskValue(item)); + } + if (typeof value === 'object') { + return Object.values(value).some(item => Inbound.hasShareableFinalMaskValue(item)); + } + if (typeof value === 'string') { + return value.length > 0; + } + return true; + } + + static serializeFinalMask(finalmask) { + if (!finalmask) { + return ''; + } + const value = typeof finalmask.toJson === 'function' ? finalmask.toJson() : finalmask; + return Inbound.hasShareableFinalMaskValue(value) ? JSON.stringify(value) : ''; + } + + // Export finalmask with the same compact JSON payload shape that + // v2rayN-compatible share links use: fm=. + static applyFinalMaskToParams(finalmask, params) { + if (!params) return; + const payload = Inbound.serializeFinalMask(finalmask); + if (payload.length > 0) { + params.set("fm", payload); + } + } + + // VMess links are a base64 JSON object, so keep the same fm payload + // under a flat property instead of a URL query string. + static applyFinalMaskToObj(finalmask, obj) { + if (!obj) return; + const payload = Inbound.serializeFinalMask(finalmask); + if (payload.length > 0) { + obj.fm = payload; + } + } + get clients() { switch (this.protocol) { case Protocols.VMESS: return this.settings.vmesses; @@ -1590,6 +1634,8 @@ class Inbound extends XrayCommonClass { Inbound.applyXhttpPaddingToObj(xhttp, obj); } + Inbound.applyFinalMaskToObj(this.stream.finalmask, obj); + if (tls === 'tls') { if (!ObjectUtil.isEmpty(this.stream.tls.sni)) { obj.sni = this.stream.tls.sni; @@ -1658,6 +1704,8 @@ class Inbound extends XrayCommonClass { break; } + Inbound.applyFinalMaskToParams(this.stream.finalmask, params); + if (security === 'tls') { params.set("security", "tls"); if (this.stream.isTls) { @@ -1761,6 +1809,8 @@ class Inbound extends XrayCommonClass { break; } + Inbound.applyFinalMaskToParams(this.stream.finalmask, params); + if (security === 'tls') { params.set("security", "tls"); if (this.stream.isTls) { @@ -1840,6 +1890,8 @@ class Inbound extends XrayCommonClass { break; } + Inbound.applyFinalMaskToParams(this.stream.finalmask, params); + if (security === 'tls') { params.set("security", "tls"); if (this.stream.isTls) { @@ -1907,6 +1959,8 @@ class Inbound extends XrayCommonClass { } } + Inbound.applyFinalMaskToParams(this.stream.finalmask, params); + const url = new URL(link); for (const [key, value] of params) { url.searchParams.set(key, value); -- cgit v1.2.3