All checks were successful
Build docker and publish / build (20.15.1) (push) Successful in 6m50s
- 创建 log_message 表用于存储客户端错误日志 - 实现客户端日志上报接口 POST /v1/common/log/message/report - 添加管理端日志查询接口 GET /v1/admin/log/message/error/list 和 GET /v1/admin/log/message/error/detail - 实现日志指纹去重和限流机制 - 完善相关模型、逻辑和文档说明
185 lines
5.5 KiB
Go
185 lines
5.5 KiB
Go
package svc
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/perfect-panel/server/internal/model/client"
|
|
"github.com/perfect-panel/server/internal/model/node"
|
|
"github.com/perfect-panel/server/pkg/device"
|
|
|
|
"github.com/perfect-panel/server/internal/config"
|
|
"github.com/perfect-panel/server/internal/model/ads"
|
|
"github.com/perfect-panel/server/internal/model/announcement"
|
|
"github.com/perfect-panel/server/internal/model/auth"
|
|
"github.com/perfect-panel/server/internal/model/coupon"
|
|
"github.com/perfect-panel/server/internal/model/document"
|
|
"github.com/perfect-panel/server/internal/model/log"
|
|
logmessage "github.com/perfect-panel/server/internal/model/logmessage"
|
|
"github.com/perfect-panel/server/internal/model/order"
|
|
"github.com/perfect-panel/server/internal/model/payment"
|
|
"github.com/perfect-panel/server/internal/model/subscribe"
|
|
"github.com/perfect-panel/server/internal/model/system"
|
|
"github.com/perfect-panel/server/internal/model/ticket"
|
|
"github.com/perfect-panel/server/internal/model/traffic"
|
|
"github.com/perfect-panel/server/internal/model/user"
|
|
"github.com/perfect-panel/server/pkg/limit"
|
|
"github.com/perfect-panel/server/pkg/nodeMultiplier"
|
|
"github.com/perfect-panel/server/pkg/orm"
|
|
|
|
tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5"
|
|
"github.com/hibiken/asynq"
|
|
"github.com/redis/go-redis/v9"
|
|
"gorm.io/gorm"
|
|
)
|
|
|
|
type ServiceContext struct {
|
|
DB *gorm.DB
|
|
Redis *redis.Client
|
|
Config config.Config
|
|
Queue *asynq.Client
|
|
ExchangeRate float64
|
|
|
|
//NodeCache *cache.NodeCacheClient
|
|
AuthModel auth.Model
|
|
AdsModel ads.Model
|
|
LogModel log.Model
|
|
LogMessageModel logmessage.Model
|
|
NodeModel node.Model
|
|
UserModel user.Model
|
|
OrderModel order.Model
|
|
ClientModel client.Model
|
|
TicketModel ticket.Model
|
|
//ServerModel server.Model
|
|
SystemModel system.Model
|
|
CouponModel coupon.Model
|
|
PaymentModel payment.Model
|
|
DocumentModel document.Model
|
|
SubscribeModel subscribe.Model
|
|
TrafficLogModel traffic.Model
|
|
AnnouncementModel announcement.Model
|
|
|
|
Restart func() error
|
|
TelegramBot *tgbotapi.BotAPI
|
|
NodeMultiplierManager *nodeMultiplier.Manager
|
|
AuthLimiter *limit.PeriodLimit
|
|
DeviceManager *device.DeviceManager
|
|
}
|
|
|
|
func NewServiceContext(c config.Config) *ServiceContext {
|
|
// gorm initialize
|
|
db, err := orm.ConnectMysql(orm.Mysql{
|
|
Config: c.MySQL,
|
|
})
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
rds := redis.NewClient(&redis.Options{
|
|
Addr: c.Redis.Host,
|
|
Password: c.Redis.Pass,
|
|
DB: c.Redis.DB,
|
|
})
|
|
err = rds.Ping(context.Background()).Err()
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
if c.Debug {
|
|
_ = rds.FlushAll(context.Background()).Err()
|
|
}
|
|
authLimiter := limit.NewPeriodLimit(86400, 15, rds, config.SendCountLimitKeyPrefix, limit.Align())
|
|
srv := &ServiceContext{
|
|
DB: db,
|
|
Redis: rds,
|
|
Config: c,
|
|
Queue: NewAsynqClient(c),
|
|
ExchangeRate: 1.0,
|
|
//NodeCache: cache.NewNodeCacheClient(rds),
|
|
AuthLimiter: authLimiter,
|
|
AdsModel: ads.NewModel(db, rds),
|
|
LogModel: log.NewModel(db),
|
|
LogMessageModel: logmessage.NewModel(db),
|
|
NodeModel: node.NewModel(db, rds),
|
|
AuthModel: auth.NewModel(db, rds),
|
|
UserModel: user.NewModel(db, rds),
|
|
OrderModel: order.NewModel(db, rds),
|
|
ClientModel: client.NewSubscribeApplicationModel(db),
|
|
TicketModel: ticket.NewModel(db, rds),
|
|
//ServerModel: server.NewModel(db, rds),
|
|
SystemModel: system.NewModel(db, rds),
|
|
CouponModel: coupon.NewModel(db, rds),
|
|
PaymentModel: payment.NewModel(db, rds),
|
|
DocumentModel: document.NewModel(db, rds),
|
|
SubscribeModel: subscribe.NewModel(db, rds),
|
|
TrafficLogModel: traffic.NewModel(db),
|
|
AnnouncementModel: announcement.NewModel(db, rds),
|
|
}
|
|
srv.DeviceManager = NewDeviceManager(srv)
|
|
return srv
|
|
|
|
}
|
|
|
|
func (srv *ServiceContext) SessionLimit() int64 {
|
|
cd := srv.Config.Site.CustomData
|
|
if cd != "" {
|
|
var obj map[string]interface{}
|
|
if json.Unmarshal([]byte(cd), &obj) == nil {
|
|
if v, ok := obj["deviceLimit"]; ok {
|
|
switch val := v.(type) {
|
|
case float64:
|
|
if val > 0 {
|
|
return int64(val)
|
|
}
|
|
case string:
|
|
if n, err := strconv.ParseInt(strings.TrimSpace(val), 10, 64); err == nil && n > 0 {
|
|
return n
|
|
}
|
|
}
|
|
}
|
|
if v, ok := obj["DeviceLimit"]; ok {
|
|
switch val := v.(type) {
|
|
case float64:
|
|
if val > 0 {
|
|
return int64(val)
|
|
}
|
|
case string:
|
|
if n, err := strconv.ParseInt(strings.TrimSpace(val), 10, 64); err == nil && n > 0 {
|
|
return n
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return srv.Config.JwtAuth.MaxSessionsPerUser
|
|
}
|
|
|
|
func (srv *ServiceContext) EnforceUserSessionLimit(ctx context.Context, userId int64, newSessionId string, max int64) error {
|
|
if max <= 0 {
|
|
return nil
|
|
}
|
|
sessionsKey := fmt.Sprintf("%s%v", config.UserSessionsKeyPrefix, userId)
|
|
now := time.Now().Unix()
|
|
if err := srv.Redis.ZAdd(ctx, sessionsKey, redis.Z{Score: float64(now), Member: newSessionId}).Err(); err != nil {
|
|
return err
|
|
}
|
|
count, err := srv.Redis.ZCard(ctx, sessionsKey).Result()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if count > max {
|
|
popped, err := srv.Redis.ZPopMin(ctx, sessionsKey, count-max).Result()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, z := range popped {
|
|
sid := fmt.Sprintf("%v", z.Member)
|
|
_ = srv.Redis.Del(ctx, fmt.Sprintf("%v:%v", config.SessionIdKey, sid)).Err()
|
|
}
|
|
}
|
|
_ = srv.Redis.Expire(ctx, sessionsKey, time.Duration(srv.Config.JwtAuth.AccessExpire)*time.Second).Err()
|
|
return nil
|
|
}
|