123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- package sub
- import (
- "bytes"
- "context"
- "fmt"
- "io/ioutil"
- "net"
- "os"
- "os/signal"
- "path/filepath"
- "strconv"
- "strings"
- "syscall"
- "time"
- "github.com/fatedier/frp/client"
- "github.com/fatedier/frp/pkg/auth"
- "github.com/fatedier/frp/pkg/config"
- "github.com/fatedier/frp/pkg/util/log"
- "github.com/fatedier/frp/pkg/util/version"
- "github.com/spf13/cobra"
- )
- const (
- CfgFileTypeIni = iota
- CfgFileTypeCmd
- )
- var (
- cfgFile string
- showVersion bool
- serverAddr string
- user string
- protocol string
- token string
- logLevel string
- logFile string
- logMaxDays int
- disableLogColor bool
- proxyName string
- localIP string
- localPort int
- remotePort int
- useEncryption bool
- useCompression bool
- customDomains string
- subDomain string
- httpUser string
- httpPwd string
- locations string
- hostHeaderRewrite string
- role string
- sk string
- multiplexer string
- serverName string
- bindAddr string
- bindPort int
- tlsEnable bool
- kcpDoneCh chan struct{}
- )
- func init() {
- rootCmd.PersistentFlags().StringVarP(&cfgFile, "config", "c", "./frpc.ini", "config file of frpc")
- rootCmd.PersistentFlags().BoolVarP(&showVersion, "version", "v", false, "version of frpc")
- kcpDoneCh = make(chan struct{})
- }
- func RegisterCommonFlags(cmd *cobra.Command) {
- cmd.PersistentFlags().StringVarP(&serverAddr, "server_addr", "s", "127.0.0.1:7000", "frp server's address")
- cmd.PersistentFlags().StringVarP(&user, "user", "u", "", "user")
- cmd.PersistentFlags().StringVarP(&protocol, "protocol", "p", "tcp", "tcp or kcp or websocket")
- cmd.PersistentFlags().StringVarP(&token, "token", "t", "", "auth token")
- cmd.PersistentFlags().StringVarP(&logLevel, "log_level", "", "info", "log level")
- cmd.PersistentFlags().StringVarP(&logFile, "log_file", "", "console", "console or file path")
- cmd.PersistentFlags().IntVarP(&logMaxDays, "log_max_days", "", 3, "log file reversed days")
- cmd.PersistentFlags().BoolVarP(&disableLogColor, "disable_log_color", "", false, "disable log color in console")
- cmd.PersistentFlags().BoolVarP(&tlsEnable, "tls_enable", "", false, "enable frpc tls")
- }
- var rootCmd = &cobra.Command{
- Use: "frpc",
- Short: "frpc is the client of frp (https://github.com/fatedier/frp)",
- RunE: func(cmd *cobra.Command, args []string) error {
- if showVersion {
- fmt.Println(version.Full())
- return nil
- }
-
- err := runClient(cfgFile)
- if err != nil {
- fmt.Println(err)
- os.Exit(1)
- }
- return nil
- },
- }
- func Execute() {
- if err := rootCmd.Execute(); err != nil {
- os.Exit(1)
- }
- }
- func handleSignal(svr *client.Service) {
- ch := make(chan os.Signal)
- signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM)
- <-ch
- svr.Close()
- time.Sleep(250 * time.Millisecond)
- close(kcpDoneCh)
- }
- func parseClientCommonCfg(fileType int, source []byte) (cfg config.ClientCommonConf, err error) {
- if fileType == CfgFileTypeIni {
- cfg, err = config.UnmarshalClientConfFromIni(source)
- } else if fileType == CfgFileTypeCmd {
- cfg, err = parseClientCommonCfgFromCmd()
- }
- if err != nil {
- return
- }
- cfg.Complete()
- err = cfg.Validate()
- if err != nil {
- err = fmt.Errorf("Parse config error: %v", err)
- return
- }
- return
- }
- func parseClientCommonCfgFromCmd() (cfg config.ClientCommonConf, err error) {
- cfg = config.GetDefaultClientConf()
- ipStr, portStr, err := net.SplitHostPort(serverAddr)
- if err != nil {
- err = fmt.Errorf("invalid server_addr: %v", err)
- return
- }
- cfg.ServerAddr = ipStr
- cfg.ServerPort, err = strconv.Atoi(portStr)
- if err != nil {
- err = fmt.Errorf("invalid server_addr: %v", err)
- return
- }
- cfg.User = user
- cfg.Protocol = protocol
- cfg.LogLevel = logLevel
- cfg.LogFile = logFile
- cfg.LogMaxDays = int64(logMaxDays)
- cfg.DisableLogColor = disableLogColor
-
- cfg.ClientConfig = auth.GetDefaultClientConf()
- cfg.Token = token
- cfg.TLSEnable = tlsEnable
- return
- }
- func runClient(cfgFilePath string) error {
- cfg, pxyCfgs, visitorCfgs, err := parseConfig(cfgFilePath)
- if err != nil {
- return err
- }
- return startService(cfg, pxyCfgs, visitorCfgs, cfgFilePath)
- }
- func parseConfig(cfgFilePath string) (
- cfg config.ClientCommonConf,
- pxyCfgs map[string]config.ProxyConf,
- visitorCfgs map[string]config.VisitorConf,
- err error,
- ) {
- var content []byte
- content, err = config.GetRenderedConfFromFile(cfgFilePath)
- if err != nil {
- return
- }
- configBuffer := bytes.NewBuffer(nil)
- configBuffer.Write(content)
-
- cfg, err = parseClientCommonCfg(CfgFileTypeIni, content)
- if err != nil {
- return
- }
-
- var buf []byte
- buf, err = getIncludeContents(cfg.IncludeConfigFiles)
- if err != nil {
- err = fmt.Errorf("getIncludeContents error: %v", err)
- return
- }
- configBuffer.WriteString("\n")
- configBuffer.Write(buf)
-
- pxyCfgs, visitorCfgs, err = config.LoadAllProxyConfsFromIni(cfg.User, configBuffer.Bytes(), cfg.Start)
- if err != nil {
- return
- }
- return
- }
- func getIncludeContents(paths []string) ([]byte, error) {
- out := bytes.NewBuffer(nil)
- for _, path := range paths {
- absDir, err := filepath.Abs(filepath.Dir(path))
- if err != nil {
- return nil, err
- }
- if _, err := os.Stat(absDir); os.IsNotExist(err) {
- return nil, err
- }
- files, err := ioutil.ReadDir(absDir)
- if err != nil {
- return nil, err
- }
- for _, fi := range files {
- if fi.IsDir() {
- continue
- }
- absFile := filepath.Join(absDir, fi.Name())
- if matched, _ := filepath.Match(filepath.Join(absDir, filepath.Base(path)), absFile); matched {
- tmpContent, err := config.GetRenderedConfFromFile(absFile)
- if err != nil {
- return nil, fmt.Errorf("render extra config %s error: %v", absFile, err)
- }
- out.Write(tmpContent)
- out.WriteString("\n")
- }
- }
- }
- return out.Bytes(), nil
- }
- func startService(
- cfg config.ClientCommonConf,
- pxyCfgs map[string]config.ProxyConf,
- visitorCfgs map[string]config.VisitorConf,
- cfgFile string,
- ) (err error) {
- log.InitLog(cfg.LogWay, cfg.LogFile, cfg.LogLevel,
- cfg.LogMaxDays, cfg.DisableLogColor)
- if cfg.DNSServer != "" {
- s := cfg.DNSServer
- if !strings.Contains(s, ":") {
- s += ":53"
- }
-
- net.DefaultResolver = &net.Resolver{
- PreferGo: true,
- Dial: func(ctx context.Context, network, address string) (net.Conn, error) {
- return net.Dial("udp", s)
- },
- }
- }
- svr, errRet := client.NewService(cfg, pxyCfgs, visitorCfgs, cfgFile)
- if errRet != nil {
- err = errRet
- return
- }
-
- if cfg.Protocol == "kcp" {
- go handleSignal(svr)
- }
- err = svr.Run()
- if err == nil && cfg.Protocol == "kcp" {
- <-kcpDoneCh
- }
- return
- }
|