| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952 |
- package sub
- import (
- "encoding/base64"
- "fmt"
- "maps"
- "net"
- "net/url"
- "slices"
- "strings"
- "time"
- "github.com/gin-gonic/gin"
- "github.com/goccy/go-json"
- "github.com/mhsanaei/3x-ui/v3/database"
- "github.com/mhsanaei/3x-ui/v3/database/model"
- "github.com/mhsanaei/3x-ui/v3/logger"
- "github.com/mhsanaei/3x-ui/v3/util/common"
- "github.com/mhsanaei/3x-ui/v3/util/random"
- "github.com/mhsanaei/3x-ui/v3/web/service"
- "github.com/mhsanaei/3x-ui/v3/xray"
- )
- // SubService provides business logic for generating subscription links and managing subscription data.
- type SubService struct {
- address string
- showInfo bool
- remarkModel string
- datepicker string
- emailInRemark bool
- inboundService service.InboundService
- settingService service.SettingService
- // nodesByID is populated per request from the Node table so
- // resolveInboundAddress can return the node's address for any
- // inbound whose NodeID is set. Keeps the per-link host derivation
- // O(1) instead of O(N) DB hits.
- nodesByID map[int]*model.Node
- }
- // NewSubService creates a new subscription service with the given configuration.
- func NewSubService(showInfo bool, remarkModel string) *SubService {
- return &SubService{
- showInfo: showInfo,
- remarkModel: remarkModel,
- }
- }
- // PrepareForRequest sets per-request state (host + nodes map) on the
- // shared SubService. Called by every entry point — GetSubs, GetJson,
- // GetClash — so resolveInboundAddress sees the right host and the
- // freshly-loaded node map regardless of which sub flavour the client
- // hit.
- func (s *SubService) PrepareForRequest(host string) {
- if !isRoutableHost(host) {
- if d := s.configuredPublicHost(); d != "" {
- host = d
- } else if isLoopbackHost(host) {
- host = "localhost"
- }
- }
- s.address = host
- s.loadNodes()
- }
- func (s *SubService) configuredPublicHost() string {
- if d, err := s.settingService.GetSubDomain(); err == nil && d != "" {
- return d
- }
- if d, err := s.settingService.GetWebDomain(); err == nil && d != "" {
- return d
- }
- return ""
- }
- func isRoutableHost(host string) bool {
- if host == "" {
- return false
- }
- if ip := net.ParseIP(strings.Trim(host, "[]")); ip != nil {
- return !ip.IsLoopback() && !ip.IsUnspecified()
- }
- return true
- }
- func isLoopbackHost(host string) bool {
- ip := net.ParseIP(strings.Trim(host, "[]"))
- return ip != nil && ip.IsLoopback()
- }
- // GetSubs retrieves subscription links for a given subscription ID and host.
- func (s *SubService) GetSubs(subId string, host string) ([]string, []string, int64, xray.ClientTraffic, error) {
- s.PrepareForRequest(host)
- var result []string
- var emails []string
- var traffic xray.ClientTraffic
- var hasEnabledClient bool
- inbounds, err := s.getInboundsBySubId(subId)
- if err != nil {
- return nil, nil, 0, traffic, err
- }
- if len(inbounds) == 0 {
- return nil, nil, 0, traffic, nil
- }
- s.datepicker, err = s.settingService.GetDatepicker()
- if err != nil {
- s.datepicker = "gregorian"
- }
- s.emailInRemark, err = s.settingService.GetSubEmailInRemark()
- if err != nil {
- s.emailInRemark = true
- }
- seenEmails := make(map[string]struct{})
- for _, inbound := range inbounds {
- clients, err := s.inboundService.GetClients(inbound)
- if err != nil {
- logger.Error("SubService - GetClients: Unable to get clients from inbound")
- }
- if clients == nil {
- continue
- }
- s.projectThroughFallbackMaster(inbound)
- for _, client := range clients {
- if client.SubID == subId {
- if client.Enable {
- hasEnabledClient = true
- }
- result = append(result, s.GetLink(inbound, client.Email))
- emails = append(emails, client.Email)
- seenEmails[client.Email] = struct{}{}
- }
- }
- }
- uniqueEmails := make([]string, 0, len(seenEmails))
- for e := range seenEmails {
- uniqueEmails = append(uniqueEmails, e)
- }
- traffic, lastOnline := s.AggregateTrafficByEmails(uniqueEmails)
- traffic.Enable = hasEnabledClient
- return result, emails, lastOnline, traffic, nil
- }
- // AggregateTrafficByEmails resolves traffic for every email in one
- // query and folds the rows into a single ClientTraffic + lastOnline.
- // xray.ClientTraffic.Email is globally unique, so a multi-inbound
- // client's single row is attached to exactly one inbound — iterating
- // per-inbound ClientStats would miss it on the others. Used by GetSubs,
- // SubClashService.GetClash, and SubJsonService.GetJson to keep the
- // sub-info header consistent across all three formats.
- func (s *SubService) AggregateTrafficByEmails(emails []string) (xray.ClientTraffic, int64) {
- var agg xray.ClientTraffic
- var lastOnline int64
- if len(emails) == 0 {
- return agg, 0
- }
- var rows []xray.ClientTraffic
- if err := database.GetDB().
- Model(&xray.ClientTraffic{}).
- Where("email IN ?", emails).
- Find(&rows).Error; err != nil {
- logger.Warning("SubService - AggregateTrafficByEmails: load by email:", err)
- return agg, 0
- }
- now := time.Now().UnixMilli()
- for i, ct := range rows {
- if ct.LastOnline > lastOnline {
- lastOnline = ct.LastOnline
- }
- if i == 0 {
- agg.Up = ct.Up
- agg.Down = ct.Down
- agg.Total = ct.Total
- agg.ExpiryTime = subscriptionExpiryFromClient(now, ct.ExpiryTime)
- continue
- }
- agg.Up += ct.Up
- agg.Down += ct.Down
- if agg.Total == 0 || ct.Total == 0 {
- agg.Total = 0
- } else {
- agg.Total += ct.Total
- }
- normalized := subscriptionExpiryFromClient(now, ct.ExpiryTime)
- if normalized != agg.ExpiryTime {
- agg.ExpiryTime = 0
- }
- }
- return agg, lastOnline
- }
- func subscriptionExpiryFromClient(nowMs, expiryTime int64) int64 {
- if expiryTime > 0 {
- return expiryTime
- }
- if expiryTime < 0 {
- return nowMs + (-expiryTime)
- }
- return 0
- }
- func (s *SubService) getInboundsBySubId(subId string) ([]*model.Inbound, error) {
- db := database.GetDB()
- var inbounds []*model.Inbound
- err := db.Model(model.Inbound{}).Preload("ClientStats").Where(`id in (
- SELECT DISTINCT inbounds.id
- FROM inbounds
- JOIN client_inbounds ON client_inbounds.inbound_id = inbounds.id
- JOIN clients ON clients.id = client_inbounds.client_id
- WHERE
- inbounds.protocol in ('vmess','vless','trojan','shadowsocks','hysteria')
- AND clients.sub_id = ? AND inbounds.enable = ?
- )`, subId, true).Find(&inbounds).Error
- if err != nil {
- return nil, err
- }
- return inbounds, nil
- }
- // projectThroughFallbackMaster mutates the inbound in place so its
- // Listen/Port/StreamSettings reflect the externally reachable master
- // when applicable. Covers both fallback mechanisms:
- // - panel-tracked: an inbound_fallbacks row where child_id = inbound.Id
- // - legacy unix-socket: inbound.Listen begins with "@" and some VLESS/
- // Trojan inbound's settings.fallbacks references that listen address
- //
- // Returns true when a projection happened; sub services call this before
- // generating links so a child VLESS-WS bound to 127.0.0.1 emits the
- // master's :443 + TLS state instead of its own loopback endpoint.
- func (s *SubService) projectThroughFallbackMaster(inbound *model.Inbound) bool {
- if inbound == nil {
- return false
- }
- db := database.GetDB()
- var master *model.Inbound
- var rule model.InboundFallback
- if err := db.Where("child_id = ?", inbound.Id).
- Order("sort_order ASC, id ASC").
- First(&rule).Error; err == nil {
- var m model.Inbound
- if err := db.Where("id = ?", rule.MasterId).First(&m).Error; err == nil {
- master = &m
- }
- }
- if master == nil && len(inbound.Listen) > 0 && inbound.Listen[0] == '@' {
- var m model.Inbound
- if err := db.Model(model.Inbound{}).
- Where("JSON_TYPE(settings, '$.fallbacks') = 'array'").
- Where("EXISTS (SELECT * FROM json_each(settings, '$.fallbacks') WHERE json_extract(value, '$.dest') = ?)", inbound.Listen).
- First(&m).Error; err == nil {
- master = &m
- }
- }
- if master == nil {
- return false
- }
- inbound.StreamSettings = mergeStreamFromMaster(inbound.StreamSettings, master.StreamSettings)
- inbound.Listen = master.Listen
- inbound.Port = master.Port
- return true
- }
- // mergeStreamFromMaster copies the master's security + tlsSettings +
- // realitySettings + externalProxy onto the child's stream so the child's
- // link advertises the master's TLS / Reality state. Transport (network
- // + ws/grpc/etc. settings) stays the child's.
- func mergeStreamFromMaster(childStream, masterStream string) string {
- var stream map[string]any
- json.Unmarshal([]byte(childStream), &stream)
- if stream == nil {
- stream = map[string]any{}
- }
- var mst map[string]any
- json.Unmarshal([]byte(masterStream), &mst)
- if mst == nil {
- return childStream
- }
- stream["security"] = mst["security"]
- if v, ok := mst["tlsSettings"]; ok {
- stream["tlsSettings"] = v
- } else {
- delete(stream, "tlsSettings")
- }
- if v, ok := mst["realitySettings"]; ok {
- stream["realitySettings"] = v
- } else {
- delete(stream, "realitySettings")
- }
- if v, ok := mst["externalProxy"]; ok {
- stream["externalProxy"] = v
- }
- out, err := json.MarshalIndent(stream, "", " ")
- if err != nil {
- return childStream
- }
- return string(out)
- }
- // GetLink dispatches to the protocol-specific generator for one (inbound, client)
- // pair. Returns "" when the inbound's protocol doesn't produce a subscription URL
- // (socks, http, mixed, wireguard, dokodemo, tunnel). The returned string may
- // contain multiple `\n`-separated URLs when the inbound has externalProxy set.
- func (s *SubService) GetLink(inbound *model.Inbound, email string) string {
- switch inbound.Protocol {
- case "vmess":
- return s.genVmessLink(inbound, email)
- case "vless":
- return s.genVlessLink(inbound, email)
- case "trojan":
- return s.genTrojanLink(inbound, email)
- case "shadowsocks":
- return s.genShadowsocksLink(inbound, email)
- case "hysteria":
- return s.genHysteriaLink(inbound, email)
- }
- return ""
- }
- // Protocol link generators are intentionally ordered as:
- // vmess -> vless -> trojan -> shadowsocks -> hysteria.
- func (s *SubService) genVmessLink(inbound *model.Inbound, email string) string {
- if inbound.Protocol != model.VMESS {
- return ""
- }
- address := s.resolveInboundAddress(inbound)
- obj := map[string]any{
- "v": "2",
- "add": address,
- "port": inbound.Port,
- "type": "none",
- }
- stream := unmarshalStreamSettings(inbound.StreamSettings)
- network, _ := stream["network"].(string)
- applyVmessNetworkParams(stream, network, obj)
- if finalmask, ok := stream["finalmask"].(map[string]any); ok {
- applyFinalMaskObj(finalmask, obj)
- }
- security, _ := stream["security"].(string)
- obj["tls"] = security
- if security == "tls" {
- applyVmessTLSParams(stream, obj)
- }
- clients, _ := s.inboundService.GetClients(inbound)
- clientIndex := findClientIndex(clients, email)
- obj["id"] = clients[clientIndex].ID
- obj["scy"] = clients[clientIndex].Security
- externalProxies, _ := stream["externalProxy"].([]any)
- if len(externalProxies) > 0 {
- return s.buildVmessExternalProxyLinks(externalProxies, obj, inbound, email)
- }
- obj["ps"] = s.genRemark(inbound, email, "")
- return buildVmessLink(obj)
- }
- func (s *SubService) genVlessLink(inbound *model.Inbound, email string) string {
- if inbound.Protocol != model.VLESS {
- return ""
- }
- address := s.resolveInboundAddress(inbound)
- stream := unmarshalStreamSettings(inbound.StreamSettings)
- clients, _ := s.inboundService.GetClients(inbound)
- clientIndex := findClientIndex(clients, email)
- uuid := clients[clientIndex].ID
- port := inbound.Port
- streamNetwork := stream["network"].(string)
- params := make(map[string]string)
- params["type"] = streamNetwork
- // Add encryption parameter for VLESS from inbound settings
- var settings map[string]any
- json.Unmarshal([]byte(inbound.Settings), &settings)
- if encryption, ok := settings["encryption"].(string); ok {
- params["encryption"] = encryption
- }
- applyShareNetworkParams(stream, streamNetwork, params)
- if finalmask, ok := stream["finalmask"].(map[string]any); ok {
- applyFinalMaskParams(finalmask, params)
- }
- security, _ := stream["security"].(string)
- switch security {
- case "tls":
- applyShareTLSParams(stream, params)
- if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
- params["flow"] = clients[clientIndex].Flow
- }
- case "reality":
- applyShareRealityParams(stream, params)
- if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
- params["flow"] = clients[clientIndex].Flow
- }
- default:
- params["security"] = "none"
- }
- externalProxies, _ := stream["externalProxy"].([]any)
- if len(externalProxies) > 0 {
- return s.buildExternalProxyURLLinks(
- externalProxies,
- params,
- security,
- func(dest string, port int) string {
- return fmt.Sprintf("vless://%s@%s:%d", uuid, dest, port)
- },
- func(ep map[string]any) string {
- return s.genRemark(inbound, email, ep["remark"].(string))
- },
- )
- }
- link := fmt.Sprintf("vless://%s@%s:%d", uuid, address, port)
- return buildLinkWithParams(link, params, s.genRemark(inbound, email, ""))
- }
- func (s *SubService) genTrojanLink(inbound *model.Inbound, email string) string {
- if inbound.Protocol != model.Trojan {
- return ""
- }
- address := s.resolveInboundAddress(inbound)
- stream := unmarshalStreamSettings(inbound.StreamSettings)
- clients, _ := s.inboundService.GetClients(inbound)
- clientIndex := findClientIndex(clients, email)
- password := encodeUserinfo(clients[clientIndex].Password)
- port := inbound.Port
- streamNetwork := stream["network"].(string)
- params := make(map[string]string)
- params["type"] = streamNetwork
- applyShareNetworkParams(stream, streamNetwork, params)
- if finalmask, ok := stream["finalmask"].(map[string]any); ok {
- applyFinalMaskParams(finalmask, params)
- }
- security, _ := stream["security"].(string)
- switch security {
- case "tls":
- applyShareTLSParams(stream, params)
- case "reality":
- applyShareRealityParams(stream, params)
- if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
- params["flow"] = clients[clientIndex].Flow
- }
- default:
- params["security"] = "none"
- }
- externalProxies, _ := stream["externalProxy"].([]any)
- if len(externalProxies) > 0 {
- return s.buildExternalProxyURLLinks(
- externalProxies,
- params,
- security,
- func(dest string, port int) string {
- return fmt.Sprintf("trojan://%s@%s:%d", password, dest, port)
- },
- func(ep map[string]any) string {
- return s.genRemark(inbound, email, ep["remark"].(string))
- },
- )
- }
- link := fmt.Sprintf("trojan://%s@%s:%d", password, address, port)
- return buildLinkWithParams(link, params, s.genRemark(inbound, email, ""))
- }
- // encodeUserinfo percent-encodes a userinfo (password/auth) value so it
- // can be safely embedded in a `scheme://<value>@host:port` URL. RFC 3986
- // allows `=` in userinfo as a sub-delim, but several Trojan and Hysteria
- // clients reject share-links where the password contains literal `/`
- // or `=` (notably the common base64-with-padding shape produced by the
- // panel). Encode them too — this matches encodeURIComponent() on the
- // frontend and round-trips cleanly through net/url's parser.
- func encodeUserinfo(s string) string {
- return strings.ReplaceAll(url.QueryEscape(s), "+", "%20")
- }
- func (s *SubService) genShadowsocksLink(inbound *model.Inbound, email string) string {
- if inbound.Protocol != model.Shadowsocks {
- return ""
- }
- address := s.resolveInboundAddress(inbound)
- stream := unmarshalStreamSettings(inbound.StreamSettings)
- clients, _ := s.inboundService.GetClients(inbound)
- var settings map[string]any
- json.Unmarshal([]byte(inbound.Settings), &settings)
- inboundPassword := settings["password"].(string)
- method := settings["method"].(string)
- clientIndex := findClientIndex(clients, email)
- streamNetwork := stream["network"].(string)
- params := make(map[string]string)
- params["type"] = streamNetwork
- applyShareNetworkParams(stream, streamNetwork, params)
- if finalmask, ok := stream["finalmask"].(map[string]any); ok {
- applyFinalMaskParams(finalmask, params)
- }
- security, _ := stream["security"].(string)
- if security == "tls" {
- applyShareTLSParams(stream, params)
- }
- encPart := fmt.Sprintf("%s:%s", method, clients[clientIndex].Password)
- if method[0] == '2' {
- encPart = fmt.Sprintf("%s:%s:%s", method, inboundPassword, clients[clientIndex].Password)
- }
- externalProxies, _ := stream["externalProxy"].([]any)
- if len(externalProxies) > 0 {
- proxyParams := cloneStringMap(params)
- proxyParams["security"] = security
- return s.buildExternalProxyURLLinks(
- externalProxies,
- proxyParams,
- security,
- func(dest string, port int) string {
- return fmt.Sprintf("ss://%s@%s:%d", base64.RawURLEncoding.EncodeToString([]byte(encPart)), dest, port)
- },
- func(ep map[string]any) string {
- return s.genRemark(inbound, email, ep["remark"].(string))
- },
- )
- }
- link := fmt.Sprintf("ss://%s@%s:%d", base64.RawURLEncoding.EncodeToString([]byte(encPart)), address, inbound.Port)
- return buildLinkWithParams(link, params, s.genRemark(inbound, email, ""))
- }
- func (s *SubService) genHysteriaLink(inbound *model.Inbound, email string) string {
- if inbound.Protocol != model.Hysteria {
- return ""
- }
- var stream map[string]any
- json.Unmarshal([]byte(inbound.StreamSettings), &stream)
- clients, _ := s.inboundService.GetClients(inbound)
- clientIndex := -1
- for i, client := range clients {
- if client.Email == email {
- clientIndex = i
- break
- }
- }
- auth := encodeUserinfo(clients[clientIndex].Auth)
- params := make(map[string]string)
- params["security"] = "tls"
- tlsSetting, _ := stream["tlsSettings"].(map[string]any)
- alpns, _ := tlsSetting["alpn"].([]any)
- var alpn []string
- for _, a := range alpns {
- alpn = append(alpn, a.(string))
- }
- if len(alpn) > 0 {
- params["alpn"] = strings.Join(alpn, ",")
- }
- if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
- params["sni"], _ = sniValue.(string)
- }
- tlsSettings, _ := searchKey(tlsSetting, "settings")
- if tlsSetting != nil {
- if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
- params["fp"], _ = fpValue.(string)
- }
- if insecure, ok := searchKey(tlsSettings, "allowInsecure"); ok {
- if insecure.(bool) {
- params["insecure"] = "1"
- }
- }
- }
- // salamander obfs (Hysteria2). The panel-side link generator already
- // emits these; keep the subscription output in sync so a client has
- // the obfs password to match the server.
- if finalmask, ok := stream["finalmask"].(map[string]any); ok {
- applyFinalMaskParams(finalmask, params)
- if udpMasks, ok := finalmask["udp"].([]any); ok {
- for _, m := range udpMasks {
- mask, _ := m.(map[string]any)
- if mask == nil || mask["type"] != "salamander" {
- continue
- }
- settings, _ := mask["settings"].(map[string]any)
- if pw, ok := settings["password"].(string); ok && pw != "" {
- params["obfs"] = "salamander"
- params["obfs-password"] = pw
- break
- }
- }
- }
- }
- var settings map[string]any
- json.Unmarshal([]byte(inbound.Settings), &settings)
- version, _ := settings["version"].(float64)
- protocol := "hysteria2"
- if int(version) == 1 {
- protocol = "hysteria"
- }
- // Fan out one link per External Proxy entry if any. Previously this
- // generator ignored `externalProxy` entirely, so the link kept the
- // server's own IP/port even when the admin configured an alternate
- // endpoint (e.g. a CDN hostname + port that forwards to the node).
- // Matches the behaviour of genVlessLink / genTrojanLink / ….
- externalProxies, _ := stream["externalProxy"].([]any)
- if len(externalProxies) > 0 {
- links := make([]string, 0, len(externalProxies))
- for _, externalProxy := range externalProxies {
- ep, ok := externalProxy.(map[string]any)
- if !ok {
- continue
- }
- dest, _ := ep["dest"].(string)
- portF, okPort := ep["port"].(float64)
- if dest == "" || !okPort {
- continue
- }
- epRemark, _ := ep["remark"].(string)
- link := fmt.Sprintf("%s://%s@%s:%d", protocol, auth, dest, int(portF))
- links = append(links, buildLinkWithParams(link, params, s.genRemark(inbound, email, epRemark)))
- }
- return strings.Join(links, "\n")
- }
- // No external proxy configured — use the inbound's resolved address so
- // node-managed inbounds get the node's host instead of the central panel's.
- link := fmt.Sprintf("%s://%s@%s:%d", protocol, auth, s.resolveInboundAddress(inbound), inbound.Port)
- return buildLinkWithParams(link, params, s.genRemark(inbound, email, ""))
- }
- // loadNodes refreshes nodesByID from the DB. Called once per request so
- // the per-inbound resolveInboundAddress lookups are pure map reads.
- // We filter to address != ” so a half-configured node row doesn't
- // accidentally produce a useless host like "https://:2053".
- func (s *SubService) loadNodes() {
- db := database.GetDB()
- var nodes []*model.Node
- if err := db.Model(&model.Node{}).Where("address != ''").Find(&nodes).Error; err != nil {
- logger.Warning("subscription: load nodes failed:", err)
- s.nodesByID = nil
- return
- }
- m := make(map[int]*model.Node, len(nodes))
- for _, n := range nodes {
- m[n.Id] = n
- }
- s.nodesByID = m
- }
- // resolveInboundAddress returns the node's address for node-managed inbounds,
- // otherwise the subscriber's host (s.address). The inbound's bind Listen is
- // deliberately ignored: it's a server-side address, not a client-reachable
- // host, so operators advertise a specific endpoint via External Proxy instead.
- func (s *SubService) resolveInboundAddress(inbound *model.Inbound) string {
- if inbound.NodeID != nil && s.nodesByID != nil {
- if n, ok := s.nodesByID[*inbound.NodeID]; ok && n.Address != "" {
- return n.Address
- }
- }
- return s.address
- }
- func findClientIndex(clients []model.Client, email string) int {
- for i, client := range clients {
- if client.Email == email {
- return i
- }
- }
- return -1
- }
- func unmarshalStreamSettings(streamSettings string) map[string]any {
- var stream map[string]any
- json.Unmarshal([]byte(streamSettings), &stream)
- return stream
- }
- func applyPathAndHostParams(settings map[string]any, params map[string]string) {
- params["path"] = settings["path"].(string)
- if host, ok := settings["host"].(string); ok && len(host) > 0 {
- params["host"] = host
- } else {
- headers, _ := settings["headers"].(map[string]any)
- params["host"] = searchHost(headers)
- }
- }
- func applyPathAndHostObj(settings map[string]any, obj map[string]any) {
- obj["path"] = settings["path"].(string)
- if host, ok := settings["host"].(string); ok && len(host) > 0 {
- obj["host"] = host
- } else {
- headers, _ := settings["headers"].(map[string]any)
- obj["host"] = searchHost(headers)
- }
- }
- func applyShareNetworkParams(stream map[string]any, streamNetwork string, params map[string]string) {
- switch streamNetwork {
- case "tcp":
- tcp, _ := stream["tcpSettings"].(map[string]any)
- header, _ := tcp["header"].(map[string]any)
- typeStr, _ := header["type"].(string)
- if typeStr == "http" {
- request := header["request"].(map[string]any)
- requestPath, _ := request["path"].([]any)
- params["path"] = requestPath[0].(string)
- host := ""
- if response, ok := header["response"].(map[string]any); ok {
- if respHeaders, ok := response["headers"].(map[string]any); ok {
- host = searchHost(respHeaders)
- }
- }
- if host == "" {
- headers, _ := request["headers"].(map[string]any)
- host = searchHost(headers)
- }
- params["host"] = host
- params["headerType"] = "http"
- }
- case "kcp":
- applyKcpShareParams(stream, params)
- case "ws":
- ws, _ := stream["wsSettings"].(map[string]any)
- applyPathAndHostParams(ws, params)
- case "grpc":
- grpc, _ := stream["grpcSettings"].(map[string]any)
- params["serviceName"] = grpc["serviceName"].(string)
- params["authority"], _ = grpc["authority"].(string)
- if grpc["multiMode"].(bool) {
- params["mode"] = "multi"
- }
- case "httpupgrade":
- httpupgrade, _ := stream["httpupgradeSettings"].(map[string]any)
- applyPathAndHostParams(httpupgrade, params)
- case "xhttp":
- xhttp, _ := stream["xhttpSettings"].(map[string]any)
- applyXhttpExtraParams(xhttp, params)
- }
- }
- // applyXhttpExtraObj copies the bidirectional xhttp settings into the
- // VMess base64 JSON link object. VMess supports arbitrary keys, so we
- // flatten the SplitHTTPConfig "extra" fields directly onto obj.
- func applyXhttpExtraObj(xhttp map[string]any, obj map[string]any) {
- if xpb, ok := xhttp["xPaddingBytes"].(string); ok && len(xpb) > 0 {
- obj["x_padding_bytes"] = xpb
- }
- maps.Copy(obj, buildXhttpExtra(xhttp))
- }
- func applyVmessNetworkParams(stream map[string]any, network string, obj map[string]any) {
- obj["net"] = network
- switch network {
- case "tcp":
- tcp, _ := stream["tcpSettings"].(map[string]any)
- header, _ := tcp["header"].(map[string]any)
- typeStr, _ := header["type"].(string)
- obj["type"] = typeStr
- if typeStr == "http" {
- request := header["request"].(map[string]any)
- requestPath, _ := request["path"].([]any)
- obj["path"] = requestPath[0].(string)
- host := ""
- if response, ok := header["response"].(map[string]any); ok {
- if respHeaders, ok := response["headers"].(map[string]any); ok {
- host = searchHost(respHeaders)
- }
- }
- if host == "" {
- headers, _ := request["headers"].(map[string]any)
- host = searchHost(headers)
- }
- obj["host"] = host
- }
- case "kcp":
- applyKcpShareObj(stream, obj)
- case "ws":
- ws, _ := stream["wsSettings"].(map[string]any)
- applyPathAndHostObj(ws, obj)
- case "grpc":
- grpc, _ := stream["grpcSettings"].(map[string]any)
- obj["path"] = grpc["serviceName"].(string)
- obj["authority"] = grpc["authority"].(string)
- if grpc["multiMode"].(bool) {
- obj["type"] = "multi"
- }
- case "httpupgrade":
- httpupgrade, _ := stream["httpupgradeSettings"].(map[string]any)
- applyPathAndHostObj(httpupgrade, obj)
- case "xhttp":
- xhttp, _ := stream["xhttpSettings"].(map[string]any)
- applyPathAndHostObj(xhttp, obj)
- if mode, ok := xhttp["mode"].(string); ok {
- obj["mode"] = mode
- }
- applyXhttpExtraObj(xhttp, obj)
- }
- }
- func applyShareTLSParams(stream map[string]any, params map[string]string) {
- params["security"] = "tls"
- tlsSetting, _ := stream["tlsSettings"].(map[string]any)
- alpns, _ := tlsSetting["alpn"].([]any)
- var alpn []string
- for _, a := range alpns {
- alpn = append(alpn, a.(string))
- }
- if len(alpn) > 0 {
- params["alpn"] = strings.Join(alpn, ",")
- }
- if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
- params["sni"], _ = sniValue.(string)
- }
- tlsSettings, _ := searchKey(tlsSetting, "settings")
- if tlsSetting != nil {
- if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
- params["fp"], _ = fpValue.(string)
- }
- if pins, ok := pinnedSha256List(tlsSettings); ok {
- params["pcs"] = strings.Join(pins, ",")
- }
- }
- }
- func applyVmessTLSParams(stream map[string]any, obj map[string]any) {
- tlsSetting, _ := stream["tlsSettings"].(map[string]any)
- alpns, _ := tlsSetting["alpn"].([]any)
- if len(alpns) > 0 {
- var alpn []string
- for _, a := range alpns {
- alpn = append(alpn, a.(string))
- }
- obj["alpn"] = strings.Join(alpn, ",")
- }
- if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
- obj["sni"], _ = sniValue.(string)
- }
- tlsSettings, _ := searchKey(tlsSetting, "settings")
- if tlsSetting != nil {
- if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
- obj["fp"], _ = fpValue.(string)
- }
- if pins, ok := pinnedSha256List(tlsSettings); ok {
- obj["pcs"] = strings.Join(pins, ",")
- }
- }
- }
- // pinnedSha256List extracts tlsSettings.settings.pinnedPeerCertSha256 as a
- // []string. The field is panel-only (stripped before the run-config reaches
- // xray-core via web/service/xray.go) but flows into share links so clients
- // can pin the server's certificate hash.
- func pinnedSha256List(tlsClientSettings any) ([]string, bool) {
- raw, ok := searchKey(tlsClientSettings, "pinnedPeerCertSha256")
- if !ok {
- return nil, false
- }
- arr, ok := raw.([]any)
- if !ok || len(arr) == 0 {
- return nil, false
- }
- out := make([]string, 0, len(arr))
- for _, v := range arr {
- s, ok := v.(string)
- if !ok || s == "" {
- continue
- }
- out = append(out, s)
- }
- if len(out) == 0 {
- return nil, false
- }
- return out, true
- }
- func applyShareRealityParams(stream map[string]any, params map[string]string) {
- params["security"] = "reality"
- realitySetting, _ := stream["realitySettings"].(map[string]any)
- realitySettings, _ := searchKey(realitySetting, "settings")
- if realitySetting != nil {
- if sniValue, ok := searchKey(realitySetting, "serverNames"); ok {
- sNames, _ := sniValue.([]any)
- params["sni"] = sNames[random.Num(len(sNames))].(string)
- }
- if pbkValue, ok := searchKey(realitySettings, "publicKey"); ok {
- params["pbk"], _ = pbkValue.(string)
- }
- if sidValue, ok := searchKey(realitySetting, "shortIds"); ok {
- shortIds, _ := sidValue.([]any)
- params["sid"] = shortIds[random.Num(len(shortIds))].(string)
- }
- if fpValue, ok := searchKey(realitySettings, "fingerprint"); ok {
- if fp, ok := fpValue.(string); ok && len(fp) > 0 {
- params["fp"] = fp
- }
- }
- if pqvValue, ok := searchKey(realitySettings, "mldsa65Verify"); ok {
- if pqv, ok := pqvValue.(string); ok && len(pqv) > 0 {
- params["pqv"] = pqv
- }
- }
- params["spx"] = "/" + random.Seq(15)
- }
- }
- func buildVmessLink(obj map[string]any) string {
- jsonStr, _ := json.MarshalIndent(obj, "", " ")
- return "vmess://" + base64.StdEncoding.EncodeToString(jsonStr)
- }
- func cloneVmessShareObj(baseObj map[string]any, newSecurity string) map[string]any {
- newObj := map[string]any{}
- for key, value := range baseObj {
- if !(newSecurity == "none" && (key == "alpn" || key == "sni" || key == "fp")) {
- newObj[key] = value
- }
- }
- return newObj
- }
- func applyExternalProxyTLSObj(ep map[string]any, obj map[string]any, security string) {
- if security != "tls" {
- return
- }
- if sni, ok := externalProxySNI(ep); ok {
- obj["sni"] = sni
- }
- if fp, ok := ep["fingerprint"].(string); ok && fp != "" {
- obj["fp"] = fp
- }
- if alpn, ok := externalProxyALPN(ep["alpn"]); ok {
- obj["alpn"] = alpn
- }
- }
- func applyExternalProxyTLSParams(ep map[string]any, params map[string]string, security string) {
- if security != "tls" {
- return
- }
- if sni, ok := externalProxySNI(ep); ok {
- params["sni"] = sni
- }
- if fp, ok := ep["fingerprint"].(string); ok && fp != "" {
- params["fp"] = fp
- }
- if alpn, ok := externalProxyALPN(ep["alpn"]); ok {
- params["alpn"] = alpn
- }
- }
- // cloneStreamForExternalProxy returns a shallow clone of stream with
- // tlsSettings (and its nested settings map) deep-copied. The external
- // proxy loop mutates tlsSettings per iteration, so without isolating
- // those maps each proxy's SNI/fingerprint/ALPN would leak into the next.
- func cloneStreamForExternalProxy(stream map[string]any) map[string]any {
- out := cloneMap(stream)
- ts, ok := out["tlsSettings"].(map[string]any)
- if !ok || ts == nil {
- return out
- }
- clonedTs := cloneMap(ts)
- if inner, ok := clonedTs["settings"].(map[string]any); ok && inner != nil {
- clonedTs["settings"] = cloneMap(inner)
- }
- out["tlsSettings"] = clonedTs
- return out
- }
- func applyExternalProxyTLSToStream(ep map[string]any, stream map[string]any, security string) {
- if security != "tls" {
- return
- }
- tlsSettings, _ := stream["tlsSettings"].(map[string]any)
- if tlsSettings == nil {
- tlsSettings = map[string]any{}
- stream["tlsSettings"] = tlsSettings
- }
- if sni, ok := externalProxySNI(ep); ok {
- tlsSettings["serverName"] = sni
- }
- if fp, ok := ep["fingerprint"].(string); ok && fp != "" {
- tlsSettings["fingerprint"] = fp
- settings, _ := tlsSettings["settings"].(map[string]any)
- if settings == nil {
- settings = map[string]any{}
- tlsSettings["settings"] = settings
- }
- settings["fingerprint"] = fp
- }
- if alpn, ok := externalProxyALPNList(ep["alpn"]); ok {
- tlsSettings["alpn"] = alpn
- }
- }
- func externalProxySNI(ep map[string]any) (string, bool) {
- if sni, ok := ep["sni"].(string); ok && sni != "" {
- return sni, true
- }
- return "", false
- }
- func externalProxyALPN(value any) (string, bool) {
- switch v := value.(type) {
- case string:
- return v, v != ""
- case []string:
- if len(v) == 0 {
- return "", false
- }
- return strings.Join(v, ","), true
- case []any:
- alpn := make([]string, 0, len(v))
- for _, item := range v {
- if s, ok := item.(string); ok && s != "" {
- alpn = append(alpn, s)
- }
- }
- if len(alpn) == 0 {
- return "", false
- }
- return strings.Join(alpn, ","), true
- default:
- return "", false
- }
- }
- func externalProxyALPNList(value any) ([]any, bool) {
- switch v := value.(type) {
- case string:
- if v == "" {
- return nil, false
- }
- parts := strings.Split(v, ",")
- out := make([]any, 0, len(parts))
- for _, part := range parts {
- if part = strings.TrimSpace(part); part != "" {
- out = append(out, part)
- }
- }
- return out, len(out) > 0
- case []string:
- out := make([]any, 0, len(v))
- for _, item := range v {
- if item != "" {
- out = append(out, item)
- }
- }
- return out, len(out) > 0
- case []any:
- out := make([]any, 0, len(v))
- for _, item := range v {
- if s, ok := item.(string); ok && s != "" {
- out = append(out, s)
- }
- }
- return out, len(out) > 0
- default:
- return nil, false
- }
- }
- func (s *SubService) buildVmessExternalProxyLinks(externalProxies []any, baseObj map[string]any, inbound *model.Inbound, email string) string {
- var links strings.Builder
- for index, externalProxy := range externalProxies {
- ep, _ := externalProxy.(map[string]any)
- newSecurity, _ := ep["forceTls"].(string)
- securityToApply := baseObj["tls"].(string)
- if newSecurity != "same" {
- securityToApply = newSecurity
- }
- newObj := cloneVmessShareObj(baseObj, newSecurity)
- newObj["ps"] = s.genRemark(inbound, email, ep["remark"].(string))
- newObj["add"] = ep["dest"].(string)
- newObj["port"] = int(ep["port"].(float64))
- if newSecurity != "same" {
- newObj["tls"] = newSecurity
- }
- applyExternalProxyTLSObj(ep, newObj, securityToApply)
- if index > 0 {
- links.WriteString("\n")
- }
- links.WriteString(buildVmessLink(newObj))
- }
- return links.String()
- }
- // buildLinkWithParams appends ?query and #fragment to a pre-built
- // scheme://userinfo@host:port string without re-parsing it. The caller
- // has already escaped userinfo via encodeUserinfo (or chosen a base64
- // alphabet with no reserved chars); a url.Parse + .String() round-trip
- // would silently decode that escaping because Go's userinfo emitter
- // leaves sub-delims (=, +, ;) literal, which breaks Trojan/Hysteria/SS
- // clients that reject those chars in the password.
- func buildLinkWithParams(link string, params map[string]string, fragment string) string {
- return appendQueryAndFragment(link, params, fragment, "", false)
- }
- // buildLinkWithParamsAndSecurity is buildLinkWithParams plus an
- // external-proxy override: the `security` key in params is replaced with
- // the supplied value, and TLS hint fields (alpn/sni/fp) are stripped when
- // the override is `none`.
- func buildLinkWithParamsAndSecurity(link string, params map[string]string, fragment, security string, omitTLSFields bool) string {
- return appendQueryAndFragment(link, params, fragment, security, omitTLSFields)
- }
- func appendQueryAndFragment(link string, params map[string]string, fragment, securityOverride string, omitTLSFields bool) string {
- var sb strings.Builder
- sb.WriteString(link)
- if len(params) > 0 {
- q := url.Values{}
- for k, v := range params {
- if securityOverride != "" && k == "security" {
- v = securityOverride
- }
- if omitTLSFields && (k == "alpn" || k == "sni" || k == "fp") {
- continue
- }
- q.Set(k, v)
- }
- encoded := q.Encode()
- if encoded != "" {
- if strings.Contains(link, "?") {
- sb.WriteByte('&')
- } else {
- sb.WriteByte('?')
- }
- sb.WriteString(encoded)
- }
- }
- if fragment != "" {
- sb.WriteByte('#')
- // Match the frontend's encodeURIComponent(remark): spaces become
- // %20 (not + as in query strings).
- sb.WriteString(strings.ReplaceAll(url.QueryEscape(fragment), "+", "%20"))
- }
- return sb.String()
- }
- func (s *SubService) buildExternalProxyURLLinks(
- externalProxies []any,
- params map[string]string,
- baseSecurity string,
- makeLink func(dest string, port int) string,
- makeRemark func(ep map[string]any) string,
- ) string {
- links := make([]string, 0, len(externalProxies))
- for _, externalProxy := range externalProxies {
- ep, _ := externalProxy.(map[string]any)
- newSecurity, _ := ep["forceTls"].(string)
- dest, _ := ep["dest"].(string)
- port := int(ep["port"].(float64))
- securityToApply := baseSecurity
- if newSecurity != "same" {
- securityToApply = newSecurity
- }
- nextParams := cloneStringMap(params)
- applyExternalProxyTLSParams(ep, nextParams, securityToApply)
- links = append(
- links,
- buildLinkWithParamsAndSecurity(
- makeLink(dest, port),
- nextParams,
- makeRemark(ep),
- securityToApply,
- newSecurity == "none",
- ),
- )
- }
- return strings.Join(links, "\n")
- }
- func cloneStringMap(source map[string]string) map[string]string {
- cloned := make(map[string]string, len(source))
- maps.Copy(cloned, source)
- return cloned
- }
- func (s *SubService) genRemark(inbound *model.Inbound, email string, extra string) string {
- separationChar := string(s.remarkModel[0])
- orderChars := s.remarkModel[1:]
- orders := map[byte]string{
- 'i': "",
- 'e': "",
- 'o': "",
- }
- if len(email) > 0 && s.emailInRemark {
- orders['e'] = email
- }
- if len(inbound.Remark) > 0 {
- orders['i'] = inbound.Remark
- }
- if len(extra) > 0 {
- orders['o'] = extra
- }
- var remark []string
- for i := 0; i < len(orderChars); i++ {
- char := orderChars[i]
- order, exists := orders[char]
- if exists && order != "" {
- remark = append(remark, order)
- }
- }
- if s.showInfo {
- statsExist := false
- var stats xray.ClientTraffic
- for _, clientStat := range inbound.ClientStats {
- if clientStat.Email == email {
- stats = clientStat
- statsExist = true
- break
- }
- }
- // Get remained days
- if statsExist {
- if !stats.Enable {
- return fmt.Sprintf("⛔️N/A%s%s", separationChar, strings.Join(remark, separationChar))
- }
- if vol := stats.Total - (stats.Up + stats.Down); vol > 0 {
- remark = append(remark, fmt.Sprintf("%s%s", common.FormatTraffic(vol), "📊"))
- }
- now := time.Now().Unix()
- switch exp := stats.ExpiryTime / 1000; {
- case exp > 0:
- remainingSeconds := exp - now
- days := remainingSeconds / 86400
- hours := (remainingSeconds % 86400) / 3600
- minutes := (remainingSeconds % 3600) / 60
- if days > 0 {
- if hours > 0 {
- remark = append(remark, fmt.Sprintf("%dD,%dH⏳", days, hours))
- } else {
- remark = append(remark, fmt.Sprintf("%dD⏳", days))
- }
- } else if hours > 0 {
- remark = append(remark, fmt.Sprintf("%dH⏳", hours))
- } else {
- remark = append(remark, fmt.Sprintf("%dM⏳", minutes))
- }
- case exp < 0:
- days := exp / -86400
- hours := (exp % -86400) / 3600
- minutes := (exp % -3600) / 60
- if days > 0 {
- if hours > 0 {
- remark = append(remark, fmt.Sprintf("%dD,%dH⏳", days, hours))
- } else {
- remark = append(remark, fmt.Sprintf("%dD⏳", days))
- }
- } else if hours > 0 {
- remark = append(remark, fmt.Sprintf("%dH⏳", hours))
- } else {
- remark = append(remark, fmt.Sprintf("%dM⏳", minutes))
- }
- }
- }
- }
- return strings.Join(remark, separationChar)
- }
- func searchKey(data any, key string) (any, bool) {
- switch val := data.(type) {
- case map[string]any:
- for k, v := range val {
- if k == key {
- return v, true
- }
- if result, ok := searchKey(v, key); ok {
- return result, true
- }
- }
- case []any:
- for _, v := range val {
- if result, ok := searchKey(v, key); ok {
- return result, true
- }
- }
- }
- return nil, false
- }
- // buildXhttpExtra walks an xhttpSettings map and returns the JSON blob
- // that goes into the URL's `extra` param (or, for VMess, the link
- // object). Carries ONLY the bidirectional fields from xray-core's
- // SplitHTTPConfig — i.e. the ones the server enforces and the client
- // must match. Strictly one-sided fields are excluded:
- //
- // - server-only (noSSEHeader, scMaxBufferedPosts, scStreamUpServerSecs,
- // serverMaxHeaderBytes) — client wouldn't read them, so emitting
- // them just bloats the URL.
- // - client-only values are included only when present in the inbound
- // JSON. Some deployments/imported configs carry them there, and the
- // subscription link is the only place clients can receive them.
- //
- // Truthy-only guards keep default inbounds emitting the same compact URL
- // they did before this helper grew.
- func buildXhttpExtra(xhttp map[string]any) map[string]any {
- if xhttp == nil {
- return nil
- }
- extra := map[string]any{}
- if xpb, ok := xhttp["xPaddingBytes"].(string); ok && len(xpb) > 0 {
- extra["xPaddingBytes"] = xpb
- }
- if obfs, ok := xhttp["xPaddingObfsMode"].(bool); ok && obfs {
- extra["xPaddingObfsMode"] = true
- for _, field := range []string{"xPaddingKey", "xPaddingHeader", "xPaddingPlacement", "xPaddingMethod"} {
- if v, ok := xhttp[field].(string); ok && len(v) > 0 {
- extra[field] = v
- }
- }
- }
- stringFields := []string{
- "uplinkHTTPMethod",
- "sessionPlacement", "sessionKey",
- "seqPlacement", "seqKey",
- "uplinkDataPlacement", "uplinkDataKey",
- "scMaxEachPostBytes", "scMinPostsIntervalMs",
- }
- for _, field := range stringFields {
- if v, ok := xhttp[field].(string); ok && len(v) > 0 {
- extra[field] = v
- }
- }
- for _, field := range []string{"uplinkChunkSize"} {
- if v, ok := nonZeroShareValue(xhttp[field]); ok {
- extra[field] = v
- }
- }
- for _, field := range []string{"noGRPCHeader"} {
- if v, ok := xhttp[field].(bool); ok && v {
- extra[field] = v
- }
- }
- for _, field := range []string{"xmux", "downloadSettings"} {
- if v, ok := nonEmptyShareObject(xhttp[field]); ok {
- extra[field] = v
- }
- }
- // Headers — emitted as the {name: value} map upstream's struct
- // expects. The server runtime ignores this field, but the client
- // (consuming the share link) honors it. Drop any "host" entry —
- // host already wins as a top-level URL param.
- if rawHeaders, ok := xhttp["headers"].(map[string]any); ok && len(rawHeaders) > 0 {
- out := map[string]any{}
- for k, v := range rawHeaders {
- if strings.EqualFold(k, "host") {
- continue
- }
- out[k] = v
- }
- if len(out) > 0 {
- extra["headers"] = out
- }
- }
- if len(extra) == 0 {
- return nil
- }
- return extra
- }
- func nonZeroShareValue(v any) (any, bool) {
- switch value := v.(type) {
- case string:
- return value, value != ""
- case int:
- return value, value != 0
- case int32:
- return value, value != 0
- case int64:
- return value, value != 0
- case float32:
- return value, value != 0
- case float64:
- return value, value != 0
- default:
- return nil, false
- }
- }
- func nonEmptyShareObject(v any) (any, bool) {
- switch value := v.(type) {
- case map[string]any:
- return value, len(value) > 0
- case map[string]string:
- return value, len(value) > 0
- case []any:
- return value, len(value) > 0
- default:
- return nil, false
- }
- }
- // applyXhttpExtraParams emits the full xhttp config into the URL query
- // params of a vless:// / trojan:// / ss:// link. Sets path/host/mode at
- // top level (xray's Build() always lets these win over `extra`) and packs
- // everything else into a JSON `extra` param. Also writes the flat
- // `x_padding_bytes` param sing-box-family clients understand.
- //
- // Without this, the admin's custom xPaddingBytes / sessionKey / etc. never
- // reach the client and handshakes are silently rejected with
- // `invalid padding (...) length: 0` — the client-visible symptom is
- // "xhttp doesn't connect" on OpenWRT / sing-box.
- //
- // Two encodings are written so every popular client can read at least one:
- //
- // - x_padding_bytes=<range> — flat param, understood by sing-box and its
- // derivatives (Podkop, OpenWRT sing-box, Karing, NekoBox, …).
- // - extra=<url-encoded-json> — full xhttp settings blob, which is how
- // xray-core clients (v2rayNG, Happ, Furious, Exclave, …) pick up the
- // bidirectional fields beyond path/host/mode.
- func applyXhttpExtraParams(xhttp map[string]any, params map[string]string) {
- if xhttp == nil {
- return
- }
- applyPathAndHostParams(xhttp, params)
- if mode, ok := xhttp["mode"].(string); ok {
- params["mode"] = mode
- }
- if xpb, ok := xhttp["xPaddingBytes"].(string); ok && len(xpb) > 0 {
- params["x_padding_bytes"] = xpb
- }
- extra := buildXhttpExtra(xhttp)
- if extra != nil {
- if b, err := json.Marshal(extra); err == nil {
- params["extra"] = string(b)
- }
- }
- }
- var kcpMaskToHeaderType = map[string]string{
- "header-dns": "dns",
- "header-dtls": "dtls",
- "header-srtp": "srtp",
- "header-utp": "utp",
- "header-wechat": "wechat-video",
- "header-wireguard": "wireguard",
- }
- var validFinalMaskUDPTypes = map[string]struct{}{
- "salamander": {},
- "mkcp-aes128gcm": {},
- "header-dns": {},
- "header-dtls": {},
- "header-srtp": {},
- "header-utp": {},
- "header-wechat": {},
- "header-wireguard": {},
- "mkcp-original": {},
- "xdns": {},
- "xicmp": {},
- "noise": {},
- "header-custom": {},
- }
- var validFinalMaskTCPTypes = map[string]struct{}{
- "header-custom": {},
- "fragment": {},
- "sudoku": {},
- }
- // applyKcpShareParams reconstructs legacy KCP share-link fields from either
- // the historical kcpSettings.header/seed shape or the current finalmask model.
- // This keeps subscription output compatible while avoiding panics when older
- // keys are absent from modern inbounds.
- func applyKcpShareParams(stream map[string]any, params map[string]string) {
- extractKcpShareFields(stream).applyToParams(params)
- }
- func applyKcpShareObj(stream map[string]any, obj map[string]any) {
- extractKcpShareFields(stream).applyToObj(obj)
- }
- type kcpShareFields struct {
- headerType string
- seed string
- mtu int
- tti int
- }
- func (f kcpShareFields) applyToParams(params map[string]string) {
- if f.headerType != "" && f.headerType != "none" {
- params["headerType"] = f.headerType
- }
- setStringParam(params, "seed", f.seed)
- setIntParam(params, "mtu", f.mtu)
- setIntParam(params, "tti", f.tti)
- }
- func (f kcpShareFields) applyToObj(obj map[string]any) {
- if f.headerType != "" && f.headerType != "none" {
- obj["type"] = f.headerType
- }
- setStringField(obj, "path", f.seed)
- setIntField(obj, "mtu", f.mtu)
- setIntField(obj, "tti", f.tti)
- }
- func extractKcpShareFields(stream map[string]any) kcpShareFields {
- fields := kcpShareFields{headerType: "none"}
- if kcp, ok := stream["kcpSettings"].(map[string]any); ok {
- if header, ok := kcp["header"].(map[string]any); ok {
- if value, ok := header["type"].(string); ok && value != "" {
- fields.headerType = value
- }
- }
- if value, ok := kcp["seed"].(string); ok && value != "" {
- fields.seed = value
- }
- if value, ok := readPositiveInt(kcp["mtu"]); ok {
- fields.mtu = value
- }
- if value, ok := readPositiveInt(kcp["tti"]); ok {
- fields.tti = value
- }
- }
- for _, rawMask := range normalizedFinalMaskUDPMasks(stream["finalmask"]) {
- mask, _ := rawMask.(map[string]any)
- if mask == nil {
- continue
- }
- maskType, _ := mask["type"].(string)
- if mapped, ok := kcpMaskToHeaderType[maskType]; ok {
- fields.headerType = mapped
- continue
- }
- switch maskType {
- case "mkcp-original":
- fields.seed = ""
- case "mkcp-aes128gcm":
- fields.seed = ""
- settings, _ := mask["settings"].(map[string]any)
- if value, ok := settings["password"].(string); ok && value != "" {
- fields.seed = value
- }
- }
- }
- return fields
- }
- func readPositiveInt(value any) (int, bool) {
- switch number := value.(type) {
- case int:
- return number, number > 0
- case int32:
- return int(number), number > 0
- case int64:
- return int(number), number > 0
- case float32:
- parsed := int(number)
- return parsed, parsed > 0
- case float64:
- parsed := int(number)
- return parsed, parsed > 0
- default:
- return 0, false
- }
- }
- func setStringParam(params map[string]string, key, value string) {
- if value == "" {
- delete(params, key)
- return
- }
- params[key] = value
- }
- func setIntParam(params map[string]string, key string, value int) {
- if value <= 0 {
- delete(params, key)
- return
- }
- params[key] = fmt.Sprintf("%d", value)
- }
- func setStringField(obj map[string]any, key, value string) {
- if value == "" {
- delete(obj, key)
- return
- }
- obj[key] = value
- }
- func setIntField(obj map[string]any, key string, value int) {
- if value <= 0 {
- delete(obj, key)
- return
- }
- obj[key] = value
- }
- // applyFinalMaskParams exports the finalmask payload as the compact
- // `fm=<json>` share-link field used by v2rayN-compatible clients.
- func applyFinalMaskParams(finalmask map[string]any, params map[string]string) {
- if fm, ok := marshalFinalMask(finalmask); ok {
- params["fm"] = fm
- }
- }
- func applyFinalMaskObj(finalmask map[string]any, obj map[string]any) {
- if fm, ok := marshalFinalMask(finalmask); ok {
- obj["fm"] = fm
- }
- }
- func marshalFinalMask(finalmask map[string]any) (string, bool) {
- normalized := normalizeFinalMask(finalmask)
- if !hasFinalMaskContent(normalized) {
- return "", false
- }
- b, err := json.Marshal(normalized)
- if err != nil || len(b) == 0 || string(b) == "null" {
- return "", false
- }
- return string(b), true
- }
- func normalizeFinalMask(finalmask map[string]any) map[string]any {
- tcpMasks := normalizedFinalMaskTCPMasks(finalmask)
- udpMasks := normalizedFinalMaskUDPMasks(finalmask)
- quicParams, hasQuicParams := finalmask["quicParams"].(map[string]any)
- if len(tcpMasks) == 0 && len(udpMasks) == 0 && !hasQuicParams {
- return nil
- }
- result := map[string]any{}
- if len(tcpMasks) > 0 {
- result["tcp"] = tcpMasks
- }
- if len(udpMasks) > 0 {
- result["udp"] = udpMasks
- }
- if hasQuicParams && len(quicParams) > 0 {
- result["quicParams"] = quicParams
- }
- return result
- }
- func normalizedFinalMaskTCPMasks(value any) []any {
- finalmask, _ := value.(map[string]any)
- if finalmask == nil {
- return nil
- }
- rawMasks, _ := finalmask["tcp"].([]any)
- if len(rawMasks) == 0 {
- return nil
- }
- normalized := make([]any, 0, len(rawMasks))
- for _, rawMask := range rawMasks {
- mask, _ := rawMask.(map[string]any)
- if mask == nil {
- continue
- }
- maskType, _ := mask["type"].(string)
- if _, ok := validFinalMaskTCPTypes[maskType]; !ok || maskType == "" {
- continue
- }
- normalizedMask := map[string]any{"type": maskType}
- if settings, ok := mask["settings"].(map[string]any); ok && len(settings) > 0 {
- normalizedMask["settings"] = settings
- }
- normalized = append(normalized, normalizedMask)
- }
- if len(normalized) == 0 {
- return nil
- }
- return normalized
- }
- func normalizedFinalMaskUDPMasks(value any) []any {
- finalmask, _ := value.(map[string]any)
- if finalmask == nil {
- return nil
- }
- rawMasks, _ := finalmask["udp"].([]any)
- if len(rawMasks) == 0 {
- return nil
- }
- normalized := make([]any, 0, len(rawMasks))
- for _, rawMask := range rawMasks {
- mask, _ := rawMask.(map[string]any)
- if mask == nil {
- continue
- }
- maskType, _ := mask["type"].(string)
- if _, ok := validFinalMaskUDPTypes[maskType]; !ok || maskType == "" {
- continue
- }
- normalizedMask := map[string]any{"type": maskType}
- if settings, ok := mask["settings"].(map[string]any); ok && len(settings) > 0 {
- normalizedMask["settings"] = settings
- }
- normalized = append(normalized, normalizedMask)
- }
- if len(normalized) == 0 {
- return nil
- }
- return normalized
- }
- func hasFinalMaskContent(value any) bool {
- switch v := value.(type) {
- case nil:
- return false
- case string:
- return len(v) > 0
- case map[string]any:
- for _, item := range v {
- if hasFinalMaskContent(item) {
- return true
- }
- }
- return false
- case []any:
- return slices.ContainsFunc(v, hasFinalMaskContent)
- default:
- return true
- }
- }
- func searchHost(headers any) string {
- data, _ := headers.(map[string]any)
- for k, v := range data {
- if strings.EqualFold(k, "host") {
- switch v.(type) {
- case []any:
- hosts, _ := v.([]any)
- if len(hosts) > 0 {
- return hosts[0].(string)
- } else {
- return ""
- }
- case any:
- return v.(string)
- }
- }
- }
- return ""
- }
- // PageData is a view model for subpage.html
- // PageData contains data for rendering the subscription information page.
- type PageData struct {
- Host string
- BasePath string
- SId string
- Enabled bool
- Download string
- Upload string
- Total string
- Used string
- Remained string
- Expire int64
- LastOnline int64
- Datepicker string
- DownloadByte int64
- UploadByte int64
- TotalByte int64
- SubUrl string
- SubJsonUrl string
- SubClashUrl string
- SubTitle string
- SubSupportUrl string
- Result []string
- Emails []string
- }
- // ResolveRequest extracts scheme and host info from request/headers consistently.
- // ResolveRequest extracts scheme, host, and header information from an HTTP request.
- func (s *SubService) ResolveRequest(c *gin.Context) (scheme string, host string, hostWithPort string, hostHeader string) {
- // scheme
- scheme = "http"
- if c.Request.TLS != nil || strings.EqualFold(c.GetHeader("X-Forwarded-Proto"), "https") {
- scheme = "https"
- }
- // base host (no port)
- if h, err := getHostFromXFH(c.GetHeader("X-Forwarded-Host")); err == nil && h != "" {
- host = h
- }
- if host == "" {
- host = c.GetHeader("X-Real-IP")
- }
- if host == "" {
- var err error
- host, _, err = net.SplitHostPort(c.Request.Host)
- if err != nil {
- host = c.Request.Host
- }
- }
- // host:port for URLs
- hostWithPort = c.GetHeader("X-Forwarded-Host")
- if hostWithPort == "" {
- hostWithPort = c.Request.Host
- }
- if hostWithPort == "" {
- hostWithPort = host
- }
- // header display host
- hostHeader = c.GetHeader("X-Forwarded-Host")
- if hostHeader == "" {
- hostHeader = c.GetHeader("X-Real-IP")
- }
- if hostHeader == "" {
- hostHeader = host
- }
- return
- }
- // BuildURLs constructs absolute subscription and JSON subscription URLs for a given subscription ID.
- // It prioritizes configured URIs, then individual settings, and finally falls back to request-derived components.
- func (s *SubService) BuildURLs(subPath, subJsonPath, subClashPath, subId string) (subURL, subJsonURL, subClashURL string) {
- if subId == "" {
- return "", "", ""
- }
- configuredSubURI, _ := s.settingService.GetSubURI()
- configuredSubJsonURI, _ := s.settingService.GetSubJsonURI()
- configuredSubClashURI, _ := s.settingService.GetSubClashURI()
- // Same base as the panel's Client Information page; s.address is the
- // subscriber's host already normalized away from any loopback/bind IP.
- base := s.settingService.BuildSubURIBase(s.address)
- subURL = s.buildSingleURL(configuredSubURI, base, subPath, subId)
- subJsonURL = s.buildSingleURL(configuredSubJsonURI, base, subJsonPath, subId)
- subClashURL = s.buildSingleURL(configuredSubClashURI, base, subClashPath, subId)
- return subURL, subJsonURL, subClashURL
- }
- // buildSingleURL constructs a single URL using configured URI or base components
- func (s *SubService) buildSingleURL(configuredURI, base, basePath, subId string) string {
- if configuredURI != "" {
- return s.joinPathWithID(configuredURI, subId)
- }
- return s.joinPathWithID(base+basePath, subId)
- }
- // joinPathWithID safely joins a base path with a subscription ID
- func (s *SubService) joinPathWithID(basePath, subId string) string {
- if strings.HasSuffix(basePath, "/") {
- return basePath + subId
- }
- return basePath + "/" + subId
- }
- // BuildPageData parses header and prepares the template view model.
- // BuildPageData constructs page data for rendering the subscription information page.
- func (s *SubService) BuildPageData(subId string, hostHeader string, traffic xray.ClientTraffic, lastOnline int64, subs []string, emails []string, subURL, subJsonURL, subClashURL string, basePath string, subTitle string, subSupportUrl string) PageData {
- download := common.FormatTraffic(traffic.Down)
- upload := common.FormatTraffic(traffic.Up)
- total := "∞"
- used := common.FormatTraffic(traffic.Up + traffic.Down)
- remained := ""
- if traffic.Total > 0 {
- total = common.FormatTraffic(traffic.Total)
- left := max(traffic.Total-(traffic.Up+traffic.Down), 0)
- remained = common.FormatTraffic(left)
- }
- datepicker := s.datepicker
- if datepicker == "" {
- datepicker = "gregorian"
- }
- return PageData{
- Host: hostHeader,
- BasePath: basePath,
- SId: subId,
- Enabled: traffic.Enable,
- Download: download,
- Upload: upload,
- Total: total,
- Used: used,
- Remained: remained,
- Expire: traffic.ExpiryTime / 1000,
- LastOnline: lastOnline,
- Datepicker: datepicker,
- DownloadByte: traffic.Down,
- UploadByte: traffic.Up,
- TotalByte: traffic.Total,
- SubUrl: subURL,
- SubJsonUrl: subJsonURL,
- SubClashUrl: subClashURL,
- SubTitle: subTitle,
- SubSupportUrl: subSupportUrl,
- Result: subs,
- Emails: emails,
- }
- }
- func getHostFromXFH(s string) (string, error) {
- if strings.Contains(s, ":") {
- realHost, _, err := net.SplitHostPort(s)
- if err != nil {
- return "", err
- }
- return realHost, nil
- }
- return s, nil
- }
|