1
0

client_groups.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  1. package service
  2. import (
  3. "encoding/json"
  4. "sort"
  5. "strings"
  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/util/common"
  9. "gorm.io/gorm"
  10. )
  11. type GroupSummary struct {
  12. Name string `json:"name"`
  13. ClientCount int `json:"clientCount"`
  14. TrafficUsed int64 `json:"trafficUsed"`
  15. Up int64 `json:"up"`
  16. Down int64 `json:"down"`
  17. }
  18. func (s *ClientService) ListGroups() ([]GroupSummary, error) {
  19. db := database.GetDB()
  20. // email is unique in both clients and client_traffics, so the LEFT JOIN
  21. // never double-counts a client's traffic.
  22. var derived []GroupSummary
  23. if err := db.Table("clients AS c").
  24. Select("c.group_name AS name, COUNT(*) AS client_count, COALESCE(SUM(ct.up + ct.down), 0) AS traffic_used, COALESCE(SUM(ct.up), 0) AS up, COALESCE(SUM(ct.down), 0) AS down").
  25. Joins("LEFT JOIN client_traffics ct ON ct.email = c.email").
  26. Where("c.group_name <> ''").
  27. Group("c.group_name").
  28. Scan(&derived).Error; err != nil {
  29. return nil, err
  30. }
  31. var stored []model.ClientGroup
  32. if err := db.Find(&stored).Error; err != nil {
  33. return nil, err
  34. }
  35. type groupAgg struct {
  36. count int
  37. up int64
  38. down int64
  39. }
  40. baseUp := make(map[string]int64, len(stored))
  41. baseDown := make(map[string]int64, len(stored))
  42. merged := make(map[string]groupAgg, len(derived)+len(stored))
  43. for _, g := range stored {
  44. merged[g.Name] = groupAgg{}
  45. baseUp[g.Name] = g.ResetUp
  46. baseDown[g.Name] = g.ResetDown
  47. }
  48. for _, g := range derived {
  49. merged[g.Name] = groupAgg{count: g.ClientCount, up: g.Up, down: g.Down}
  50. }
  51. out := make([]GroupSummary, 0, len(merged))
  52. for name, agg := range merged {
  53. up := agg.up - baseUp[name]
  54. if up < 0 {
  55. up = 0
  56. }
  57. down := agg.down - baseDown[name]
  58. if down < 0 {
  59. down = 0
  60. }
  61. out = append(out, GroupSummary{Name: name, ClientCount: agg.count, TrafficUsed: up + down, Up: up, Down: down})
  62. }
  63. sort.Slice(out, func(i, j int) bool {
  64. return strings.ToLower(out[i].Name) < strings.ToLower(out[j].Name)
  65. })
  66. return out, nil
  67. }
  68. // adjustGroupBaselinesForRemovedTraffic shifts group baselines down by the clients'
  69. // current counters so ListGroups totals survive a traffic reset or client delete (#5675).
  70. func adjustGroupBaselinesForRemovedTraffic(tx *gorm.DB, emails []string) error {
  71. if len(emails) == 0 {
  72. return nil
  73. }
  74. type groupDelta struct {
  75. Name string
  76. Up int64
  77. Down int64
  78. }
  79. totals := make(map[string]*groupDelta)
  80. for _, batch := range chunkStrings(emails, sqlInChunk) {
  81. var part []groupDelta
  82. if err := tx.Table("clients AS c").
  83. Select("c.group_name AS name, COALESCE(SUM(ct.up), 0) AS up, COALESCE(SUM(ct.down), 0) AS down").
  84. Joins("JOIN client_traffics ct ON ct.email = c.email").
  85. Where("c.group_name <> '' AND c.email IN ?", batch).
  86. Group("c.group_name").
  87. Scan(&part).Error; err != nil {
  88. return err
  89. }
  90. for i := range part {
  91. if agg, ok := totals[part[i].Name]; ok {
  92. agg.Up += part[i].Up
  93. agg.Down += part[i].Down
  94. } else {
  95. totals[part[i].Name] = &part[i]
  96. }
  97. }
  98. }
  99. for name, d := range totals {
  100. if d.Up == 0 && d.Down == 0 {
  101. continue
  102. }
  103. res := tx.Model(&model.ClientGroup{}).Where("name = ?", name).Updates(map[string]any{
  104. "reset_up": gorm.Expr("reset_up - ?", d.Up),
  105. "reset_down": gorm.Expr("reset_down - ?", d.Down),
  106. })
  107. if res.Error != nil {
  108. return res.Error
  109. }
  110. if res.RowsAffected == 0 {
  111. if err := tx.Create(&model.ClientGroup{Name: name, ResetUp: -d.Up, ResetDown: -d.Down}).Error; err != nil {
  112. return err
  113. }
  114. }
  115. }
  116. return nil
  117. }
  118. func (s *ClientService) EmailsByGroup(name string) ([]string, error) {
  119. name = strings.TrimSpace(name)
  120. if name == "" {
  121. return []string{}, nil
  122. }
  123. db := database.GetDB()
  124. var emails []string
  125. if err := db.Model(&model.ClientRecord{}).
  126. Where("group_name = ?", name).
  127. Order("email ASC").
  128. Pluck("email", &emails).Error; err != nil {
  129. return nil, err
  130. }
  131. if emails == nil {
  132. emails = []string{}
  133. }
  134. return emails, nil
  135. }
  136. func (s *ClientService) ResetGroupTraffic(name string) error {
  137. name = strings.TrimSpace(name)
  138. if name == "" {
  139. return common.NewError("group name is required")
  140. }
  141. db := database.GetDB()
  142. var agg struct {
  143. Up int64
  144. Down int64
  145. }
  146. if err := db.Table("clients AS c").
  147. Select("COALESCE(SUM(ct.up), 0) AS up, COALESCE(SUM(ct.down), 0) AS down").
  148. Joins("LEFT JOIN client_traffics ct ON ct.email = c.email").
  149. Where("c.group_name = ?", name).
  150. Scan(&agg).Error; err != nil {
  151. return err
  152. }
  153. var count int64
  154. if err := db.Model(&model.ClientGroup{}).Where("name = ?", name).Count(&count).Error; err != nil {
  155. return err
  156. }
  157. if count == 0 {
  158. return db.Create(&model.ClientGroup{Name: name, ResetUp: agg.Up, ResetDown: agg.Down}).Error
  159. }
  160. return db.Model(&model.ClientGroup{}).Where("name = ?", name).
  161. Updates(map[string]any{"reset_up": agg.Up, "reset_down": agg.Down}).Error
  162. }
  163. func (s *ClientService) CreateGroup(name string) error {
  164. name = strings.TrimSpace(name)
  165. if name == "" {
  166. return common.NewError("group name is required")
  167. }
  168. db := database.GetDB()
  169. var count int64
  170. if err := db.Model(&model.ClientGroup{}).Where("name = ?", name).Count(&count).Error; err != nil {
  171. return err
  172. }
  173. if count > 0 {
  174. return common.NewError("group already exists")
  175. }
  176. return db.Create(&model.ClientGroup{Name: name}).Error
  177. }
  178. func (s *ClientService) RenameGroup(oldName, newName string) (int, error) {
  179. oldName = strings.TrimSpace(oldName)
  180. newName = strings.TrimSpace(newName)
  181. if oldName == "" {
  182. return 0, common.NewError("old group name is required")
  183. }
  184. if newName == "" {
  185. return 0, common.NewError("new group name is required")
  186. }
  187. if oldName == newName {
  188. return 0, nil
  189. }
  190. return s.replaceGroupValue(oldName, newName)
  191. }
  192. func (s *ClientService) DeleteGroup(name string) (int, error) {
  193. name = strings.TrimSpace(name)
  194. if name == "" {
  195. return 0, common.NewError("group name is required")
  196. }
  197. return s.replaceGroupValue(name, "")
  198. }
  199. func (s *ClientService) RemoveFromGroup(emails []string) (int, error) {
  200. return s.AddToGroup(emails, "")
  201. }
  202. func (s *ClientService) AddToGroup(emails []string, group string) (int, error) {
  203. group = strings.TrimSpace(group)
  204. if len(emails) == 0 {
  205. return 0, nil
  206. }
  207. db := database.GetDB()
  208. if group != "" {
  209. var exists int64
  210. if err := db.Model(&model.ClientGroup{}).Where("name = ?", group).Count(&exists).Error; err != nil {
  211. return 0, err
  212. }
  213. if exists == 0 {
  214. var derived int64
  215. if err := db.Model(&model.ClientRecord{}).Where("group_name = ?", group).Count(&derived).Error; err != nil {
  216. return 0, err
  217. }
  218. if derived == 0 {
  219. if err := db.Create(&model.ClientGroup{Name: group}).Error; err != nil {
  220. return 0, err
  221. }
  222. }
  223. }
  224. }
  225. var records []model.ClientRecord
  226. for _, batch := range chunkStrings(emails, sqlInChunk) {
  227. var rows []model.ClientRecord
  228. if err := db.Where("email IN ?", batch).Find(&rows).Error; err != nil {
  229. return 0, err
  230. }
  231. records = append(records, rows...)
  232. }
  233. if len(records) == 0 {
  234. return 0, nil
  235. }
  236. affectedEmails := make([]string, 0, len(records))
  237. for _, r := range records {
  238. affectedEmails = append(affectedEmails, r.Email)
  239. }
  240. tx := db.Begin()
  241. for _, batch := range chunkStrings(affectedEmails, sqlInChunk) {
  242. if err := tx.Model(&model.ClientRecord{}).
  243. Where("email IN ?", batch).
  244. UpdateColumn("group_name", group).Error; err != nil {
  245. tx.Rollback()
  246. return 0, err
  247. }
  248. }
  249. var inboundIDs []int
  250. inboundIDSeen := make(map[int]struct{})
  251. for _, batch := range chunkStrings(affectedEmails, sqlInChunk) {
  252. var ids []int
  253. if err := tx.Table("client_inbounds").
  254. Joins("JOIN clients ON clients.id = client_inbounds.client_id").
  255. Where("clients.email IN ?", batch).
  256. Distinct("client_inbounds.inbound_id").
  257. Pluck("inbound_id", &ids).Error; err != nil {
  258. tx.Rollback()
  259. return 0, err
  260. }
  261. for _, id := range ids {
  262. if _, ok := inboundIDSeen[id]; !ok {
  263. inboundIDSeen[id] = struct{}{}
  264. inboundIDs = append(inboundIDs, id)
  265. }
  266. }
  267. }
  268. emailSet := make(map[string]struct{}, len(affectedEmails))
  269. for _, e := range affectedEmails {
  270. emailSet[e] = struct{}{}
  271. }
  272. for _, ibID := range inboundIDs {
  273. var ib model.Inbound
  274. if err := tx.First(&ib, ibID).Error; err != nil {
  275. tx.Rollback()
  276. return 0, err
  277. }
  278. var settings map[string]any
  279. if err := json.Unmarshal([]byte(ib.Settings), &settings); err != nil {
  280. continue
  281. }
  282. clients, ok := settings["clients"].([]any)
  283. if !ok {
  284. continue
  285. }
  286. modified := false
  287. for i := range clients {
  288. cm, ok := clients[i].(map[string]any)
  289. if !ok {
  290. continue
  291. }
  292. email, _ := cm["email"].(string)
  293. if _, hit := emailSet[email]; !hit {
  294. continue
  295. }
  296. if group == "" {
  297. delete(cm, "group")
  298. } else {
  299. cm["group"] = group
  300. }
  301. clients[i] = cm
  302. modified = true
  303. }
  304. if modified {
  305. settings["clients"] = clients
  306. newSettings, err := json.Marshal(settings)
  307. if err != nil {
  308. continue
  309. }
  310. ib.Settings = string(newSettings)
  311. if err := tx.Save(&ib).Error; err != nil {
  312. tx.Rollback()
  313. return 0, err
  314. }
  315. }
  316. }
  317. if err := tx.Commit().Error; err != nil {
  318. return 0, err
  319. }
  320. return len(records), nil
  321. }
  322. func (s *ClientService) replaceGroupValue(oldName, newName string) (int, error) {
  323. db := database.GetDB()
  324. if newName == "" {
  325. if err := db.Where("name = ?", oldName).Delete(&model.ClientGroup{}).Error; err != nil {
  326. return 0, err
  327. }
  328. } else {
  329. if err := db.Model(&model.ClientGroup{}).Where("name = ?", oldName).Update("name", newName).Error; err != nil {
  330. return 0, err
  331. }
  332. }
  333. var records []model.ClientRecord
  334. if err := db.Where("group_name = ?", oldName).Find(&records).Error; err != nil {
  335. return 0, err
  336. }
  337. if len(records) == 0 {
  338. return 0, nil
  339. }
  340. affectedEmails := make([]string, 0, len(records))
  341. for _, r := range records {
  342. affectedEmails = append(affectedEmails, r.Email)
  343. }
  344. tx := db.Begin()
  345. if err := tx.Model(&model.ClientRecord{}).
  346. Where("group_name = ?", oldName).
  347. UpdateColumn("group_name", newName).Error; err != nil {
  348. tx.Rollback()
  349. return 0, err
  350. }
  351. var inboundIDs []int
  352. inboundIDSeen := make(map[int]struct{})
  353. for _, batch := range chunkStrings(affectedEmails, sqlInChunk) {
  354. var ids []int
  355. if err := tx.Table("client_inbounds").
  356. Joins("JOIN clients ON clients.id = client_inbounds.client_id").
  357. Where("clients.email IN ?", batch).
  358. Distinct("client_inbounds.inbound_id").
  359. Pluck("inbound_id", &ids).Error; err != nil {
  360. tx.Rollback()
  361. return 0, err
  362. }
  363. for _, id := range ids {
  364. if _, ok := inboundIDSeen[id]; !ok {
  365. inboundIDSeen[id] = struct{}{}
  366. inboundIDs = append(inboundIDs, id)
  367. }
  368. }
  369. }
  370. for _, ibID := range inboundIDs {
  371. var ib model.Inbound
  372. if err := tx.First(&ib, ibID).Error; err != nil {
  373. tx.Rollback()
  374. return 0, err
  375. }
  376. var settings map[string]any
  377. if err := json.Unmarshal([]byte(ib.Settings), &settings); err != nil {
  378. continue
  379. }
  380. clients, ok := settings["clients"].([]any)
  381. if !ok {
  382. continue
  383. }
  384. modified := false
  385. for i := range clients {
  386. cm, ok := clients[i].(map[string]any)
  387. if !ok {
  388. continue
  389. }
  390. if g, ok := cm["group"].(string); ok && g == oldName {
  391. if newName == "" {
  392. delete(cm, "group")
  393. } else {
  394. cm["group"] = newName
  395. }
  396. clients[i] = cm
  397. modified = true
  398. }
  399. }
  400. if modified {
  401. settings["clients"] = clients
  402. newSettings, err := json.Marshal(settings)
  403. if err != nil {
  404. continue
  405. }
  406. ib.Settings = string(newSettings)
  407. if err := tx.Save(&ib).Error; err != nil {
  408. tx.Rollback()
  409. return 0, err
  410. }
  411. }
  412. }
  413. if err := tx.Commit().Error; err != nil {
  414. return 0, err
  415. }
  416. return len(records), nil
  417. }