host_sub.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. package sub
  2. import (
  3. "encoding/json"
  4. "maps"
  5. "slices"
  6. "github.com/mhsanaei/3x-ui/v3/internal/database"
  7. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  8. "github.com/mhsanaei/3x-ui/v3/internal/logger"
  9. )
  10. // hostEndpoints loads an inbound's enabled hosts for the given subscription
  11. // format ("raw"|"json"|"clash") and returns them as externalProxy-shaped maps so
  12. // the existing per-format renderers can fan out one link/proxy per host. Returns
  13. // nil when the inbound has no applicable host — the caller then uses the legacy
  14. // inbound/externalProxy path, preserving byte-identical output for zero-host
  15. // inbounds.
  16. func (s *SubService) hostEndpoints(inbound *model.Inbound, format string) []map[string]any {
  17. var hosts []*model.Host
  18. if err := database.GetDB().
  19. Where("inbound_id = ? AND is_disabled = ?", inbound.Id, false).
  20. Order("sort_order asc, id asc").
  21. Find(&hosts).Error; err != nil {
  22. logger.Warning("SubService - hostEndpoints:", err)
  23. return nil
  24. }
  25. if len(hosts) == 0 {
  26. return nil
  27. }
  28. defaultDest := s.resolveInboundAddress(inbound)
  29. eps := make([]map[string]any, 0, len(hosts))
  30. for _, h := range hosts {
  31. if slices.Contains(h.ExcludeFromSubTypes, format) {
  32. continue
  33. }
  34. eps = append(eps, hostToExternalProxyMap(h, defaultDest, inbound.Port))
  35. }
  36. return eps
  37. }
  38. // hostToExternalProxyMap projects a Host onto the externalProxy entry shape the
  39. // raw/json/clash renderers already consume. Address/port fall back to the
  40. // inbound's own when the host leaves them blank (override-only host).
  41. func hostToExternalProxyMap(h *model.Host, defaultDest string, defaultPort int) map[string]any {
  42. dest := h.Address
  43. if dest == "" {
  44. dest = defaultDest
  45. }
  46. port := h.Port
  47. if port == 0 {
  48. port = defaultPort
  49. }
  50. ep := map[string]any{
  51. "forceTls": hostSecurityToForceTls(h.Security),
  52. "dest": dest,
  53. "port": float64(port),
  54. "remark": h.Remark,
  55. // Marks this as a host (not a legacy externalProxy) entry so host-only
  56. // behaviors (e.g. reality SNI/fp override) apply without touching the
  57. // legacy externalProxy path. Not emitted into output.
  58. "isHost": true,
  59. }
  60. sni := h.Sni
  61. if h.OverrideSniFromAddress {
  62. sni = dest
  63. }
  64. if !h.KeepSniBlank && sni != "" {
  65. ep["sni"] = sni
  66. }
  67. if h.Fingerprint != "" {
  68. ep["fingerprint"] = h.Fingerprint
  69. }
  70. if len(h.Alpn) > 0 {
  71. ep["alpn"] = stringsToAnySlice(h.Alpn)
  72. }
  73. if len(h.PinnedPeerCertSha256) > 0 {
  74. ep["pinnedPeerCertSha256"] = stringsToAnySlice(h.PinnedPeerCertSha256)
  75. }
  76. if h.EchConfigList != "" {
  77. ep["echConfigList"] = h.EchConfigList
  78. }
  79. if h.AllowInsecure {
  80. ep["allowInsecure"] = true
  81. }
  82. if h.HostHeader != "" {
  83. ep["hostHeader"] = h.HostHeader
  84. }
  85. if h.Path != "" {
  86. ep["path"] = h.Path
  87. }
  88. if h.MihomoIpVersion != "" {
  89. ep["mihomoIpVersion"] = h.MihomoIpVersion
  90. }
  91. if h.SockoptParams != "" {
  92. ep["sockoptParams"] = h.SockoptParams
  93. }
  94. if h.MuxParams != "" {
  95. ep["muxParams"] = h.MuxParams
  96. }
  97. if h.FinalMask != "" {
  98. ep["finalMask"] = h.FinalMask
  99. }
  100. return ep
  101. }
  102. // hostMuxOverride returns a host's muxParams when it is valid JSON, else "".
  103. // Used to override the JSON outbound's mux for that host.
  104. func hostMuxOverride(ep map[string]any) string {
  105. mp, ok := ep["muxParams"].(string)
  106. if ok && mp != "" && json.Valid([]byte(mp)) {
  107. return mp
  108. }
  109. return ""
  110. }
  111. // applyHostStreamOverrides injects a host's free-JSON stream overrides into the
  112. // per-host stream the JSON/Clash renderers build: sockoptParams (re-added since
  113. // the base stream strips sockopt) and finalMask. No-op for legacy externalProxy
  114. // entries (which never carry these keys), so existing output is unchanged.
  115. func applyHostStreamOverrides(ep map[string]any, stream map[string]any) {
  116. if sp, ok := ep["sockoptParams"].(string); ok && sp != "" {
  117. var sockopt map[string]any
  118. if json.Unmarshal([]byte(sp), &sockopt) == nil && len(sockopt) > 0 {
  119. stream["sockopt"] = sockopt
  120. }
  121. }
  122. // Host finalmask: merge the host's masks into the stream's finalmask (the
  123. // JSON renderer consumes streamSettings["finalmask"]; clash ignores it).
  124. if fm, ok := ep["finalMask"].(string); ok && fm != "" {
  125. var masks map[string]any
  126. if json.Unmarshal([]byte(fm), &masks) == nil && len(masks) > 0 {
  127. merged := mergeFinalMask(stream["finalmask"], masks)
  128. if len(merged) > 0 {
  129. stream["finalmask"] = merged
  130. }
  131. }
  132. }
  133. // Reality SNI override (host only): JSON realityData reads serverNames and
  134. // clash reads serverName, so set both forms.
  135. if isHostEndpoint(ep) {
  136. if sec, _ := stream["security"].(string); sec == "reality" {
  137. if rs, ok := stream["realitySettings"].(map[string]any); ok && rs != nil {
  138. if sni, ok := externalProxySNI(ep); ok {
  139. rs["serverName"] = sni
  140. rs["serverNames"] = []any{sni}
  141. }
  142. }
  143. }
  144. }
  145. }
  146. // hostSecurityToForceTls maps Host.Security onto the externalProxy forceTls
  147. // vocabulary. "reality"/"same"/"" all keep the inbound's base security ("same")
  148. // — reality parameters can only come from the inbound itself.
  149. func hostSecurityToForceTls(security string) string {
  150. switch security {
  151. case "tls", "none":
  152. return security
  153. default:
  154. return "same"
  155. }
  156. }
  157. func stringsToAnySlice(in []string) []any {
  158. out := make([]any, 0, len(in))
  159. for _, s := range in {
  160. if s != "" {
  161. out = append(out, s)
  162. }
  163. }
  164. return out
  165. }
  166. // injectExternalProxy rewrites the inbound's StreamSettings so its externalProxy
  167. // array is exactly eps. Host endpoints win over any legacy externalProxy.
  168. func injectExternalProxy(inbound *model.Inbound, eps []map[string]any) {
  169. stream := unmarshalStreamSettings(inbound.StreamSettings)
  170. if stream == nil {
  171. stream = map[string]any{}
  172. }
  173. arr := make([]any, len(eps))
  174. for i := range eps {
  175. arr[i] = eps[i]
  176. }
  177. stream["externalProxy"] = arr
  178. if b, err := json.Marshal(stream); err == nil {
  179. inbound.StreamSettings = string(b)
  180. }
  181. }
  182. // linkFromHosts renders a (possibly multi-line) raw link for one client using
  183. // the given host endpoints. It renders ONLY the hosts: an empty eps yields ""
  184. // (no legacy fallback) — the caller decides when to take the legacy path. That
  185. // separation is what makes the zero-hosts fallback mutation-testable.
  186. func (s *SubService) linkFromHosts(inbound *model.Inbound, client model.Client, eps []map[string]any) string {
  187. if len(eps) == 0 {
  188. return ""
  189. }
  190. // Clone each ep before expanding its remark template: the eps slice is
  191. // shared across all clients of this inbound, so the rendered (per-client)
  192. // remark must not leak into the next client's links.
  193. rendered := make([]map[string]any, len(eps))
  194. for i, ep := range eps {
  195. cp := maps.Clone(ep)
  196. s.renderHostRemark(inbound, client, cp)
  197. rendered[i] = cp
  198. }
  199. clone := *inbound
  200. injectExternalProxy(&clone, rendered)
  201. return s.GetLink(&clone, client.Email)
  202. }
  203. // renderHostRemark expands a host endpoint's {{VAR}} remark template for one
  204. // client in place and marks it final, so the downstream link/proxy/config
  205. // renderers emit it verbatim (via endpointRemark) instead of re-composing it.
  206. // No-op for non-host endpoints (legacy externalProxy / synthetic default), so
  207. // their output stays byte-identical.
  208. func (s *SubService) renderHostRemark(inbound *model.Inbound, client model.Client, ep map[string]any) {
  209. if !isHostEndpoint(ep) {
  210. return
  211. }
  212. tmpl, _ := ep["remark"].(string)
  213. ep["remark"] = s.genHostRemark(inbound, client, tmpl)
  214. ep["remarkFinal"] = true
  215. }
  216. // endpointRemark returns the remark to stamp on an endpoint's link/proxy/config
  217. // entry. A host endpoint whose template was pre-expanded by renderHostRemark
  218. // carries remarkFinal and is used verbatim; every other entry flows through the
  219. // standard genRemark composition unchanged.
  220. func (s *SubService) endpointRemark(inbound *model.Inbound, email string, ep map[string]any) string {
  221. if ep != nil {
  222. if final, _ := ep["remarkFinal"].(bool); final {
  223. r, _ := ep["remark"].(string)
  224. return r
  225. }
  226. }
  227. var extra string
  228. if ep != nil {
  229. extra, _ = ep["remark"].(string)
  230. }
  231. return s.genRemark(inbound, email, extra)
  232. }
  233. // applyEndpointHostPath overrides the transport host header / path for a host
  234. // endpoint. It is a no-op for legacy externalProxy entries (which never carry
  235. // hostHeader/path) and only replaces keys the transport already emits, so it
  236. // cannot add spurious params to e.g. a tcp link.
  237. func applyEndpointHostPath(e ShareEndpoint, params map[string]string) {
  238. if e.ep == nil {
  239. return
  240. }
  241. if h, ok := e.ep["hostHeader"].(string); ok && h != "" {
  242. if _, exists := params["host"]; exists {
  243. params["host"] = h
  244. }
  245. }
  246. if p, ok := e.ep["path"].(string); ok && p != "" {
  247. if _, exists := params["path"]; exists {
  248. params["path"] = p
  249. }
  250. }
  251. }
  252. // isHostEndpoint reports whether ep was synthesized from a Host (vs a legacy
  253. // externalProxy entry), so host-only overrides stay off the legacy path.
  254. func isHostEndpoint(ep map[string]any) bool {
  255. v, _ := ep["isHost"].(bool)
  256. return v
  257. }
  258. // applyEndpointRealityParams overrides a reality link's SNI + fingerprint from a
  259. // host (reality's pbk/sid are inherited from the inbound, so they aren't touched).
  260. // Host-only: legacy externalProxy reality links are unchanged.
  261. func applyEndpointRealityParams(e ShareEndpoint, params map[string]string, security string) {
  262. if security != "reality" || e.ep == nil || !isHostEndpoint(e.ep) {
  263. return
  264. }
  265. if sni, ok := externalProxySNI(e.ep); ok {
  266. params["sni"] = sni
  267. }
  268. if fp, ok := e.ep["fingerprint"].(string); ok && fp != "" {
  269. params["fp"] = fp
  270. }
  271. }
  272. // applyEndpointAllowInsecure adds allowInsecure=1 to a TLS/Reality link when the
  273. // host opts into skipping cert verification. No-op for legacy externalProxy
  274. // entries (which never carry the key) and for plaintext (none) endpoints.
  275. func applyEndpointAllowInsecure(e ShareEndpoint, params map[string]string, security string) {
  276. if e.ep == nil || security == "none" {
  277. return
  278. }
  279. if ai, ok := e.ep["allowInsecure"].(bool); ok && ai {
  280. params["allowInsecure"] = "1"
  281. }
  282. }
  283. // applyEndpointHostPathObj is applyEndpointHostPath for the VMess object form.
  284. func applyEndpointHostPathObj(e ShareEndpoint, obj map[string]any) {
  285. if e.ep == nil {
  286. return
  287. }
  288. if h, ok := e.ep["hostHeader"].(string); ok && h != "" {
  289. if _, exists := obj["host"]; exists {
  290. obj["host"] = h
  291. }
  292. }
  293. if p, ok := e.ep["path"].(string); ok && p != "" {
  294. if _, exists := obj["path"]; exists {
  295. obj["path"] = p
  296. }
  297. }
  298. }