123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646 |
- package service
- import (
- "encoding/json"
- "fmt"
- "time"
- "x-ui/database"
- "x-ui/database/model"
- "x-ui/logger"
- "x-ui/util/common"
- "x-ui/xray"
- "gorm.io/gorm"
- )
- type InboundService struct {
- }
- func (s *InboundService) GetInbounds(userId int) ([]*model.Inbound, error) {
- db := database.GetDB()
- var inbounds []*model.Inbound
- err := db.Model(model.Inbound{}).Preload("ClientStats").Where("user_id = ?", userId).Find(&inbounds).Error
- if err != nil && err != gorm.ErrRecordNotFound {
- return nil, err
- }
- return inbounds, nil
- }
- func (s *InboundService) GetAllInbounds() ([]*model.Inbound, error) {
- db := database.GetDB()
- var inbounds []*model.Inbound
- err := db.Model(model.Inbound{}).Preload("ClientStats").Find(&inbounds).Error
- if err != nil && err != gorm.ErrRecordNotFound {
- return nil, err
- }
- return inbounds, nil
- }
- func (s *InboundService) checkPortExist(port int, ignoreId int) (bool, error) {
- db := database.GetDB()
- db = db.Model(model.Inbound{}).Where("port = ?", port)
- if ignoreId > 0 {
- db = db.Where("id != ?", ignoreId)
- }
- var count int64
- err := db.Count(&count).Error
- if err != nil {
- return false, err
- }
- return count > 0, nil
- }
- func (s *InboundService) getClients(inbound *model.Inbound) ([]model.Client, error) {
- settings := map[string][]model.Client{}
- json.Unmarshal([]byte(inbound.Settings), &settings)
- if settings == nil {
- return nil, fmt.Errorf("setting is null")
- }
- clients := settings["clients"]
- if clients == nil {
- return nil, nil
- }
- return clients, nil
- }
- func (s *InboundService) checkEmailsExist(emails map[string]bool, ignoreId int) (string, error) {
- db := database.GetDB()
- var inbounds []*model.Inbound
- db = db.Model(model.Inbound{}).Where("Protocol in ?", []model.Protocol{model.VMess, model.VLESS, model.Trojan})
- if ignoreId > 0 {
- db = db.Where("id != ?", ignoreId)
- }
- db = db.Find(&inbounds)
- if db.Error != nil {
- return "", db.Error
- }
- for _, inbound := range inbounds {
- clients, err := s.getClients(inbound)
- if err != nil {
- return "", err
- }
- for _, client := range clients {
- if emails[client.Email] {
- return client.Email, nil
- }
- }
- }
- return "", nil
- }
- func (s *InboundService) checkEmailExistForInbound(inbound *model.Inbound) (string, error) {
- clients, err := s.getClients(inbound)
- if err != nil {
- return "", err
- }
- emails := make(map[string]bool)
- for _, client := range clients {
- if client.Email != "" {
- if emails[client.Email] {
- return client.Email, nil
- }
- emails[client.Email] = true
- }
- }
- return s.checkEmailsExist(emails, inbound.Id)
- }
- func (s *InboundService) AddInbound(inbound *model.Inbound) (*model.Inbound, error) {
- exist, err := s.checkPortExist(inbound.Port, 0)
- if err != nil {
- return inbound, err
- }
- if exist {
- return inbound, common.NewError("Port already exists:", inbound.Port)
- }
- existEmail, err := s.checkEmailExistForInbound(inbound)
- if err != nil {
- return inbound, err
- }
- if existEmail != "" {
- return inbound, common.NewError("Duplicate email:", existEmail)
- }
- clients, err := s.getClients(inbound)
- if err != nil {
- return inbound, err
- }
- db := database.GetDB()
- err = db.Save(inbound).Error
- if err == nil {
- for _, client := range clients {
- s.AddClientStat(inbound.Id, &client)
- }
- }
- return inbound, err
- }
- func (s *InboundService) AddInbounds(inbounds []*model.Inbound) error {
- for _, inbound := range inbounds {
- exist, err := s.checkPortExist(inbound.Port, 0)
- if err != nil {
- return err
- }
- if exist {
- return common.NewError("Port already exists:", inbound.Port)
- }
- }
- db := database.GetDB()
- tx := db.Begin()
- var err error
- defer func() {
- if err == nil {
- tx.Commit()
- } else {
- tx.Rollback()
- }
- }()
- for _, inbound := range inbounds {
- err = tx.Save(inbound).Error
- if err != nil {
- return err
- }
- }
- return nil
- }
- func (s *InboundService) DelInbound(id int) error {
- db := database.GetDB()
- err := db.Where("inbound_id = ?", id).Delete(xray.ClientTraffic{}).Error
- if err != nil {
- return err
- }
- inbound, err := s.GetInbound(id)
- if err != nil {
- return err
- }
- clients, err := s.getClients(inbound)
- if err != nil {
- return err
- }
- for _, client := range clients {
- err := s.DelClientIPs(db, client.Email)
- if err != nil {
- return err
- }
- }
- return db.Delete(model.Inbound{}, id).Error
- }
- func (s *InboundService) GetInbound(id int) (*model.Inbound, error) {
- db := database.GetDB()
- inbound := &model.Inbound{}
- err := db.Model(model.Inbound{}).First(inbound, id).Error
- if err != nil {
- return nil, err
- }
- return inbound, nil
- }
- func (s *InboundService) UpdateInbound(inbound *model.Inbound) (*model.Inbound, error) {
- exist, err := s.checkPortExist(inbound.Port, inbound.Id)
- if err != nil {
- return inbound, err
- }
- if exist {
- return inbound, common.NewError("Port already exists:", inbound.Port)
- }
- existEmail, err := s.checkEmailExistForInbound(inbound)
- if err != nil {
- return inbound, err
- }
- if existEmail != "" {
- return inbound, common.NewError("Duplicate email:", existEmail)
- }
- oldInbound, err := s.GetInbound(inbound.Id)
- if err != nil {
- return inbound, err
- }
- oldInbound.Up = inbound.Up
- oldInbound.Down = inbound.Down
- oldInbound.Total = inbound.Total
- oldInbound.Remark = inbound.Remark
- oldInbound.Enable = inbound.Enable
- oldInbound.ExpiryTime = inbound.ExpiryTime
- oldInbound.Listen = inbound.Listen
- oldInbound.Port = inbound.Port
- oldInbound.Protocol = inbound.Protocol
- oldInbound.Settings = inbound.Settings
- oldInbound.StreamSettings = inbound.StreamSettings
- oldInbound.Sniffing = inbound.Sniffing
- oldInbound.Tag = fmt.Sprintf("inbound-%v", inbound.Port)
- db := database.GetDB()
- return inbound, db.Save(oldInbound).Error
- }
- func (s *InboundService) AddInboundClient(inbound *model.Inbound) error {
- existEmail, err := s.checkEmailExistForInbound(inbound)
- if err != nil {
- return err
- }
- if existEmail != "" {
- return common.NewError("Duplicate email:", existEmail)
- }
- clients, err := s.getClients(inbound)
- if err != nil {
- return err
- }
- oldInbound, err := s.GetInbound(inbound.Id)
- if err != nil {
- return err
- }
- oldClients, err := s.getClients(oldInbound)
- if err != nil {
- return err
- }
- oldInbound.Settings = inbound.Settings
- if len(clients[len(clients)-1].Email) > 0 {
- s.AddClientStat(inbound.Id, &clients[len(clients)-1])
- }
- for i := len(oldClients); i < len(clients); i++ {
- if len(clients[i].Email) > 0 {
- s.AddClientStat(inbound.Id, &clients[i])
- }
- }
- db := database.GetDB()
- return db.Save(oldInbound).Error
- }
- func (s *InboundService) DelInboundClient(inbound *model.Inbound, email string) error {
- db := database.GetDB()
- err := s.DelClientStat(db, email)
- if err != nil {
- logger.Error("Delete stats Data Error")
- return err
- }
- oldInbound, err := s.GetInbound(inbound.Id)
- if err != nil {
- logger.Error("Load Old Data Error")
- return err
- }
- oldInbound.Settings = inbound.Settings
- err = s.DelClientIPs(db, email)
- if err != nil {
- logger.Error("Error in delete client IPs")
- return err
- }
- return db.Save(oldInbound).Error
- }
- func (s *InboundService) UpdateInboundClient(inbound *model.Inbound, index int) error {
- existEmail, err := s.checkEmailExistForInbound(inbound)
- if err != nil {
- return err
- }
- if existEmail != "" {
- return common.NewError("Duplicate email:", existEmail)
- }
- clients, err := s.getClients(inbound)
- if err != nil {
- return err
- }
- oldInbound, err := s.GetInbound(inbound.Id)
- if err != nil {
- return err
- }
- oldClients, err := s.getClients(oldInbound)
- if err != nil {
- return err
- }
- oldInbound.Settings = inbound.Settings
- db := database.GetDB()
- if len(clients[index].Email) > 0 {
- if len(oldClients[index].Email) > 0 {
- err = s.UpdateClientStat(oldClients[index].Email, &clients[index])
- if err != nil {
- return err
- }
- err = s.UpdateClientIPs(db, oldClients[index].Email, clients[index].Email)
- if err != nil {
- return err
- }
- } else {
- s.AddClientStat(inbound.Id, &clients[index])
- }
- } else {
- err = s.DelClientStat(db, oldClients[index].Email)
- if err != nil {
- return err
- }
- err = s.DelClientIPs(db, oldClients[index].Email)
- if err != nil {
- return err
- }
- }
- return db.Save(oldInbound).Error
- }
- func (s *InboundService) AddTraffic(traffics []*xray.Traffic) (err error) {
- if len(traffics) == 0 {
- return nil
- }
- db := database.GetDB()
- db = db.Model(model.Inbound{})
- tx := db.Begin()
- defer func() {
- if err != nil {
- tx.Rollback()
- } else {
- tx.Commit()
- }
- }()
- for _, traffic := range traffics {
- if traffic.IsInbound {
- err = tx.Where("tag = ?", traffic.Tag).
- UpdateColumns(map[string]interface{}{
- "up": gorm.Expr("up + ?", traffic.Up),
- "down": gorm.Expr("down + ?", traffic.Down)}).Error
- if err != nil {
- return
- }
- }
- }
- return
- }
- func (s *InboundService) AddClientTraffic(traffics []*xray.ClientTraffic) (err error) {
- if len(traffics) == 0 {
- return nil
- }
- db := database.GetDB()
- dbInbound := db.Model(model.Inbound{})
- db = db.Model(xray.ClientTraffic{})
- tx := db.Begin()
- defer func() {
- if err != nil {
- tx.Rollback()
- } else {
- tx.Commit()
- }
- }()
- txInbound := dbInbound.Begin()
- defer func() {
- if err != nil {
- txInbound.Rollback()
- } else {
- txInbound.Commit()
- }
- }()
- for _, traffic := range traffics {
- inbound := &model.Inbound{}
- client := &xray.ClientTraffic{}
- err := tx.Where("email = ?", traffic.Email).First(client).Error
- if err != nil {
- if err == gorm.ErrRecordNotFound {
- logger.Warning(err, traffic.Email)
- }
- continue
- }
- err = txInbound.Where("id=?", client.InboundId).First(inbound).Error
- if err != nil {
- if err == gorm.ErrRecordNotFound {
- logger.Warning(err, traffic.Email)
- }
- continue
- }
- // get settings clients
- settings := map[string][]model.Client{}
- json.Unmarshal([]byte(inbound.Settings), &settings)
- clients := settings["clients"]
- for _, client := range clients {
- if traffic.Email == client.Email {
- traffic.ExpiryTime = client.ExpiryTime
- traffic.Total = client.TotalGB
- }
- }
- if tx.Where("inbound_id = ? and email = ?", inbound.Id, traffic.Email).
- UpdateColumns(map[string]interface{}{
- "enable": true,
- "expiry_time": traffic.ExpiryTime,
- "total": traffic.Total,
- "up": gorm.Expr("up + ?", traffic.Up),
- "down": gorm.Expr("down + ?", traffic.Down)}).RowsAffected == 0 {
- err = tx.Create(traffic).Error
- }
- if err != nil {
- logger.Warning("AddClientTraffic update data ", err)
- continue
- }
- }
- return
- }
- func (s *InboundService) DisableInvalidInbounds() (int64, error) {
- db := database.GetDB()
- now := time.Now().Unix() * 1000
- result := db.Model(model.Inbound{}).
- Where("((total > 0 and up + down >= total) or (expiry_time > 0 and expiry_time <= ?)) and enable = ?", now, true).
- Update("enable", false)
- err := result.Error
- count := result.RowsAffected
- return count, err
- }
- func (s *InboundService) DisableInvalidClients() (int64, error) {
- db := database.GetDB()
- now := time.Now().Unix() * 1000
- result := db.Model(xray.ClientTraffic{}).
- Where("((total > 0 and up + down >= total) or (expiry_time > 0 and expiry_time <= ?)) and enable = ?", now, true).
- Update("enable", false)
- err := result.Error
- count := result.RowsAffected
- return count, err
- }
- func (s *InboundService) AddClientStat(inboundId int, client *model.Client) error {
- db := database.GetDB()
- clientTraffic := xray.ClientTraffic{}
- clientTraffic.InboundId = inboundId
- clientTraffic.Email = client.Email
- clientTraffic.Total = client.TotalGB
- clientTraffic.ExpiryTime = client.ExpiryTime
- clientTraffic.Enable = true
- clientTraffic.Up = 0
- clientTraffic.Down = 0
- result := db.Create(&clientTraffic)
- err := result.Error
- if err != nil {
- return err
- }
- return nil
- }
- func (s *InboundService) UpdateClientStat(email string, client *model.Client) error {
- db := database.GetDB()
- result := db.Model(xray.ClientTraffic{}).
- Where("email = ?", email).
- Updates(map[string]interface{}{
- "enable": true,
- "email": client.Email,
- "total": client.TotalGB,
- "expiry_time": client.ExpiryTime})
- err := result.Error
- if err != nil {
- return err
- }
- return nil
- }
- func (s *InboundService) UpdateClientIPs(tx *gorm.DB, oldEmail string, newEmail string) error {
- return tx.Model(model.InboundClientIps{}).Where("client_email = ?", oldEmail).Update("client_email", newEmail).Error
- }
- func (s *InboundService) DelClientStat(tx *gorm.DB, email string) error {
- return tx.Where("email = ?", email).Delete(xray.ClientTraffic{}).Error
- }
- func (s *InboundService) DelClientIPs(tx *gorm.DB, email string) error {
- logger.Warning(email)
- return tx.Where("client_email = ?", email).Delete(model.InboundClientIps{}).Error
- }
- func (s *InboundService) ResetClientTraffic(id int, clientEmail string) error {
- db := database.GetDB()
- result := db.Model(xray.ClientTraffic{}).
- Where("inbound_id = ? and email = ?", id, clientEmail).
- Updates(map[string]interface{}{"enable": true, "up": 0, "down": 0})
- err := result.Error
- if err != nil {
- return err
- }
- return nil
- }
- func (s *InboundService) GetClientTrafficTgBot(tguname string) (traffic []*xray.ClientTraffic, err error) {
- db := database.GetDB()
- var traffics []*xray.ClientTraffic
- err = db.Model(xray.ClientTraffic{}).Where("email like ?", "%@"+tguname).Find(&traffics).Error
- if err != nil {
- if err == gorm.ErrRecordNotFound {
- logger.Warning(err)
- return nil, err
- }
- }
- return traffics, err
- }
- func (s *InboundService) GetClientTrafficByEmail(email string) (traffic []*xray.ClientTraffic, err error) {
- db := database.GetDB()
- var traffics []*xray.ClientTraffic
- err = db.Model(xray.ClientTraffic{}).Where("email like ?", "%"+email+"%").Find(&traffics).Error
- if err != nil {
- if err == gorm.ErrRecordNotFound {
- logger.Warning(err)
- return nil, err
- }
- }
- return traffics, err
- }
- func (s *InboundService) SearchClientTraffic(query string) (traffic *xray.ClientTraffic, err error) {
- db := database.GetDB()
- inbound := &model.Inbound{}
- traffic = &xray.ClientTraffic{}
- err = db.Model(model.Inbound{}).Where("settings like ?", "%\""+query+"\"%").First(inbound).Error
- if err != nil {
- if err == gorm.ErrRecordNotFound {
- logger.Warning(err)
- return nil, err
- }
- }
- traffic.InboundId = inbound.Id
- // get settings clients
- settings := map[string][]model.Client{}
- json.Unmarshal([]byte(inbound.Settings), &settings)
- clients := settings["clients"]
- for _, client := range clients {
- if client.ID == query && client.Email != "" {
- traffic.Email = client.Email
- break
- }
- if client.Password == query && client.Email != "" {
- traffic.Email = client.Email
- break
- }
- }
- if traffic.Email == "" {
- return nil, err
- }
- err = db.Model(xray.ClientTraffic{}).Where("email = ?", traffic.Email).First(traffic).Error
- if err != nil {
- logger.Warning(err)
- return nil, err
- }
- return traffic, err
- }
- func (s *InboundService) GetInboundClientIps(clientEmail string) (string, error) {
- db := database.GetDB()
- InboundClientIps := &model.InboundClientIps{}
- err := db.Model(model.InboundClientIps{}).Where("client_email = ?", clientEmail).First(InboundClientIps).Error
- if err != nil {
- return "", err
- }
- return InboundClientIps.Ips, nil
- }
- func (s *InboundService) ClearClientIps(clientEmail string) error {
- db := database.GetDB()
- result := db.Model(model.InboundClientIps{}).
- Where("client_email = ?", clientEmail).
- Update("ips", "")
- err := result.Error
- if err != nil {
- return err
- }
- return nil
- }
- func (s *InboundService) SearchInbounds(query string) ([]*model.Inbound, error) {
- db := database.GetDB()
- var inbounds []*model.Inbound
- err := db.Model(model.Inbound{}).Preload("ClientStats").Where("remark like ?", "%"+query+"%").Find(&inbounds).Error
- if err != nil && err != gorm.ErrRecordNotFound {
- return nil, err
- }
- return inbounds, nil
- }
|