* fix(database): correct name entry for SingBox in initialization script * fix(purchase): update gift amount deduction logic and handle zero-amount order status * feat: add type and default fields to rule group requests and update related logic * feat(rule): implement logic to set a default rule group during creation and update * fix(rule): add type and default fields to rule group model and update related logic * feat(proxy): enhance proxy group handling and sorting logic * refactor(proxy): replace hardcoded group names with constants for better maintainability * fix(proxy): update group selection logic to skip empty and default names * feat(proxy): enhance proxy and group handling with new configuration options * feat(surge): add Surge adapter support and enhance subscription URL handling * feat(traffic): implement traffic reset logic for subscription cycles * feat(auth): improve email and mobile config unmarshalling with default values * fix(auth) upbind email not update * fix(order) discount set default 1 * fix(order) discount set default 1 * fix: refactor surfboard proxy handling and enhance configuration template * fix(renewal) discount set default 1 * feat(loon): add Loon configuration template and enhance proxy handling * feat(subscription): update user subscription status based on expiration time * fix(renewal): update subscription retrieval method to use token instead of order ID * feat(order): enhance order processing logic with improved error handling and user subscription management * fix(order): improve code quality and fix critical bugs in order processing logic - Fix inconsistent logging calls across all order logic files - Fix critical gift amount deduction logic bug in renewal process - Fix variable shadowing errors in database transactions - Add comprehensive Go-standard documentation comments - Improve log prefix consistency for better debugging - Remove redundant discount validation code * fix(docker): add build argument for version in Docker image build process * feat(version): add endpoint to retrieve application version information * fix(auth): improve user authentication method logic and update user cache * feat(user): add ordering functionality to user list retrieval * fix(RevenueStatistics) fill list * fix(UserStatistics) fill list * fix(user): implement user cache clearing after auth method operations * fix(auth): enhance OAuth login logic with improved request handling and user registration flow * fix(user): implement sorting for authentication methods based on priority * fix(user): correct ordering clause for user retrieval based on filter * refactor(user): streamline cache management and enhance cache clearing logic * feat(logs) set logs volume in develop * fix(handler): implement browser interception to deny access for specific user agents * fix(resetTraffic) reset daily server * refactor(trojan): remove unused parameter and clean up logging in slice * fix(middleware): add domain length check and improve user-agent handling * fix(middleware): reorder domain processing and enhance user-agent handling * fix(resetTraffic): update subscription reset logic to use expire_time for monthly and yearly checks * fix(scheduler): update reset traffic task schedule to run daily at 00:30 * fix(traffic): enhance traffic reset logic for subscriptions and adjust status checks * fix(activateOrder): update traffic reset logic to include reset day check * feat(marketing): add batch email task management API and logic * feat(application): implement CRUD operations for subscribe applications * feat(types): add user agent limit and list to subscription configuration * feat(application): update subscription application requests to include structured download links * feat(application): add scheme field and download link handling to subscribe application * feat(application): add endpoint to retrieve client information * feat(application): move DownloadLink and SubscribeApplication types to types.api * feat(application): add DownloadLink and SubscribeClient types, update client response structure * feat(application): remove ProxyTemplate field from application API * feat(application): implement adapter for client configuration and add preview template functionality * feat(application): move DownloadLink type to types.api and remove from common.api * feat(application): update PreviewSubscribeTemplate to return structured response * feat(application): remove ProxyTemplate field from application API * feat(application): enhance cache key generation for user list and server data * feat(subscribe): add ClearCache method to manage subscription cache invalidation * feat(payment): add Description field to PaymentMethodDetail response * feat(subscribe): update next reset time calculation to use ExpireTime * feat(purchase): include handling fee in total amount calculation * feat(subscribe): add V2SubscribeHandler and logic for enhanced subscription management * feat(subscribe): add output format configuration to subscription adapter * feat(application): default data --------- Co-authored-by: Chang lue Tsen <tension@ppanel.dev> Co-authored-by: NoWay <Bob455668@hotmail.com>
286 lines
6.9 KiB
Go
286 lines
6.9 KiB
Go
package user
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/perfect-panel/server/pkg/logger"
|
|
)
|
|
|
|
type CacheKeyGenerator interface {
|
|
GetCacheKeys() []string
|
|
}
|
|
|
|
type CacheManager interface {
|
|
ClearCache(ctx context.Context, keys ...string) error
|
|
ClearModelCache(ctx context.Context, models ...CacheKeyGenerator) error
|
|
}
|
|
|
|
type UserCacheManager struct {
|
|
model *defaultUserModel
|
|
}
|
|
|
|
func NewUserCacheManager(model *defaultUserModel) *UserCacheManager {
|
|
return &UserCacheManager{
|
|
model: model,
|
|
}
|
|
}
|
|
|
|
func (c *UserCacheManager) ClearCache(ctx context.Context, keys ...string) error {
|
|
if len(keys) == 0 {
|
|
return nil
|
|
}
|
|
return c.model.CachedConn.DelCacheCtx(ctx, keys...)
|
|
}
|
|
|
|
func (c *UserCacheManager) ClearModelCache(ctx context.Context, models ...CacheKeyGenerator) error {
|
|
var allKeys []string
|
|
for _, model := range models {
|
|
if model != nil {
|
|
allKeys = append(allKeys, model.GetCacheKeys()...)
|
|
}
|
|
}
|
|
return c.ClearCache(ctx, allKeys...)
|
|
}
|
|
|
|
func (u *User) GetCacheKeys() []string {
|
|
if u == nil {
|
|
return []string{}
|
|
}
|
|
keys := []string{
|
|
fmt.Sprintf("%s%d", cacheUserIdPrefix, u.Id),
|
|
}
|
|
|
|
for _, auth := range u.AuthMethods {
|
|
if auth.AuthType == "email" {
|
|
keys = append(keys, fmt.Sprintf("%s%s", cacheUserEmailPrefix, auth.AuthIdentifier))
|
|
break
|
|
}
|
|
}
|
|
return keys
|
|
}
|
|
|
|
func (s *Subscribe) GetCacheKeys() []string {
|
|
if s == nil {
|
|
return []string{}
|
|
}
|
|
keys := []string{}
|
|
|
|
if s.Token != "" {
|
|
keys = append(keys, fmt.Sprintf("%s%s", cacheUserSubscribeTokenPrefix, s.Token))
|
|
}
|
|
if s.UserId != 0 {
|
|
keys = append(keys, fmt.Sprintf("%s%d", cacheUserSubscribeUserPrefix, s.UserId))
|
|
}
|
|
if s.Id != 0 {
|
|
keys = append(keys, fmt.Sprintf("%s%d", cacheUserSubscribeIdPrefix, s.Id))
|
|
}
|
|
return keys
|
|
}
|
|
|
|
func (s *Subscribe) GetExtendedCacheKeys(model *defaultUserModel) []string {
|
|
keys := s.GetCacheKeys()
|
|
|
|
if s.SubscribeId != 0 && model != nil {
|
|
serverKeys := model.getServerRelatedCacheKeys(s.SubscribeId)
|
|
keys = append(keys, serverKeys...)
|
|
}
|
|
|
|
return keys
|
|
}
|
|
|
|
func (d *Device) GetCacheKeys() []string {
|
|
if d == nil {
|
|
return []string{}
|
|
}
|
|
keys := []string{}
|
|
|
|
if d.Id != 0 {
|
|
keys = append(keys, fmt.Sprintf("%s%d", cacheUserDeviceIdPrefix, d.Id))
|
|
}
|
|
if d.Identifier != "" {
|
|
keys = append(keys, fmt.Sprintf("%s%s", cacheUserDeviceNumberPrefix, d.Identifier))
|
|
}
|
|
return keys
|
|
}
|
|
|
|
func (a *AuthMethods) GetCacheKeys() []string {
|
|
if a == nil {
|
|
return []string{}
|
|
}
|
|
keys := []string{}
|
|
|
|
if a.UserId != 0 {
|
|
keys = append(keys, fmt.Sprintf("%s%d", cacheUserIdPrefix, a.UserId))
|
|
}
|
|
if a.AuthType == "email" && a.AuthIdentifier != "" {
|
|
keys = append(keys, fmt.Sprintf("%s%s", cacheUserEmailPrefix, a.AuthIdentifier))
|
|
}
|
|
return keys
|
|
}
|
|
|
|
func (m *defaultUserModel) GetCacheManager() *UserCacheManager {
|
|
return NewUserCacheManager(m)
|
|
}
|
|
|
|
func (m *defaultUserModel) getServerRelatedCacheKeys(subscribeId int64) []string {
|
|
// 这里复用了 model.go 中的逻辑,但简化了实现
|
|
keys := []string{}
|
|
|
|
if subscribeId == 0 {
|
|
return keys
|
|
}
|
|
|
|
// 这里需要从 getSubscribeCacheKey 方法中提取服务器相关的逻辑
|
|
// 为了避免重复查询,我们可以在需要时才获取
|
|
// 或者可以将这个逻辑移到一个统一的地方
|
|
|
|
return keys
|
|
}
|
|
|
|
func (m *defaultUserModel) ClearUserCache(ctx context.Context, users ...*User) error {
|
|
cacheManager := m.GetCacheManager()
|
|
models := make([]CacheKeyGenerator, len(users))
|
|
for i, user := range users {
|
|
models[i] = user
|
|
}
|
|
return cacheManager.ClearModelCache(ctx, models...)
|
|
}
|
|
|
|
func (m *defaultUserModel) ClearSubscribeCacheByModels(ctx context.Context, subscribes ...*Subscribe) error {
|
|
cacheManager := m.GetCacheManager()
|
|
models := make([]CacheKeyGenerator, len(subscribes))
|
|
for i, subscribe := range subscribes {
|
|
models[i] = subscribe
|
|
}
|
|
return cacheManager.ClearModelCache(ctx, models...)
|
|
}
|
|
|
|
func (m *defaultUserModel) ClearDeviceCache(ctx context.Context, devices ...*Device) error {
|
|
cacheManager := m.GetCacheManager()
|
|
models := make([]CacheKeyGenerator, len(devices))
|
|
for i, device := range devices {
|
|
models[i] = device
|
|
}
|
|
return cacheManager.ClearModelCache(ctx, models...)
|
|
}
|
|
|
|
func (m *defaultUserModel) ClearAuthMethodCache(ctx context.Context, authMethods ...*AuthMethods) error {
|
|
cacheManager := m.GetCacheManager()
|
|
models := make([]CacheKeyGenerator, len(authMethods))
|
|
for i, auth := range authMethods {
|
|
models[i] = auth
|
|
}
|
|
return cacheManager.ClearModelCache(ctx, models...)
|
|
}
|
|
|
|
func (m *defaultUserModel) BatchClearRelatedCache(ctx context.Context, user *User) error {
|
|
if user == nil {
|
|
return nil
|
|
}
|
|
|
|
cacheManager := m.GetCacheManager()
|
|
|
|
var allModels []CacheKeyGenerator
|
|
allModels = append(allModels, user)
|
|
|
|
for _, auth := range user.AuthMethods {
|
|
allModels = append(allModels, &auth)
|
|
}
|
|
|
|
for _, device := range user.UserDevices {
|
|
allModels = append(allModels, &device)
|
|
}
|
|
|
|
subscribes, err := m.QueryUserSubscribe(ctx, user.Id)
|
|
if err != nil {
|
|
logger.Errorf("failed to query user subscribes for cache clearing: %v", err)
|
|
} else {
|
|
for _, sub := range subscribes {
|
|
subModel := &Subscribe{
|
|
Id: sub.Id,
|
|
UserId: sub.UserId,
|
|
Token: sub.Token,
|
|
SubscribeId: sub.SubscribeId,
|
|
}
|
|
allModels = append(allModels, subModel)
|
|
}
|
|
}
|
|
|
|
return cacheManager.ClearModelCache(ctx, allModels...)
|
|
}
|
|
|
|
func (m *defaultUserModel) CacheInvalidationHandler(ctx context.Context, operation string, modelType string, model interface{}) error {
|
|
switch operation {
|
|
case "create", "update", "delete":
|
|
switch modelType {
|
|
case "user":
|
|
if user, ok := model.(*User); ok {
|
|
return m.BatchClearRelatedCache(ctx, user)
|
|
}
|
|
case "subscribe":
|
|
if subscribe, ok := model.(*Subscribe); ok {
|
|
return m.ClearSubscribeCacheByModels(ctx, subscribe)
|
|
}
|
|
case "device":
|
|
if device, ok := model.(*Device); ok {
|
|
return m.ClearDeviceCache(ctx, device)
|
|
}
|
|
case "authmethod":
|
|
if authMethod, ok := model.(*AuthMethods); ok {
|
|
return m.ClearAuthMethodCache(ctx, authMethod)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *customUserModel) GetRelatedCacheKeys(ctx context.Context, modelType string, modelId int64) ([]string, error) {
|
|
var keys []string
|
|
|
|
switch modelType {
|
|
case "user":
|
|
user, err := m.FindOne(ctx, modelId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
keys = append(keys, user.GetCacheKeys()...)
|
|
|
|
auths, err := m.FindUserAuthMethods(ctx, modelId)
|
|
if err == nil {
|
|
for _, auth := range auths {
|
|
keys = append(keys, auth.GetCacheKeys()...)
|
|
}
|
|
}
|
|
|
|
subscribes, err := m.QueryUserSubscribe(ctx, modelId)
|
|
if err == nil {
|
|
for _, sub := range subscribes {
|
|
subModel := &Subscribe{
|
|
Id: sub.Id,
|
|
UserId: sub.UserId,
|
|
Token: sub.Token,
|
|
SubscribeId: sub.SubscribeId,
|
|
}
|
|
keys = append(keys, subModel.GetCacheKeys()...)
|
|
}
|
|
}
|
|
|
|
case "subscribe":
|
|
subscribe, err := m.FindOneSubscribe(ctx, modelId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
keys = append(keys, subscribe.GetCacheKeys()...)
|
|
|
|
case "device":
|
|
device, err := m.FindOneDevice(ctx, modelId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
keys = append(keys, device.GetCacheKeys()...)
|
|
}
|
|
|
|
return keys, nil
|
|
}
|