mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2025-03-01 01:20:49 +03:00
418 lines
10 KiB
Go
418 lines
10 KiB
Go
![]() |
package service
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"time"
|
||
|
"x-ui/database"
|
||
|
"encoding/json"
|
||
|
"x-ui/database/model"
|
||
|
"x-ui/util/common"
|
||
|
"x-ui/xray"
|
||
|
"x-ui/logger"
|
||
|
|
||
|
"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})
|
||
|
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("端口已存在:", inbound.Port)
|
||
|
}
|
||
|
|
||
|
existEmail, err := s.checkEmailExistForInbound(inbound)
|
||
|
if err != nil {
|
||
|
return inbound, err
|
||
|
}
|
||
|
if existEmail != "" {
|
||
|
return inbound, common.NewError("Duplicate email:", existEmail)
|
||
|
}
|
||
|
|
||
|
db := database.GetDB()
|
||
|
|
||
|
err = db.Save(inbound).Error
|
||
|
if err == nil {
|
||
|
s.UpdateClientStat(inbound.Id,inbound.Settings)
|
||
|
}
|
||
|
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("端口已存在:", 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()
|
||
|
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("端口已存在:", 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)
|
||
|
|
||
|
s.UpdateClientStat(inbound.Id,inbound.Settings)
|
||
|
db := database.GetDB()
|
||
|
return inbound, 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).
|
||
|
UpdateColumn("up", gorm.Expr("up + ?", traffic.Up)).
|
||
|
UpdateColumn("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{}
|
||
|
|
||
|
err := txInbound.Where("settings like ?", "%" + traffic.Email + "%").First(inbound).Error
|
||
|
traffic.InboundId = inbound.Id
|
||
|
if err != nil {
|
||
|
if err == gorm.ErrRecordNotFound {
|
||
|
// delete removed client record
|
||
|
clientErr := s.DelClientStat(tx, traffic.Email)
|
||
|
logger.Warning(err, traffic.Email,clientErr)
|
||
|
|
||
|
}
|
||
|
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 = ?", inbound.Id).Where("email = ?", traffic.Email).
|
||
|
UpdateColumn("enable", true).
|
||
|
UpdateColumn("expiry_time", traffic.ExpiryTime).
|
||
|
UpdateColumn("total",traffic.Total).
|
||
|
UpdateColumn("up", gorm.Expr("up + ?", traffic.Up)).
|
||
|
UpdateColumn("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) UpdateClientStat(inboundId int, inboundSettings string) (error) {
|
||
|
db := database.GetDB()
|
||
|
|
||
|
// get settings clients
|
||
|
settings := map[string][]model.Client{}
|
||
|
json.Unmarshal([]byte(inboundSettings), &settings)
|
||
|
clients := settings["clients"]
|
||
|
for _, client := range clients {
|
||
|
result := db.Model(xray.ClientTraffic{}).
|
||
|
Where("inbound_id = ? and email = ?", inboundId, client.Email).
|
||
|
Updates(map[string]interface{}{"enable": true, "total": client.TotalGB, "expiry_time": client.ExpiryTime})
|
||
|
if result.RowsAffected == 0 {
|
||
|
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
|
||
|
db.Create(&clientTraffic)
|
||
|
}
|
||
|
err := result.Error
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
func (s *InboundService) DelClientStat(tx *gorm.DB, email string) error {
|
||
|
return tx.Where("email = ?", email).Delete(xray.ClientTraffic{}).Error
|
||
|
}
|
||
|
|
||
|
func (s *InboundService) ResetClientTraffic(clientEmail string) (error) {
|
||
|
db := database.GetDB()
|
||
|
|
||
|
result := db.Model(xray.ClientTraffic{}).
|
||
|
Where("email = ?", clientEmail).
|
||
|
Update("up", 0).
|
||
|
Update("down", 0)
|
||
|
|
||
|
err := result.Error
|
||
|
|
||
|
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
func (s *InboundService) GetClientTrafficById(uuid string) (traffic *xray.ClientTraffic, err error) {
|
||
|
db := database.GetDB()
|
||
|
inbound := &model.Inbound{}
|
||
|
traffic = &xray.ClientTraffic{}
|
||
|
|
||
|
err = db.Model(model.Inbound{}).Where("settings like ?", "%" + uuid + "%").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 uuid == client.ID {
|
||
|
traffic.Email = client.Email
|
||
|
}
|
||
|
}
|
||
|
err = db.Model(xray.ClientTraffic{}).Where("email = ?", traffic.Email).First(traffic).Error
|
||
|
if err != nil {
|
||
|
logger.Warning(err)
|
||
|
return nil, err
|
||
|
}
|
||
|
return traffic, err
|
||
|
}
|