client_portable.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. package service
  2. import (
  3. "strings"
  4. "time"
  5. "github.com/google/uuid"
  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/xray"
  9. "gorm.io/gorm"
  10. )
  11. // ExportAll returns every client in the same {client, inboundIds} shape that
  12. // /add and /bulkCreate accept, so an exported file round-trips straight back
  13. // through Import. Clients with no inbound attachment are included with an empty
  14. // inboundIds list so an export taken before DeleteOrphans can restore them.
  15. func (s *ClientService) ExportAll() ([]ClientCreatePayload, error) {
  16. db := database.GetDB()
  17. var rows []model.ClientRecord
  18. if err := db.Order("id ASC").Find(&rows).Error; err != nil {
  19. return nil, err
  20. }
  21. out := make([]ClientCreatePayload, 0, len(rows))
  22. if len(rows) == 0 {
  23. return out, nil
  24. }
  25. ids := make([]int, 0, len(rows))
  26. for i := range rows {
  27. ids = append(ids, rows[i].Id)
  28. }
  29. attachments := make(map[int][]int, len(rows))
  30. for _, batch := range chunkInts(ids, sqlInChunk) {
  31. var links []model.ClientInbound
  32. if err := db.Where("client_id IN ?", batch).Order("inbound_id ASC").Find(&links).Error; err != nil {
  33. return nil, err
  34. }
  35. for _, l := range links {
  36. attachments[l.ClientId] = append(attachments[l.ClientId], l.InboundId)
  37. }
  38. }
  39. for i := range rows {
  40. client := rows[i].ToClient()
  41. // The per-inbound flow_override is the reliable flow for multi-inbound
  42. // clients; the canonical column can be left stale by SyncInbound (#4792).
  43. if flow, err := s.EffectiveFlow(db, rows[i].Id); err == nil && flow != "" {
  44. client.Flow = flow
  45. }
  46. out = append(out, ClientCreatePayload{
  47. Client: *client,
  48. InboundIds: attachments[rows[i].Id],
  49. })
  50. }
  51. return out, nil
  52. }
  53. // ImportClients recreates clients from an exported list. Items that carry
  54. // inboundIds go through the normal BulkCreate path (added to every inbound and
  55. // pushed to xray); items with no inboundIds are restored as bare records so an
  56. // orphan-inclusive export round-trips. Existing emails are never overwritten —
  57. // they are reported in Skipped. The boolean reports whether xray needs a restart.
  58. func (s *ClientService) ImportClients(inboundSvc *InboundService, items []ClientCreatePayload) (BulkCreateResult, bool, error) {
  59. result := BulkCreateResult{}
  60. if len(items) == 0 {
  61. return result, false, nil
  62. }
  63. attached := make([]ClientCreatePayload, 0, len(items))
  64. orphans := make([]ClientCreatePayload, 0)
  65. for i := range items {
  66. if len(items[i].InboundIds) > 0 {
  67. attached = append(attached, items[i])
  68. } else {
  69. orphans = append(orphans, items[i])
  70. }
  71. }
  72. skip := func(email, reason string) {
  73. if strings.TrimSpace(email) == "" {
  74. email = "(missing email)"
  75. }
  76. result.Skipped = append(result.Skipped, BulkCreateReport{Email: email, Reason: reason})
  77. }
  78. needRestart := false
  79. if len(attached) > 0 {
  80. sub, nr, err := s.BulkCreate(inboundSvc, attached)
  81. if err != nil {
  82. return result, needRestart, err
  83. }
  84. needRestart = needRestart || nr
  85. result.Created += sub.Created
  86. result.Skipped = append(result.Skipped, sub.Skipped...)
  87. }
  88. db := database.GetDB()
  89. for i := range orphans {
  90. client := orphans[i].Client
  91. email := strings.TrimSpace(client.Email)
  92. if email == "" {
  93. skip("", "client email is required")
  94. continue
  95. }
  96. if verr := validateClientEmail(email); verr != nil {
  97. skip(email, verr.Error())
  98. continue
  99. }
  100. if verr := validateClientSubID(client.SubID); verr != nil {
  101. skip(email, verr.Error())
  102. continue
  103. }
  104. // An existing record (in the DB or just created from the attached set
  105. // above) always wins — import never clobbers a live client.
  106. var taken int64
  107. if err := db.Model(&model.ClientRecord{}).Where("email = ?", email).Count(&taken).Error; err != nil {
  108. return result, needRestart, err
  109. }
  110. if taken > 0 {
  111. skip(email, "email already in use: "+email)
  112. continue
  113. }
  114. client.Email = email
  115. if client.SubID == "" {
  116. client.SubID = uuid.NewString()
  117. }
  118. if client.SubID != "" {
  119. var subTaken int64
  120. if err := db.Model(&model.ClientRecord{}).
  121. Where("sub_id = ? AND email <> ?", client.SubID, email).
  122. Count(&subTaken).Error; err != nil {
  123. return result, needRestart, err
  124. }
  125. if subTaken > 0 {
  126. skip(email, "subId already in use: "+client.SubID)
  127. continue
  128. }
  129. }
  130. if !client.Enable {
  131. client.Enable = true
  132. }
  133. now := time.Now().UnixMilli()
  134. if client.CreatedAt == 0 {
  135. client.CreatedAt = now
  136. }
  137. client.UpdatedAt = now
  138. if err := db.Create(client.ToRecord()).Error; err != nil {
  139. skip(email, err.Error())
  140. continue
  141. }
  142. result.Created++
  143. }
  144. return result, needRestart, nil
  145. }
  146. // DeleteOrphans removes every client that is not attached to any inbound,
  147. // together with its traffic rows, IP log, and external links. It mirrors the
  148. // cleanup the single-client Delete performs, batched into one transaction.
  149. // Returns the number of clients deleted.
  150. func (s *ClientService) DeleteOrphans() (int, error) {
  151. db := database.GetDB()
  152. sub := database.GetDB().Table("client_inbounds").Select("client_id")
  153. var rows []model.ClientRecord
  154. if err := db.Where("id NOT IN (?)", sub).Order("id ASC").Find(&rows).Error; err != nil {
  155. return 0, err
  156. }
  157. if len(rows) == 0 {
  158. return 0, nil
  159. }
  160. ids := make([]int, 0, len(rows))
  161. emails := make([]string, 0, len(rows))
  162. for i := range rows {
  163. ids = append(ids, rows[i].Id)
  164. if rows[i].Email != "" {
  165. emails = append(emails, rows[i].Email)
  166. }
  167. }
  168. tombstoneClientEmails(emails)
  169. if err := runSerializedTx(func(tx *gorm.DB) error {
  170. for _, batch := range chunkInts(ids, sqlInChunk) {
  171. if e := tx.Where("client_id IN ?", batch).Delete(&model.ClientInbound{}).Error; e != nil {
  172. return e
  173. }
  174. if e := tx.Where("client_id IN ?", batch).Delete(&model.ClientExternalLink{}).Error; e != nil {
  175. return e
  176. }
  177. }
  178. if len(emails) > 0 {
  179. for _, batch := range chunkStrings(emails, sqlInChunk) {
  180. if e := tx.Where("email IN ?", batch).Delete(&xray.ClientTraffic{}).Error; e != nil {
  181. return e
  182. }
  183. if e := tx.Where("client_email IN ?", batch).Delete(&model.InboundClientIps{}).Error; e != nil {
  184. return e
  185. }
  186. }
  187. if e := clearGlobalTraffic(tx, emails...); e != nil {
  188. return e
  189. }
  190. }
  191. for _, batch := range chunkInts(ids, sqlInChunk) {
  192. if e := tx.Where("id IN ?", batch).Delete(&model.ClientRecord{}).Error; e != nil {
  193. return e
  194. }
  195. }
  196. return nil
  197. }); err != nil {
  198. return 0, err
  199. }
  200. return len(ids), nil
  201. }