diff --git a/backend/api/api.go b/backend/api/api.go
index 0457b04..9e77827 100644
--- a/backend/api/api.go
+++ b/backend/api/api.go
@@ -3,7 +3,6 @@ package api
import (
"s-ui/logger"
"s-ui/service"
- "s-ui/singbox"
"s-ui/util"
"strconv"
"strings"
@@ -21,7 +20,6 @@ type APIHandler struct {
service.PanelService
service.StatsService
service.ServerService
- singbox.Controller
}
func NewAPIHandler(g *gin.RouterGroup) {
@@ -92,7 +90,7 @@ func (a *APIHandler) postHandler(c *gin.Context) {
err = a.PanelService.RestartPanel(3)
jsonMsg(c, "restartApp", err)
case "restartSb":
- err = a.Controller.Restart()
+ err = a.ConfigService.RestartCore()
jsonMsg(c, "restartSb", err)
case "linkConvert":
link := c.Request.FormValue("link")
@@ -156,10 +154,9 @@ func (a *APIHandler) getHandler(c *gin.Context) {
onlines, err := a.StatsService.GetOnlines()
jsonObj(c, onlines, err)
case "logs":
- service := c.Query("s")
count := c.Query("c")
level := c.Query("l")
- logs := a.ServerService.GetLogs(service, count, level)
+ logs := a.ServerService.GetLogs(count, level)
jsonObj(c, logs, nil)
case "changes":
actor := c.Query("a")
@@ -188,7 +185,7 @@ func (a *APIHandler) loadData(c *gin.Context) (interface{}, error) {
sysInfo := a.ServerService.GetSingboxInfo()
if sysInfo["running"] == false {
- logs := a.ServerService.GetLogs("sing-box", "1", "debug")
+ logs := a.ServerService.GetLogs("1", "debug")
if len(logs) > 0 {
data["lastLog"] = logs[0]
}
diff --git a/backend/app/app.go b/backend/app/app.go
index 1b4c767..f68b110 100644
--- a/backend/app/app.go
+++ b/backend/app/app.go
@@ -3,6 +3,7 @@ package app
import (
"log"
"s-ui/config"
+ "s-ui/core"
"s-ui/cronjob"
"s-ui/database"
"s-ui/logger"
@@ -18,6 +19,8 @@ type APP struct {
webServer *web.Server
subServer *sub.Server
cronJob *cronjob.CronJob
+ logger *logging.Logger
+ core *core.Core
}
func NewApp() *APP {
@@ -34,11 +37,13 @@ func (a *APP) Init() error {
return err
}
- a.cronJob = cronjob.NewCronJob()
+ a.core = core.NewCore()
+
+ a.cronJob = cronjob.NewCronJob(a.core)
a.webServer = web.NewServer()
a.subServer = sub.NewServer()
- configService := service.NewConfigService()
+ configService := service.NewConfigService(a.core)
err = configService.InitConfig()
if err != nil {
return err
@@ -72,6 +77,11 @@ func (a *APP) Start() error {
return err
}
+ err = a.core.Start()
+ if err != nil {
+ logger.Error(err)
+ }
+
return nil
}
@@ -106,3 +116,7 @@ func (a *APP) RestartApp() {
a.Stop()
a.Start()
}
+
+func (a *APP) GetCore() *core.Core {
+ return a.core
+}
diff --git a/backend/core/box.go b/backend/core/box.go
new file mode 100644
index 0000000..3287e73
--- /dev/null
+++ b/backend/core/box.go
@@ -0,0 +1,412 @@
+package core
+
+import (
+ "context"
+ "fmt"
+ "io"
+ "os"
+ "s-ui/util/common"
+ "time"
+
+ "github.com/sagernet/sing-box/adapter"
+ "github.com/sagernet/sing-box/adapter/endpoint"
+ "github.com/sagernet/sing-box/adapter/inbound"
+ "github.com/sagernet/sing-box/adapter/outbound"
+ "github.com/sagernet/sing-box/common/dialer"
+ "github.com/sagernet/sing-box/common/taskmonitor"
+ C "github.com/sagernet/sing-box/constant"
+ "github.com/sagernet/sing-box/experimental/cachefile"
+ "github.com/sagernet/sing-box/experimental/libbox/platform"
+ "github.com/sagernet/sing-box/log"
+ "github.com/sagernet/sing-box/option"
+ "github.com/sagernet/sing-box/protocol/direct"
+ "github.com/sagernet/sing-box/route"
+ sbCommon "github.com/sagernet/sing/common"
+ F "github.com/sagernet/sing/common/format"
+ "github.com/sagernet/sing/common/ntp"
+ "github.com/sagernet/sing/service"
+ "github.com/sagernet/sing/service/pause"
+)
+
+var _ adapter.Service = (*Box)(nil)
+
+type Box struct {
+ createdAt time.Time
+ logFactory log.Factory
+ logger log.ContextLogger
+ network *route.NetworkManager
+ endpoint *endpoint.Manager
+ inbound *inbound.Manager
+ outbound *outbound.Manager
+ connection *route.ConnectionManager
+ router *route.Router
+ services []adapter.LifecycleService
+ connTracker *ConnTracker
+ done chan struct{}
+}
+
+type Options struct {
+ option.Options
+ Context context.Context
+}
+
+func Context(
+ ctx context.Context,
+ inboundRegistry adapter.InboundRegistry,
+ outboundRegistry adapter.OutboundRegistry,
+ endpointRegistry adapter.EndpointRegistry,
+) context.Context {
+ if service.FromContext[option.InboundOptionsRegistry](ctx) == nil ||
+ service.FromContext[adapter.InboundRegistry](ctx) == nil {
+ ctx = service.ContextWith[option.InboundOptionsRegistry](ctx, inboundRegistry)
+ ctx = service.ContextWith[adapter.InboundRegistry](ctx, inboundRegistry)
+ }
+ if service.FromContext[option.OutboundOptionsRegistry](ctx) == nil ||
+ service.FromContext[adapter.OutboundRegistry](ctx) == nil {
+ ctx = service.ContextWith[option.OutboundOptionsRegistry](ctx, outboundRegistry)
+ ctx = service.ContextWith[adapter.OutboundRegistry](ctx, outboundRegistry)
+ }
+ if service.FromContext[option.EndpointOptionsRegistry](ctx) == nil ||
+ service.FromContext[adapter.EndpointRegistry](ctx) == nil {
+ ctx = service.ContextWith[option.EndpointOptionsRegistry](ctx, endpointRegistry)
+ ctx = service.ContextWith[adapter.EndpointRegistry](ctx, endpointRegistry)
+ }
+ return ctx
+}
+
+func NewBox(options Options) (*Box, error) {
+ var err error
+ createdAt := time.Now()
+ ctx := options.Context
+ if ctx == nil {
+ ctx = context.Background()
+ }
+ ctx = service.ContextWithDefaultRegistry(ctx)
+
+ endpointRegistry := service.FromContext[adapter.EndpointRegistry](ctx)
+ inboundRegistry := service.FromContext[adapter.InboundRegistry](ctx)
+ outboundRegistry := service.FromContext[adapter.OutboundRegistry](ctx)
+
+ if endpointRegistry == nil {
+ return nil, common.NewError("missing endpoint registry in context")
+ }
+ if inboundRegistry == nil {
+ return nil, common.NewError("missing inbound registry in context")
+ }
+ if outboundRegistry == nil {
+ return nil, common.NewError("missing outbound registry in context")
+ }
+
+ ctx = pause.WithDefaultManager(ctx)
+ experimentalOptions := sbCommon.PtrValueOrDefault(options.Experimental)
+ var needCacheFile bool
+ if experimentalOptions.CacheFile != nil && experimentalOptions.CacheFile.Enabled {
+ needCacheFile = true
+ }
+ platformInterface := service.FromContext[platform.Interface](ctx)
+ var defaultLogWriter io.Writer
+ if platformInterface != nil {
+ defaultLogWriter = io.Discard
+ }
+ var logFactory log.Factory
+ if factory == nil {
+ logFactory, err = NewFactory(log.Options{
+ Context: ctx,
+ Options: sbCommon.PtrValueOrDefault(options.Log),
+ DefaultWriter: defaultLogWriter,
+ BaseTime: createdAt,
+ })
+ if err != nil {
+ return nil, common.NewError("create log factory", err)
+ }
+ factory = logFactory
+ } else {
+ logFactory = factory
+ }
+
+ routeOptions := sbCommon.PtrValueOrDefault(options.Route)
+ endpointManager := endpoint.NewManager(logFactory.NewLogger("endpoint"), endpointRegistry)
+ inboundManager := inbound.NewManager(logFactory.NewLogger("inbound"), inboundRegistry, endpointManager)
+ outboundManager := outbound.NewManager(logFactory.NewLogger("outbound"), outboundRegistry, endpointManager, routeOptions.Final)
+ service.MustRegister[adapter.EndpointManager](ctx, endpointManager)
+ service.MustRegister[adapter.InboundManager](ctx, inboundManager)
+ service.MustRegister[adapter.OutboundManager](ctx, outboundManager)
+
+ networkManager, err := route.NewNetworkManager(ctx, logFactory.NewLogger("network"), routeOptions)
+ if err != nil {
+ return nil, common.NewError("initialize network manager", err)
+ }
+ service.MustRegister[adapter.NetworkManager](ctx, networkManager)
+ connectionManager := route.NewConnectionManager(logFactory.NewLogger("connection"))
+ service.MustRegister[adapter.ConnectionManager](ctx, connectionManager)
+ router, err := route.NewRouter(ctx, logFactory, routeOptions, sbCommon.PtrValueOrDefault(options.DNS))
+ if err != nil {
+ return nil, common.NewError("initialize router", err)
+ }
+ for i, endpointOptions := range options.Endpoints {
+ var tag string
+ if endpointOptions.Tag != "" {
+ tag = endpointOptions.Tag
+ } else {
+ tag = F.ToString(i)
+ }
+ err = endpointManager.Create(ctx,
+ router,
+ logFactory.NewLogger(F.ToString("endpoint/", endpointOptions.Type, "[", tag, "]")),
+ tag,
+ endpointOptions.Type,
+ endpointOptions.Options,
+ )
+ if err != nil {
+ return nil, common.NewError("initialize endpoint["+string(i)+"] "+tag, err)
+ }
+ }
+ for i, inboundOptions := range options.Inbounds {
+ var tag string
+ if inboundOptions.Tag != "" {
+ tag = inboundOptions.Tag
+ } else {
+ tag = F.ToString(i)
+ }
+ err = inboundManager.Create(ctx,
+ router,
+ logFactory.NewLogger(F.ToString("inbound/", inboundOptions.Type, "[", tag, "]")),
+ tag,
+ inboundOptions.Type,
+ inboundOptions.Options,
+ )
+ if err != nil {
+ return nil, common.NewError("initialize inbound[", i, "] ", tag, err)
+ }
+ }
+ for i, outboundOptions := range options.Outbounds {
+ var tag string
+ if outboundOptions.Tag != "" {
+ tag = outboundOptions.Tag
+ } else {
+ tag = F.ToString(i)
+ }
+ outboundCtx := ctx
+ if tag != "" {
+ // TODO: remove this
+ outboundCtx = adapter.WithContext(outboundCtx, &adapter.InboundContext{
+ Outbound: tag,
+ })
+ }
+ err = outboundManager.Create(
+ outboundCtx,
+ router,
+ logFactory.NewLogger(F.ToString("outbound/", outboundOptions.Type, "[", tag, "]")),
+ tag,
+ outboundOptions.Type,
+ outboundOptions.Options,
+ )
+ if err != nil {
+ return nil, common.NewError("initialize outbound["+string(i)+"] "+tag, err)
+ }
+ }
+ outboundManager.Initialize(sbCommon.Must1(
+ direct.NewOutbound(
+ ctx,
+ router,
+ logFactory.NewLogger("outbound/direct"),
+ "direct",
+ option.DirectOutboundOptions{},
+ ),
+ ))
+ if platformInterface != nil {
+ err = platformInterface.Initialize(networkManager)
+ if err != nil {
+ return nil, common.NewError("initialize platform interface", err)
+ }
+ }
+ if connTracker == nil {
+ connTracker = NewConnTracker()
+ }
+ router.SetTracker(connTracker)
+
+ var services []adapter.LifecycleService
+
+ if needCacheFile {
+ cacheFile := cachefile.New(ctx, sbCommon.PtrValueOrDefault(experimentalOptions.CacheFile))
+ service.MustRegister[adapter.CacheFile](ctx, cacheFile)
+ services = append(services, cacheFile)
+ }
+ ntpOptions := sbCommon.PtrValueOrDefault(options.NTP)
+ if ntpOptions.Enabled {
+ ntpDialer, err := dialer.New(ctx, ntpOptions.DialerOptions)
+ if err != nil {
+ return nil, common.NewError(err, "create NTP service")
+ }
+ timeService := ntp.NewService(ntp.Options{
+ Context: ctx,
+ Dialer: ntpDialer,
+ Logger: logFactory.NewLogger("ntp"),
+ Server: ntpOptions.ServerOptions.Build(),
+ Interval: time.Duration(ntpOptions.Interval),
+ WriteToSystem: ntpOptions.WriteToSystem,
+ })
+ service.MustRegister[ntp.TimeService](ctx, timeService)
+ services = append(services, adapter.NewLifecycleService(timeService, "ntp service"))
+ }
+ return &Box{
+ network: networkManager,
+ endpoint: endpointManager,
+ inbound: inboundManager,
+ outbound: outboundManager,
+ connection: connectionManager,
+ router: router,
+ createdAt: createdAt,
+ logFactory: logFactory,
+ logger: logFactory.Logger(),
+ services: services,
+ connTracker: connTracker,
+ done: make(chan struct{}),
+ }, nil
+}
+
+func (s *Box) PreStart() error {
+ err := s.preStart()
+ if err != nil {
+ // TODO: remove catch error
+ defer func() {
+ v := recover()
+ if v != nil {
+ s.logger.Error(err.Error())
+ s.logger.Error("panic on early close: " + fmt.Sprint(v))
+ }
+ }()
+ s.Close()
+ return err
+ }
+ s.logger.Info("sing-box pre-started (", F.Seconds(time.Since(s.createdAt).Seconds()), "s)")
+ return nil
+}
+
+func (s *Box) Start() error {
+ err := s.start()
+ if err != nil {
+ // TODO: remove catch error
+ defer func() {
+ v := recover()
+ if v != nil {
+ s.logger.Debug(err.Error())
+ s.logger.Error("panic on early start: " + fmt.Sprint(v))
+ }
+ }()
+ s.Close()
+ return err
+ }
+ s.logger.Info("sing-box started (", F.Seconds(time.Since(s.createdAt).Seconds()), "s)")
+ return nil
+}
+
+func (s *Box) preStart() error {
+ monitor := taskmonitor.New(s.logger, C.StartTimeout)
+ monitor.Start("start logger")
+ err := s.logFactory.Start()
+ monitor.Finish()
+ if err != nil {
+ return common.NewError(err, "start logger")
+ }
+ err = adapter.StartNamed(adapter.StartStateInitialize, s.services) // cache-file
+ if err != nil {
+ return err
+ }
+ err = adapter.Start(adapter.StartStateInitialize, s.network, s.connection, s.router, s.outbound, s.inbound, s.endpoint)
+ if err != nil {
+ return err
+ }
+ err = adapter.Start(adapter.StartStateStart, s.outbound, s.network, s.connection, s.router)
+ if err != nil {
+ return err
+ }
+ return nil
+}
+
+func (s *Box) start() error {
+ err := s.preStart()
+ if err != nil {
+ return err
+ }
+ err = adapter.StartNamed(adapter.StartStateStart, s.services)
+ if err != nil {
+ return err
+ }
+ err = s.inbound.Start(adapter.StartStateStart)
+ if err != nil {
+ return err
+ }
+ err = adapter.Start(adapter.StartStateStart, s.endpoint)
+ if err != nil {
+ return err
+ }
+ err = adapter.Start(adapter.StartStatePostStart, s.outbound, s.network, s.connection, s.router, s.inbound, s.endpoint)
+ if err != nil {
+ return err
+ }
+ err = adapter.StartNamed(adapter.StartStatePostStart, s.services)
+ if err != nil {
+ return err
+ }
+ err = adapter.Start(adapter.StartStateStarted, s.network, s.connection, s.router, s.outbound, s.inbound, s.endpoint)
+ if err != nil {
+ return err
+ }
+ err = adapter.StartNamed(adapter.StartStateStarted, s.services)
+ if err != nil {
+ return err
+ }
+ return nil
+}
+
+func (s *Box) Close() error {
+ select {
+ case <-s.done:
+ return os.ErrClosed
+ default:
+ close(s.done)
+ }
+ err := sbCommon.Close(
+ s.inbound, s.outbound, s.router, s.connection, s.network,
+ )
+ for _, lifecycleService := range s.services {
+ err1 := lifecycleService.Close()
+ if err1 != nil {
+ s.logger.Debug(lifecycleService.Name(), " close error: ", err1)
+ }
+ }
+ err1 := s.logFactory.Close()
+ if err1 != nil {
+ s.logger.Debug("logger close error: ", err1)
+ }
+ return err
+}
+
+func (s *Box) Uptime() uint32 {
+ return uint32(time.Now().Sub(s.createdAt).Seconds())
+}
+
+func (s *Box) Network() adapter.NetworkManager {
+ return s.network
+}
+
+func (s *Box) Router() adapter.Router {
+ return s.router
+}
+
+func (s *Box) Inbound() adapter.InboundManager {
+ return s.inbound
+}
+
+func (s *Box) Outbound() adapter.OutboundManager {
+ return s.outbound
+}
+
+func (s *Box) Endpoint() adapter.EndpointManager {
+ return s.endpoint
+}
+
+func (s *Box) ConnTracker() *ConnTracker {
+ return s.connTracker
+}
diff --git a/backend/core/conntracker.go b/backend/core/conntracker.go
new file mode 100644
index 0000000..5f99adc
--- /dev/null
+++ b/backend/core/conntracker.go
@@ -0,0 +1,145 @@
+package core
+
+import (
+ "context"
+ "net"
+ "s-ui/database/model"
+ "sync"
+ "time"
+
+ "github.com/sagernet/sing-box/adapter"
+ "github.com/sagernet/sing/common/atomic"
+ "github.com/sagernet/sing/common/bufio"
+ "github.com/sagernet/sing/common/network"
+)
+
+type Counter struct {
+ read *atomic.Int64
+ write *atomic.Int64
+}
+
+type ConnTracker struct {
+ access sync.Mutex
+ createdAt time.Time
+ inbounds map[string]Counter
+ outbounds map[string]Counter
+ users map[string]Counter
+}
+
+func NewConnTracker() *ConnTracker {
+ return &ConnTracker{
+ createdAt: time.Now(),
+ inbounds: make(map[string]Counter),
+ outbounds: make(map[string]Counter),
+ users: make(map[string]Counter),
+ }
+}
+
+func (c *ConnTracker) getReadCounters(inbound string, outbound string, user string) ([]*atomic.Int64, []*atomic.Int64) {
+ var readCounter []*atomic.Int64
+ var writeCounter []*atomic.Int64
+ c.access.Lock()
+ if inbound != "" {
+ readCounter = append(readCounter, c.loadOrCreateCounter(&c.inbounds, inbound).read)
+ writeCounter = append(writeCounter, c.inbounds[inbound].write)
+ }
+ if outbound != "" {
+ readCounter = append(readCounter, c.loadOrCreateCounter(&c.outbounds, outbound).read)
+ writeCounter = append(writeCounter, c.outbounds[outbound].write)
+ }
+ if user != "" {
+ readCounter = append(readCounter, c.loadOrCreateCounter(&c.users, user).read)
+ writeCounter = append(writeCounter, c.users[user].write)
+ }
+ c.access.Unlock()
+ return readCounter, writeCounter
+}
+
+func (c *ConnTracker) loadOrCreateCounter(obj *map[string]Counter, name string) Counter {
+ counter, loaded := (*obj)[name]
+ if loaded {
+ return counter
+ }
+ counter = Counter{read: &atomic.Int64{}, write: &atomic.Int64{}}
+ (*obj)[name] = counter
+ return counter
+}
+
+func (c *ConnTracker) RoutedConnection(ctx context.Context, conn net.Conn, metadata adapter.InboundContext, matchedRule adapter.Rule, matchOutbound adapter.Outbound) net.Conn {
+ readCounter, writeCounter := c.getReadCounters(metadata.Inbound, matchOutbound.Tag(), metadata.User)
+ return bufio.NewInt64CounterConn(conn, readCounter, writeCounter)
+}
+
+func (c *ConnTracker) RoutedPacketConnection(ctx context.Context, conn network.PacketConn, metadata adapter.InboundContext, matchedRule adapter.Rule, matchOutbound adapter.Outbound) network.PacketConn {
+ readCounter, writeCounter := c.getReadCounters(metadata.Inbound, matchOutbound.Tag(), metadata.User)
+ return bufio.NewInt64CounterPacketConn(conn, readCounter, writeCounter)
+}
+
+func (c *ConnTracker) GetStats() *[]model.Stats {
+ c.access.Lock()
+ defer c.access.Unlock()
+
+ dt := time.Now().Unix()
+
+ s := []model.Stats{}
+ for inbound, counter := range c.inbounds {
+ down := counter.write.Swap(0)
+ up := counter.read.Swap(0)
+ if down > 0 || up > 0 {
+ s = append(s, model.Stats{
+ DateTime: dt,
+ Resource: "inbound",
+ Tag: inbound,
+ Direction: false,
+ Traffic: down,
+ }, model.Stats{
+ DateTime: dt,
+ Resource: "inbound",
+ Tag: inbound,
+ Direction: true,
+ Traffic: up,
+ })
+ }
+ }
+
+ for outbound, counter := range c.outbounds {
+ down := counter.write.Swap(0)
+ up := counter.read.Swap(0)
+ if down > 0 || up > 0 {
+ s = append(s, model.Stats{
+ DateTime: dt,
+ Resource: "outbound",
+ Tag: outbound,
+ Direction: false,
+ Traffic: down,
+ }, model.Stats{
+ DateTime: dt,
+ Resource: "outbound",
+ Tag: outbound,
+ Direction: true,
+ Traffic: up,
+ })
+ }
+ }
+
+ for user, counter := range c.users {
+ down := counter.write.Swap(0)
+ up := counter.read.Swap(0)
+ if down > 0 || up > 0 {
+ s = append(s, model.Stats{
+ DateTime: dt,
+ Resource: "user",
+ Tag: user,
+ Direction: false,
+ Traffic: down,
+ }, model.Stats{
+ DateTime: dt,
+ Resource: "user",
+ Tag: user,
+ Direction: true,
+ Traffic: up,
+ })
+ }
+ }
+ return &s
+}
diff --git a/backend/core/endpoint.go b/backend/core/endpoint.go
new file mode 100644
index 0000000..4f0be03
--- /dev/null
+++ b/backend/core/endpoint.go
@@ -0,0 +1,109 @@
+package core
+
+import (
+ "s-ui/logger"
+ "s-ui/util/common"
+
+ "github.com/sagernet/sing-box/option"
+)
+
+func (c *Core) AddInbound(config []byte) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ var err error
+ var inbound_config option.Inbound
+ err = inbound_config.UnmarshalJSONContext(globalCtx, config)
+ if err != nil {
+ return err
+ }
+
+ err = inbound_manager.Create(
+ globalCtx,
+ router,
+ factory.NewLogger("inbound/"+inbound_config.Type+"["+inbound_config.Tag+"]"),
+ inbound_config.Tag,
+ inbound_config.Type,
+ inbound_config.Options)
+ if err != nil {
+ return err
+ }
+
+ return nil
+}
+
+func (c *Core) RemoveInbound(tag string) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ logger.Info("remove inbound: ", tag)
+ return inbound_manager.Remove(tag)
+}
+
+func (c *Core) AddOutbound(config []byte) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ var err error
+ var outbound_config option.Outbound
+
+ err = outbound_config.UnmarshalJSONContext(globalCtx, config)
+ if err != nil {
+ return err
+ }
+
+ err = outbound_manager.Create(
+ globalCtx,
+ router,
+ factory.NewLogger("outbound/"+outbound_config.Type+"["+outbound_config.Tag+"]"),
+ outbound_config.Tag,
+ outbound_config.Type,
+ outbound_config)
+ if err != nil {
+ return err
+ }
+
+ return nil
+}
+
+func (c *Core) RemoveOutbound(tag string) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ logger.Info("remove outbound: ", tag)
+ return outbound_manager.Remove(tag)
+}
+
+func (c *Core) AddEndpoint(config []byte) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ var err error
+ var endpoint_config option.Endpoint
+
+ err = endpoint_config.UnmarshalJSONContext(globalCtx, config)
+ if err != nil {
+ return err
+ }
+
+ err = endpoint_manager.Create(
+ globalCtx,
+ router,
+ factory.NewLogger("endpoint/"+endpoint_config.Type+"["+endpoint_config.Tag+"]"),
+ endpoint_config.Tag,
+ endpoint_config.Type,
+ endpoint_config)
+ if err != nil {
+ return err
+ }
+
+ return nil
+}
+
+func (c *Core) RemoveEndpoint(tag string) error {
+ if !c.isRunning {
+ return common.NewError("sing-box is not running")
+ }
+ logger.Info("remove endpoint: ", tag)
+ return endpoint_manager.Remove(tag)
+}
diff --git a/backend/core/log.go b/backend/core/log.go
new file mode 100644
index 0000000..46ac701
--- /dev/null
+++ b/backend/core/log.go
@@ -0,0 +1,236 @@
+package core
+
+import (
+ "context"
+ "io"
+ "os"
+ suiLog "s-ui/logger"
+
+ "github.com/sagernet/sing-box/log"
+ "github.com/sagernet/sing/common"
+ F "github.com/sagernet/sing/common/format"
+ "github.com/sagernet/sing/common/observable"
+ "github.com/sagernet/sing/service/filemanager"
+)
+
+type PlatformWriter struct{}
+
+func (p PlatformWriter) DisableColors() bool {
+ return true
+}
+func (p PlatformWriter) WriteMessage(level log.Level, message string) {
+ switch level {
+ case log.LevelInfo:
+ suiLog.Info(message)
+ case log.LevelWarn:
+ suiLog.Warning(message)
+ case log.LevelPanic:
+ case log.LevelFatal:
+ case log.LevelError:
+ suiLog.Error(message)
+ default:
+ suiLog.Debug(message)
+ }
+}
+
+func NewFactory(options log.Options) (log.Factory, error) {
+ logOptions := options.Options
+
+ if logOptions.Disabled {
+ return log.NewNOPFactory(), nil
+ }
+
+ var logWriter io.Writer
+ var logFilePath string
+
+ switch logOptions.Output {
+ case "":
+ logWriter = options.DefaultWriter
+ if logWriter == nil {
+ logWriter = os.Stderr
+ }
+ case "stderr":
+ logWriter = os.Stderr
+ case "stdout":
+ logWriter = os.Stdout
+ default:
+ logFilePath = logOptions.Output
+ }
+ logFormatter := log.Formatter{
+ BaseTime: options.BaseTime,
+ DisableColors: logOptions.DisableColor || logFilePath != "",
+ DisableTimestamp: !logOptions.Timestamp && logFilePath != "",
+ FullTimestamp: logOptions.Timestamp,
+ TimestampFormat: "-0700 2006-01-02 15:04:05",
+ }
+ factory := NewDefaultFactory(
+ options.Context,
+ logFormatter,
+ logWriter,
+ logFilePath,
+ )
+ if logOptions.Level != "" {
+ logLevel, err := log.ParseLevel(logOptions.Level)
+ if err != nil {
+ return nil, common.Error("parse log level", err)
+ }
+ factory.SetLevel(logLevel)
+ } else {
+ factory.SetLevel(log.LevelTrace)
+ }
+ return factory, nil
+}
+
+var _ log.Factory = (*defaultFactory)(nil)
+
+type defaultFactory struct {
+ ctx context.Context
+ formatter log.Formatter
+ writer io.Writer
+ file *os.File
+ filePath string
+ level log.Level
+ subscriber *observable.Subscriber[log.Entry]
+ observer *observable.Observer[log.Entry]
+}
+
+func NewDefaultFactory(
+ ctx context.Context,
+ formatter log.Formatter,
+ writer io.Writer,
+ filePath string,
+) log.ObservableFactory {
+ factory := &defaultFactory{
+ ctx: ctx,
+ formatter: formatter,
+ writer: writer,
+ filePath: filePath,
+ level: log.LevelTrace,
+ subscriber: observable.NewSubscriber[log.Entry](128),
+ }
+ return factory
+}
+
+func (f *defaultFactory) Start() error {
+ if f.filePath != "" {
+ logFile, err := filemanager.OpenFile(f.ctx, f.filePath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0o644)
+ if err != nil {
+ return err
+ }
+ f.writer = logFile
+ f.file = logFile
+ }
+ return nil
+}
+
+func (f *defaultFactory) Close() error {
+ return common.Close(
+ common.PtrOrNil(f.file),
+ f.subscriber,
+ )
+}
+
+func (f *defaultFactory) Level() log.Level {
+ return f.level
+}
+
+func (f *defaultFactory) SetLevel(level log.Level) {
+ f.level = level
+}
+
+func (f *defaultFactory) Logger() log.ContextLogger {
+ return f.NewLogger("")
+}
+
+func (f *defaultFactory) NewLogger(tag string) log.ContextLogger {
+ return &observableLogger{f, tag}
+}
+
+func (f *defaultFactory) Subscribe() (subscription observable.Subscription[log.Entry], done <-chan struct{}, err error) {
+ return f.observer.Subscribe()
+}
+
+func (f *defaultFactory) UnSubscribe(sub observable.Subscription[log.Entry]) {
+ f.observer.UnSubscribe(sub)
+}
+
+type observableLogger struct {
+ *defaultFactory
+ tag string
+}
+
+func (l *observableLogger) Log(ctx context.Context, level log.Level, args []any) {
+ level = log.OverrideLevelFromContext(level, ctx)
+ if level > l.level {
+ return
+ }
+ msg := F.ToString(args...)
+ switch level {
+ case log.LevelInfo:
+ suiLog.Info(l.tag, msg)
+ case log.LevelWarn:
+ suiLog.Warning(l.tag, msg)
+ case log.LevelPanic:
+ case log.LevelFatal:
+ case log.LevelError:
+ suiLog.Error(l.tag, msg)
+ default:
+ suiLog.Debug(l.tag, msg)
+ }
+}
+
+func (l *observableLogger) Trace(args ...any) {
+ l.TraceContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Debug(args ...any) {
+ l.DebugContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Info(args ...any) {
+ l.InfoContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Warn(args ...any) {
+ l.WarnContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Error(args ...any) {
+ l.ErrorContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Fatal(args ...any) {
+ l.FatalContext(context.Background(), args...)
+}
+
+func (l *observableLogger) Panic(args ...any) {
+ l.PanicContext(context.Background(), args...)
+}
+
+func (l *observableLogger) TraceContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelTrace, args)
+}
+
+func (l *observableLogger) DebugContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelDebug, args)
+}
+
+func (l *observableLogger) InfoContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelInfo, args)
+}
+
+func (l *observableLogger) WarnContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelWarn, args)
+}
+
+func (l *observableLogger) ErrorContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelError, args)
+}
+
+func (l *observableLogger) FatalContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelFatal, args)
+}
+
+func (l *observableLogger) PanicContext(ctx context.Context, args ...any) {
+ l.Log(ctx, log.LevelPanic, args)
+}
diff --git a/backend/core/main.go b/backend/core/main.go
new file mode 100644
index 0000000..ac62a43
--- /dev/null
+++ b/backend/core/main.go
@@ -0,0 +1,107 @@
+package core
+
+import (
+ "context"
+ "os"
+ "s-ui/config"
+ "s-ui/logger"
+
+ sb "github.com/sagernet/sing-box"
+ "github.com/sagernet/sing-box/adapter"
+ _ "github.com/sagernet/sing-box/experimental/clashapi"
+ _ "github.com/sagernet/sing-box/experimental/v2rayapi"
+ "github.com/sagernet/sing-box/log"
+ "github.com/sagernet/sing-box/option"
+ _ "github.com/sagernet/sing-box/transport/v2rayquic"
+ _ "github.com/sagernet/sing-dns/quic"
+ "github.com/sagernet/sing/service"
+)
+
+var (
+ globalCtx context.Context
+ inbound_manager adapter.InboundManager
+ outbound_manager adapter.OutboundManager
+ endpoint_manager adapter.EndpointManager
+ router adapter.Router
+ connTracker *ConnTracker
+ factory log.Factory
+)
+
+type Core struct {
+ isRunning bool
+ instance *Box
+}
+
+func NewCore() *Core {
+ globalCtx = context.Background()
+ globalCtx = sb.Context(globalCtx, inboundRegistry(), outboundRegistry(), EndpointRegistry())
+ return &Core{
+ isRunning: false,
+ instance: nil,
+ }
+}
+
+func (c *Core) GetCtx() context.Context {
+ return globalCtx
+}
+
+func (c *Core) GetInstance() *Box {
+ return c.instance
+}
+
+func (c *Core) Start() error {
+ filepath := config.GetBinFolderPath() + "/config.json"
+ configFile, err := os.ReadFile(filepath)
+ if err != nil {
+ return err
+ }
+ var opt option.Options
+ err = opt.UnmarshalJSONContext(globalCtx, configFile)
+ if err != nil {
+ logger.Error("Unmarshal config err:", err.Error())
+ }
+
+ c.instance, err = NewBox(Options{
+ Context: globalCtx,
+ Options: opt,
+ })
+ if err != nil {
+ return err
+ }
+
+ err = c.instance.Start()
+ if err != nil {
+ return err
+ }
+
+ globalCtx = service.ContextWith(globalCtx, c)
+ inbound_manager = service.FromContext[adapter.InboundManager](globalCtx)
+ outbound_manager = service.FromContext[adapter.OutboundManager](globalCtx)
+ endpoint_manager = service.FromContext[adapter.EndpointManager](globalCtx)
+ router = service.FromContext[adapter.Router](globalCtx)
+
+ c.isRunning = true
+ return nil
+}
+
+func (c *Core) Stop() error {
+ if c.isRunning {
+ c.isRunning = false
+ return c.instance.Close()
+ }
+ return nil
+}
+
+func (c *Core) Restart() error {
+ err := c.Stop()
+ if err != nil {
+ logger.Error("stop sing-box err:", err.Error())
+ return err
+ }
+ logger.Info("sing-box stopped")
+ return c.Start()
+}
+
+func (c *Core) IsRunning() bool {
+ return c.isRunning
+}
diff --git a/backend/core/register.go b/backend/core/register.go
new file mode 100644
index 0000000..aaf3e01
--- /dev/null
+++ b/backend/core/register.go
@@ -0,0 +1,94 @@
+package core
+
+import (
+ "github.com/sagernet/sing-box/adapter/endpoint"
+ "github.com/sagernet/sing-box/adapter/inbound"
+ "github.com/sagernet/sing-box/adapter/outbound"
+ "github.com/sagernet/sing-box/protocol/block"
+ "github.com/sagernet/sing-box/protocol/direct"
+ "github.com/sagernet/sing-box/protocol/dns"
+ "github.com/sagernet/sing-box/protocol/group"
+ "github.com/sagernet/sing-box/protocol/http"
+ "github.com/sagernet/sing-box/protocol/hysteria"
+ "github.com/sagernet/sing-box/protocol/hysteria2"
+ "github.com/sagernet/sing-box/protocol/mixed"
+ "github.com/sagernet/sing-box/protocol/naive"
+ _ "github.com/sagernet/sing-box/protocol/naive/quic"
+ "github.com/sagernet/sing-box/protocol/redirect"
+ "github.com/sagernet/sing-box/protocol/shadowsocks"
+ "github.com/sagernet/sing-box/protocol/shadowtls"
+ "github.com/sagernet/sing-box/protocol/socks"
+ "github.com/sagernet/sing-box/protocol/ssh"
+ "github.com/sagernet/sing-box/protocol/tor"
+ "github.com/sagernet/sing-box/protocol/trojan"
+ "github.com/sagernet/sing-box/protocol/tuic"
+ "github.com/sagernet/sing-box/protocol/tun"
+ "github.com/sagernet/sing-box/protocol/vless"
+ "github.com/sagernet/sing-box/protocol/vmess"
+ "github.com/sagernet/sing-box/protocol/wireguard"
+ _ "github.com/sagernet/sing-box/transport/v2rayquic"
+ _ "github.com/sagernet/sing-dns/quic"
+)
+
+func inboundRegistry() *inbound.Registry {
+ registry := inbound.NewRegistry()
+
+ tun.RegisterInbound(registry)
+ redirect.RegisterRedirect(registry)
+ redirect.RegisterTProxy(registry)
+ direct.RegisterInbound(registry)
+
+ socks.RegisterInbound(registry)
+ http.RegisterInbound(registry)
+ mixed.RegisterInbound(registry)
+
+ shadowsocks.RegisterInbound(registry)
+ vmess.RegisterInbound(registry)
+ trojan.RegisterInbound(registry)
+ naive.RegisterInbound(registry)
+ shadowtls.RegisterInbound(registry)
+ vless.RegisterInbound(registry)
+
+ hysteria.RegisterInbound(registry)
+ tuic.RegisterInbound(registry)
+ hysteria2.RegisterInbound(registry)
+
+ return registry
+}
+
+func outboundRegistry() *outbound.Registry {
+ registry := outbound.NewRegistry()
+
+ direct.RegisterOutbound(registry)
+
+ block.RegisterOutbound(registry)
+ dns.RegisterOutbound(registry)
+
+ group.RegisterSelector(registry)
+ group.RegisterURLTest(registry)
+
+ socks.RegisterOutbound(registry)
+ http.RegisterOutbound(registry)
+ shadowsocks.RegisterOutbound(registry)
+ vmess.RegisterOutbound(registry)
+ trojan.RegisterOutbound(registry)
+ tor.RegisterOutbound(registry)
+ ssh.RegisterOutbound(registry)
+ shadowtls.RegisterOutbound(registry)
+ vless.RegisterOutbound(registry)
+
+ hysteria.RegisterOutbound(registry)
+ tuic.RegisterOutbound(registry)
+ hysteria2.RegisterOutbound(registry)
+ wireguard.RegisterOutbound(registry)
+
+ return registry
+}
+
+func EndpointRegistry() *endpoint.Registry {
+ registry := endpoint.NewRegistry()
+
+ wireguard.RegisterEndpoint(registry)
+
+ return registry
+}
diff --git a/backend/cronjob/cronJob.go b/backend/cronjob/cronJob.go
index df5853c..0a73f44 100644
--- a/backend/cronjob/cronJob.go
+++ b/backend/cronjob/cronJob.go
@@ -1,6 +1,7 @@
package cronjob
import (
+ "s-ui/core"
"time"
"github.com/robfig/cron/v3"
@@ -8,10 +9,13 @@ import (
type CronJob struct {
cron *cron.Cron
+ Core *core.Core
}
-func NewCronJob() *CronJob {
- return &CronJob{}
+func NewCronJob(c *core.Core) *CronJob {
+ return &CronJob{
+ Core: c,
+ }
}
func (c *CronJob) Start(loc *time.Location, trafficAge int) error {
diff --git a/backend/cronjob/statsJob.go b/backend/cronjob/statsJob.go
index 84a4124..dcf72f7 100644
--- a/backend/cronjob/statsJob.go
+++ b/backend/cronjob/statsJob.go
@@ -6,15 +6,15 @@ import (
)
type StatsJob struct {
- service.SingBoxService
+ service.StatsService
}
func NewStatsJob() *StatsJob {
- return new(StatsJob)
+ return &StatsJob{}
}
func (s *StatsJob) Run() {
- err := s.SingBoxService.GetStats()
+ err := s.StatsService.SaveStats()
if err != nil {
logger.Warning("Get stats failed: ", err)
return
diff --git a/backend/go.mod b/backend/go.mod
index 9574084..d99f888 100644
--- a/backend/go.mod
+++ b/backend/go.mod
@@ -7,62 +7,121 @@ require (
github.com/gin-gonic/gin v1.10.0
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7
github.com/robfig/cron/v3 v3.0.1
+ github.com/sagernet/sing v0.6.0-beta.8
+ github.com/sagernet/sing-box v1.11.0-beta.11
+ github.com/sagernet/sing-dns v0.4.0-beta.1
github.com/shirou/gopsutil/v3 v3.24.5
- github.com/v2fly/v2ray-core/v5 v5.17.1
- google.golang.org/grpc v1.67.1
+ golang.zx2c4.com/wireguard/wgctrl v0.0.0-20230429144221-925a1e7659e6
gorm.io/driver/sqlite v1.5.6
gorm.io/gorm v1.25.12
)
+require google.golang.org/grpc v1.67.1 // indirect
+
require (
- github.com/adrg/xdg v0.5.0 // indirect
+ github.com/ajg/form v1.5.1 // indirect
+ github.com/andybalholm/brotli v1.0.6 // indirect
github.com/bytedance/sonic v1.12.3 // indirect
github.com/bytedance/sonic/loader v0.2.1 // indirect
+ github.com/caddyserver/certmagic v0.20.0 // indirect
+ github.com/cloudflare/circl v1.3.7 // indirect
github.com/cloudwego/base64x v0.1.4 // indirect
github.com/cloudwego/iasm v0.2.0 // indirect
+ github.com/cretz/bine v0.2.0 // indirect
+ github.com/fsnotify/fsnotify v1.7.0 // indirect
github.com/gabriel-vasile/mimetype v1.4.6 // indirect
github.com/gin-contrib/sessions v1.0.1
github.com/gin-contrib/sse v0.1.0 // indirect
+ github.com/go-chi/chi/v5 v5.1.0 // indirect
+ github.com/go-chi/render v1.0.3 // indirect
github.com/go-ole/go-ole v1.3.0 // indirect
github.com/go-playground/locales v0.14.1 // indirect
github.com/go-playground/universal-translator v0.18.1 // indirect
github.com/go-playground/validator/v10 v10.22.1 // indirect
+ github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect
+ github.com/gobwas/httphead v0.1.0 // indirect
+ github.com/gobwas/pool v0.2.1 // indirect
github.com/goccy/go-json v0.10.3 // indirect
+ github.com/gofrs/uuid/v5 v5.3.0 // indirect
github.com/golang/protobuf v1.5.4 // indirect
+ github.com/google/btree v1.1.3 // indirect
+ github.com/google/go-cmp v0.6.0 // indirect
github.com/google/pprof v0.0.0-20231101202521-4ca4178f5c7a // indirect
github.com/gorilla/context v1.1.2 // indirect
github.com/gorilla/securecookie v1.1.2 // indirect
github.com/gorilla/sessions v1.4.0 // indirect
+ github.com/hashicorp/yamux v0.1.2 // indirect
github.com/jinzhu/inflection v1.0.0 // indirect
github.com/jinzhu/now v1.1.5 // indirect
+ github.com/josharian/native v1.1.0 // indirect
github.com/json-iterator/go v1.1.12 // indirect
+ github.com/klauspost/compress v1.17.7 // indirect
github.com/klauspost/cpuid/v2 v2.2.8 // indirect
- github.com/kr/text v0.2.0 // indirect
github.com/leodido/go-urn v1.4.0 // indirect
+ github.com/libdns/alidns v1.0.3 // indirect
+ github.com/libdns/cloudflare v0.1.1 // indirect
+ github.com/libdns/libdns v0.2.2 // indirect; indiresct
+ github.com/logrusorgru/aurora v2.0.3+incompatible // indirect
github.com/lufia/plan9stats v0.0.0-20240909124753-873cd0166683 // indirect
github.com/mattn/go-isatty v0.0.20 // indirect
github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect
+ github.com/mdlayher/netlink v1.7.2 // indirect
+ github.com/mdlayher/socket v0.4.1 // indirect
+ github.com/metacubex/tfo-go v0.0.0-20241006021335-daedaf0ca7aa // indirect
+ github.com/mholt/acmez v1.2.0 // indirect
+ github.com/miekg/dns v1.1.62 // indirect
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
github.com/modern-go/reflect2 v1.0.2 // indirect
- github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e // indirect
+ github.com/onsi/ginkgo/v2 v2.10.0 // indirect
+ github.com/oschwald/maxminddb-golang v1.12.0 // indirect
github.com/pelletier/go-toml/v2 v2.2.3 // indirect
- github.com/pires/go-proxyproto v0.8.0 // indirect
github.com/power-devops/perfstat v0.0.0-20240221224432-82ca36839d55 // indirect
+ github.com/quic-go/qpack v0.4.0 // indirect
+ github.com/quic-go/qtls-go1-20 v0.4.1 // indirect
+ github.com/sagernet/bbolt v0.0.0-20231014093535-ea5cb2fe9f0a // indirect
+ github.com/sagernet/cloudflare-tls v0.0.0-20231208171750-a4483c1b7cd1 // indirect
+ github.com/sagernet/cors v1.2.1 // indirect
+ github.com/sagernet/fswatch v0.1.1 // indirect
+ github.com/sagernet/gvisor v0.0.0-20241123041152-536d05261cff // indirect
+ github.com/sagernet/netlink v0.0.0-20240612041022-b9a21c07ac6a // indirect
+ github.com/sagernet/nftables v0.3.0-beta.4 // indirect
+ github.com/sagernet/quic-go v0.48.2-beta.1 // indirect
+ github.com/sagernet/reality v0.0.0-20230406110435-ee17307e7691 // indirect
+ github.com/sagernet/sing-mux v0.3.0-alpha.1 // indirect
+ github.com/sagernet/sing-quic v0.4.0-alpha.4 // indirect
+ github.com/sagernet/sing-shadowsocks v0.2.7 // indirect
+ github.com/sagernet/sing-shadowsocks2 v0.2.0 // indirect
+ github.com/sagernet/sing-shadowtls v0.2.0-alpha.2 // indirect
+ github.com/sagernet/sing-tun v0.6.0-beta.6 // indirect
+ github.com/sagernet/sing-vmess v0.2.0-beta.1 // indirect
+ github.com/sagernet/smux v0.0.0-20231208180855-7041f6ea79e7 // indirect
+ github.com/sagernet/utls v1.6.7 // indirect
+ github.com/sagernet/wireguard-go v0.0.1-beta.5 // indirect
+ github.com/sagernet/ws v0.0.0-20231204124109-acfe8907c854 // indirect
github.com/shoenig/go-m1cpu v0.1.6 // indirect
github.com/tklauser/go-sysconf v0.3.14 // indirect
github.com/tklauser/numcpus v0.9.0 // indirect
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
github.com/ugorji/go/codec v1.2.12 // indirect
+ github.com/vishvananda/netns v0.0.4 // indirect
github.com/yusufpapurcu/wmi v1.2.4 // indirect
+ github.com/zeebo/blake3 v0.2.3 // indirect
+ go.uber.org/multierr v1.11.0 // indirect
+ go.uber.org/zap v1.27.0 // indirect
go4.org/netipx v0.0.0-20231129151722-fdeea329fbba // indirect
golang.org/x/arch v0.11.0 // indirect
- golang.org/x/crypto v0.28.0 // indirect
+ golang.org/x/crypto v0.29.0 // indirect
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 // indirect
- golang.org/x/net v0.30.0 // indirect
- golang.org/x/sys v0.26.0 // indirect
- golang.org/x/text v0.19.0 // indirect
+ golang.org/x/mod v0.20.0 // indirect
+ golang.org/x/net v0.31.0 // indirect
+ golang.org/x/sync v0.9.0 // indirect
+ golang.org/x/sys v0.27.0 // indirect
+ golang.org/x/text v0.20.0 // indirect
+ golang.org/x/time v0.7.0 // indirect
+ golang.org/x/tools v0.24.0 // indirect
+ golang.zx2c4.com/wintun v0.0.0-20230126152724-0fa3db229ce2 // indirect
google.golang.org/genproto/googleapis/rpc v0.0.0-20241015192408-796eee8c2d53 // indirect
google.golang.org/protobuf v1.35.1 // indirect
- gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
+ lukechampine.com/blake3 v1.3.0 // indirect
)
diff --git a/backend/go.sum b/backend/go.sum
index 47c3ce0..8083f22 100644
--- a/backend/go.sum
+++ b/backend/go.sum
@@ -1,34 +1,27 @@
-github.com/adrg/xdg v0.5.0 h1:dDaZvhMXatArP1NPHhnfaQUqWBLBsmx1h1HXQdMoFCY=
-github.com/adrg/xdg v0.5.0/go.mod h1:dDdY4M4DF9Rjy4kHPeNL+ilVF+p2lK8IdM9/rTSGcI4=
-github.com/aead/cmac v0.0.0-20160719120800-7af84192f0b1 h1:+JkXLHME8vLJafGhOH4aoV2Iu8bR55nU6iKMVfYVLjY=
-github.com/aead/cmac v0.0.0-20160719120800-7af84192f0b1/go.mod h1:nuudZmJhzWtx2212z+pkuy7B6nkBqa+xwNXZHL1j8cg=
github.com/ajg/form v1.5.1 h1:t9c7v8JUKu/XxOGBU0yjNpaMloxGEJhUkqFRq0ibGeU=
github.com/ajg/form v1.5.1/go.mod h1:uL1WgH+h2mgNtvBq0339dVnzXdBETtL2LeUXaIv25UY=
github.com/andybalholm/brotli v1.0.6 h1:Yf9fFpf49Zrxb9NlQaluyE92/+X7UVHlhMNJN2sxfOI=
github.com/andybalholm/brotli v1.0.6/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
-github.com/boljen/go-bitmap v0.0.0-20151001105940-23cd2fb0ce7d h1:zsO4lp+bjv5XvPTF58Vq+qgmZEYZttJK+CWtSZhKenI=
-github.com/boljen/go-bitmap v0.0.0-20151001105940-23cd2fb0ce7d/go.mod h1:f1iKL6ZhUWvbk7PdWVmOaak10o86cqMUYEmn1CZNGEI=
-github.com/bufbuild/protocompile v0.10.0 h1:+jW/wnLMLxaCEG8AX9lD0bQ5v9h1RUiMKOBOT5ll9dM=
-github.com/bufbuild/protocompile v0.10.0/go.mod h1:G9qQIQo0xZ6Uyj6CMNz0saGmx2so+KONo8/KrELABiY=
github.com/bytedance/sonic v1.12.3 h1:W2MGa7RCU1QTeYRTPE3+88mVC0yXmsRQRChiyVocVjU=
github.com/bytedance/sonic v1.12.3/go.mod h1:B8Gt/XvtZ3Fqj+iSKMypzymZxw/FVwgIGKzMzT9r/rk=
github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
github.com/bytedance/sonic/loader v0.2.1 h1:1GgorWTqf12TA8mma4DDSbaQigE2wOgQo7iCjjJv3+E=
github.com/bytedance/sonic/loader v0.2.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
+github.com/caddyserver/certmagic v0.20.0 h1:bTw7LcEZAh9ucYCRXyCpIrSAGplplI0vGYJ4BpCQ/Fc=
+github.com/caddyserver/certmagic v0.20.0/go.mod h1:N4sXgpICQUskEWpj7zVzvWD41p3NYacrNoZYiRM2jTg=
github.com/cloudflare/circl v1.3.7 h1:qlCDlTPz2n9fu58M0Nh1J/JzcFpfgkFHHX3O35r5vcU=
github.com/cloudflare/circl v1.3.7/go.mod h1:sRTcRWXGLrKw6yIGJ+l7amYJFfAXbZG0kBSc8r4zxgA=
github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y=
github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w=
github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg=
github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY=
-github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
+github.com/cretz/bine v0.2.0 h1:8GiDRGlTgz+o8H9DSnsl+5MeBK4HsExxgl6WgzOCuZo=
+github.com/cretz/bine v0.2.0/go.mod h1:WU4o9QR9wWp8AVKtTM1XD5vUHkEqnf2vVSo6dBqbetI=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
-github.com/dgryski/go-metro v0.0.0-20211217172704-adc40b04c140 h1:y7y0Oa6UawqTFPCDw9JG6pdKt4F9pAhHv0B7FMGaGD0=
-github.com/dgryski/go-metro v0.0.0-20211217172704-adc40b04c140/go.mod h1:c9O8+fpSOX1DM8cPNSkX/qsBWdkD4yd2dpciOWQjpBw=
-github.com/ebfe/bcrypt_pbkdf v0.0.0-20140212075826-3c8d2dcb253a h1:YtdtTUN1iH97s+6PUjLnaiKSQj4oG1/EZ3N9bx6g4kU=
-github.com/ebfe/bcrypt_pbkdf v0.0.0-20140212075826-3c8d2dcb253a/go.mod h1:/CZpbhAusDOobpcb9yubw46kdYjq0zRC0Wpg9a9zFQM=
+github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
+github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
github.com/gabriel-vasile/mimetype v1.4.6 h1:3+PzJTKLkvgjeTbts6msPJt4DixhT4YtFNf1gtGe3zc=
github.com/gabriel-vasile/mimetype v1.4.6/go.mod h1:JX1qVKqZd40hUPpAfiNTe0Sne7hdfKSbOqqmkq8GCXc=
github.com/gin-contrib/gzip v1.0.1 h1:HQ8ENHODeLY7a4g1Au/46Z92bdGFl74OhxcZble9WJE=
@@ -43,6 +36,8 @@ github.com/go-chi/chi/v5 v5.1.0 h1:acVI1TYaD+hhedDJ3r54HyA6sExp3HfXq7QWEEY/xMw=
github.com/go-chi/chi/v5 v5.1.0/go.mod h1:DslCQbL2OYiznFReuXYUmQ2hGd1aDpCnlMNITLSKoi8=
github.com/go-chi/render v1.0.3 h1:AsXqd2a1/INaIfUSKq3G5uA8weYx20FOsM7uSoCyyt4=
github.com/go-chi/render v1.0.3/go.mod h1:/gr3hVkmYR0YlEy3LxCuVRFzEu9Ruok+gFqbIofjao0=
+github.com/go-logr/logr v1.2.4 h1:g01GSCwiDw2xSZfjJ2/T9M+S6pFdcNtFYsp+Y43HYDQ=
+github.com/go-logr/logr v1.2.4/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0=
github.com/go-ole/go-ole v1.3.0 h1:Dt6ye7+vXGIKZ7Xtk4s6/xVdGDQynvom7xCFEdWr6uE=
github.com/go-ole/go-ole v1.3.0/go.mod h1:5LS6F96DhAwUc7C+1HLexzMXY1xGRSryjyPPKW6zv78=
@@ -56,23 +51,23 @@ github.com/go-playground/validator/v10 v10.22.1 h1:40JcKH+bBNGFczGuoBYgX4I6m/i27
github.com/go-playground/validator/v10 v10.22.1/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM=
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI=
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls=
+github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU=
+github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM=
+github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og=
+github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw=
github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA=
github.com/goccy/go-json v0.10.3/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M=
-github.com/golang-collections/go-datastructures v0.0.0-20150211160725-59788d5eb259 h1:ZHJ7+IGpuOXtVf6Zk/a3WuHQgkC+vXwaqfUBDFwahtI=
-github.com/golang-collections/go-datastructures v0.0.0-20150211160725-59788d5eb259/go.mod h1:9Qcha0gTWLw//0VNka1Cbnjvg3pNKGFdAm7E9sBabxE=
-github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc=
-github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
+github.com/gofrs/uuid/v5 v5.3.0 h1:m0mUMr+oVYUdxpMLgSYCZiXe7PuVPnI94+OMeVBNedk=
+github.com/gofrs/uuid/v5 v5.3.0/go.mod h1:CDOjlDMVAtN56jqyRUZh58JT31Tiw7/oQyEXZV+9bD8=
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
-github.com/google/btree v1.1.2 h1:xf4v41cLI2Z6FxbKm+8Bu+m8ifhj15JuZ9sa0jZCMUU=
-github.com/google/btree v1.1.2/go.mod h1:qOPhT0dTNdNzV6Z/lhRX0YXUafgPLFUh+gZMl761Gm4=
+github.com/google/btree v1.1.3 h1:CVpQJjYgC4VbzxeGVHfvZrv1ctoYCAI8vbl07Fcxlyg=
+github.com/google/btree v1.1.3/go.mod h1:qOPhT0dTNdNzV6Z/lhRX0YXUafgPLFUh+gZMl761Gm4=
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0=
github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
-github.com/google/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8=
-github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo=
github.com/google/pprof v0.0.0-20231101202521-4ca4178f5c7a h1:fEBsGL/sjAuJrgah5XqmmYsTLzJp/TO9Lhy39gkverk=
github.com/google/pprof v0.0.0-20231101202521-4ca4178f5c7a/go.mod h1:czg5+yv1E0ZGTi6S6vVK1mke0fV+FaUhNGcd6VRS9Ik=
github.com/gorilla/context v1.1.2 h1:WRkNAv2uoa03QNIc1A6u4O7DAGMUVoopZhkiXWA2V1o=
@@ -81,87 +76,134 @@ github.com/gorilla/securecookie v1.1.2 h1:YCIWL56dvtr73r6715mJs5ZvhtnY73hBvEF8kX
github.com/gorilla/securecookie v1.1.2/go.mod h1:NfCASbcHqRSY+3a8tlWJwsQap2VX5pwzwo4h3eOamfo=
github.com/gorilla/sessions v1.4.0 h1:kpIYOp/oi6MG/p5PgxApU8srsSw9tuFbt46Lt7auzqQ=
github.com/gorilla/sessions v1.4.0/go.mod h1:FLWm50oby91+hl7p/wRxDth9bWSuk0qVL2emc7lT5ik=
-github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg=
-github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
-github.com/jhump/protoreflect v1.16.0 h1:54fZg+49widqXYQ0b+usAFHbMkBGR4PpXrsHc8+TBDg=
-github.com/jhump/protoreflect v1.16.0/go.mod h1:oYPd7nPvcBw/5wlDfm/AVmU9zH9BgqGCI469pGxfj/8=
+github.com/hashicorp/yamux v0.1.2 h1:XtB8kyFOyHXYVFnwT5C3+Bdo8gArse7j2AQ0DA0Uey8=
+github.com/hashicorp/yamux v0.1.2/go.mod h1:C+zze2n6e/7wshOZep2A70/aQU6QBRWJO/G6FT1wIns=
github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ=
github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
+github.com/josharian/native v1.1.0 h1:uuaP0hAbW7Y4l0ZRQ6C9zfb7Mg1mbFKry/xzDAfmtLA=
+github.com/josharian/native v1.1.0/go.mod h1:7X/raswPFr05uY3HiLlYeyQntB6OO7E/d2Cu7qoaN2w=
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
github.com/klauspost/compress v1.17.7 h1:ehO88t2UGzQK66LMdE8tibEd1ErmzZjNEqWkjLAKQQg=
github.com/klauspost/compress v1.17.7/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw=
github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
+github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c=
github.com/klauspost/cpuid/v2 v2.2.8 h1:+StwCXwm9PdpiEkPyzBXIy+M9KUb4ODm0Zarf1kS5BM=
github.com/klauspost/cpuid/v2 v2.2.8/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws=
-github.com/klauspost/reedsolomon v1.11.7 h1:9uaHU0slncktTEEg4+7Vl7q7XUNMBUOK4R9gnKhMjAU=
-github.com/klauspost/reedsolomon v1.11.7/go.mod h1:4bXRN+cVzMdml6ti7qLouuYi32KHJ5MGv0Qd8a47h6A=
github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M=
-github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
-github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ=
github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI=
+github.com/libdns/alidns v1.0.3 h1:LFHuGnbseq5+HCeGa1aW8awyX/4M2psB9962fdD2+yQ=
+github.com/libdns/alidns v1.0.3/go.mod h1:e18uAG6GanfRhcJj6/tps2rCMzQJaYVcGKT+ELjdjGE=
+github.com/libdns/cloudflare v0.1.1 h1:FVPfWwP8zZCqj268LZjmkDleXlHPlFU9KC4OJ3yn054=
+github.com/libdns/cloudflare v0.1.1/go.mod h1:9VK91idpOjg6v7/WbjkEW49bSCxj00ALesIFDhJ8PBU=
+github.com/libdns/libdns v0.2.0/go.mod h1:yQCXzk1lEZmmCPa857bnk4TsOiqYasqpyOEeSObbb40=
+github.com/libdns/libdns v0.2.2 h1:O6ws7bAfRPaBsgAYt8MDe2HcNBGC29hkZ9MX2eUSX3s=
+github.com/libdns/libdns v0.2.2/go.mod h1:4Bj9+5CQiNMVGf87wjX4CY3HQJypUHRuLvlsfsZqLWQ=
+github.com/logrusorgru/aurora v2.0.3+incompatible h1:tOpm7WcpBTn4fjmVfgpQq0EfczGlG91VSDkswnjF5A8=
+github.com/logrusorgru/aurora v2.0.3+incompatible/go.mod h1:7rIyQOR62GCctdiQpZ/zOJlFyk6y+94wXzv6RNZgaR4=
github.com/lufia/plan9stats v0.0.0-20240909124753-873cd0166683 h1:7UMa6KCCMjZEMDtTVdcGu0B1GmmC7QJKiCCjyTAWQy0=
github.com/lufia/plan9stats v0.0.0-20240909124753-873cd0166683/go.mod h1:ilwx/Dta8jXAgpFYFvSWEMwxmbWXyiUHkd5FwyKhb5k=
-github.com/lunixbochs/struc v0.0.0-20200707160740-784aaebc1d40 h1:EnfXoSqDfSNJv0VBNqY/88RNnhSGYkrHaO0mmFGbVsc=
-github.com/lunixbochs/struc v0.0.0-20200707160740-784aaebc1d40/go.mod h1:vy1vK6wD6j7xX6O6hXe621WabdtNkou2h7uRtTfRMyg=
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
github.com/mattn/go-sqlite3 v2.0.3+incompatible h1:gXHsfypPkaMZrKbD5209QV9jbUTJKjyR5WD3HYQSd+U=
github.com/mattn/go-sqlite3 v2.0.3+incompatible/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
+github.com/mdlayher/netlink v1.7.2 h1:/UtM3ofJap7Vl4QWCPDGXY8d3GIY2UGSDbK+QWmY8/g=
+github.com/mdlayher/netlink v1.7.2/go.mod h1:xraEF7uJbxLhc5fpHL4cPe221LI2bdttWlU+ZGLfQSw=
+github.com/mdlayher/socket v0.4.1 h1:eM9y2/jlbs1M615oshPQOHZzj6R6wMT7bX5NPiQvn2U=
+github.com/mdlayher/socket v0.4.1/go.mod h1:cAqeGjoufqdxWkD7DkpyS+wcefOtmu5OQ8KuoJGIReA=
+github.com/metacubex/tfo-go v0.0.0-20241006021335-daedaf0ca7aa h1:9mcjV+RGZVC3reJBNDjjNPyS8PmFG97zq56X7WNaFO4=
+github.com/metacubex/tfo-go v0.0.0-20241006021335-daedaf0ca7aa/go.mod h1:4tLB5c8U0CxpkFM+AJJB77jEaVDbLH5XQvy42vAGsWw=
+github.com/mholt/acmez v1.2.0 h1:1hhLxSgY5FvH5HCnGUuwbKY2VQVo8IU7rxXKSnZ7F30=
+github.com/mholt/acmez v1.2.0/go.mod h1:VT9YwH1xgNX1kmYY89gY8xPJC84BFAisjo8Egigt4kE=
+github.com/miekg/dns v1.1.62 h1:cN8OuEF1/x5Rq6Np+h1epln8OiyPWV+lROx9LxcGgIQ=
+github.com/miekg/dns v1.1.62/go.mod h1:mvDlcItzm+br7MToIKqkglaGhlFMHJ9DTNNWONWXbNQ=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
-github.com/mustafaturan/bus v1.0.2 h1:2x3ErwZ0uUPwwZ5ZZoknEQprdaxr68Yl3mY8jDye1Ws=
-github.com/mustafaturan/bus v1.0.2/go.mod h1:h7gfehm8TThv4Dcaa+wDQG7r7j6p74v+7ftr0Rq9i1Q=
-github.com/mustafaturan/monoton v1.0.0 h1:8SCej+JiNn0lyps7V+Jzc1CRAkDR4EZPWrTupQ61YCQ=
-github.com/mustafaturan/monoton v1.0.0/go.mod h1:FOnE7NV3s3EWPXb8/7+/OSdiMBbdlkV0Lz8p1dc+vy8=
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs=
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno=
github.com/onsi/ginkgo/v2 v2.10.0 h1:sfUl4qgLdvkChZrWCYndY2EAu9BRIw1YphNAzy1VNWs=
github.com/onsi/ginkgo/v2 v2.10.0/go.mod h1:UDQOh5wbQUlMnkLfVaIUMtQ1Vus92oM+P2JX1aulgcE=
+github.com/onsi/gomega v1.27.7 h1:fVih9JD6ogIiHUN6ePK7HJidyEDpWGVB5mzM7cWNXoU=
+github.com/onsi/gomega v1.27.7/go.mod h1:1p8OOlwo2iUUDsHnOrjE5UKYJ+e3W8eQ3qSlRahPmr4=
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7 h1:lDH9UUVJtmYCjyT0CI4q8xvlXPxeZ0gYCVvWbmPlp88=
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk=
-github.com/patrickmn/go-cache v2.1.0+incompatible h1:HRMgzkcYKYpi3C8ajMPV8OFXaaRUnok+kx1WdO15EQc=
-github.com/patrickmn/go-cache v2.1.0+incompatible/go.mod h1:3Qf8kWWT7OJRJbdiICTKqZju1ZixQ/KpMGzzAfe6+WQ=
-github.com/pelletier/go-toml v1.9.5 h1:4yBQzkHv+7BHq2PQUZF3Mx0IYxG7LsP222s7Agd3ve8=
-github.com/pelletier/go-toml v1.9.5/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c=
+github.com/oschwald/maxminddb-golang v1.12.0 h1:9FnTOD0YOhP7DGxGsq4glzpGy5+w7pq50AS6wALUMYs=
+github.com/oschwald/maxminddb-golang v1.12.0/go.mod h1:q0Nob5lTCqyQ8WT6FYgS1L7PXKVVbgiymefNwIjPzgY=
github.com/pelletier/go-toml/v2 v2.2.3 h1:YmeHyLY8mFWbdkNWwpr+qIL2bEqT0o95WSdkNHvL12M=
github.com/pelletier/go-toml/v2 v2.2.3/go.mod h1:MfCQTFTvCcUyyvvwm1+G6H/jORL20Xlb6rzQu9GuUkc=
-github.com/pion/dtls/v2 v2.2.12 h1:KP7H5/c1EiVAAKUmXyCzPiQe5+bCJrpOeKg/L05dunk=
-github.com/pion/dtls/v2 v2.2.12/go.mod h1:d9SYc9fch0CqK90mRk1dC7AkzzpwJj6u2GU3u+9pqFE=
-github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY=
-github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms=
-github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA=
-github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8=
-github.com/pion/sctp v1.8.7 h1:JnABvFakZueGAn4KU/4PSKg+GWbF6QWbKTWZOSGJjXw=
-github.com/pion/sctp v1.8.7/go.mod h1:g1Ul+ARqZq5JEmoFy87Q/4CePtKnTJ1QCL9dBBdN6AU=
-github.com/pion/transport/v2 v2.2.10 h1:ucLBLE8nuxiHfvkFKnkDQRYWYfp8ejf4YBOPfaQpw6Q=
-github.com/pion/transport/v2 v2.2.10/go.mod h1:sq1kSLWs+cHW9E+2fJP95QudkzbK7wscs8yYgQToO5E=
-github.com/pires/go-proxyproto v0.8.0 h1:5unRmEAPbHXHuLjDg01CxJWf91cw3lKHc/0xzKpXEe0=
-github.com/pires/go-proxyproto v0.8.0/go.mod h1:iknsfgnH8EkjrMeMyvfKByp9TiBZCKZM0jx2xmKqnVY=
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/power-devops/perfstat v0.0.0-20240221224432-82ca36839d55 h1:o4JXh1EVt9k/+g42oCprj/FisM4qX9L3sZB3upGN2ZU=
github.com/power-devops/perfstat v0.0.0-20240221224432-82ca36839d55/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE=
-github.com/quic-go/quic-go v0.46.0 h1:uuwLClEEyk1DNvchH8uCByQVjo3yKL9opKulExNDs7Y=
-github.com/quic-go/quic-go v0.46.0/go.mod h1:1dLehS7TIR64+vxGR70GDcatWTOtMX2PUtnKsjbTurI=
-github.com/refraction-networking/utls v1.6.7 h1:zVJ7sP1dJx/WtVuITug3qYUq034cDq9B2MR1K67ULZM=
-github.com/refraction-networking/utls v1.6.7/go.mod h1:BC3O4vQzye5hqpmDTWUqi4P5DDhzJfkV1tdqtawQIH0=
-github.com/riobard/go-bloom v0.0.0-20200614022211-cdc8013cb5b3 h1:f/FNXud6gA3MNr8meMVVGxhp+QBTqY91tM8HjEuMjGg=
-github.com/riobard/go-bloom v0.0.0-20200614022211-cdc8013cb5b3/go.mod h1:HgjTstvQsPGkxUsCd2KWxErBblirPizecHcpD3ffK+s=
+github.com/quic-go/qpack v0.4.0 h1:Cr9BXA1sQS2SmDUWjSofMPNKmvF6IiIfDRmgU0w1ZCo=
+github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A=
+github.com/quic-go/qtls-go1-20 v0.4.1 h1:D33340mCNDAIKBqXuAvexTNMUByrYmFYVfKfDN5nfFs=
+github.com/quic-go/qtls-go1-20 v0.4.1/go.mod h1:X9Nh97ZL80Z+bX/gUXMbipO6OxdiDi58b/fMC9mAL+k=
github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs=
github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro=
-github.com/secure-io/siv-go v0.0.0-20180922214919-5ff40651e2c4 h1:zOjq+1/uLzn/Xo40stbvjIY/yehG0+mfmlsiEmc0xmQ=
-github.com/secure-io/siv-go v0.0.0-20180922214919-5ff40651e2c4/go.mod h1:aI+8yClBW+1uovkHw6HM01YXnYB8vohtB9C83wzx34E=
-github.com/seiflotfy/cuckoofilter v0.0.0-20220411075957-e3b120b3f5fb h1:XfLJSPIOUX+osiMraVgIrMR27uMXnRJWGm1+GL8/63U=
-github.com/seiflotfy/cuckoofilter v0.0.0-20220411075957-e3b120b3f5fb/go.mod h1:bR6DqgcAl1zTcOX8/pE2Qkj9XO00eCNqmKb7lXP8EAg=
+github.com/sagernet/bbolt v0.0.0-20231014093535-ea5cb2fe9f0a h1:+NkI2670SQpQWvkkD2QgdTuzQG263YZ+2emfpeyGqW0=
+github.com/sagernet/bbolt v0.0.0-20231014093535-ea5cb2fe9f0a/go.mod h1:63s7jpZqcDAIpj8oI/1v4Izok+npJOHACFCU6+huCkM=
+github.com/sagernet/cloudflare-tls v0.0.0-20231208171750-a4483c1b7cd1 h1:YbmpqPQEMdlk9oFSKYWRqVuu9qzNiOayIonKmv1gCXY=
+github.com/sagernet/cloudflare-tls v0.0.0-20231208171750-a4483c1b7cd1/go.mod h1:J2yAxTFPDjrDPhuAi9aWFz2L3ox9it4qAluBBbN0H5k=
+github.com/sagernet/cors v1.2.1 h1:Cv5Z8y9YSD6Gm+qSpNrL3LO4lD3eQVvbFYJSG7JCMHQ=
+github.com/sagernet/cors v1.2.1/go.mod h1:O64VyOjjhrkLmQIjF4KGRrJO/5dVXFdpEmCW/eISRAI=
+github.com/sagernet/fswatch v0.1.1 h1:YqID+93B7VRfqIH3PArW/XpJv5H4OLEVWDfProGoRQs=
+github.com/sagernet/fswatch v0.1.1/go.mod h1:nz85laH0mkQqJfaOrqPpkwtU1znMFNVTpT/5oRsVz/o=
+github.com/sagernet/gvisor v0.0.0-20241123041152-536d05261cff h1:mlohw3360Wg1BNGook/UHnISXhUx4Gd/3tVLs5T0nSs=
+github.com/sagernet/gvisor v0.0.0-20241123041152-536d05261cff/go.mod h1:ehZwnT2UpmOWAHFL48XdBhnd4Qu4hN2O3Ji0us3ZHMw=
+github.com/sagernet/netlink v0.0.0-20240612041022-b9a21c07ac6a h1:ObwtHN2VpqE0ZNjr6sGeT00J8uU7JF4cNUdb44/Duis=
+github.com/sagernet/netlink v0.0.0-20240612041022-b9a21c07ac6a/go.mod h1:xLnfdiJbSp8rNqYEdIW/6eDO4mVoogml14Bh2hSiFpM=
+github.com/sagernet/nftables v0.3.0-beta.4 h1:kbULlAwAC3jvdGAC1P5Fa3GSxVwQJibNenDW2zaXr8I=
+github.com/sagernet/nftables v0.3.0-beta.4/go.mod h1:OQXAjvjNGGFxaTgVCSTRIhYB5/llyVDeapVoENYBDS8=
+github.com/sagernet/quic-go v0.48.2-beta.1 h1:W0plrLWa1XtOWDTdX3CJwxmQuxkya12nN5BRGZ87kEg=
+github.com/sagernet/quic-go v0.48.2-beta.1/go.mod h1:1WgdDIVD1Gybp40JTWketeSfKA/+or9YMLaG5VeTk4k=
+github.com/sagernet/reality v0.0.0-20230406110435-ee17307e7691 h1:5Th31OC6yj8byLGkEnIYp6grlXfo1QYUfiYFGjewIdc=
+github.com/sagernet/reality v0.0.0-20230406110435-ee17307e7691/go.mod h1:B8lp4WkQ1PwNnrVMM6KyuFR20pU8jYBD+A4EhJovEXU=
+github.com/sagernet/sing v0.2.18/go.mod h1:OL6k2F0vHmEzXz2KW19qQzu172FDgSbUSODylighuVo=
+github.com/sagernet/sing v0.6.0-beta.5 h1:RD2j8WmJsvAbbBkAlJWaiYmnd+v/JohBiweoew7kMwo=
+github.com/sagernet/sing v0.6.0-beta.5/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
+github.com/sagernet/sing v0.6.0-beta.8 h1:PoxDdN7y8D4oImT3cQ05Sq1ZYnYsJberkUkIEHIGwWE=
+github.com/sagernet/sing v0.6.0-beta.8/go.mod h1:ARkL0gM13/Iv5VCZmci/NuoOlePoIsW0m7BWfln/Hak=
+github.com/sagernet/sing-box v1.11.0-beta.6 h1:MPdL2Yem/xM0RhejCO7krYvl1Zbd1zkSjKluKpHnHPQ=
+github.com/sagernet/sing-box v1.11.0-beta.6/go.mod h1:6dO5V0A37cLlhvKnxCmZinSpZXz7ZSk11x3rgI+xH1I=
+github.com/sagernet/sing-box v1.11.0-beta.11 h1:bVR0n3oQ3hGcuc/CSS7axsOeRNCRlCGkYVOKl0wxbsw=
+github.com/sagernet/sing-box v1.11.0-beta.11/go.mod h1:GZnZUzUHZ6Bgm7D/i8unNORv3537u1s03tLXFdxCRpg=
+github.com/sagernet/sing-dns v0.4.0-beta.1 h1:W1XkdhigwxDOMgMDVB+9kdomCpb7ExsZfB4acPcTZFY=
+github.com/sagernet/sing-dns v0.4.0-beta.1/go.mod h1:8wuFcoFkWM4vJuQyg8e97LyvDwe0/Vl7G839WLcKDs8=
+github.com/sagernet/sing-mux v0.3.0-alpha.1 h1:IgNX5bJBpL41gGbp05pdDOvh/b5eUQ6cv9240+Ngipg=
+github.com/sagernet/sing-mux v0.3.0-alpha.1/go.mod h1:FTcImmdfW38Lz7b+HQ+mxxOth1lz4ao8uEnz+MwIJQE=
+github.com/sagernet/sing-quic v0.4.0-alpha.4 h1:P9xAx3nIfcqb9M8jfgs0uLm+VxCcaY++FCqaBfHY3dQ=
+github.com/sagernet/sing-quic v0.4.0-alpha.4/go.mod h1:h5RkKTmUhudJKzK7c87FPXD5w1bJjVyxMN9+opZcctA=
+github.com/sagernet/sing-shadowsocks v0.2.7 h1:zaopR1tbHEw5Nk6FAkM05wCslV6ahVegEZaKMv9ipx8=
+github.com/sagernet/sing-shadowsocks v0.2.7/go.mod h1:0rIKJZBR65Qi0zwdKezt4s57y/Tl1ofkaq6NlkzVuyE=
+github.com/sagernet/sing-shadowsocks2 v0.2.0 h1:wpZNs6wKnR7mh1wV9OHwOyUr21VkS3wKFHi+8XwgADg=
+github.com/sagernet/sing-shadowsocks2 v0.2.0/go.mod h1:RnXS0lExcDAovvDeniJ4IKa2IuChrdipolPYWBv9hWQ=
+github.com/sagernet/sing-shadowtls v0.2.0-alpha.2 h1:RPrpgAdkP5td0vLfS5ldvYosFjSsZtRPxiyLV6jyKg0=
+github.com/sagernet/sing-shadowtls v0.2.0-alpha.2/go.mod h1:0j5XlzKxaWRIEjc1uiSKmVoWb0k+L9QgZVb876+thZA=
+github.com/sagernet/sing-tun v0.6.0-beta.2 h1:GK7r2jWKm7RhlJGTq4QadgFcebQia1c3BO3OlYMcQJ0=
+github.com/sagernet/sing-tun v0.6.0-beta.2/go.mod h1:fisFCbC4Vfb6HqQNcwPJi2CDK2bf0Xapyz3j3t4cnHE=
+github.com/sagernet/sing-tun v0.6.0-beta.6 h1:xaIHoH78MqTSvZqQ4SQto8pC1A+X4qXReDRNaC8DQeI=
+github.com/sagernet/sing-tun v0.6.0-beta.6/go.mod h1:fisFCbC4Vfb6HqQNcwPJi2CDK2bf0Xapyz3j3t4cnHE=
+github.com/sagernet/sing-vmess v0.2.0-beta.1 h1:5sXQ23uwNlZuDvygzi0dFtnG0Csm/SNqTjAHXJkpuj4=
+github.com/sagernet/sing-vmess v0.2.0-beta.1/go.mod h1:fLyE1emIcvQ5DV8reFWnufquZ7MkCSYM5ThodsR9NrQ=
+github.com/sagernet/smux v0.0.0-20231208180855-7041f6ea79e7 h1:DImB4lELfQhplLTxeq2z31Fpv8CQqqrUwTbrIRumZqQ=
+github.com/sagernet/smux v0.0.0-20231208180855-7041f6ea79e7/go.mod h1:FP9X2xjT/Az1EsG/orYYoC+5MojWnuI7hrffz8fGwwo=
+github.com/sagernet/utls v1.6.7 h1:Ep3+aJ8FUGGta+II2IEVNUc3EDhaRCZINWkj/LloIA8=
+github.com/sagernet/utls v1.6.7/go.mod h1:Uua1TKO/FFuAhLr9rkaVnnrTmmiItzDjv1BUb2+ERwM=
+github.com/sagernet/wireguard-go v0.0.1-beta.4 h1:8uyM5fxfEXdu4RH05uOK+v25i3lTNdCYMPSAUJ14FnI=
+github.com/sagernet/wireguard-go v0.0.1-beta.4/go.mod h1:jGXij2Gn2wbrWuYNUmmNhf1dwcZtvyAvQoe8Xd8MbUo=
+github.com/sagernet/wireguard-go v0.0.1-beta.5 h1:aBEsxJUMEONwOZqKPIkuAcv4zJV5p6XlzEN04CF0FXc=
+github.com/sagernet/wireguard-go v0.0.1-beta.5/go.mod h1:jGXij2Gn2wbrWuYNUmmNhf1dwcZtvyAvQoe8Xd8MbUo=
+github.com/sagernet/ws v0.0.0-20231204124109-acfe8907c854 h1:6uUiZcDRnZSAegryaUGwPC/Fj13JSHwiTftrXhMmYOc=
+github.com/sagernet/ws v0.0.0-20231204124109-acfe8907c854/go.mod h1:LtfoSK3+NG57tvnVEHgcuBW9ujgE8enPSgzgwStwCAA=
github.com/shirou/gopsutil/v3 v3.24.5 h1:i0t8kL+kQTvpAYToeuiVk3TgDeKOFioZO3Ztz/iZ9pI=
github.com/shirou/gopsutil/v3 v3.24.5/go.mod h1:bsoOS1aStSs9ErQ1WWfxllSeS1K5D+U30r2NfcubMVk=
github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFtM=
@@ -172,6 +214,7 @@ github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
+github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
@@ -186,53 +229,65 @@ github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE=
github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg=
-github.com/v2fly/BrowserBridge v0.0.0-20210430233438-0570fc1d7d08 h1:4Yh46CVE3k/lPq6hUbEdbB1u1anRBXLewm3k+L0iOMc=
-github.com/v2fly/BrowserBridge v0.0.0-20210430233438-0570fc1d7d08/go.mod h1:KAuQNm+LWQCOFqdBcUgihPzRpVXRKzGbTNhfEfRZ4wY=
-github.com/v2fly/VSign v0.0.0-20201108000810-e2adc24bf848 h1:p1UzXK6VAutXFFQMnre66h7g1BjRKUnLv0HfmmRoz7w=
-github.com/v2fly/VSign v0.0.0-20201108000810-e2adc24bf848/go.mod h1:p80Bv154ZtrGpXMN15slDCqc9UGmfBuUzheDFBYaW/M=
-github.com/v2fly/ss-bloomring v0.0.0-20210312155135-28617310f63e h1:5QefA066A1tF8gHIiADmOVOV5LS43gt3ONnlEl3xkwI=
-github.com/v2fly/ss-bloomring v0.0.0-20210312155135-28617310f63e/go.mod h1:5t19P9LBIrNamL6AcMQOncg/r10y3Pc01AbHeMhwlpU=
-github.com/v2fly/v2ray-core/v5 v5.17.1 h1:IIMMtmRdaG5HTYNn6VX1xKULknJl7nhkSFnmoTb5TDQ=
-github.com/v2fly/v2ray-core/v5 v5.17.1/go.mod h1:IhDN0rhXJnNcs9jUuC5sILTGCT2L+4yr0+tfD8ZVuL8=
-github.com/vincent-petithory/dataurl v1.0.0 h1:cXw+kPto8NLuJtlMsI152irrVw9fRDX8AbShPRpg2CI=
-github.com/vincent-petithory/dataurl v1.0.0/go.mod h1:FHafX5vmDzyP+1CQATJn7WFKc9CvnvxyvZy6I1MrG/U=
-github.com/xiaokangwang/VLite v0.0.0-20220418190619-cff95160a432 h1:I/ATawgO2RerCq9ACwL0wBB8xNXZdE3J+93MCEHReRs=
-github.com/xiaokangwang/VLite v0.0.0-20220418190619-cff95160a432/go.mod h1:QN7Go2ftTVfx0aCTh9RXHV8pkpi0FtmbwQw40dy61wQ=
-github.com/xtaci/smux v1.5.24 h1:77emW9dtnOxxOQ5ltR+8BbsX1kzcOxQ5gB+aaV9hXOY=
-github.com/xtaci/smux v1.5.24/go.mod h1:OMlQbT5vcgl2gb49mFkYo6SMf+zP3rcjcwQz7ZU7IGY=
+github.com/vishvananda/netns v0.0.4 h1:Oeaw1EM2JMxD51g9uhtC0D7erkIjgmj8+JZc26m1YX8=
+github.com/vishvananda/netns v0.0.4/go.mod h1:SpkAiCQRtJ6TvvxPnOSyH3BMl6unz3xZlaprSwhNNJM=
github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0=
github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0=
-go.starlark.net v0.0.0-20230612165344-9532f5667272 h1:2/wtqS591wZyD2OsClsVBKRPEvBsQt/Js+fsCiYhwu8=
-go.starlark.net v0.0.0-20230612165344-9532f5667272/go.mod h1:jxU+3+j+71eXOW14274+SmmuW82qJzl6iZSeqEtTGds=
-go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU=
-go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc=
+github.com/zeebo/assert v1.1.0 h1:hU1L1vLTHsnO8x8c9KAR5GmM5QscxHg5RNU5z5qbUWY=
+github.com/zeebo/assert v1.1.0/go.mod h1:Pq9JiuJQpG8JLJdtkwrJESF0Foym2/D9XMU5ciN/wJ0=
+github.com/zeebo/blake3 v0.2.3 h1:TFoLXsjeXqRNFxSbk35Dk4YtszE/MQQGK10BH4ptoTg=
+github.com/zeebo/blake3 v0.2.3/go.mod h1:mjJjZpnsyIVtVgTOSpJ9vmRE4wgDeyt2HU3qXvvKCaQ=
+github.com/zeebo/pcg v1.0.1 h1:lyqfGeWiv4ahac6ttHs+I5hwtH/+1mrhlCtVNQM2kHo=
+github.com/zeebo/pcg v1.0.1/go.mod h1:09F0S9iiKrwn9rlI5yjLkmrug154/YRW6KnnXVDM/l4=
+go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
+go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
+go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
+go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
+go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8=
+go.uber.org/zap v1.27.0/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E=
go4.org/netipx v0.0.0-20231129151722-fdeea329fbba h1:0b9z3AuHCjxk0x/opv64kcgZLBseWJUpBw5I82+2U4M=
go4.org/netipx v0.0.0-20231129151722-fdeea329fbba/go.mod h1:PLyyIXexvUFg3Owu6p/WfdlivPbZJsZdgWZlrGope/Y=
golang.org/x/arch v0.11.0 h1:KXV8WWKCXm6tRpLirl2szsO5j/oOODwZf4hATmGVNs4=
golang.org/x/arch v0.11.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys=
-golang.org/x/crypto v0.28.0 h1:GBDwsMXVQi34v5CCYUm2jkJvu4cbtru2U4TN2PSyQnw=
-golang.org/x/crypto v0.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U=
+golang.org/x/crypto v0.0.0-20210513164829-c07d793c2f9a/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8=
+golang.org/x/crypto v0.29.0 h1:L5SG1JTTXupVV3n6sUqMTeWbjAyfPwoda2DLX8J8FrQ=
+golang.org/x/crypto v0.29.0/go.mod h1:+F4F4N5hv6v38hfeYwTdx20oUvLLc+QfrE9Ax9HtgRg=
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 h1:2dVuKD2vS7b0QIHQbpyTISPd0LeHDbnYEryqj5Q1ug8=
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56/go.mod h1:M4RDyNAINzryxdtnbRXRL/OHtkFuWGRjvuhBJpk2IlY=
-golang.org/x/mod v0.19.0 h1:fEdghXQSo20giMthA7cd28ZC+jts4amQ3YMXiP5oMQ8=
-golang.org/x/mod v0.19.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
-golang.org/x/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4=
-golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU=
-golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ=
-golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
+golang.org/x/mod v0.20.0 h1:utOm6MM3R3dnawAiJgn0y+xvuYRsm1RKM/4giyfDgV0=
+golang.org/x/mod v0.20.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
+golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
+golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
+golang.org/x/net v0.31.0 h1:68CPQngjLL0r2AlUKiSxtQFKvzRVbnzLwMUn5SzcLHo=
+golang.org/x/net v0.31.0/go.mod h1:P4fl1q7dY2hnZFxEk4pPSkDHF+QqjitcnDjUQyMM+pM=
+golang.org/x/sync v0.9.0 h1:fEo0HyrW1GIgZdpbhCRO0PkJajUS5H9IFUztCgEo2jQ=
+golang.org/x/sync v0.9.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo=
-golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
-golang.org/x/text v0.19.0 h1:kTxAhCbGbxhK0IwgSKiMO5awPoDQ0RpfiVYBfK860YM=
-golang.org/x/text v0.19.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY=
-golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk=
-golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM=
-golang.org/x/tools v0.23.0 h1:SGsXPZ+2l4JsgaCKkx+FQ9YZ5XEtA1GZYuoDjenLjvg=
-golang.org/x/tools v0.23.0/go.mod h1:pnu6ufv6vQkll6szChhK3C3L/ruaIv5eBeztNG8wtsI=
+golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
+golang.org/x/sys v0.27.0 h1:wBqf8DvsY9Y/2P8gAfPDEYNuS30J4lPHJxXSb/nJZ+s=
+golang.org/x/sys v0.27.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
+golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
+golang.org/x/term v0.26.0 h1:WEQa6V3Gja/BhNxg540hBip/kkaYtRg3cxg4oXSw4AU=
+golang.org/x/term v0.26.0/go.mod h1:Si5m1o57C5nBNQo5z1iq+XDijt21BDBDp2bK0QI8e3E=
+golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/text v0.20.0 h1:gK/Kv2otX8gz+wn7Rmb3vT96ZwuoxnQlY+HlJVj7Qug=
+golang.org/x/text v0.20.0/go.mod h1:D4IsuqiFMhST5bX19pQ9ikHC2GsaKyk/oF+pn3ducp4=
+golang.org/x/time v0.7.0 h1:ntUhktv3OPE6TgYxXWv9vKvUSJyIFJlyohwbkEwPrKQ=
+golang.org/x/time v0.7.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM=
+golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.24.0 h1:J1shsA93PJUEVaUSaay7UXAyE8aimq3GW0pjlolpa24=
+golang.org/x/tools v0.24.0/go.mod h1:YhNqVBIfWHdzvTLs0d8LCuMhkKUgSUKldakyV7W/WDQ=
+golang.zx2c4.com/wintun v0.0.0-20230126152724-0fa3db229ce2 h1:B82qJJgjvYKsXS9jeunTOisW56dUokqW/FOteYJJ/yg=
+golang.zx2c4.com/wintun v0.0.0-20230126152724-0fa3db229ce2/go.mod h1:deeaetjYA+DHMHg+sMSMI58GrEteJUUzzw7en6TJQcI=
+golang.zx2c4.com/wireguard/wgctrl v0.0.0-20230429144221-925a1e7659e6 h1:CawjfCvYQH2OU3/TnxLx97WDSUDRABfT18pCOYwc2GE=
+golang.zx2c4.com/wireguard/wgctrl v0.0.0-20230429144221-925a1e7659e6/go.mod h1:3rxYc4HtVcSG9gVaTs2GEBdehh+sYPOwKtyUWEOTb80=
google.golang.org/genproto/googleapis/rpc v0.0.0-20241015192408-796eee8c2d53 h1:X58yt85/IXCx0Y3ZwN6sEIKZzQtDEYaBWrDvErdXrRE=
google.golang.org/genproto/googleapis/rpc v0.0.0-20241015192408-796eee8c2d53/go.mod h1:GX3210XPVPUjJbTUbvwI8f2IpZDMZuPJWDzDuebbviI=
google.golang.org/grpc v1.67.1 h1:zWnc1Vrcno+lHZCOofnIMvycFcc0QRGIzm9dhnDX68E=
@@ -249,8 +304,6 @@ gorm.io/driver/sqlite v1.5.6 h1:fO/X46qn5NUEEOZtnjJRWRzZMe8nqJiQ9E+0hi+hKQE=
gorm.io/driver/sqlite v1.5.6/go.mod h1:U+J8craQU6Fzkcvu8oLeAQmi50TkwPEhHDEjQZXDah4=
gorm.io/gorm v1.25.12 h1:I0u8i2hWQItBq1WfE0o2+WuL9+8L21K9e2HHSTE/0f8=
gorm.io/gorm v1.25.12/go.mod h1:xh7N7RHfYlNc5EmcI/El95gXusucDrQnHXe0+CgWcLQ=
-gvisor.dev/gvisor v0.0.0-20231020174304-b8a429915ff1 h1:qDCwdCWECGnwQSQC01Dpnp09fRHxJs9PbktotUqG+hs=
-gvisor.dev/gvisor v0.0.0-20231020174304-b8a429915ff1/go.mod h1:8hmigyCdYtw5xJGfQDJzSH5Ju8XEIDBnpyi8+O6GRt8=
lukechampine.com/blake3 v1.3.0 h1:sJ3XhFINmHSrYCgl958hscfIa3bw8x4DqMP3u1YvoYE=
lukechampine.com/blake3 v1.3.0/go.mod h1:0OFRp7fBtAylGVCO40o87sbupkyIGgbpv1+M1k1LM6k=
nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50=
diff --git a/backend/logger/logger.go b/backend/logger/logger.go
index a9d7a0c..6e30b4b 100644
--- a/backend/logger/logger.go
+++ b/backend/logger/logger.go
@@ -43,6 +43,10 @@ func InitLogger(level logging.Level) {
logger = newLogger
}
+func GetLogger() *logging.Logger {
+ return logger
+}
+
func Debug(args ...interface{}) {
logger.Debug(args...)
addToBuffer("DEBUG", fmt.Sprint(args...))
diff --git a/backend/service/config.go b/backend/service/config.go
index 91daab0..e816f77 100644
--- a/backend/service/config.go
+++ b/backend/service/config.go
@@ -4,23 +4,24 @@ import (
"encoding/json"
"os"
"s-ui/config"
+ "s-ui/core"
"s-ui/database"
"s-ui/database/model"
"s-ui/logger"
- "s-ui/singbox"
"strconv"
"time"
)
-var ApiAddr string
-var LastUpdate int64
-var IsSystemd bool
+var (
+ LastUpdate int64
+ IsSystemd bool
+ corePtr *core.Core
+)
type ConfigService struct {
ClientService
TlsService
InDataService
- singbox.Controller
SettingService
}
@@ -34,7 +35,8 @@ type SingBoxConfig struct {
Experimental json.RawMessage `json:"experimental"`
}
-func NewConfigService() *ConfigService {
+func NewConfigService(core *core.Core) *ConfigService {
+ corePtr = core
return &ConfigService{}
}
@@ -64,7 +66,7 @@ func (s *ConfigService) InitConfig() error {
return err
}
- return s.RefreshApiAddr(&singboxConfig)
+ return nil
}
func (s *ConfigService) GetConfig() (*SingBoxConfig, error) {
@@ -149,6 +151,7 @@ func (s *ConfigService) SaveChanges(changes map[string]string, loginUser string)
return err
}
}
+ needRestart := false
if len(configChanges) > 0 {
singboxConfig, err := s.GetConfig()
if err != nil {
@@ -163,36 +166,114 @@ func (s *ConfigService) SaveChanges(changes map[string]string, loginUser string)
if err != nil {
return err
}
+ needRestart = true
case "log":
newConfig.Log = rawObject
+ needRestart = true
case "dns":
newConfig.Dns = rawObject
+ needRestart = true
case "ntp":
newConfig.Ntp = rawObject
+ needRestart = true
case "route":
newConfig.Route = rawObject
+ needRestart = true
case "experimental":
newConfig.Experimental = rawObject
+ needRestart = true
case "inbounds":
if change.Action == "edit" {
+ var object map[string]interface{}
+ err = json.Unmarshal(newConfig.Inbounds[change.Index], &object)
+ if err != nil {
+ return err
+ }
+ if tag, ok := object["tag"].(string); ok {
+ err = corePtr.RemoveInbound(tag)
+ if err == nil {
+ err = corePtr.AddInbound(rawObject)
+ if err != nil {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
newConfig.Inbounds[change.Index] = rawObject
} else if change.Action == "del" {
+ var object map[string]interface{}
+ err = json.Unmarshal(newConfig.Inbounds[change.Index], &object)
+ if err != nil {
+ return err
+ }
+ if tag, ok := object["tag"].(string); ok {
+ err = corePtr.RemoveInbound(tag)
+ if err != nil {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
newConfig.Inbounds = append(newConfig.Inbounds[:change.Index], newConfig.Inbounds[change.Index+1:]...)
} else {
newConfig.Inbounds = append(newConfig.Inbounds, rawObject)
+ err = corePtr.AddInbound(rawObject)
+ if err != nil {
+ logger.Debug(err)
+ needRestart = true
+ }
}
case "outbounds":
if change.Action == "edit" {
+ var object map[string]interface{}
+ err = json.Unmarshal(newConfig.Outbounds[change.Index], &object)
+ if err != nil {
+ return err
+ }
+ if tag, ok := object["tag"].(string); ok {
+ err = corePtr.RemoveOutbound(tag)
+ if err == nil {
+ err = corePtr.AddOutbound(rawObject)
+ if err != nil {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
newConfig.Outbounds[change.Index] = rawObject
} else if change.Action == "del" {
+ var object map[string]interface{}
+ err = json.Unmarshal(newConfig.Outbounds[change.Index], &object)
+ if err != nil {
+ return err
+ }
+ if tag, ok := object["tag"].(string); ok {
+ err = corePtr.RemoveOutbound(tag)
+ if err != nil {
+ needRestart = true
+ }
+ } else {
+ needRestart = true
+ }
newConfig.Outbounds = append(newConfig.Outbounds[:change.Index], newConfig.Outbounds[change.Index+1:]...)
} else {
+ err = corePtr.AddOutbound(rawObject)
+ if err != nil {
+ logger.Debug(err)
+ needRestart = true
+ }
newConfig.Outbounds = append(newConfig.Outbounds, rawObject)
}
}
}
- err = s.Save(&newConfig)
+ err = s.Save(&newConfig, needRestart)
if err != nil {
return err
}
@@ -238,7 +319,7 @@ func (s *ConfigService) CheckChanges(lu string) (bool, error) {
}
}
-func (s *ConfigService) Save(singboxConfig *SingBoxConfig) error {
+func (s *ConfigService) Save(singboxConfig *SingBoxConfig, needRestart bool) error {
configPath := config.GetBinFolderPath()
_, err := os.Stat(configPath + "/config.json")
if os.IsNotExist(err) {
@@ -260,39 +341,14 @@ func (s *ConfigService) Save(singboxConfig *SingBoxConfig) error {
return err
}
- s.RefreshApiAddr(singboxConfig)
- s.Controller.Restart()
-
- return nil
-}
-
-func (s *ConfigService) RefreshApiAddr(singboxConfig *SingBoxConfig) error {
- Env_API := config.GetEnvApi()
- if len(Env_API) > 0 {
- ApiAddr = Env_API
- } else {
- var err error
- if singboxConfig == nil {
- singboxConfig, err = s.GetConfig()
- if err != nil {
- return err
- }
-
- }
-
- var experimental struct {
- V2rayApi struct {
- Listen string `json:"listen"`
- Stats interface{} `jaon:"stats"`
- } `json:"v2ray_api"`
- }
- err = json.Unmarshal(singboxConfig.Experimental, &experimental)
+ if needRestart {
+ err = corePtr.Restart()
if err != nil {
return err
}
-
- ApiAddr = experimental.V2rayApi.Listen
}
+ // s.Controller.Restart()
+
return nil
}
@@ -348,7 +404,7 @@ func (s *ConfigService) DepleteClients() error {
singboxConfig.Inbounds[inbound_index] = modifiedInbound
}
- err = s.Save(singboxConfig)
+ err = s.Save(singboxConfig, true)
if err != nil {
return err
}
@@ -381,3 +437,7 @@ func (s *ConfigService) GetChanges(actor string, chngKey string, count string) [
}
return chngs
}
+
+func (s *ConfigService) RestartCore() error {
+ return corePtr.Restart()
+}
diff --git a/backend/service/server.go b/backend/service/server.go
index 33bc5a0..8ba1459 100644
--- a/backend/service/server.go
+++ b/backend/service/server.go
@@ -1,24 +1,24 @@
package service
import (
- "bytes"
+ "encoding/base64"
"os"
- "os/exec"
"runtime"
"s-ui/config"
"s-ui/logger"
"strconv"
"strings"
+ "time"
+ "github.com/sagernet/sing-box/common/tls"
"github.com/shirou/gopsutil/v3/cpu"
"github.com/shirou/gopsutil/v3/host"
"github.com/shirou/gopsutil/v3/mem"
"github.com/shirou/gopsutil/v3/net"
+ "golang.zx2c4.com/wireguard/wgctrl/wgtypes"
)
-type ServerService struct {
- SingBoxService
-}
+type ServerService struct{}
func (s *ServerService) GetStatus(request string) *map[string]interface{} {
status := make(map[string]interface{}, 0)
@@ -91,15 +91,21 @@ func (s *ServerService) GetNetInfo() map[string]interface{} {
}
func (s *ServerService) GetSingboxInfo() map[string]interface{} {
- info := make(map[string]interface{}, 0)
- sysStats, err := s.SingBoxService.GetSysStats()
- if err == nil {
- info["running"] = true
- info["stats"] = sysStats
- } else {
- info["running"] = s.SingBoxService.IsRunning()
+ var rtm runtime.MemStats
+ runtime.ReadMemStats(&rtm)
+ isRunning := corePtr.IsRunning()
+ uptime := uint32(0)
+ if isRunning {
+ uptime = corePtr.GetInstance().Uptime()
+ }
+ return map[string]interface{}{
+ "running": isRunning,
+ "stats": map[string]interface{}{
+ "NumGoroutine": uint32(runtime.NumGoroutine()),
+ "Alloc": rtm.Alloc,
+ "Uptime": uptime,
+ },
}
- return info
}
func (s *ServerService) GetSystemInfo() map[string]interface{} {
@@ -139,48 +145,50 @@ func (s *ServerService) GetSystemInfo() map[string]interface{} {
return info
}
-func (s *ServerService) GetLogs(service string, count string, level string) []string {
+func (s *ServerService) GetLogs(count string, level string) []string {
c, _ := strconv.Atoi(count)
-
- if service == "s-ui" {
- return logger.GetLogs(c, level)
- }
- var lines []string
- var cmdArgs []string
- if IsSystemd {
- cmdArgs = []string{"journalctl", "-u", service, "--no-pager", "-n", count, "-p", level}
- } else {
- cmdArgs = []string{"tail", "/logs/" + service + ".log", "-n", count}
- }
- // Run the command
- cmd := exec.Command(cmdArgs[0], cmdArgs[1:]...)
- var out bytes.Buffer
- cmd.Stdout = &out
- err := cmd.Run()
- if err != nil {
- return []string{"Failed to get logs!", err.Error()}
- }
- lines = strings.Split(out.String(), "\n")
-
- return lines
+ return logger.GetLogs(c, level)
}
func (s *ServerService) GenKeypair(keyType string, options string) []string {
if len(keyType) == 0 {
return []string{"No keypair to generate"}
}
- sbExec := s.GetBinaryPath()
- cmdArgs := []string{"generate", keyType + "-keypair"}
- if keyType == "tls" || keyType == "ech" {
- cmdArgs = append(cmdArgs, options)
+
+ switch keyType {
+ case "ech":
+ return s.generateECHKeyPair(options)
+ case "tls":
+ return s.generateTLSKeyPair(options)
+ case "reality":
+ return s.generateRealityKeyPair()
}
- // Run the command
- cmd := exec.Command(sbExec, cmdArgs...)
- var out bytes.Buffer
- cmd.Stdout = &out
- err := cmd.Run()
- if err != nil {
- return []string{"Failed to generate keypair"}
- }
- return strings.Split(out.String(), "\n")
+
+ return []string{"Failed to generate keypair"}
+}
+
+func (s *ServerService) generateECHKeyPair(options string) []string {
+ parts := strings.Split(options, ",")
+ configPem, keyPem, err := tls.ECHKeygenDefault(parts[0], parts[1] == "true")
+ if err != nil {
+ return []string{"Failed to generate ECH keypair: ", err.Error()}
+ }
+ return append(strings.Split(configPem, "\n"), strings.Split(keyPem, "\n")...)
+}
+
+func (s *ServerService) generateTLSKeyPair(serverName string) []string {
+ privateKeyPem, publicKeyPem, err := tls.GenerateKeyPair(time.Now, serverName, time.Now().AddDate(0, 12, 0))
+ if err != nil {
+ return []string{"Failed to generate TLS keypair: ", err.Error()}
+ }
+ return append(strings.Split(string(privateKeyPem), "\n"), strings.Split(string(publicKeyPem), "\n")...)
+}
+
+func (s *ServerService) generateRealityKeyPair() []string {
+ privateKey, err := wgtypes.GeneratePrivateKey()
+ if err != nil {
+ return []string{"Failed to generate Reality keypair: ", err.Error()}
+ }
+ publicKey := privateKey.PublicKey()
+ return []string{"PrivateKey: " + base64.RawURLEncoding.EncodeToString(privateKey[:]), "PublicKey: " + base64.RawURLEncoding.EncodeToString(publicKey[:])}
}
diff --git a/backend/service/sinxbox.go b/backend/service/sinxbox.go
deleted file mode 100644
index ce64a4f..0000000
--- a/backend/service/sinxbox.go
+++ /dev/null
@@ -1,45 +0,0 @@
-package service
-
-import (
- "s-ui/singbox"
-)
-
-type SingBoxService struct {
- singbox.V2rayAPI
- singbox.Controller
- StatsService
-}
-
-func (s *SingBoxService) GetStats() error {
- s.V2rayAPI.Init(ApiAddr)
- defer s.V2rayAPI.Close()
- stats, err := s.V2rayAPI.GetStats(true)
- if err != nil {
- return err
- }
- err = s.StatsService.SaveStats(stats)
- if err != nil {
- return err
- }
-
- return nil
-}
-
-func (s *SingBoxService) GetSysStats() (*map[string]interface{}, error) {
- err := s.V2rayAPI.Init(ApiAddr)
- if err != nil {
- return nil, err
- }
- defer s.V2rayAPI.Close()
- resp, err := s.V2rayAPI.GetSysStats()
- if err != nil {
- return nil, err
- }
-
- result := make(map[string]interface{})
- result["NumGoroutine"] = resp.NumGoroutine
- result["Alloc"] = resp.Alloc
- result["Uptime"] = resp.Uptime
-
- return &result, nil
-}
diff --git a/backend/service/stats.go b/backend/service/stats.go
index fd7bf36..2ff547c 100644
--- a/backend/service/stats.go
+++ b/backend/service/stats.go
@@ -19,18 +19,22 @@ var onlineResources = &onlines{}
type StatsService struct {
}
-func (s *StatsService) SaveStats(stats []*model.Stats) error {
- var err error
+func (s *StatsService) SaveStats() error {
+ if !corePtr.IsRunning() {
+ return nil
+ }
+ stats := corePtr.GetInstance().ConnTracker().GetStats()
// Reset onlines
onlineResources.Inbound = nil
onlineResources.Outbound = nil
onlineResources.User = nil
- if len(stats) == 0 {
+ if len(*stats) == 0 {
return nil
}
+ var err error
db := database.GetDB()
tx := db.Begin()
defer func() {
@@ -41,7 +45,7 @@ func (s *StatsService) SaveStats(stats []*model.Stats) error {
}
}()
- for _, stat := range stats {
+ for _, stat := range *stats {
if stat.Resource == "user" {
if stat.Direction {
err = tx.Model(model.Client{}).Where("name = ?", stat.Tag).
diff --git a/backend/singbox/controller.go b/backend/singbox/controller.go
deleted file mode 100644
index d5ca5b3..0000000
--- a/backend/singbox/controller.go
+++ /dev/null
@@ -1,54 +0,0 @@
-package singbox
-
-import (
- "errors"
- "io/fs"
- "os"
- "os/exec"
- "s-ui/config"
- "strings"
-)
-
-var serviceName = "sing-box"
-
-type Controller struct {
-}
-
-func (s *Controller) GetBinaryName() string {
- return "sing-box"
-}
-
-func (s *Controller) GetBinaryPath() string {
- return config.GetBinFolderPath() + "/" + s.GetBinaryName()
-}
-
-func (s *Controller) GetConfigPath() string {
- return config.GetBinFolderPath() + "/config.json"
-}
-
-func (s *Controller) IsRunning() bool {
- cmd := exec.Command("pgrep", "sing-box")
- output, err := cmd.Output()
- if err != nil {
- return false
- }
-
- // If pgrep found the Controller, its output will not be empty
- return strings.TrimSpace(string(output)) != ""
-}
-
-func (s *Controller) signalSingbox(signal string) error {
- return os.WriteFile(config.GetBinFolderPath()+"/signal", []byte(signal), fs.ModePerm)
-}
-
-func (s *Controller) Restart() error {
- return s.signalSingbox("restart")
-}
-
-func (s *Controller) Stop() error {
- if !s.IsRunning() {
- return errors.New("Sing-Box is not running")
- }
-
- return s.signalSingbox("stop")
-}
diff --git a/backend/singbox/v2rayApi.go b/backend/singbox/v2rayApi.go
deleted file mode 100644
index 571edf9..0000000
--- a/backend/singbox/v2rayApi.go
+++ /dev/null
@@ -1,96 +0,0 @@
-package singbox
-
-import (
- "context"
-
- "regexp"
- "s-ui/database/model"
- "s-ui/util/common"
- "time"
-
- statsService "github.com/v2fly/v2ray-core/v5/app/stats/command"
- "google.golang.org/grpc"
- "google.golang.org/grpc/credentials/insecure"
-)
-
-type V2rayAPI struct {
- StatsServiceClient *statsService.StatsServiceClient
- grpcClient *grpc.ClientConn
- isConnected bool
-}
-
-func (v *V2rayAPI) Init(ApiAddr string) (err error) {
- if len(ApiAddr) == 0 {
- return common.NewError("The api address is wrong: ", ApiAddr)
- }
- v.grpcClient, err = grpc.NewClient(ApiAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
- if err != nil {
- return err
- }
- v.isConnected = true
-
- ssClient := statsService.NewStatsServiceClient(v.grpcClient)
-
- v.StatsServiceClient = &ssClient
-
- return
-}
-
-func (v *V2rayAPI) Close() {
- v.grpcClient.Close()
- v.StatsServiceClient = nil
- v.isConnected = false
-}
-
-func (v *V2rayAPI) GetStats(reset bool) ([]*model.Stats, error) {
- if v.grpcClient == nil {
- return nil, common.NewError("v2ray api is not initialized")
- }
- var trafficRegex = regexp.MustCompile("(inbound|outbound|user)>>>([^>]+)>>>traffic>>>(downlink|uplink)")
-
- client := *v.StatsServiceClient
- ctx, cancel := context.WithTimeout(context.Background(), time.Second*10)
- defer cancel()
- request := &statsService.QueryStatsRequest{
- Reset_: reset,
- }
- resp, err := client.QueryStats(ctx, request)
- if err != nil {
- return nil, err
- }
-
- dt := time.Now().Unix()
- stats := make([]*model.Stats, 0)
- for _, stat := range resp.GetStat() {
- if stat.Value > 0 {
- matchs := trafficRegex.FindStringSubmatch(stat.Name)
- if len(matchs) > 3 {
- stat := model.Stats{
- DateTime: dt,
- Resource: matchs[1],
- Tag: matchs[2],
- Direction: matchs[3] == "uplink",
- Traffic: stat.Value,
- }
- stats = append(stats, &stat)
- }
- }
- }
-
- return stats, nil
-}
-
-func (v *V2rayAPI) GetSysStats() (*statsService.SysStatsResponse, error) {
- if v.grpcClient == nil {
- return nil, common.NewError("v2ray api is not initialized")
- }
- client := *v.StatsServiceClient
- ctx, cancel := context.WithTimeout(context.Background(), time.Second*10)
- defer cancel()
- request := &statsService.SysStatsRequest{}
- resp, err := client.GetSysStats(ctx, request)
- if err != nil {
- return nil, err
- }
- return resp, nil
-}
diff --git a/frontend/src/components/Main.vue b/frontend/src/components/Main.vue
index 583d480..9fa97c0 100644
--- a/frontend/src/components/Main.vue
+++ b/frontend/src/components/Main.vue
@@ -2,7 +2,6 @@
@@ -93,7 +92,7 @@
v{{ tilesData.sys?.appVersion }}
-
+
{{ $t('basic.log.title') + " - S-UI" }}
@@ -110,12 +109,6 @@
{{ $t('yes') }}
{{ $t('no') }}
-
-
- {{ $t('basic.log.title') + " - Sing-Box" }}
-
-
-
{{ $t('actions.restartSb') }}
@@ -244,16 +237,13 @@ onBeforeUnmount(() => {
const logModal = ref({
visible: false,
- logType: "s-ui"
})
-const openLogs = (logType: string) => {
- logModal.value.logType = logType
+const openLogs = () => {
logModal.value.visible = true
}
const closeLogs = () => {
- logModal.value.logType = "s-ui"
logModal.value.visible = false
}
diff --git a/frontend/src/components/tls/Ech.vue b/frontend/src/components/tls/Ech.vue
index 16255fa..3485417 100644
--- a/frontend/src/components/tls/Ech.vue
+++ b/frontend/src/components/tls/Ech.vue
@@ -93,7 +93,10 @@ export default {
methods: {
async genECH(){
this.loading = true
- const msg = await HttpUtils.get('api/keypairs', { k: "ech", o: this.iTls.server_name?? "''" })
+ const msg = await HttpUtils.get('api/keypairs', {
+ k: "ech",
+ o: this.iTls.server_name?? "''" + "," + this.iTls.ech.pq_signature_schemes_enabled?? false
+ })
this.loading = false
if (msg.success && this.iTls.ech && this.oTls.ech) {
this.iTls.ech.key_path=undefined
diff --git a/frontend/src/layouts/modals/Logs.vue b/frontend/src/layouts/modals/Logs.vue
index e2dc9ef..8497981 100644
--- a/frontend/src/layouts/modals/Logs.vue
+++ b/frontend/src/layouts/modals/Logs.vue
@@ -3,7 +3,7 @@
- {{ $t('basic.log.title') + " - " + (logType == 's-ui'? "S-UI" : "Sing-Box") }}
+ {{ $t('basic.log.title') }}
@@ -51,7 +51,7 @@
import HttpUtils from '@/plugins/httputil';
export default {
- props: ['logType', 'visible'],
+ props: ['visible'],
data() {
return {
loading: false,
@@ -69,7 +69,7 @@ export default {
methods: {
async loadData() {
this.loading = true
- const data = await HttpUtils.get('api/logs',{ s: this.$props.logType, c: this.logCount, l: this.logLevel })
+ const data = await HttpUtils.get('api/logs',{ c: this.logCount, l: this.logLevel })
if (data.success) {
this.lines = data.obj?? []
this.loading = false