host_sub.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  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.VerifyPeerCertByName != "" {
  80. ep["verifyPeerCertByName"] = h.VerifyPeerCertByName
  81. }
  82. if h.AllowInsecure {
  83. ep["allowInsecure"] = true
  84. }
  85. if h.HostHeader != "" {
  86. ep["hostHeader"] = h.HostHeader
  87. }
  88. if h.Path != "" {
  89. ep["path"] = h.Path
  90. }
  91. if h.MihomoIpVersion != "" {
  92. ep["mihomoIpVersion"] = h.MihomoIpVersion
  93. }
  94. if h.SockoptParams != "" {
  95. ep["sockoptParams"] = h.SockoptParams
  96. }
  97. if h.MuxParams != "" {
  98. ep["muxParams"] = h.MuxParams
  99. }
  100. if h.FinalMask != "" {
  101. ep["finalMask"] = h.FinalMask
  102. }
  103. return ep
  104. }
  105. // hostMuxOverride returns a host's muxParams when it is valid JSON, else "".
  106. // Used to override the JSON outbound's mux for that host.
  107. func hostMuxOverride(ep map[string]any) string {
  108. mp, ok := ep["muxParams"].(string)
  109. if ok && mp != "" && json.Valid([]byte(mp)) {
  110. return mp
  111. }
  112. return ""
  113. }
  114. // applyHostStreamOverrides injects a host's free-JSON stream overrides into the
  115. // per-host stream the JSON/Clash renderers build: sockoptParams (re-added since
  116. // the base stream strips sockopt) and finalMask. No-op for legacy externalProxy
  117. // entries (which never carry these keys), so existing output is unchanged.
  118. func applyHostStreamOverrides(ep map[string]any, stream map[string]any) {
  119. if sp, ok := ep["sockoptParams"].(string); ok && sp != "" {
  120. var sockopt map[string]any
  121. if json.Unmarshal([]byte(sp), &sockopt) == nil && len(sockopt) > 0 {
  122. stream["sockopt"] = sockopt
  123. }
  124. }
  125. // Host finalmask: merge the host's masks into the stream's finalmask (the
  126. // JSON renderer consumes streamSettings["finalmask"]; clash ignores it).
  127. if fm, ok := ep["finalMask"].(string); ok && fm != "" {
  128. var masks map[string]any
  129. if json.Unmarshal([]byte(fm), &masks) == nil && len(masks) > 0 {
  130. merged := mergeFinalMask(stream["finalmask"], masks)
  131. if len(merged) > 0 {
  132. stream["finalmask"] = merged
  133. }
  134. }
  135. }
  136. // Reality SNI override (host only): JSON realityData reads serverNames and
  137. // clash reads serverName, so set both forms.
  138. if isHostEndpoint(ep) {
  139. if sec, _ := stream["security"].(string); sec == "reality" {
  140. if rs, ok := stream["realitySettings"].(map[string]any); ok && rs != nil {
  141. if sni, ok := externalProxySNI(ep); ok {
  142. rs["serverName"] = sni
  143. rs["serverNames"] = []any{sni}
  144. }
  145. }
  146. }
  147. }
  148. }
  149. // hostSecurityToForceTls maps Host.Security onto the externalProxy forceTls
  150. // vocabulary. "reality"/"same"/"" all keep the inbound's base security ("same")
  151. // — reality parameters can only come from the inbound itself.
  152. func hostSecurityToForceTls(security string) string {
  153. switch security {
  154. case "tls", "none":
  155. return security
  156. default:
  157. return "same"
  158. }
  159. }
  160. func stringsToAnySlice(in []string) []any {
  161. out := make([]any, 0, len(in))
  162. for _, s := range in {
  163. if s != "" {
  164. out = append(out, s)
  165. }
  166. }
  167. return out
  168. }
  169. // injectExternalProxy rewrites the inbound's StreamSettings so its externalProxy
  170. // array is exactly eps. Host endpoints win over any legacy externalProxy.
  171. func injectExternalProxy(inbound *model.Inbound, eps []map[string]any) {
  172. stream := unmarshalStreamSettings(inbound.StreamSettings)
  173. if stream == nil {
  174. stream = map[string]any{}
  175. }
  176. arr := make([]any, len(eps))
  177. for i := range eps {
  178. arr[i] = eps[i]
  179. }
  180. stream["externalProxy"] = arr
  181. if b, err := json.Marshal(stream); err == nil {
  182. inbound.StreamSettings = string(b)
  183. }
  184. }
  185. // linkFromHosts renders a (possibly multi-line) raw link for one client using
  186. // the given host endpoints. It renders ONLY the hosts: an empty eps yields ""
  187. // (no legacy fallback) — the caller decides when to take the legacy path. That
  188. // separation is what makes the zero-hosts fallback mutation-testable.
  189. func (s *SubService) linkFromHosts(inbound *model.Inbound, client model.Client, eps []map[string]any) string {
  190. if len(eps) == 0 {
  191. return ""
  192. }
  193. stream := unmarshalStreamSettings(inbound.StreamSettings)
  194. transport, _ := stream["network"].(string)
  195. // Clone each ep before expanding its remark template: the eps slice is
  196. // shared across all clients of this inbound, so the rendered (per-client)
  197. // remark must not leak into the next client's links.
  198. rendered := make([]map[string]any, len(eps))
  199. for i, ep := range eps {
  200. cp := maps.Clone(ep)
  201. s.renderHostRemark(inbound, client, cp, transport)
  202. rendered[i] = cp
  203. }
  204. clone := *inbound
  205. injectExternalProxy(&clone, rendered)
  206. return s.GetLink(&clone, client.Email)
  207. }
  208. // renderHostRemark expands a host endpoint's {{VAR}} remark template for one
  209. // client in place and marks it final, so the downstream link/proxy/config
  210. // renderers emit it verbatim (via endpointRemark) instead of re-composing it.
  211. // No-op for non-host endpoints (legacy externalProxy / synthetic default), so
  212. // their output stays byte-identical.
  213. func (s *SubService) renderHostRemark(inbound *model.Inbound, client model.Client, ep map[string]any, transport string) {
  214. if !isHostEndpoint(ep) {
  215. return
  216. }
  217. tmpl, _ := ep["remark"].(string)
  218. ep["remark"] = s.genHostRemark(inbound, client, tmpl, transport)
  219. ep["remarkFinal"] = true
  220. }
  221. // endpointRemark returns the remark to stamp on an endpoint's link/proxy/config
  222. // entry. A host endpoint whose template was pre-expanded by renderHostRemark
  223. // carries remarkFinal and is used verbatim; every other entry flows through the
  224. // standard genRemark composition unchanged.
  225. func (s *SubService) endpointRemark(inbound *model.Inbound, email string, ep map[string]any, transport string) string {
  226. if ep != nil {
  227. if final, _ := ep["remarkFinal"].(bool); final {
  228. r, _ := ep["remark"].(string)
  229. return r
  230. }
  231. }
  232. var extra string
  233. if ep != nil {
  234. extra, _ = ep["remark"].(string)
  235. }
  236. return s.genRemark(inbound, email, extra, transport)
  237. }
  238. // applyEndpointHostPath overrides the transport host header / path for a host
  239. // endpoint. It is a no-op for legacy externalProxy entries (which never carry
  240. // hostHeader/path) and only replaces keys the transport already emits, so it
  241. // cannot add spurious params to e.g. a tcp link.
  242. func applyEndpointHostPath(e ShareEndpoint, params map[string]string) {
  243. if e.ep == nil {
  244. return
  245. }
  246. if h, ok := e.ep["hostHeader"].(string); ok && h != "" {
  247. if _, exists := params["host"]; exists {
  248. params["host"] = h
  249. }
  250. }
  251. if p, ok := e.ep["path"].(string); ok && p != "" {
  252. if _, exists := params["path"]; exists {
  253. params["path"] = p
  254. }
  255. }
  256. }
  257. // isHostEndpoint reports whether ep was synthesized from a Host (vs a legacy
  258. // externalProxy entry), so host-only overrides stay off the legacy path.
  259. func isHostEndpoint(ep map[string]any) bool {
  260. v, _ := ep["isHost"].(bool)
  261. return v
  262. }
  263. // applyEndpointRealityParams overrides a reality link's SNI + fingerprint from a
  264. // host (reality's pbk/sid are inherited from the inbound, so they aren't touched).
  265. // Host-only: legacy externalProxy reality links are unchanged.
  266. func applyEndpointRealityParams(e ShareEndpoint, params map[string]string, security string) {
  267. if security != "reality" || e.ep == nil || !isHostEndpoint(e.ep) {
  268. return
  269. }
  270. if sni, ok := externalProxySNI(e.ep); ok {
  271. params["sni"] = sni
  272. }
  273. if fp, ok := e.ep["fingerprint"].(string); ok && fp != "" {
  274. params["fp"] = fp
  275. }
  276. }
  277. // applyEndpointAllowInsecure adds allowInsecure=1 to a TLS/Reality link when the
  278. // host opts into skipping cert verification. No-op for legacy externalProxy
  279. // entries (which never carry the key) and for plaintext (none) endpoints.
  280. func applyEndpointAllowInsecure(e ShareEndpoint, params map[string]string, security string) {
  281. if e.ep == nil || security == "none" {
  282. return
  283. }
  284. if ai, ok := e.ep["allowInsecure"].(bool); ok && ai {
  285. params["allowInsecure"] = "1"
  286. }
  287. }
  288. // applyEndpointHostPathObj is applyEndpointHostPath for the VMess object form.
  289. func applyEndpointHostPathObj(e ShareEndpoint, obj map[string]any) {
  290. if e.ep == nil {
  291. return
  292. }
  293. if h, ok := e.ep["hostHeader"].(string); ok && h != "" {
  294. if _, exists := obj["host"]; exists {
  295. obj["host"] = h
  296. }
  297. }
  298. if p, ok := e.ep["path"].(string); ok && p != "" {
  299. if _, exists := obj["path"]; exists {
  300. obj["path"] = p
  301. }
  302. }
  303. }