1
0

client_groups.go 11 KB

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