client_lookup.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. package service
  2. import (
  3. "encoding/json"
  4. "strings"
  5. "github.com/mhsanaei/3x-ui/v3/internal/database"
  6. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  7. "github.com/mhsanaei/3x-ui/v3/internal/xray"
  8. "gorm.io/gorm"
  9. )
  10. func (s *ClientService) GetRecordByEmail(tx *gorm.DB, email string) (*model.ClientRecord, error) {
  11. if tx == nil {
  12. tx = database.GetDB()
  13. }
  14. row := &model.ClientRecord{}
  15. err := tx.Where("email = ?", email).First(row).Error
  16. if err != nil {
  17. return nil, err
  18. }
  19. return row, nil
  20. }
  21. // EffectiveFlow returns the client's flow from the first flow-capable inbound
  22. // it is attached to (lowest inbound_id with a non-empty flow_override). The
  23. // canonical clients.Flow column is unreliable for multi-inbound clients: a
  24. // non-flow inbound (Hysteria, WS, gRPC, …) carries an empty flow and, when its
  25. // SyncInbound runs last, overwrites the column to "" even though a VLESS Reality
  26. // inbound stored a real flow. The per-inbound flow_override is always correct,
  27. // so derive the display flow from it (order-independent). See issue #4792.
  28. func (s *ClientService) EffectiveFlow(tx *gorm.DB, recordId int) (string, error) {
  29. if tx == nil {
  30. tx = database.GetDB()
  31. }
  32. var flows []string
  33. err := tx.Model(&model.ClientInbound{}).
  34. Where("client_id = ? AND flow_override <> ?", recordId, "").
  35. Order("inbound_id ASC").
  36. Limit(1).
  37. Pluck("flow_override", &flows).Error
  38. if err != nil {
  39. return "", err
  40. }
  41. if len(flows) == 0 {
  42. return "", nil
  43. }
  44. return flows[0], nil
  45. }
  46. // EffectiveFlowsByEmails resolves the intended flow (non-empty flow_override,
  47. // lowest inbound_id first — same rule as EffectiveFlow) for many clients in one
  48. // query, keyed by email. Emails absent from the result carry no flow anywhere.
  49. // Batched so flow restoration on an inbound with many clients is O(1) queries
  50. // instead of O(clients). Used to restore a stripped flow onto an inbound that
  51. // has just become flow-eligible.
  52. func (s *ClientService) EffectiveFlowsByEmails(tx *gorm.DB, emails []string) (map[string]string, error) {
  53. if tx == nil {
  54. tx = database.GetDB()
  55. }
  56. out := make(map[string]string, len(emails))
  57. if len(emails) == 0 {
  58. return out, nil
  59. }
  60. type row struct {
  61. Email string
  62. Flow string `gorm:"column:flow_override"`
  63. }
  64. for _, batch := range chunkStrings(emails, sqlInChunk) {
  65. var rows []row
  66. err := tx.Table("client_inbounds").
  67. Select("clients.email AS email, client_inbounds.flow_override AS flow_override").
  68. Joins("JOIN clients ON clients.id = client_inbounds.client_id").
  69. Where("clients.email IN ? AND client_inbounds.flow_override <> ?", batch, "").
  70. Order("client_inbounds.inbound_id ASC").
  71. Scan(&rows).Error
  72. if err != nil {
  73. return nil, err
  74. }
  75. for _, r := range rows {
  76. if _, seen := out[r.Email]; !seen { // ordered by inbound_id ASC → first = lowest
  77. out[r.Email] = r.Flow
  78. }
  79. }
  80. }
  81. return out, nil
  82. }
  83. func (s *ClientService) GetInboundIdsForEmail(tx *gorm.DB, email string) ([]int, error) {
  84. if tx == nil {
  85. tx = database.GetDB()
  86. }
  87. var ids []int
  88. err := tx.Table("client_inbounds").
  89. Select("client_inbounds.inbound_id").
  90. Joins("JOIN clients ON clients.id = client_inbounds.client_id").
  91. Where("clients.email = ?", email).
  92. Scan(&ids).Error
  93. if err != nil {
  94. return nil, err
  95. }
  96. return ids, nil
  97. }
  98. func (s *ClientService) GetByID(id int) (*model.ClientRecord, error) {
  99. row := &model.ClientRecord{}
  100. if err := database.GetDB().Where("id = ?", id).First(row).Error; err != nil {
  101. return nil, err
  102. }
  103. return row, nil
  104. }
  105. func (s *ClientService) GetInboundIdsForRecord(id int) ([]int, error) {
  106. var ids []int
  107. err := database.GetDB().Table("client_inbounds").
  108. Where("client_id = ?", id).
  109. Order("inbound_id ASC").
  110. Pluck("inbound_id", &ids).Error
  111. if err != nil {
  112. return nil, err
  113. }
  114. return ids, nil
  115. }
  116. func (s *ClientService) List() ([]ClientWithAttachments, error) {
  117. db := database.GetDB()
  118. var rows []model.ClientRecord
  119. if err := db.Order("id ASC").Find(&rows).Error; err != nil {
  120. return nil, err
  121. }
  122. if len(rows) == 0 {
  123. return []ClientWithAttachments{}, nil
  124. }
  125. clientIds := make([]int, 0, len(rows))
  126. emails := make([]string, 0, len(rows))
  127. for i := range rows {
  128. clientIds = append(clientIds, rows[i].Id)
  129. if rows[i].Email != "" {
  130. emails = append(emails, rows[i].Email)
  131. }
  132. }
  133. attachments := make(map[int][]int, len(rows))
  134. for _, batch := range chunkInts(clientIds, sqlInChunk) {
  135. var links []model.ClientInbound
  136. if err := db.Where("client_id IN ?", batch).Find(&links).Error; err != nil {
  137. return nil, err
  138. }
  139. for _, l := range links {
  140. attachments[l.ClientId] = append(attachments[l.ClientId], l.InboundId)
  141. }
  142. }
  143. trafficByEmail := make(map[string]*xray.ClientTraffic, len(emails))
  144. if len(emails) > 0 {
  145. var stats []xray.ClientTraffic
  146. for _, batch := range chunkStrings(emails, sqlInChunk) {
  147. var batchStats []xray.ClientTraffic
  148. if err := db.Where("email IN ?", batch).Find(&batchStats).Error; err != nil {
  149. return nil, err
  150. }
  151. stats = append(stats, batchStats...)
  152. }
  153. overlayGlobalTrafficValues(db, stats)
  154. for i := range stats {
  155. trafficByEmail[stats[i].Email] = &stats[i]
  156. }
  157. }
  158. out := make([]ClientWithAttachments, 0, len(rows))
  159. for i := range rows {
  160. out = append(out, ClientWithAttachments{
  161. ClientRecord: rows[i],
  162. InboundIds: attachments[rows[i].Id],
  163. Traffic: trafficByEmail[rows[i].Email],
  164. })
  165. }
  166. return out, nil
  167. }
  168. func (s *ClientService) HasPendingNode(inboundSvc *InboundService, email string) bool {
  169. if strings.TrimSpace(email) == "" {
  170. return false
  171. }
  172. ids, err := s.GetInboundIdsForEmail(nil, email)
  173. if err != nil {
  174. return false
  175. }
  176. return inboundSvc.AnyNodePending(ids)
  177. }
  178. // findInboundIdsByClientEmail returns every inbound whose settings.clients[]
  179. // JSON contains an entry with the given email. Driver-portable (no JSON
  180. // operators) by parsing in Go — fine for the rare fallback path.
  181. func (s *ClientService) findInboundIdsByClientEmail(email string) ([]int, error) {
  182. var inbounds []model.Inbound
  183. if err := database.GetDB().
  184. Select("id, settings").
  185. Where("settings LIKE ?", "%"+email+"%").
  186. Find(&inbounds).Error; err != nil {
  187. return nil, err
  188. }
  189. out := make([]int, 0, len(inbounds))
  190. for _, ib := range inbounds {
  191. var settings map[string]any
  192. if err := json.Unmarshal([]byte(ib.Settings), &settings); err != nil {
  193. continue
  194. }
  195. clients, ok := settings["clients"].([]any)
  196. if !ok {
  197. continue
  198. }
  199. for _, c := range clients {
  200. cm, ok := c.(map[string]any)
  201. if !ok {
  202. continue
  203. }
  204. if cEmail, _ := cm["email"].(string); cEmail == email {
  205. out = append(out, ib.Id)
  206. break
  207. }
  208. }
  209. }
  210. return out, nil
  211. }