From 03d7320a442cc816f5c09bdf86d72e86df9635e8 Mon Sep 17 00:00:00 2001 From: cthuang Date: Tue, 29 Sep 2020 10:47:50 +0100 Subject: [PATCH] TUN-3430: Copy flags to configure proxy to run subcommand, print relevant tunnel flags in help --- cmd/cloudflared/tunnel/cmd.go | 433 +++++++++++++------------- cmd/cloudflared/tunnel/subcommands.go | 121 ++++--- 2 files changed, 294 insertions(+), 260 deletions(-) diff --git a/cmd/cloudflared/tunnel/cmd.go b/cmd/cloudflared/tunnel/cmd.go index 80638d09d75..8d8bfeb92ce 100644 --- a/cmd/cloudflared/tunnel/cmd.go +++ b/cmd/cloudflared/tunnel/cmd.go @@ -753,26 +753,11 @@ func appendFlags(flags []cli.Flag, extraFlags ...cli.Flag) []cli.Flag { } func tunnelFlags(shouldHide bool) []cli.Flag { - flags := []cli.Flag{ - &cli.StringFlag{ - Name: "config", - Usage: "Specifies a config file in YAML format.", - Value: config.FindDefaultConfigPath(), - Hidden: shouldHide, - }, - altsrc.NewDurationFlag(&cli.DurationFlag{ - Name: "autoupdate-freq", - Usage: fmt.Sprintf("Autoupdate frequency. Default is %v.", updater.DefaultCheckUpdateFreq), - Value: updater.DefaultCheckUpdateFreq, - Hidden: shouldHide, - }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "no-autoupdate", - Usage: "Disable periodic check for updates, restarting the server with the new version.", - EnvVars: []string{"NO_AUTOUPDATE"}, - Value: false, - Hidden: shouldHide, - }), + flags := configureCloudflaredFlags(shouldHide) + flags = append(flags, configureProxyFlags(shouldHide)...) + flags = append(flags, configureLoggingFlags(shouldHide)...) + flags = append(flags, configureProxyDNSFlags(shouldHide)...) + flags = append(flags, []cli.Flag{ altsrc.NewBoolFlag(&cli.BoolFlag{ Name: "is-autoupdated", Usage: "Signal the new process that Argo Tunnel client has been autoupdated", @@ -791,49 +776,12 @@ func tunnelFlags(shouldHide bool) []cli.Flag { EnvVars: []string{"TUNNEL_CACERT"}, Hidden: true, }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "no-tls-verify", - Usage: "Disables TLS verification of the certificate presented by your origin. Will allow any certificate from the origin to be accepted. Note: The connection from your machine to Cloudflare's Edge is still encrypted.", - EnvVars: []string{"NO_TLS_VERIFY"}, - Hidden: shouldHide, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "origincert", - Usage: "Path to the certificate generated for your origin when you run cloudflared login.", - EnvVars: []string{"TUNNEL_ORIGIN_CERT"}, - Value: findDefaultOriginCertPath(), - Hidden: shouldHide, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: tlsconfig.OriginCAPoolFlag, - Usage: "Path to the CA for the certificate of your origin. This option should be used only if your certificate is not signed by Cloudflare.", - EnvVars: []string{"TUNNEL_ORIGIN_CA_POOL"}, - Hidden: shouldHide, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "unix-socket", - Usage: "Path to unix socket to use instead of --url", - EnvVars: []string{"TUNNEL_UNIX_SOCKET"}, - Hidden: shouldHide, - }), altsrc.NewStringFlag(&cli.StringFlag{ Name: "hostname", Usage: "Set a hostname on a Cloudflare zone to route traffic through this tunnel.", EnvVars: []string{"TUNNEL_HOSTNAME"}, Hidden: shouldHide, }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "http-host-header", - Usage: "Sets the HTTP Host header for the local webserver.", - EnvVars: []string{"TUNNEL_HTTP_HOST_HEADER"}, - Hidden: shouldHide, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "origin-server-name", - Usage: "Hostname on the origin server certificate.", - EnvVars: []string{"TUNNEL_ORIGIN_SERVER_NAME"}, - Hidden: shouldHide, - }), altsrc.NewStringFlag(&cli.StringFlag{ Name: "id", Usage: "A unique identifier used to tie connections to this tunnel instance.", @@ -871,13 +819,6 @@ func tunnelFlags(shouldHide bool) []cli.Flag { Value: "https://api.cloudflare.com/client/v4", Hidden: true, }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "metrics", - Value: "localhost:", - Usage: "Listen address for metrics reporting.", - EnvVars: []string{"TUNNEL_METRICS"}, - Hidden: shouldHide, - }), altsrc.NewDurationFlag(&cli.DurationFlag{ Name: "metrics-update-freq", Usage: "Frequency to update tunnel metrics", @@ -903,21 +844,6 @@ func tunnelFlags(shouldHide bool) []cli.Flag { Value: 5, Hidden: true, }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "loglevel", - Value: "info", - Usage: "Application logging level {fatal, error, info, debug}. " + debugLevelWarning, - EnvVars: []string{"TUNNEL_LOGLEVEL"}, - Hidden: shouldHide, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "transport-loglevel", - Aliases: []string{"proto-loglevel"}, // This flag used to be called proto-loglevel - Value: "fatal", - Usage: "Transport logging level(previously called protocol logging level) {fatal, error, info, debug}", - EnvVars: []string{"TUNNEL_PROTO_LOGLEVEL", "TUNNEL_TRANSPORT_LOGLEVEL"}, - Hidden: shouldHide, - }), altsrc.NewUintFlag(&cli.UintFlag{ Name: "retries", Value: 5, @@ -925,41 +851,129 @@ func tunnelFlags(shouldHide bool) []cli.Flag { EnvVars: []string{"TUNNEL_RETRIES"}, Hidden: shouldHide, }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "pidfile", - Usage: "Write the application's PID to this file after first successful connection.", - EnvVars: []string{"TUNNEL_PIDFILE"}, + altsrc.NewIntFlag(&cli.IntFlag{ + Name: "ha-connections", + Value: 4, + Hidden: true, + }), + altsrc.NewDurationFlag(&cli.DurationFlag{ + Name: "grace-period", + Usage: "Duration to accept new requests after cloudflared receives first SIGINT/SIGTERM. A second SIGINT/SIGTERM will force cloudflared to shutdown immediately.", + Value: time.Second * 30, + EnvVars: []string{"TUNNEL_GRACE_PERIOD"}, + Hidden: true, + }), + altsrc.NewUintFlag(&cli.UintFlag{ + Name: "compression-quality", + Value: 0, + Usage: "(beta) Use cross-stream compression instead HTTP compression. 0-off, 1-low, 2-medium, >=3-high.", + EnvVars: []string{"TUNNEL_COMPRESSION_LEVEL"}, Hidden: shouldHide, }), + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "use-reconnect-token", + Usage: "Test reestablishing connections with the new 'reconnect token' flow.", + Value: true, + EnvVars: []string{"TUNNEL_USE_RECONNECT_TOKEN"}, + Hidden: true, + }), + altsrc.NewDurationFlag(&cli.DurationFlag{ + Name: "dial-edge-timeout", + Usage: "Maximum wait time to set up a connection with the edge", + Value: time.Second * 15, + EnvVars: []string{"DIAL_EDGE_TIMEOUT"}, + Hidden: true, + }), + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "stdin-control", + Usage: "Control the process using commands sent through stdin", + EnvVars: []string{"STDIN-CONTROL"}, + Hidden: true, + Value: false, + }), altsrc.NewStringFlag(&cli.StringFlag{ - Name: "logfile", - Usage: "Save application log to this file for reporting issues.", - EnvVars: []string{"TUNNEL_LOGFILE"}, + Name: "name", + Aliases: []string{"n"}, + EnvVars: []string{"TUNNEL_NAME"}, + Usage: "Stable name to identify the tunnel. Using this flag will create, route and run a tunnel. For production usage, execute each command separately", + }), + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: uiFlag, + Usage: "Launch tunnel UI. Tunnel logs are scrollable via 'j', 'k', or arrow keys.", + Value: false, + Hidden: shouldHide, + }), + }...) + + return flags +} + +// Flags in tunnel command that is relevant to run subcommand +func configureCloudflaredFlags(shouldHide bool) []cli.Flag { + return []cli.Flag{ + &cli.StringFlag{ + Name: "config", + Usage: "Specifies a config file in YAML format.", + Value: config.FindDefaultConfigPath(), + Hidden: shouldHide, + }, + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "origincert", + Usage: "Path to the certificate generated for your origin when you run cloudflared login.", + EnvVars: []string{"TUNNEL_ORIGIN_CERT"}, + Value: findDefaultOriginCertPath(), + Hidden: shouldHide, + }), + altsrc.NewDurationFlag(&cli.DurationFlag{ + Name: "autoupdate-freq", + Usage: fmt.Sprintf("Autoupdate frequency. Default is %v.", updater.DefaultCheckUpdateFreq), + Value: updater.DefaultCheckUpdateFreq, + Hidden: shouldHide, + }), + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "no-autoupdate", + Usage: "Disable periodic check for updates, restarting the server with the new version.", + EnvVars: []string{"NO_AUTOUPDATE"}, + Value: false, Hidden: shouldHide, }), altsrc.NewStringFlag(&cli.StringFlag{ - Name: logDirectoryFlag, - Usage: "Save application log to this directory for reporting issues.", - EnvVars: []string{"TUNNEL_LOGDIRECTORY"}, + Name: "metrics", + Value: "localhost:", + Usage: "Listen address for metrics reporting.", + EnvVars: []string{"TUNNEL_METRICS"}, Hidden: shouldHide, }), - altsrc.NewIntFlag(&cli.IntFlag{ - Name: "ha-connections", - Value: 4, - Hidden: true, + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "pidfile", + Usage: "Write the application's PID to this file after first successful connection.", + EnvVars: []string{"TUNNEL_PIDFILE"}, + Hidden: shouldHide, }), + } +} + +func configureProxyFlags(shouldHide bool) []cli.Flag { + flags := []cli.Flag{ altsrc.NewStringFlag(&cli.StringFlag{ - Name: "proxy-address", - Usage: "Listen address for the proxy.", - Value: "127.0.0.1", - EnvVars: []string{"TUNNEL_PROXY_ADDRESS"}, + Name: "url", + Value: "http://localhost:8080", + Usage: "Connect to the local webserver at `URL`.", + EnvVars: []string{"TUNNEL_URL"}, Hidden: shouldHide, }), - altsrc.NewIntFlag(&cli.IntFlag{ - Name: "proxy-port", - Usage: "Listen port for the proxy.", - Value: 0, - EnvVars: []string{"TUNNEL_PROXY_PORT"}, + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "hello-world", + Value: false, + Usage: "Run Hello World Server", + EnvVars: []string{"TUNNEL_HELLO_WORLD"}, + Hidden: shouldHide, + }), + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: socks5Flag, + Usage: "specify if this tunnel is running as a SOCK5 Server", + EnvVars: []string{"TUNNEL_SOCKS"}, + Value: false, Hidden: shouldHide, }), altsrc.NewDurationFlag(&cli.DurationFlag{ @@ -1009,52 +1023,34 @@ func tunnelFlags(shouldHide bool) []cli.Flag { Value: time.Second * 90, Hidden: shouldHide, }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "proxy-dns", - Usage: "Run a DNS over HTTPS proxy server.", - EnvVars: []string{"TUNNEL_DNS"}, - Hidden: shouldHide, - }), - altsrc.NewIntFlag(&cli.IntFlag{ - Name: "proxy-dns-port", - Value: 53, - Usage: "Listen on given port for the DNS over HTTPS proxy server.", - EnvVars: []string{"TUNNEL_DNS_PORT"}, + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "http-host-header", + Usage: "Sets the HTTP Host header for the local webserver.", + EnvVars: []string{"TUNNEL_HTTP_HOST_HEADER"}, Hidden: shouldHide, }), altsrc.NewStringFlag(&cli.StringFlag{ - Name: "proxy-dns-address", - Usage: "Listen address for the DNS over HTTPS proxy server.", - Value: "localhost", - EnvVars: []string{"TUNNEL_DNS_ADDRESS"}, + Name: "origin-server-name", + Usage: "Hostname on the origin server certificate.", + EnvVars: []string{"TUNNEL_ORIGIN_SERVER_NAME"}, Hidden: shouldHide, }), - altsrc.NewStringSliceFlag(&cli.StringSliceFlag{ - Name: "proxy-dns-upstream", - Usage: "Upstream endpoint URL, you can specify multiple endpoints for redundancy.", - Value: cli.NewStringSlice("https://1.1.1.1/dns-query", "https://1.0.0.1/dns-query"), - EnvVars: []string{"TUNNEL_DNS_UPSTREAM"}, + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "unix-socket", + Usage: "Path to unix socket to use instead of --url", + EnvVars: []string{"TUNNEL_UNIX_SOCKET"}, Hidden: shouldHide, }), - altsrc.NewStringSliceFlag(&cli.StringSliceFlag{ - Name: "proxy-dns-bootstrap", - Usage: "bootstrap endpoint URL, you can specify multiple endpoints for redundancy.", - Value: cli.NewStringSlice("https://162.159.36.1/dns-query", "https://162.159.46.1/dns-query", "https://[2606:4700:4700::1111]/dns-query", "https://[2606:4700:4700::1001]/dns-query"), - EnvVars: []string{"TUNNEL_DNS_BOOTSTRAP"}, + altsrc.NewStringFlag(&cli.StringFlag{ + Name: tlsconfig.OriginCAPoolFlag, + Usage: "Path to the CA for the certificate of your origin. This option should be used only if your certificate is not signed by Cloudflare.", + EnvVars: []string{"TUNNEL_ORIGIN_CA_POOL"}, Hidden: shouldHide, }), - altsrc.NewDurationFlag(&cli.DurationFlag{ - Name: "grace-period", - Usage: "Duration to accept new requests after cloudflared receives first SIGINT/SIGTERM. A second SIGINT/SIGTERM will force cloudflared to shutdown immediately.", - Value: time.Second * 30, - EnvVars: []string{"TUNNEL_GRACE_PERIOD"}, - Hidden: true, - }), - altsrc.NewUintFlag(&cli.UintFlag{ - Name: "compression-quality", - Value: 0, - Usage: "(beta) Use cross-stream compression instead HTTP compression. 0-off, 1-low, 2-medium, >=3-high.", - EnvVars: []string{"TUNNEL_COMPRESSION_LEVEL"}, + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "no-tls-verify", + Usage: "Disables TLS verification of the certificate presented by your origin. Will allow any certificate from the origin to be accepted. Note: The connection from your machine to Cloudflare's Edge is still encrypted.", + EnvVars: []string{"NO_TLS_VERIFY"}, Hidden: shouldHide, }), altsrc.NewBoolFlag(&cli.BoolFlag{ @@ -1063,72 +1059,10 @@ func tunnelFlags(shouldHide bool) []cli.Flag { EnvVars: []string{"TUNNEL_NO_CHUNKED_ENCODING"}, Hidden: shouldHide, }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "trace-output", - Usage: "Name of trace output file, generated when cloudflared stops.", - EnvVars: []string{"TUNNEL_TRACE_OUTPUT"}, - Hidden: shouldHide, - }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "use-reconnect-token", - Usage: "Test reestablishing connections with the new 'reconnect token' flow.", - Value: true, - EnvVars: []string{"TUNNEL_USE_RECONNECT_TOKEN"}, - Hidden: true, - }), - altsrc.NewDurationFlag(&cli.DurationFlag{ - Name: "dial-edge-timeout", - Usage: "Maximum wait time to set up a connection with the edge", - Value: time.Second * 15, - EnvVars: []string{"DIAL_EDGE_TIMEOUT"}, - Hidden: true, - }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "stdin-control", - Usage: "Control the process using commands sent through stdin", - EnvVars: []string{"STDIN-CONTROL"}, - Hidden: true, - Value: false, - }), - altsrc.NewStringFlag(&cli.StringFlag{ - Name: "name", - Aliases: []string{"n"}, - EnvVars: []string{"TUNNEL_NAME"}, - Usage: "Stable name to identify the tunnel. Using this flag will create, route and run a tunnel. For production usage, execute each command separately", - }), - altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: uiFlag, - Usage: "Launch tunnel UI. Tunnel logs are scrollable via 'j', 'k', or arrow keys.", - Value: false, - Hidden: shouldHide, - }), - urlFlag(shouldHide), - helloWorldFlag(shouldHide), - createSocks5Flag(shouldHide), } return append(flags, sshFlags(shouldHide)...) } -func urlFlag(shouldHide bool) cli.Flag { - return altsrc.NewStringFlag(&cli.StringFlag{ - Name: "url", - Value: "http://localhost:8080", - Usage: "Connect to the local webserver at `URL`.", - EnvVars: []string{"TUNNEL_URL"}, - Hidden: shouldHide, - }) -} - -func helloWorldFlag(shouldHide bool) cli.Flag { - return altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: "hello-world", - Value: false, - Usage: "Run Hello World Server", - EnvVars: []string{"TUNNEL_HELLO_WORLD"}, - Hidden: shouldHide, - }) -} - func sshFlags(shouldHide bool) []cli.Flag { return []cli.Flag{ altsrc.NewStringFlag(&cli.StringFlag{ @@ -1206,17 +1140,98 @@ func sshFlags(shouldHide bool) []cli.Flag { EnvVars: []string{"TUNNEL_BASTION"}, Hidden: shouldHide, }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "proxy-address", + Usage: "Listen address for the proxy.", + Value: "127.0.0.1", + EnvVars: []string{"TUNNEL_PROXY_ADDRESS"}, + Hidden: shouldHide, + }), + altsrc.NewIntFlag(&cli.IntFlag{ + Name: "proxy-port", + Usage: "Listen port for the proxy.", + Value: 0, + EnvVars: []string{"TUNNEL_PROXY_PORT"}, + Hidden: shouldHide, + }), } } -func createSocks5Flag(shouldHide bool) cli.Flag { - return altsrc.NewBoolFlag(&cli.BoolFlag{ - Name: socks5Flag, - Usage: "specify if this tunnel is running as a SOCK5 Server", - EnvVars: []string{"TUNNEL_SOCKS"}, - Value: false, - Hidden: shouldHide, - }) +func configureLoggingFlags(shouldHide bool) []cli.Flag { + return []cli.Flag{ + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "loglevel", + Value: "info", + Usage: "Application logging level {fatal, error, info, debug}. " + debugLevelWarning, + EnvVars: []string{"TUNNEL_LOGLEVEL"}, + Hidden: shouldHide, + }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "transport-loglevel", + Aliases: []string{"proto-loglevel"}, // This flag used to be called proto-loglevel + Value: "fatal", + Usage: "Transport logging level(previously called protocol logging level) {fatal, error, info, debug}", + EnvVars: []string{"TUNNEL_PROTO_LOGLEVEL", "TUNNEL_TRANSPORT_LOGLEVEL"}, + Hidden: shouldHide, + }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "logfile", + Usage: "Save application log to this file for reporting issues.", + EnvVars: []string{"TUNNEL_LOGFILE"}, + Hidden: shouldHide, + }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: logDirectoryFlag, + Usage: "Save application log to this directory for reporting issues.", + EnvVars: []string{"TUNNEL_LOGDIRECTORY"}, + Hidden: shouldHide, + }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "trace-output", + Usage: "Name of trace output file, generated when cloudflared stops.", + EnvVars: []string{"TUNNEL_TRACE_OUTPUT"}, + Hidden: shouldHide, + }), + } +} + +func configureProxyDNSFlags(shouldHide bool) []cli.Flag { + return []cli.Flag{ + altsrc.NewBoolFlag(&cli.BoolFlag{ + Name: "proxy-dns", + Usage: "Run a DNS over HTTPS proxy server.", + EnvVars: []string{"TUNNEL_DNS"}, + Hidden: shouldHide, + }), + altsrc.NewIntFlag(&cli.IntFlag{ + Name: "proxy-dns-port", + Value: 53, + Usage: "Listen on given port for the DNS over HTTPS proxy server.", + EnvVars: []string{"TUNNEL_DNS_PORT"}, + Hidden: shouldHide, + }), + altsrc.NewStringFlag(&cli.StringFlag{ + Name: "proxy-dns-address", + Usage: "Listen address for the DNS over HTTPS proxy server.", + Value: "localhost", + EnvVars: []string{"TUNNEL_DNS_ADDRESS"}, + Hidden: shouldHide, + }), + altsrc.NewStringSliceFlag(&cli.StringSliceFlag{ + Name: "proxy-dns-upstream", + Usage: "Upstream endpoint URL, you can specify multiple endpoints for redundancy.", + Value: cli.NewStringSlice("https://1.1.1.1/dns-query", "https://1.0.0.1/dns-query"), + EnvVars: []string{"TUNNEL_DNS_UPSTREAM"}, + Hidden: shouldHide, + }), + altsrc.NewStringSliceFlag(&cli.StringSliceFlag{ + Name: "proxy-dns-bootstrap", + Usage: "bootstrap endpoint URL, you can specify multiple endpoints for redundancy.", + Value: cli.NewStringSlice("https://162.159.36.1/dns-query", "https://162.159.46.1/dns-query", "https://[2606:4700:4700::1111]/dns-query", "https://[2606:4700:4700::1001]/dns-query"), + EnvVars: []string{"TUNNEL_DNS_BOOTSTRAP"}, + Hidden: shouldHide, + }), + } } func stdinControl(reconnectCh chan origin.ReconnectSignal, logger logger.Service) { diff --git a/cmd/cloudflared/tunnel/subcommands.go b/cmd/cloudflared/tunnel/subcommands.go index d02a6b91127..581e29b2262 100644 --- a/cmd/cloudflared/tunnel/subcommands.go +++ b/cmd/cloudflared/tunnel/subcommands.go @@ -85,11 +85,14 @@ var ( func buildCreateCommand() *cli.Command { return &cli.Command{ - Name: "create", - Action: cliutil.ErrorHandler(createCommand), - Usage: "Create a new tunnel with given name", - ArgsUsage: "TUNNEL-NAME", - Flags: []cli.Flag{outputFormatFlag}, + Name: "create", + Action: cliutil.ErrorHandler(createCommand), + Usage: "Create a new tunnel with given name", + UsageText: "cloudflared tunnel [tunnel command options] create [create command options]", + Description: "cloudflared tunnel create example will create a tunnel named example, and generate a credential file to run the tunnel", + ArgsUsage: "TUNNEL-NAME", + Flags: []cli.Flag{outputFormatFlag}, + CustomHelpTemplate: commandHelpTemplate(), } } @@ -149,11 +152,14 @@ func validFilePath(path string) bool { func buildListCommand() *cli.Command { return &cli.Command{ - Name: "list", - Action: cliutil.ErrorHandler(listCommand), - Usage: "List existing tunnels", - ArgsUsage: " ", - Flags: []cli.Flag{outputFormatFlag, showDeletedFlag, listNameFlag, listExistedAtFlag, listIDFlag, showRecentlyDisconnected}, + Name: "list", + Action: cliutil.ErrorHandler(listCommand), + Usage: "List existing tunnels", + UsageText: "cloudflared tunnel [tunnel command options] list [subcommand options]", + Description: "cloudflared tunnel list will return all active tunnels, their created time and associated connections. Use -d flag to include deleted tunnels. See the list of options to filter the list", + ArgsUsage: " ", + Flags: []cli.Flag{outputFormatFlag, showDeletedFlag, listNameFlag, listExistedAtFlag, listIDFlag, showRecentlyDisconnected}, + CustomHelpTemplate: commandHelpTemplate(), } } @@ -253,11 +259,14 @@ func fmtConnections(connections []tunnelstore.Connection, showRecentlyDisconnect func buildDeleteCommand() *cli.Command { return &cli.Command{ - Name: "delete", - Action: cliutil.ErrorHandler(deleteCommand), - Usage: "Delete existing tunnel by UUID or name", - ArgsUsage: "TUNNEL", - Flags: []cli.Flag{credentialsFileFlag, forceDeleteFlag}, + Name: "delete", + Action: cliutil.ErrorHandler(deleteCommand), + Usage: "Delete existing tunnel by UUID or name", + UsageText: "cloudflared tunnel [tunnel command options] delete [subcommand options]", + Description: "cloudflared tunnel delete will delete tunnels with the given tunnel UUIDs or names. A tunnel cannot be deleted if it has active connections. To delete the tunnel unconditionally, use -f flag.", + ArgsUsage: "TUNNEL", + Flags: []cli.Flag{credentialsFileFlag, forceDeleteFlag}, + CustomHelpTemplate: commandHelpTemplate(), } } @@ -296,45 +305,23 @@ func buildRunCommand() *cli.Command { flags := []cli.Flag{ forceFlag, credentialsFileFlag, - urlFlag(false), - helloWorldFlag(false), - createSocks5Flag(false), } - flags = append(flags, sshFlags(false)...) - var runCommandHelpTemplate = `NAME: - {{.HelpName}} - {{.Usage}} - -USAGE: - {{.UsageText}} - -DESCRIPTION: - {{.Description}} - -TUNNEL COMMAND OPTIONS: - See cloudflared tunnel -h - -RUN COMMAND OPTIONS: - {{range .VisibleFlags}}{{.}} - {{end}} -` + flags = append(flags, configureProxyFlags(false)...) return &cli.Command{ Name: "run", Action: cliutil.ErrorHandler(runCommand), Usage: "Proxy a local web server by running the given tunnel", - UsageText: "cloudflared tunnel [tunnel command options] run [run command options]", + UsageText: "cloudflared tunnel [tunnel command options] run [subcommand options]", ArgsUsage: "TUNNEL", - Description: `Runs the tunnel identified by name or UUUD, creating a highly available connection + Description: `Runs the tunnel identified by name or UUUD, creating highly available connections between your server and the Cloudflare edge. This command requires the tunnel credentials file created when "cloudflared tunnel create" was run, - however it does not need access to cert.pem from "cloudflared login". If you experience problems running - the tunnel, "cloudflared tunnel cleanup" may help by removing any old connection records. - - All the flags from the tunnel command are available, note that they have to be specified before the run command. There are flags defined both in tunnel and run command. The one in run command will take precedence. - For example cloudflared tunnel --url localhost:3000 run --url localhost:5000 will proxy requests to localhost:5000. + however it does not need access to cert.pem from "cloudflared login" if you identify the tunnel by UUID. + If you experience other problems running gitthe tunnel, "cloudflared tunnel cleanup" may help by removing any old connection records. `, Flags: flags, - CustomHelpTemplate: runCommandHelpTemplate, + CustomHelpTemplate: commandHelpTemplate(), } } @@ -357,10 +344,13 @@ func runCommand(c *cli.Context) error { func buildCleanupCommand() *cli.Command { return &cli.Command{ - Name: "cleanup", - Action: cliutil.ErrorHandler(cleanupCommand), - Usage: "Cleanup tunnel connections", - ArgsUsage: "TUNNEL", + Name: "cleanup", + Action: cliutil.ErrorHandler(cleanupCommand), + Usage: "Cleanup tunnel connections", + UsageText: "cloudflared tunnel [tunnel command options] cleanup [subcommand options]", + Description: "Delete connections for tunnels with the given UUIDs or names.", + ArgsUsage: "TUNNEL", + CustomHelpTemplate: commandHelpTemplate(), } } @@ -384,16 +374,18 @@ func cleanupCommand(c *cli.Context) error { func buildRouteCommand() *cli.Command { return &cli.Command{ - Name: "route", - Action: cliutil.ErrorHandler(routeCommand), - Usage: "Define what hostname or load balancer can route to this tunnel", + Name: "route", + Action: cliutil.ErrorHandler(routeCommand), + Usage: "Define what hostname or load balancer can route to this tunnel", + UsageText: "cloudflared tunnel [tunnel command options] route [subcommand options]", Description: `The route defines what hostname or load balancer will proxy requests to this tunnel. To route a hostname by creating a CNAME to tunnel's address: cloudflared tunnel route dns To use this tunnel as a load balancer origin, creating pool and load balancer if necessary: cloudflared tunnel route lb `, - ArgsUsage: "dns|lb TUNNEL HOSTNAME [LB-POOL]", + ArgsUsage: "dns|lb TUNNEL HOSTNAME [LB-POOL]", + CustomHelpTemplate: commandHelpTemplate(), } } @@ -501,3 +493,30 @@ func routeCommand(c *cli.Context) error { sc.logger.Infof(res.SuccessSummary()) return nil } + +func commandHelpTemplate() string { + var parentFlagsHelp string + for _, f := range configureCloudflaredFlags(false) { + parentFlagsHelp += fmt.Sprintf(" %s\n\t", f) + } + for _, f := range configureLoggingFlags(false) { + parentFlagsHelp += fmt.Sprintf(" %s\n\t", f) + } + const template = `NAME: + {{.HelpName}} - {{.Usage}} + +USAGE: + {{.UsageText}} + +DESCRIPTION: + {{.Description}} + +TUNNEL COMMAND OPTIONS: + %s + +SUBCOMMAND COMMAND OPTIONS: + {{range .VisibleFlags}}{{.}} + {{end}} +` + return fmt.Sprintf(template, parentFlagsHelp) +}