host_sub.go 10 KB

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