* 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>
154 lines
4.9 KiB
Go
154 lines
4.9 KiB
Go
package adapter
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
var tpl = `
|
|
{{- range $n := .Proxies }}
|
|
{{- $dn := urlquery (default "node" $n.Name) -}}
|
|
{{- $sni := default $n.Host $n.SNI -}}
|
|
|
|
{{- if eq $n.Type "shadowsocks" -}}
|
|
{{- $userinfo := b64enc (print $n.Method ":" $.UserInfo.Password) -}}
|
|
{{- printf "ss://%s@%s:%v#%s" $userinfo $n.Host $n.Port $dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if eq $n.Type "trojan" -}}
|
|
{{- $qs := "security=tls" -}}
|
|
{{- if $sni }}{{ $qs = printf "%s&sni=%s" $qs (urlquery $sni) }}{{ end -}}
|
|
{{- if $n.AllowInsecure }}{{ $qs = printf "%s&allowInsecure=%v" $qs $n.AllowInsecure }}{{ end -}}
|
|
{{- if $n.Fingerprint }}{{ $qs = printf "%s&fp=%s" $qs (urlquery $n.Fingerprint) }}{{ end -}}
|
|
{{- printf "trojan://%s@%s:%v?%s#%s" $.UserInfo.Password $n.Host $n.Port $qs $dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if eq $n.Type "vless" -}}
|
|
{{- $qs := "encryption=none" -}}
|
|
{{- if $n.RealityPublicKey -}}
|
|
{{- $qs = printf "%s&security=reality" $qs -}}
|
|
{{- $qs = printf "%s&pbk=%s" $qs (urlquery $n.RealityPublicKey) -}}
|
|
{{- if $n.RealityShortId }}{{ $qs = printf "%s&sid=%s" $qs (urlquery $n.RealityShortId) }}{{ end -}}
|
|
{{- else -}}
|
|
{{- if or $n.SNI $n.Fingerprint $n.AllowInsecure }}
|
|
{{- $qs = printf "%s&security=tls" $qs -}}
|
|
{{- end -}}
|
|
{{- end -}}
|
|
{{- if $n.SNI }}{{ $qs = printf "%s&sni=%s" $qs (urlquery $n.SNI) }}{{ end -}}
|
|
{{- if $n.AllowInsecure }}{{ $qs = printf "%s&allowInsecure=%v" $qs $n.AllowInsecure }}{{ end -}}
|
|
{{- if $n.Fingerprint }}{{ $qs = printf "%s&fp=%s" $qs (urlquery $n.Fingerprint) }}{{ end -}}
|
|
{{- if $n.Network }}{{ $qs = printf "%s&type=%s" $qs $n.Network }}{{ end -}}
|
|
{{- if $n.Path }}{{ $qs = printf "%s&path=%s" $qs (urlquery $n.Path) }}{{ end -}}
|
|
{{- if $n.ServiceName }}{{ $qs = printf "%s&serviceName=%s" $qs (urlquery $n.ServiceName) }}{{ end -}}
|
|
{{- if $n.Flow }}{{ $qs = printf "%s&flow=%s" $qs (urlquery $n.Flow) }}{{ end -}}
|
|
{{- printf "vless://%s@%s:%v?%s#%s" $n.ServerKey $n.Host $n.Port $qs $dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if eq $n.Type "vmess" -}}
|
|
{{- $obj := dict
|
|
"v" "2"
|
|
"ps" $n.Name
|
|
"add" $n.Host
|
|
"port" $n.Port
|
|
"id" $n.ServerKey
|
|
"aid" 0
|
|
"net" (or $n.Network "tcp")
|
|
"type" "none"
|
|
"path" (or $n.Path "")
|
|
"host" $n.Host
|
|
-}}
|
|
{{- if or $n.SNI $n.Fingerprint $n.AllowInsecure }}{{ set $obj "tls" "tls" }}{{ end -}}
|
|
{{- if $n.SNI }}{{ set $obj "sni" $n.SNI }}{{ end -}}
|
|
{{- if $n.Fingerprint }}{{ set $obj "fp" $n.Fingerprint }}{{ end -}}
|
|
{{- printf "vmess://%s" (b64enc (toJson $obj)) -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if or (eq $n.Type "hysteria2") (eq $n.Type "hy2") -}}
|
|
{{- $qs := "" -}}
|
|
{{- if $n.SNI }}{{ $qs = printf "sni=%s" (urlquery $n.SNI) }}{{ end -}}
|
|
{{- if $n.AllowInsecure }}{{ $qs = printf "%s&insecure=%v" $qs $n.AllowInsecure }}{{ end -}}
|
|
{{- if $n.ObfsPassword }}{{ $qs = printf "%s&obfs-password=%s" $qs (urlquery $n.ObfsPassword) }}{{ end -}}
|
|
{{- printf "hy2://%s@%s:%v%s#%s"
|
|
$.UserInfo.Password
|
|
$n.Host
|
|
$n.Port
|
|
(ternary (gt (len $qs) 0) (print "?" $qs) "")
|
|
$dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if eq $n.Type "tuic" -}}
|
|
{{- $qs := "" -}}
|
|
{{- if $n.SNI }}{{ $qs = printf "sni=%s" (urlquery $n.SNI) }}{{ end -}}
|
|
{{- if $n.AllowInsecure }}{{ $qs = printf "%s&insecure=%v" $qs $n.AllowInsecure }}{{ end -}}
|
|
{{- printf "tuic://%s:%s@%s:%v%s#%s"
|
|
$n.ServerKey
|
|
$.UserInfo.Password
|
|
$n.Host
|
|
$n.Port
|
|
(ternary (gt (len $qs) 0) (print "?" $qs) "")
|
|
$dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- if eq $n.Type "anytls" -}}
|
|
{{- $qs := "" -}}
|
|
{{- if $n.SNI }}{{ $qs = printf "sni=%s" (urlquery $n.SNI) }}{{ end -}}
|
|
{{- printf "anytls://%s@%s:%v%s#%s"
|
|
$.UserInfo.Password
|
|
$n.Host
|
|
$n.Port
|
|
(ternary (gt (len $qs) 0) (print "?" $qs) "")
|
|
$dn -}}
|
|
{{- "\n" -}}
|
|
{{- end -}}
|
|
|
|
{{- end }}
|
|
`
|
|
|
|
func TestClient_Build(t *testing.T) {
|
|
client := &Client{
|
|
SiteName: "TestSite",
|
|
SubscribeName: "TestSubscribe",
|
|
ClientTemplate: tpl,
|
|
Proxies: []Proxy{
|
|
{
|
|
Name: "TestShadowSocks",
|
|
Type: "shadowsocks",
|
|
Host: "127.0.0.1",
|
|
Port: 1234,
|
|
Method: "aes-256-gcm",
|
|
},
|
|
{
|
|
Name: "TestTrojan",
|
|
Type: "trojan",
|
|
Host: "example.com",
|
|
Port: 443,
|
|
AllowInsecure: true,
|
|
Security: "tls",
|
|
Transport: "tcp",
|
|
SNI: "v1-dy.ixigua.com",
|
|
},
|
|
},
|
|
UserInfo: User{
|
|
Password: "testpassword",
|
|
ExpiredAt: time.Now().Add(24 * time.Hour),
|
|
Download: 1000000,
|
|
Upload: 500000,
|
|
Traffic: 1500000,
|
|
SubscribeURL: "https://example.com/subscribe",
|
|
},
|
|
}
|
|
buf, err := client.Build()
|
|
if err != nil {
|
|
t.Fatalf("Failed to build client: %v", err)
|
|
}
|
|
|
|
t.Logf("[测试] 输出: %s", buf)
|
|
|
|
}
|