* add: device login * update: global config * add: User transmission interface encryption * update: get global config * update: User transmission interface encryption * add: get device list * add: SecretIsEmpty Message * update: device middleware * add: query user subscribe node list * fix bug: query device list * fix bug: unbind device * update: device login * fix bug: The ad table is missing the description field * fix bug:page size is zero * update: Device Middleware * fix bug: Site custom data update failed
196 lines
6.5 KiB
Go
196 lines
6.5 KiB
Go
package subscribe
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/perfect-panel/server/internal/model/node"
|
|
"github.com/perfect-panel/server/internal/model/user"
|
|
"github.com/perfect-panel/server/internal/svc"
|
|
"github.com/perfect-panel/server/internal/types"
|
|
"github.com/perfect-panel/server/pkg/constant"
|
|
"github.com/perfect-panel/server/pkg/logger"
|
|
"github.com/perfect-panel/server/pkg/tool"
|
|
"github.com/perfect-panel/server/pkg/xerr"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type QueryUserSubscribeNodeListLogic struct {
|
|
logger.Logger
|
|
ctx context.Context
|
|
svcCtx *svc.ServiceContext
|
|
}
|
|
|
|
// Get user subscribe node info
|
|
func NewQueryUserSubscribeNodeListLogic(ctx context.Context, svcCtx *svc.ServiceContext) *QueryUserSubscribeNodeListLogic {
|
|
return &QueryUserSubscribeNodeListLogic{
|
|
Logger: logger.WithContext(ctx),
|
|
ctx: ctx,
|
|
svcCtx: svcCtx,
|
|
}
|
|
}
|
|
|
|
func (l *QueryUserSubscribeNodeListLogic) QueryUserSubscribeNodeList() (resp *types.QueryUserSubscribeNodeListResponse, err error) {
|
|
u, ok := l.ctx.Value(constant.CtxKeyUser).(*user.User)
|
|
if !ok {
|
|
logger.Error("current user is not found in context")
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.InvalidAccess), "Invalid Access")
|
|
}
|
|
|
|
userSubscribes, err := l.svcCtx.UserModel.QueryUserSubscribe(l.ctx, u.Id, 1, 2)
|
|
if err != nil {
|
|
logger.Errorw("failed to query user subscribe", logger.Field("error", err.Error()), logger.Field("user_id", u.Id))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "DB_ERROR")
|
|
}
|
|
|
|
resp = &types.QueryUserSubscribeNodeListResponse{}
|
|
for _, us := range userSubscribes {
|
|
userSubscribe, err := l.getUserSubscribe(us.Token)
|
|
if err != nil {
|
|
l.Errorw("[SubscribeLogic] Get user subscribe failed", logger.Field("error", err.Error()), logger.Field("token", userSubscribe.Token))
|
|
return nil, err
|
|
}
|
|
nodes, err := l.getServers(userSubscribe)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
userSubscribeInfo := types.UserSubscribeInfo{
|
|
Id: userSubscribe.Id,
|
|
Nodes: nodes,
|
|
Traffic: userSubscribe.Traffic,
|
|
Upload: userSubscribe.Upload,
|
|
Download: userSubscribe.Download,
|
|
Token: userSubscribe.Token,
|
|
UserId: userSubscribe.UserId,
|
|
OrderId: userSubscribe.OrderId,
|
|
SubscribeId: userSubscribe.SubscribeId,
|
|
StartTime: userSubscribe.StartTime.Unix(),
|
|
ExpireTime: userSubscribe.ExpireTime.Unix(),
|
|
Status: userSubscribe.Status,
|
|
CreatedAt: userSubscribe.CreatedAt.Unix(),
|
|
UpdatedAt: userSubscribe.UpdatedAt.Unix(),
|
|
}
|
|
|
|
if userSubscribe.FinishedAt != nil {
|
|
userSubscribeInfo.FinishedAt = userSubscribe.FinishedAt.Unix()
|
|
}
|
|
|
|
if l.svcCtx.Config.Register.EnableTrial && l.svcCtx.Config.Register.TrialSubscribe == userSubscribe.SubscribeId {
|
|
userSubscribeInfo.IsTryOut = true
|
|
}
|
|
|
|
resp.List = append(resp.List, userSubscribeInfo)
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (l *QueryUserSubscribeNodeListLogic) getServers(userSub *user.Subscribe) (userSubscribeNodes []*types.UserSubscribeNodeInfo, err error) {
|
|
userSubscribeNodes = make([]*types.UserSubscribeNodeInfo, 0)
|
|
if l.isSubscriptionExpired(userSub) {
|
|
return l.createExpiredServers(), nil
|
|
}
|
|
|
|
subDetails, err := l.svcCtx.SubscribeModel.FindOne(l.ctx, userSub.SubscribeId)
|
|
if err != nil {
|
|
l.Errorw("[Generate Subscribe]find subscribe details error: %v", logger.Field("error", err.Error()))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "find subscribe details error: %v", err.Error())
|
|
}
|
|
nodeIds := tool.StringToInt64Slice(subDetails.Nodes)
|
|
tags := strings.Split(subDetails.NodeTags, ",")
|
|
|
|
l.Debugf("[Generate Subscribe]nodes: %v, NodeTags: %v", nodeIds, tags)
|
|
|
|
enable := true
|
|
|
|
_, nodes, err := l.svcCtx.NodeModel.FilterNodeList(l.ctx, &node.FilterNodeParams{
|
|
Page: 0,
|
|
Size: 1000,
|
|
NodeId: nodeIds,
|
|
Enabled: &enable, // Only get enabled nodes
|
|
})
|
|
|
|
if len(nodes) > 0 {
|
|
var serverMapIds = make(map[int64]*node.Server)
|
|
for _, n := range nodes {
|
|
serverMapIds[n.ServerId] = nil
|
|
}
|
|
var serverIds []int64
|
|
for k := range serverMapIds {
|
|
serverIds = append(serverIds, k)
|
|
}
|
|
|
|
servers, err := l.svcCtx.NodeModel.QueryServerList(l.ctx, serverIds)
|
|
if err != nil {
|
|
l.Errorw("[Generate Subscribe]find server details error: %v", logger.Field("error", err.Error()))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "find server details error: %v", err.Error())
|
|
}
|
|
|
|
for _, s := range servers {
|
|
serverMapIds[s.Id] = s
|
|
}
|
|
|
|
for _, n := range nodes {
|
|
server := serverMapIds[n.ServerId]
|
|
if server == nil {
|
|
continue
|
|
}
|
|
userSubscribeNode := &types.UserSubscribeNodeInfo{
|
|
Id: n.Id,
|
|
Name: n.Name,
|
|
Uuid: userSub.UUID,
|
|
Protocol: n.Protocol,
|
|
Port: n.Port,
|
|
Address: n.Address,
|
|
Tags: strings.Split(n.Tags, ","),
|
|
Country: server.Country,
|
|
City: server.City,
|
|
CreatedAt: n.CreatedAt.Unix(),
|
|
}
|
|
userSubscribeNodes = append(userSubscribeNodes, userSubscribeNode)
|
|
}
|
|
}
|
|
|
|
l.Debugf("[Query Subscribe]found servers: %v", len(nodes))
|
|
|
|
if err != nil {
|
|
l.Errorw("[Generate Subscribe]find server details error: %v", logger.Field("error", err.Error()))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "find server details error: %v", err.Error())
|
|
}
|
|
logger.Debugf("[Generate Subscribe]found servers: %v", len(nodes))
|
|
return userSubscribeNodes, nil
|
|
}
|
|
|
|
func (l *QueryUserSubscribeNodeListLogic) isSubscriptionExpired(userSub *user.Subscribe) bool {
|
|
return userSub.ExpireTime.Unix() < time.Now().Unix() && userSub.ExpireTime.Unix() != 0
|
|
}
|
|
|
|
func (l *QueryUserSubscribeNodeListLogic) createExpiredServers() []*types.UserSubscribeNodeInfo {
|
|
return nil
|
|
}
|
|
|
|
func (l *QueryUserSubscribeNodeListLogic) getFirstHostLine() string {
|
|
host := l.svcCtx.Config.Host
|
|
lines := strings.Split(host, "\n")
|
|
if len(lines) > 0 {
|
|
return lines[0]
|
|
}
|
|
return host
|
|
}
|
|
func (l *QueryUserSubscribeNodeListLogic) getUserSubscribe(token string) (*user.Subscribe, error) {
|
|
userSub, err := l.svcCtx.UserModel.FindOneSubscribeByToken(l.ctx, token)
|
|
if err != nil {
|
|
l.Infow("[Generate Subscribe]find subscribe error: %v", logger.Field("error", err.Error()), logger.Field("token", token))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "find subscribe error: %v", err.Error())
|
|
}
|
|
|
|
// Ignore expiration check
|
|
//if userSub.Status > 1 {
|
|
// l.Infow("[Generate Subscribe]subscribe is not available", logger.Field("status", int(userSub.Status)), logger.Field("token", token))
|
|
// return nil, errors.Wrapf(xerr.NewErrCode(xerr.SubscribeNotAvailable), "subscribe is not available")
|
|
//}
|
|
|
|
return userSub, nil
|
|
}
|