各种配置项修复,优化到后台管理端配置
All checks were successful
Build docker and publish / build (20.15.1) (push) Successful in 7m59s

This commit is contained in:
shanshanzhong 2026-03-04 20:03:03 -08:00
parent 2215df8c0b
commit 3594097d47
2 changed files with 214 additions and 4 deletions

View File

@ -3,7 +3,9 @@ package user
import ( import (
"context" "context"
"encoding/json" "encoding/json"
"fmt"
"sort" "sort"
"strings"
"github.com/perfect-panel/server/pkg/constant" "github.com/perfect-panel/server/pkg/constant"
"github.com/perfect-panel/server/pkg/kutt" "github.com/perfect-panel/server/pkg/kutt"
@ -16,6 +18,7 @@ import (
"github.com/perfect-panel/server/pkg/logger" "github.com/perfect-panel/server/pkg/logger"
"github.com/perfect-panel/server/pkg/phone" "github.com/perfect-panel/server/pkg/phone"
"github.com/perfect-panel/server/pkg/tool" "github.com/perfect-panel/server/pkg/tool"
"gorm.io/gorm"
) )
type QueryUserInfoLogic struct { type QueryUserInfoLogic struct {
@ -41,6 +44,7 @@ func (l *QueryUserInfoLogic) QueryUserInfo() (resp *types.User, err error) {
return nil, errors.Wrapf(xerr.NewErrCode(xerr.InvalidAccess), "Invalid Access") return nil, errors.Wrapf(xerr.NewErrCode(xerr.InvalidAccess), "Invalid Access")
} }
tool.DeepCopy(resp, u) tool.DeepCopy(resp, u)
ownerEmailMethod := l.fillFamilyContext(resp, u.Id)
var userMethods []types.UserAuthMethod var userMethods []types.UserAuthMethod
for _, method := range resp.AuthMethods { for _, method := range resp.AuthMethods {
@ -56,11 +60,9 @@ func (l *QueryUserInfoLogic) QueryUserInfo() (resp *types.User, err error) {
} }
userMethods = append(userMethods, item) userMethods = append(userMethods, item)
} }
userMethods = appendFamilyOwnerEmailIfNeeded(userMethods, resp.FamilyJoined, ownerEmailMethod)
// 按照指定顺序排序email第一位mobile第二位其他按原顺序 sortUserAuthMethodsByPriority(userMethods)
sort.Slice(userMethods, func(i, j int) bool {
return getAuthTypePriority(userMethods[i].AuthType) < getAuthTypePriority(userMethods[j].AuthType)
})
resp.AuthMethods = userMethods resp.AuthMethods = userMethods
@ -75,6 +77,109 @@ func (l *QueryUserInfoLogic) QueryUserInfo() (resp *types.User, err error) {
return resp, nil return resp, nil
} }
func (l *QueryUserInfoLogic) fillFamilyContext(resp *types.User, userId int64) *user.AuthMethods {
type familyRelation struct {
FamilyId int64
Role uint8
FamilyStatus uint8
OwnerUserId int64
MaxMembers int64
}
var relation familyRelation
relationErr := l.svcCtx.DB.WithContext(l.ctx).
Table("user_family_member").
Select("user_family_member.family_id, user_family_member.role, user_family.status as family_status, user_family.owner_user_id, user_family.max_members").
Joins("JOIN user_family ON user_family.id = user_family_member.family_id AND user_family.deleted_at IS NULL").
Where("user_family_member.user_id = ? AND user_family_member.deleted_at IS NULL AND user_family_member.status = ?", userId, user.FamilyMemberActive).
First(&relation).Error
if relationErr != nil {
if !errors.Is(relationErr, gorm.ErrRecordNotFound) {
l.Errorw("query family relation failed", logger.Field("user_id", userId), logger.Field("error", relationErr.Error()))
}
return nil
}
resp.FamilyJoined = true
resp.FamilyId = relation.FamilyId
resp.FamilyRole = relation.Role
resp.FamilyRoleName = getFamilyRoleName(relation.Role)
resp.FamilyOwnerUserId = relation.OwnerUserId
resp.FamilyStatus = getFamilyStatusName(relation.FamilyStatus)
resp.FamilyMaxMembers = relation.MaxMembers
var activeMemberCount int64
countErr := l.svcCtx.DB.WithContext(l.ctx).
Table("user_family_member").
Where("family_id = ? AND status = ? AND deleted_at IS NULL", relation.FamilyId, user.FamilyMemberActive).
Count(&activeMemberCount).Error
if countErr != nil {
l.Errorw("count family members failed", logger.Field("family_id", relation.FamilyId), logger.Field("error", countErr.Error()))
} else {
resp.FamilyMemberCount = activeMemberCount
}
ownerEmailMethod, ownerEmailErr := l.svcCtx.UserModel.FindUserAuthMethodByUserId(l.ctx, "email", relation.OwnerUserId)
if ownerEmailErr != nil {
if !errors.Is(ownerEmailErr, gorm.ErrRecordNotFound) {
l.Errorw("query family owner email failed", logger.Field("owner_user_id", relation.OwnerUserId), logger.Field("error", ownerEmailErr.Error()))
}
return nil
}
return ownerEmailMethod
}
func appendFamilyOwnerEmailIfNeeded(methods []types.UserAuthMethod, familyJoined bool, ownerEmailMethod *user.AuthMethods) []types.UserAuthMethod {
if !familyJoined || ownerEmailMethod == nil {
return methods
}
ownerEmail := strings.TrimSpace(ownerEmailMethod.AuthIdentifier)
if ownerEmail == "" {
return methods
}
if hasEmailAuthMethod(methods) {
return methods
}
return append(methods, types.UserAuthMethod{
AuthType: "email",
AuthIdentifier: ownerEmail,
Verified: ownerEmailMethod.Verified,
})
}
func hasEmailAuthMethod(methods []types.UserAuthMethod) bool {
for _, method := range methods {
if strings.EqualFold(strings.TrimSpace(method.AuthType), "email") && strings.TrimSpace(method.AuthIdentifier) != "" {
return true
}
}
return false
}
func sortUserAuthMethodsByPriority(methods []types.UserAuthMethod) {
sort.SliceStable(methods, func(i, j int) bool {
return getAuthTypePriority(methods[i].AuthType) < getAuthTypePriority(methods[j].AuthType)
})
}
func getFamilyRoleName(role uint8) string {
switch role {
case user.FamilyRoleOwner:
return "owner"
case user.FamilyRoleMember:
return "member"
default:
return fmt.Sprintf("role_%d", role)
}
}
func getFamilyStatusName(status uint8) string {
if status == user.FamilyStatusActive {
return "active"
}
return "disabled"
}
// customData 用于解析 SiteConfig.CustomData JSON 字段 // customData 用于解析 SiteConfig.CustomData JSON 字段
// 包含从自定义数据中提取所需的配置项 // 包含从自定义数据中提取所需的配置项
type customData struct { type customData struct {

View File

@ -0,0 +1,105 @@
package user
import (
"testing"
modelUser "github.com/perfect-panel/server/internal/model/user"
"github.com/perfect-panel/server/internal/types"
"github.com/stretchr/testify/require"
)
func TestAppendFamilyOwnerEmailIfNeeded(t *testing.T) {
testCases := []struct {
name string
methods []types.UserAuthMethod
familyJoined bool
ownerEmailMethod *modelUser.AuthMethods
wantMethodCount int
wantEmailCount int
wantFirstAuthType string
wantFirstAuthValue string
}{
{
name: "inject owner email when member has no email",
methods: []types.UserAuthMethod{
{AuthType: "device", AuthIdentifier: "dev-1", Verified: true},
},
familyJoined: true,
ownerEmailMethod: &modelUser.AuthMethods{AuthType: "email", AuthIdentifier: "owner@example.com", Verified: true},
wantMethodCount: 2,
wantEmailCount: 1,
wantFirstAuthType: "email",
wantFirstAuthValue: "owner@example.com",
},
{
name: "do not inject when member already has email",
methods: []types.UserAuthMethod{
{AuthType: "email", AuthIdentifier: "member@example.com", Verified: true},
{AuthType: "device", AuthIdentifier: "dev-1", Verified: true},
},
familyJoined: true,
ownerEmailMethod: &modelUser.AuthMethods{AuthType: "email", AuthIdentifier: "owner@example.com", Verified: true},
wantMethodCount: 2,
wantEmailCount: 1,
wantFirstAuthType: "email",
wantFirstAuthValue: "member@example.com",
},
{
name: "do not inject when owner has no email",
methods: []types.UserAuthMethod{
{AuthType: "device", AuthIdentifier: "dev-1", Verified: true},
},
familyJoined: true,
ownerEmailMethod: &modelUser.AuthMethods{AuthType: "email", AuthIdentifier: "", Verified: true},
wantMethodCount: 1,
wantEmailCount: 0,
wantFirstAuthType: "device",
},
{
name: "do not inject for non active family relationship",
methods: []types.UserAuthMethod{
{AuthType: "device", AuthIdentifier: "dev-1", Verified: true},
},
familyJoined: false,
ownerEmailMethod: &modelUser.AuthMethods{AuthType: "email", AuthIdentifier: "owner@example.com", Verified: true},
wantMethodCount: 1,
wantEmailCount: 0,
wantFirstAuthType: "device",
},
{
name: "sort keeps injected email at first position",
methods: []types.UserAuthMethod{
{AuthType: "mobile", AuthIdentifier: "+1234567890", Verified: true},
{AuthType: "device", AuthIdentifier: "dev-1", Verified: true},
},
familyJoined: true,
ownerEmailMethod: &modelUser.AuthMethods{AuthType: "email", AuthIdentifier: "owner@example.com", Verified: true},
wantMethodCount: 3,
wantEmailCount: 1,
wantFirstAuthType: "email",
wantFirstAuthValue: "owner@example.com",
},
}
for _, testCase := range testCases {
t.Run(testCase.name, func(t *testing.T) {
finalMethods := appendFamilyOwnerEmailIfNeeded(testCase.methods, testCase.familyJoined, testCase.ownerEmailMethod)
sortUserAuthMethodsByPriority(finalMethods)
require.Len(t, finalMethods, testCase.wantMethodCount)
emailCount := 0
for _, method := range finalMethods {
if method.AuthType == "email" {
emailCount++
}
}
require.Equal(t, testCase.wantEmailCount, emailCount)
require.Equal(t, testCase.wantFirstAuthType, finalMethods[0].AuthType)
if testCase.wantFirstAuthValue != "" {
require.Equal(t, testCase.wantFirstAuthValue, finalMethods[0].AuthIdentifier)
}
})
}
}