123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 |
- // Copyright 2018 fatedier, fatedier@gmail.com
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package sub
- import (
- "context"
- "fmt"
- "net"
- "os"
- "os/signal"
- "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{}
- )
- var services = make(map[string]*client.Service)
- func GetUids() (uids []string) {
- keys := make([]string, 0, len(services))
- for k := range services {
- keys = append(keys, k)
- }
- return keys
- }
- func IsRunning(uid string) (running bool) {
- return services[uid] != nil
- }
- func getServiceByUid(uid string) (svr *client.Service) {
- return services[uid]
- }
- func putServiceByUid(uid string, svr *client.Service) {
- services[uid] = svr
- }
- func delServiceByUid(uid string) {
- delete(services, uid)
- }
- func Close(uid string) (ret bool) {
- svr := getServiceByUid(uid)
- if svr != nil {
- svr.Close()
- delServiceByUid(uid)
- return true
- }
- return false
- }
- 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
- }
- // Do not show command usage here.
- 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.GracefulClose(500 * time.Millisecond)
- close(kcpDoneCh)
- }
- 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
- // Only token authentication is supported in cmd mode
- cfg.ClientConfig = auth.GetDefaultClientConf()
- cfg.Token = token
- cfg.TLSEnable = tlsEnable
- cfg.Complete()
- if err = cfg.Validate(); err != nil {
- err = fmt.Errorf("Parse config error: %v", err)
- return
- }
- return
- }
- 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
- }
- err = cfg.Validate()
- if err != nil {
- return
- }
- return
- }
- func RunClient(cfgFilePath string) (err error, svr *client.Service) {
- var content []byte
- content, err = config.GetRenderedConfFromFile(cfgFilePath)
- if err != nil {
- return
- }
- cfg, err := parseClientCommonCfg(CfgFileTypeIni, content)
- if err != nil {
- return
- }
- pxyCfgs, visitorCfgs, err := config.LoadAllProxyConfsFromIni(cfg.User, content, cfg.Start)
- if err != nil {
- return
- }
- return startService(cfg, pxyCfgs, visitorCfgs, cfgFilePath)
- }
- func RunClientWithUid(uid string, cfgFilePath string) (err error, svr *client.Service) {
- var content []byte
- content, err = config.GetRenderedConfFromFile(cfgFilePath)
- if err != nil {
- return
- }
- cfg, err := parseClientCommonCfg(CfgFileTypeIni, content)
- if err != nil {
- return
- }
- pxyCfgs, visitorCfgs, err := config.LoadAllProxyConfsFromIni(cfg.User, content, cfg.Start)
- if err != nil {
- return
- }
- return startServiceWithUid(uid, cfg, pxyCfgs, visitorCfgs)
- }
- func RunClientByContent(uid string, cfgContent string) (err error, svr *client.Service) {
- content, err := config.RenderContent([]byte(cfgContent))
- if err != nil {
- return
- }
- cfg, err := parseClientCommonCfg(CfgFileTypeIni, content)
- if err != nil {
- return
- }
- pxyCfgs, visitorCfgs, err := config.LoadAllProxyConfsFromIni(cfg.User, content, cfg.Start)
- if err != nil {
- return
- }
- return startServiceWithUid(uid, cfg, pxyCfgs, visitorCfgs)
- }
- func runClient(cfgFilePath string) error {
- cfg, pxyCfgs, visitorCfgs, err := config.ParseClientConfig(cfgFilePath)
- if err != nil {
- return err
- }
- err, _ = startService(cfg, pxyCfgs, visitorCfgs, cfgFilePath)
- return err
- }
- func startServiceWithUid(
- uid string,
- cfg config.ClientCommonConf,
- pxyCfgs map[string]config.ProxyConf,
- visitorCfgs map[string]config.VisitorConf,
- ) (err error, svr *client.Service) {
- defer delServiceByUid(uid)
- log.InitLog(cfg.LogWay, cfg.LogFile, cfg.LogLevel,
- cfg.LogMaxDays, cfg.DisableLogColor)
- if cfg.DNSServer != "" {
- s := cfg.DNSServer
- if !strings.Contains(s, ":") {
- s += ":53"
- }
- // Change default dns server for frpc
- 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
- }
- // Capture the exit signal if we use kcp.
- if cfg.Protocol == "kcp" {
- go handleSignal(svr)
- }
- putServiceByUid(uid, svr)
- err = svr.Run()
- if err == nil && cfg.Protocol == "kcp" {
- <-kcpDoneCh
- }
- return err, svr
- }
- func startService(
- cfg config.ClientCommonConf,
- pxyCfgs map[string]config.ProxyConf,
- visitorCfgs map[string]config.VisitorConf,
- cfgFile string,
- ) (err error, svr *client.Service) {
- log.InitLog(cfg.LogWay, cfg.LogFile, cfg.LogLevel,
- cfg.LogMaxDays, cfg.DisableLogColor)
- if cfg.DNSServer != "" {
- s := cfg.DNSServer
- if !strings.Contains(s, ":") {
- s += ":53"
- }
- // Change default dns server for frpc
- 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
- }
- // Capture the exit signal if we use kcp.
- if cfg.Protocol == "kcp" {
- go handleSignal(svr)
- }
- err = svr.Run()
- if err == nil && cfg.Protocol == "kcp" {
- <-kcpDoneCh
- }
- return err, svr
- }
|