package cmd import ( "fmt" "github.com/sirupsen/logrus" "github.com/spf13/cobra" "github.com/v2rayA/dae/cmd/internal" "github.com/v2rayA/dae/component/control" "github.com/v2rayA/dae/config" "github.com/v2rayA/dae/pkg/config_parser" "github.com/v2rayA/dae/pkg/logger" "os" "os/signal" "syscall" ) var ( cfgFile string runCmd = &cobra.Command{ Use: "run", Short: "Run dae in the foreground", Run: func(cmd *cobra.Command, args []string) { if cfgFile == "" { logrus.Fatalln("Argument \"--config\" or \"-c\" is required but not provided.") } if err := Run(); err != nil { logrus.Fatalln(err) } }, } ) func init() { runCmd.PersistentFlags().StringVarP(&cfgFile, "config", "c", "", "config file") } func Run() (err error) { logrus.SetLevel(logrus.DebugLevel) log := logger.NewLogger(2) // Read config from --config cfgFile. param, err := readConfig() if err != nil { return fmt.Errorf("readConfig: %w", err) } // Require "sudo" if necessary. internal.AutoSu() // Resolve subscriptions to nodes. nodeList := make([]string, len(param.Node)) copy(nodeList, param.Node) for _, sub := range param.Subscription { nodes, err := internal.ResolveSubscription(log, sub) if err != nil { log.Warnf(`failed to resolve subscription "%v": %v`, sub, err) } nodeList = append(nodeList, nodes...) } if len(param.Global.LanInterface) == 0 && len(param.Global.WanInterface) == 0 { return fmt.Errorf("LanInterface and WanInterface cannot both be empty") } // New ControlPlane. bindLan := len(param.Global.LanInterface) != 0 bindWan := len(param.Global.WanInterface) != 0 t, err := control.NewControlPlane( log, nodeList, param.Group, ¶m.Routing, param.Global.DnsUpstream, param.Global.CheckUrl, param.Global.CheckInterval, bindLan, bindWan, ) if err != nil { return err } // Bind to links. for _, ifname := range param.Global.LanInterface { if err = t.BindLan(ifname); err != nil { return fmt.Errorf("BindLan: %v: %w", ifname, err) } } for _, ifname := range param.Global.WanInterface { if err = t.BindWan(ifname); err != nil { return fmt.Errorf("BindWan: %v: %w", ifname, err) } } // Serve tproxy TCP/UDP server util signals. sigs := make(chan os.Signal, 1) signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM, syscall.SIGHUP, syscall.SIGQUIT, syscall.SIGKILL, syscall.SIGILL) go func() { if err := t.ListenAndServe(param.Global.TproxyPort); err != nil { log.Errorln("ListenAndServe:", err) sigs <- nil } }() <-sigs if e := t.Close(); e != nil { return fmt.Errorf("close control plane: %w", e) } return nil } func readConfig() (params *config.Params, err error) { b, err := os.ReadFile(cfgFile) if err != nil { return nil, err } sections, err := config_parser.Parse(string(b)) if err != nil { return nil, fmt.Errorf("\n%w", err) } if params, err = config.New(sections); err != nil { return nil, err } return params, nil }