* 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>
138 lines
3.8 KiB
Go
138 lines
3.8 KiB
Go
package adapter
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/perfect-panel/server/internal/model/server"
|
|
"github.com/perfect-panel/server/pkg/logger"
|
|
"github.com/perfect-panel/server/pkg/random"
|
|
)
|
|
|
|
type Adapter struct {
|
|
SiteName string // 站点名称
|
|
Servers []*server.Server // 服务器列表
|
|
UserInfo User // 用户信息
|
|
ClientTemplate string // 客户端配置模板
|
|
OutputFormat string // 输出格式,默认是 base64
|
|
SubscribeName string // 订阅名称
|
|
}
|
|
|
|
type Option func(*Adapter)
|
|
|
|
// WithServers 设置服务器列表
|
|
func WithServers(servers []*server.Server) Option {
|
|
return func(opts *Adapter) {
|
|
opts.Servers = servers
|
|
}
|
|
}
|
|
|
|
// WithUserInfo 设置用户信息
|
|
func WithUserInfo(user User) Option {
|
|
return func(opts *Adapter) {
|
|
opts.UserInfo = user
|
|
}
|
|
}
|
|
|
|
// WithOutputFormat 设置输出格式
|
|
func WithOutputFormat(format string) Option {
|
|
return func(opts *Adapter) {
|
|
opts.OutputFormat = format
|
|
}
|
|
}
|
|
|
|
// WithSiteName 设置站点名称
|
|
func WithSiteName(name string) Option {
|
|
return func(opts *Adapter) {
|
|
opts.SiteName = name
|
|
}
|
|
}
|
|
|
|
// WithSubscribeName 设置订阅名称
|
|
func WithSubscribeName(name string) Option {
|
|
return func(opts *Adapter) {
|
|
opts.SubscribeName = name
|
|
}
|
|
}
|
|
|
|
func NewAdapter(tpl string, opts ...Option) *Adapter {
|
|
adapter := &Adapter{
|
|
Servers: []*server.Server{},
|
|
UserInfo: User{},
|
|
ClientTemplate: tpl,
|
|
OutputFormat: "base64", // 默认输出格式
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
opt(adapter)
|
|
}
|
|
|
|
return adapter
|
|
}
|
|
|
|
func (adapter *Adapter) Client() (*Client, error) {
|
|
client := &Client{
|
|
SiteName: adapter.SiteName,
|
|
SubscribeName: adapter.SubscribeName,
|
|
ClientTemplate: adapter.ClientTemplate,
|
|
OutputFormat: adapter.OutputFormat,
|
|
Proxies: []Proxy{},
|
|
UserInfo: adapter.UserInfo,
|
|
}
|
|
|
|
proxies, err := adapter.Proxies(adapter.Servers)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
client.Proxies = proxies
|
|
return client, nil
|
|
}
|
|
|
|
func (adapter *Adapter) Proxies(servers []*server.Server) ([]Proxy, error) {
|
|
var proxies []Proxy
|
|
for _, srv := range servers {
|
|
switch srv.RelayMode {
|
|
case server.RelayModeAll:
|
|
var relays []server.NodeRelay
|
|
if err := json.Unmarshal([]byte(srv.RelayNode), &relays); err != nil {
|
|
logger.Errorw("Unmarshal RelayNode", logger.Field("error", err.Error()), logger.Field("node", srv.Name), logger.Field("relayNode", srv.RelayNode))
|
|
continue
|
|
}
|
|
for _, relay := range relays {
|
|
proxy, err := adapterProxy(*srv, relay.Host, uint64(relay.Port))
|
|
if err != nil {
|
|
logger.Errorw("Adapter Proxy", logger.Field("error", err.Error()), logger.Field("node", srv.Name), logger.Field("relayNode", relay))
|
|
continue
|
|
}
|
|
proxies = append(proxies, proxy)
|
|
}
|
|
|
|
case server.RelayModeRandom:
|
|
var relays []server.NodeRelay
|
|
if err := json.Unmarshal([]byte(srv.RelayNode), &relays); err != nil {
|
|
logger.Errorw("Unmarshal RelayNode", logger.Field("error", err.Error()), logger.Field("node", srv.Name), logger.Field("relayNode", srv.RelayNode))
|
|
continue
|
|
}
|
|
randNum := random.RandomInRange(0, len(relays)-1)
|
|
relay := relays[randNum]
|
|
proxy, err := adapterProxy(*srv, relay.Host, uint64(relay.Port))
|
|
if err != nil {
|
|
logger.Errorw("Adapter Proxy", logger.Field("error", err.Error()), logger.Field("node", srv.Name), logger.Field("relayNode", relay))
|
|
continue
|
|
}
|
|
proxies = append(proxies, proxy)
|
|
|
|
case server.RelayModeNone:
|
|
proxy, err := adapterProxy(*srv, srv.ServerAddr, 0)
|
|
if err != nil {
|
|
logger.Errorw("Adapter Proxy", logger.Field("error", err.Error()), logger.Field("node", srv.Name), logger.Field("serverAddr", srv.ServerAddr))
|
|
continue
|
|
}
|
|
proxies = append(proxies, proxy)
|
|
default:
|
|
logger.Errorw("Unknown RelayMode", logger.Field("node", srv.Name), logger.Field("relayMode", srv.RelayMode))
|
|
}
|
|
|
|
}
|
|
return proxies, nil
|
|
}
|