inbound.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738
  1. package service
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "time"
  6. "x-ui/database"
  7. "x-ui/database/model"
  8. "x-ui/logger"
  9. "x-ui/util/common"
  10. "x-ui/xray"
  11. "gorm.io/gorm"
  12. )
  13. type InboundService struct {
  14. }
  15. func (s *InboundService) GetInbounds(userId int) ([]*model.Inbound, error) {
  16. db := database.GetDB()
  17. var inbounds []*model.Inbound
  18. err := db.Model(model.Inbound{}).Preload("ClientStats").Where("user_id = ?", userId).Find(&inbounds).Error
  19. if err != nil && err != gorm.ErrRecordNotFound {
  20. return nil, err
  21. }
  22. return inbounds, nil
  23. }
  24. func (s *InboundService) GetAllInbounds() ([]*model.Inbound, error) {
  25. db := database.GetDB()
  26. var inbounds []*model.Inbound
  27. err := db.Model(model.Inbound{}).Preload("ClientStats").Find(&inbounds).Error
  28. if err != nil && err != gorm.ErrRecordNotFound {
  29. return nil, err
  30. }
  31. return inbounds, nil
  32. }
  33. func (s *InboundService) checkPortExist(port int, ignoreId int) (bool, error) {
  34. db := database.GetDB()
  35. db = db.Model(model.Inbound{}).Where("port = ?", port)
  36. if ignoreId > 0 {
  37. db = db.Where("id != ?", ignoreId)
  38. }
  39. var count int64
  40. err := db.Count(&count).Error
  41. if err != nil {
  42. return false, err
  43. }
  44. return count > 0, nil
  45. }
  46. func (s *InboundService) getClients(inbound *model.Inbound) ([]model.Client, error) {
  47. settings := map[string][]model.Client{}
  48. json.Unmarshal([]byte(inbound.Settings), &settings)
  49. if settings == nil {
  50. return nil, fmt.Errorf("setting is null")
  51. }
  52. clients := settings["clients"]
  53. if clients == nil {
  54. return nil, nil
  55. }
  56. return clients, nil
  57. }
  58. func (s *InboundService) checkEmailsExist(emails map[string]bool, ignoreId int) (string, error) {
  59. db := database.GetDB()
  60. var inbounds []*model.Inbound
  61. db = db.Model(model.Inbound{}).Where("Protocol in ?", []model.Protocol{model.VMess, model.VLESS, model.Trojan})
  62. if ignoreId > 0 {
  63. db = db.Where("id != ?", ignoreId)
  64. }
  65. db = db.Find(&inbounds)
  66. if db.Error != nil {
  67. return "", db.Error
  68. }
  69. for _, inbound := range inbounds {
  70. clients, err := s.getClients(inbound)
  71. if err != nil {
  72. return "", err
  73. }
  74. for _, client := range clients {
  75. if emails[client.Email] {
  76. return client.Email, nil
  77. }
  78. }
  79. }
  80. return "", nil
  81. }
  82. func (s *InboundService) checkEmailExistForInbound(inbound *model.Inbound) (string, error) {
  83. clients, err := s.getClients(inbound)
  84. if err != nil {
  85. return "", err
  86. }
  87. emails := make(map[string]bool)
  88. for _, client := range clients {
  89. if client.Email != "" {
  90. if emails[client.Email] {
  91. return client.Email, nil
  92. }
  93. emails[client.Email] = true
  94. }
  95. }
  96. return s.checkEmailsExist(emails, inbound.Id)
  97. }
  98. func (s *InboundService) AddInbound(inbound *model.Inbound) (*model.Inbound, error) {
  99. exist, err := s.checkPortExist(inbound.Port, 0)
  100. if err != nil {
  101. return inbound, err
  102. }
  103. if exist {
  104. return inbound, common.NewError("Port already exists:", inbound.Port)
  105. }
  106. existEmail, err := s.checkEmailExistForInbound(inbound)
  107. if err != nil {
  108. return inbound, err
  109. }
  110. if existEmail != "" {
  111. return inbound, common.NewError("Duplicate email:", existEmail)
  112. }
  113. clients, err := s.getClients(inbound)
  114. if err != nil {
  115. return inbound, err
  116. }
  117. db := database.GetDB()
  118. err = db.Save(inbound).Error
  119. if err == nil {
  120. for _, client := range clients {
  121. s.AddClientStat(inbound.Id, &client)
  122. }
  123. }
  124. return inbound, err
  125. }
  126. func (s *InboundService) AddInbounds(inbounds []*model.Inbound) error {
  127. for _, inbound := range inbounds {
  128. exist, err := s.checkPortExist(inbound.Port, 0)
  129. if err != nil {
  130. return err
  131. }
  132. if exist {
  133. return common.NewError("Port already exists:", inbound.Port)
  134. }
  135. }
  136. db := database.GetDB()
  137. tx := db.Begin()
  138. var err error
  139. defer func() {
  140. if err == nil {
  141. tx.Commit()
  142. } else {
  143. tx.Rollback()
  144. }
  145. }()
  146. for _, inbound := range inbounds {
  147. err = tx.Save(inbound).Error
  148. if err != nil {
  149. return err
  150. }
  151. }
  152. return nil
  153. }
  154. func (s *InboundService) DelInbound(id int) error {
  155. db := database.GetDB()
  156. err := db.Where("inbound_id = ?", id).Delete(xray.ClientTraffic{}).Error
  157. if err != nil {
  158. return err
  159. }
  160. inbound, err := s.GetInbound(id)
  161. if err != nil {
  162. return err
  163. }
  164. clients, err := s.getClients(inbound)
  165. if err != nil {
  166. return err
  167. }
  168. for _, client := range clients {
  169. err := s.DelClientIPs(db, client.Email)
  170. if err != nil {
  171. return err
  172. }
  173. }
  174. return db.Delete(model.Inbound{}, id).Error
  175. }
  176. func (s *InboundService) GetInbound(id int) (*model.Inbound, error) {
  177. db := database.GetDB()
  178. inbound := &model.Inbound{}
  179. err := db.Model(model.Inbound{}).First(inbound, id).Error
  180. if err != nil {
  181. return nil, err
  182. }
  183. return inbound, nil
  184. }
  185. func (s *InboundService) UpdateInbound(inbound *model.Inbound) (*model.Inbound, error) {
  186. exist, err := s.checkPortExist(inbound.Port, inbound.Id)
  187. if err != nil {
  188. return inbound, err
  189. }
  190. if exist {
  191. return inbound, common.NewError("Port already exists:", inbound.Port)
  192. }
  193. existEmail, err := s.checkEmailExistForInbound(inbound)
  194. if err != nil {
  195. return inbound, err
  196. }
  197. if existEmail != "" {
  198. return inbound, common.NewError("Duplicate email:", existEmail)
  199. }
  200. oldInbound, err := s.GetInbound(inbound.Id)
  201. if err != nil {
  202. return inbound, err
  203. }
  204. oldInbound.Up = inbound.Up
  205. oldInbound.Down = inbound.Down
  206. oldInbound.Total = inbound.Total
  207. oldInbound.Remark = inbound.Remark
  208. oldInbound.Enable = inbound.Enable
  209. oldInbound.ExpiryTime = inbound.ExpiryTime
  210. oldInbound.Listen = inbound.Listen
  211. oldInbound.Port = inbound.Port
  212. oldInbound.Protocol = inbound.Protocol
  213. oldInbound.Settings = inbound.Settings
  214. oldInbound.StreamSettings = inbound.StreamSettings
  215. oldInbound.Sniffing = inbound.Sniffing
  216. oldInbound.Tag = fmt.Sprintf("inbound-%v", inbound.Port)
  217. db := database.GetDB()
  218. return inbound, db.Save(oldInbound).Error
  219. }
  220. func (s *InboundService) AddInboundClient(inbound *model.Inbound) error {
  221. existEmail, err := s.checkEmailExistForInbound(inbound)
  222. if err != nil {
  223. return err
  224. }
  225. if existEmail != "" {
  226. return common.NewError("Duplicate email:", existEmail)
  227. }
  228. clients, err := s.getClients(inbound)
  229. if err != nil {
  230. return err
  231. }
  232. oldInbound, err := s.GetInbound(inbound.Id)
  233. if err != nil {
  234. return err
  235. }
  236. oldClients, err := s.getClients(oldInbound)
  237. if err != nil {
  238. return err
  239. }
  240. oldInbound.Settings = inbound.Settings
  241. if len(clients[len(clients)-1].Email) > 0 {
  242. s.AddClientStat(inbound.Id, &clients[len(clients)-1])
  243. }
  244. for i := len(oldClients); i < len(clients); i++ {
  245. if len(clients[i].Email) > 0 {
  246. s.AddClientStat(inbound.Id, &clients[i])
  247. }
  248. }
  249. db := database.GetDB()
  250. return db.Save(oldInbound).Error
  251. }
  252. func (s *InboundService) DelInboundClient(inbound *model.Inbound, email string) error {
  253. db := database.GetDB()
  254. err := s.DelClientStat(db, email)
  255. if err != nil {
  256. logger.Error("Delete stats Data Error")
  257. return err
  258. }
  259. oldInbound, err := s.GetInbound(inbound.Id)
  260. if err != nil {
  261. logger.Error("Load Old Data Error")
  262. return err
  263. }
  264. oldInbound.Settings = inbound.Settings
  265. err = s.DelClientIPs(db, email)
  266. if err != nil {
  267. logger.Error("Error in delete client IPs")
  268. return err
  269. }
  270. return db.Save(oldInbound).Error
  271. }
  272. func (s *InboundService) UpdateInboundClient(inbound *model.Inbound, index int) error {
  273. existEmail, err := s.checkEmailExistForInbound(inbound)
  274. if err != nil {
  275. return err
  276. }
  277. if existEmail != "" {
  278. return common.NewError("Duplicate email:", existEmail)
  279. }
  280. clients, err := s.getClients(inbound)
  281. if err != nil {
  282. return err
  283. }
  284. oldInbound, err := s.GetInbound(inbound.Id)
  285. if err != nil {
  286. return err
  287. }
  288. oldClients, err := s.getClients(oldInbound)
  289. if err != nil {
  290. return err
  291. }
  292. oldInbound.Settings = inbound.Settings
  293. db := database.GetDB()
  294. if len(clients[index].Email) > 0 {
  295. if len(oldClients[index].Email) > 0 {
  296. err = s.UpdateClientStat(oldClients[index].Email, &clients[index])
  297. if err != nil {
  298. return err
  299. }
  300. err = s.UpdateClientIPs(db, oldClients[index].Email, clients[index].Email)
  301. if err != nil {
  302. return err
  303. }
  304. } else {
  305. s.AddClientStat(inbound.Id, &clients[index])
  306. }
  307. } else {
  308. err = s.DelClientStat(db, oldClients[index].Email)
  309. if err != nil {
  310. return err
  311. }
  312. err = s.DelClientIPs(db, oldClients[index].Email)
  313. if err != nil {
  314. return err
  315. }
  316. }
  317. return db.Save(oldInbound).Error
  318. }
  319. func (s *InboundService) AddTraffic(traffics []*xray.Traffic) (err error) {
  320. if len(traffics) == 0 {
  321. return nil
  322. }
  323. db := database.GetDB()
  324. db = db.Model(model.Inbound{})
  325. tx := db.Begin()
  326. defer func() {
  327. if err != nil {
  328. tx.Rollback()
  329. } else {
  330. tx.Commit()
  331. }
  332. }()
  333. for _, traffic := range traffics {
  334. if traffic.IsInbound {
  335. err = tx.Where("tag = ?", traffic.Tag).
  336. UpdateColumns(map[string]interface{}{
  337. "up": gorm.Expr("up + ?", traffic.Up),
  338. "down": gorm.Expr("down + ?", traffic.Down)}).Error
  339. if err != nil {
  340. return
  341. }
  342. }
  343. }
  344. return
  345. }
  346. func (s *InboundService) AddClientTraffic(traffics []*xray.ClientTraffic) (err error) {
  347. if len(traffics) == 0 {
  348. return nil
  349. }
  350. traffics, err = s.adjustTraffics(traffics)
  351. if err != nil {
  352. return err
  353. }
  354. db := database.GetDB()
  355. db = db.Model(xray.ClientTraffic{})
  356. tx := db.Begin()
  357. defer func() {
  358. if err != nil {
  359. tx.Rollback()
  360. } else {
  361. tx.Commit()
  362. }
  363. }()
  364. err = tx.Save(traffics).Error
  365. if err != nil {
  366. logger.Warning("AddClientTraffic update data ", err)
  367. }
  368. return nil
  369. }
  370. func (s *InboundService) adjustTraffics(traffics []*xray.ClientTraffic) (full_traffics []*xray.ClientTraffic, err error) {
  371. db := database.GetDB()
  372. dbInbound := db.Model(model.Inbound{})
  373. txInbound := dbInbound.Begin()
  374. defer func() {
  375. if err != nil {
  376. txInbound.Rollback()
  377. } else {
  378. txInbound.Commit()
  379. }
  380. }()
  381. for _, traffic := range traffics {
  382. inbound := &model.Inbound{}
  383. client_traffic := &xray.ClientTraffic{}
  384. err := db.Model(xray.ClientTraffic{}).Where("email = ?", traffic.Email).First(client_traffic).Error
  385. if err != nil {
  386. if err == gorm.ErrRecordNotFound {
  387. logger.Warning(err, traffic.Email)
  388. }
  389. continue
  390. }
  391. client_traffic.Up += traffic.Up
  392. client_traffic.Down += traffic.Down
  393. err = txInbound.Where("id=?", client_traffic.InboundId).First(inbound).Error
  394. if err != nil {
  395. if err == gorm.ErrRecordNotFound {
  396. logger.Warning(err, traffic.Email)
  397. }
  398. continue
  399. }
  400. // get clients
  401. clients, err := s.getClients(inbound)
  402. needUpdate := false
  403. if err == nil {
  404. for client_index, client := range clients {
  405. if traffic.Email == client.Email {
  406. if client.ExpiryTime < 0 {
  407. clients[client_index].ExpiryTime = (time.Now().Unix() * 1000) - client.ExpiryTime
  408. needUpdate = true
  409. }
  410. client_traffic.ExpiryTime = client.ExpiryTime
  411. client_traffic.Total = client.TotalGB
  412. break
  413. }
  414. }
  415. }
  416. if needUpdate {
  417. settings := map[string]interface{}{}
  418. json.Unmarshal([]byte(inbound.Settings), &settings)
  419. // Convert clients to []interface to update clients in settings
  420. var clientsInterface []interface{}
  421. for _, c := range clients {
  422. clientsInterface = append(clientsInterface, interface{}(c))
  423. }
  424. settings["clients"] = clientsInterface
  425. modifiedSettings, err := json.MarshalIndent(settings, "", " ")
  426. if err != nil {
  427. return nil, err
  428. }
  429. err = txInbound.Where("id=?", inbound.Id).Update("settings", string(modifiedSettings)).Error
  430. if err != nil {
  431. return nil, err
  432. }
  433. }
  434. full_traffics = append(full_traffics, client_traffic)
  435. }
  436. return full_traffics, nil
  437. }
  438. func (s *InboundService) DisableInvalidInbounds() (int64, error) {
  439. db := database.GetDB()
  440. now := time.Now().Unix() * 1000
  441. result := db.Model(model.Inbound{}).
  442. Where("((total > 0 and up + down >= total) or (expiry_time > 0 and expiry_time <= ?)) and enable = ?", now, true).
  443. Update("enable", false)
  444. err := result.Error
  445. count := result.RowsAffected
  446. return count, err
  447. }
  448. func (s *InboundService) RemoveOrphanedTraffics() {
  449. db := database.GetDB()
  450. db.Exec(`
  451. DELETE FROM client_traffics
  452. WHERE email NOT IN (
  453. SELECT JSON_EXTRACT(client.value, '$.email')
  454. FROM inbounds,
  455. JSON_EACH(JSON_EXTRACT(inbounds.settings, '$.clients')) AS client
  456. )
  457. `)
  458. }
  459. func (s *InboundService) DisableInvalidClients() (int64, error) {
  460. db := database.GetDB()
  461. now := time.Now().Unix() * 1000
  462. result := db.Model(xray.ClientTraffic{}).
  463. Where("((total > 0 and up + down >= total) or (expiry_time > 0 and expiry_time <= ?)) and enable = ?", now, true).
  464. Update("enable", false)
  465. err := result.Error
  466. count := result.RowsAffected
  467. return count, err
  468. }
  469. func (s *InboundService) AddClientStat(inboundId int, client *model.Client) error {
  470. db := database.GetDB()
  471. clientTraffic := xray.ClientTraffic{}
  472. clientTraffic.InboundId = inboundId
  473. clientTraffic.Email = client.Email
  474. clientTraffic.Total = client.TotalGB
  475. clientTraffic.ExpiryTime = client.ExpiryTime
  476. clientTraffic.Enable = true
  477. clientTraffic.Up = 0
  478. clientTraffic.Down = 0
  479. result := db.Create(&clientTraffic)
  480. err := result.Error
  481. if err != nil {
  482. return err
  483. }
  484. return nil
  485. }
  486. func (s *InboundService) UpdateClientStat(email string, client *model.Client) error {
  487. db := database.GetDB()
  488. result := db.Model(xray.ClientTraffic{}).
  489. Where("email = ?", email).
  490. Updates(map[string]interface{}{
  491. "enable": true,
  492. "email": client.Email,
  493. "total": client.TotalGB,
  494. "expiry_time": client.ExpiryTime})
  495. err := result.Error
  496. if err != nil {
  497. return err
  498. }
  499. return nil
  500. }
  501. func (s *InboundService) UpdateClientIPs(tx *gorm.DB, oldEmail string, newEmail string) error {
  502. return tx.Model(model.InboundClientIps{}).Where("client_email = ?", oldEmail).Update("client_email", newEmail).Error
  503. }
  504. func (s *InboundService) DelClientStat(tx *gorm.DB, email string) error {
  505. return tx.Where("email = ?", email).Delete(xray.ClientTraffic{}).Error
  506. }
  507. func (s *InboundService) DelClientIPs(tx *gorm.DB, email string) error {
  508. logger.Warning(email)
  509. return tx.Where("client_email = ?", email).Delete(model.InboundClientIps{}).Error
  510. }
  511. func (s *InboundService) ResetClientTraffic(id int, clientEmail string) error {
  512. db := database.GetDB()
  513. result := db.Model(xray.ClientTraffic{}).
  514. Where("inbound_id = ? and email = ?", id, clientEmail).
  515. Updates(map[string]interface{}{"enable": true, "up": 0, "down": 0})
  516. err := result.Error
  517. if err != nil {
  518. return err
  519. }
  520. return nil
  521. }
  522. func (s *InboundService) ResetAllClientTraffics(id int) error {
  523. db := database.GetDB()
  524. result := db.Model(xray.ClientTraffic{}).
  525. Where("inbound_id = ?", id).
  526. Updates(map[string]interface{}{"enable": true, "up": 0, "down": 0})
  527. err := result.Error
  528. if err != nil {
  529. return err
  530. }
  531. return nil
  532. }
  533. func (s *InboundService) ResetAllTraffics() error {
  534. db := database.GetDB()
  535. result := db.Model(model.Inbound{}).
  536. Where("user_id > ?", 0).
  537. Updates(map[string]interface{}{"up": 0, "down": 0})
  538. err := result.Error
  539. if err != nil {
  540. return err
  541. }
  542. return nil
  543. }
  544. func (s *InboundService) GetClientTrafficTgBot(tguname string) ([]*xray.ClientTraffic, error) {
  545. db := database.GetDB()
  546. var inbounds []*model.Inbound
  547. err := db.Model(model.Inbound{}).Where("settings like ?", fmt.Sprintf(`%%"tgId": "%s"%%`, tguname)).Find(&inbounds).Error
  548. if err != nil && err != gorm.ErrRecordNotFound {
  549. return nil, err
  550. }
  551. var emails []string
  552. for _, inbound := range inbounds {
  553. clients, err := s.getClients(inbound)
  554. if err != nil {
  555. logger.Error("Unable to get clients from inbound")
  556. }
  557. for _, client := range clients {
  558. if client.TgID == tguname {
  559. emails = append(emails, client.Email)
  560. }
  561. }
  562. }
  563. var traffics []*xray.ClientTraffic
  564. err = db.Model(xray.ClientTraffic{}).Where("email IN ?", emails).Find(&traffics).Error
  565. if err != nil {
  566. if err == gorm.ErrRecordNotFound {
  567. logger.Warning(err)
  568. return nil, err
  569. }
  570. }
  571. return traffics, err
  572. }
  573. func (s *InboundService) GetClientTrafficByEmail(email string) (traffic []*xray.ClientTraffic, err error) {
  574. db := database.GetDB()
  575. var traffics []*xray.ClientTraffic
  576. err = db.Model(xray.ClientTraffic{}).Where("email like ?", "%"+email+"%").Find(&traffics).Error
  577. if err != nil {
  578. if err == gorm.ErrRecordNotFound {
  579. logger.Warning(err)
  580. return nil, err
  581. }
  582. }
  583. return traffics, err
  584. }
  585. func (s *InboundService) SearchClientTraffic(query string) (traffic *xray.ClientTraffic, err error) {
  586. db := database.GetDB()
  587. inbound := &model.Inbound{}
  588. traffic = &xray.ClientTraffic{}
  589. err = db.Model(model.Inbound{}).Where("settings like ?", "%\""+query+"\"%").First(inbound).Error
  590. if err != nil {
  591. if err == gorm.ErrRecordNotFound {
  592. logger.Warning(err)
  593. return nil, err
  594. }
  595. }
  596. traffic.InboundId = inbound.Id
  597. // get settings clients
  598. settings := map[string][]model.Client{}
  599. json.Unmarshal([]byte(inbound.Settings), &settings)
  600. clients := settings["clients"]
  601. for _, client := range clients {
  602. if client.ID == query && client.Email != "" {
  603. traffic.Email = client.Email
  604. break
  605. }
  606. if client.Password == query && client.Email != "" {
  607. traffic.Email = client.Email
  608. break
  609. }
  610. }
  611. if traffic.Email == "" {
  612. return nil, err
  613. }
  614. err = db.Model(xray.ClientTraffic{}).Where("email = ?", traffic.Email).First(traffic).Error
  615. if err != nil {
  616. logger.Warning(err)
  617. return nil, err
  618. }
  619. return traffic, err
  620. }
  621. func (s *InboundService) GetInboundClientIps(clientEmail string) (string, error) {
  622. db := database.GetDB()
  623. InboundClientIps := &model.InboundClientIps{}
  624. err := db.Model(model.InboundClientIps{}).Where("client_email = ?", clientEmail).First(InboundClientIps).Error
  625. if err != nil {
  626. return "", err
  627. }
  628. return InboundClientIps.Ips, nil
  629. }
  630. func (s *InboundService) ClearClientIps(clientEmail string) error {
  631. db := database.GetDB()
  632. result := db.Model(model.InboundClientIps{}).
  633. Where("client_email = ?", clientEmail).
  634. Update("ips", "")
  635. err := result.Error
  636. if err != nil {
  637. return err
  638. }
  639. return nil
  640. }
  641. func (s *InboundService) SearchInbounds(query string) ([]*model.Inbound, error) {
  642. db := database.GetDB()
  643. var inbounds []*model.Inbound
  644. err := db.Model(model.Inbound{}).Preload("ClientStats").Where("remark like ?", "%"+query+"%").Find(&inbounds).Error
  645. if err != nil && err != gorm.ErrRecordNotFound {
  646. return nil, err
  647. }
  648. return inbounds, nil
  649. }