2023-03-13 21:43:28 -04:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/peterbourgon/ff/v3/ffcli"
|
|
|
|
"tailscale.com/ipn"
|
2023-08-09 10:06:58 -04:00
|
|
|
"tailscale.com/tailcfg"
|
2023-03-13 21:43:28 -04:00
|
|
|
)
|
|
|
|
|
2023-08-17 11:47:35 -04:00
|
|
|
var funnelCmd = func() *ffcli.Command {
|
|
|
|
se := &serveEnv{lc: &localClient}
|
2023-10-17 09:32:17 -07:00
|
|
|
// previously used to serve legacy newFunnelCommand unless useWIPCode is true
|
|
|
|
// change is limited to make a revert easier and full cleanup to come after the relase.
|
|
|
|
// TODO(tylersmalley): cleanup and removal of newFunnelCommand as of 2023-10-16
|
|
|
|
return newServeV2Command(se, funnel)
|
2023-08-17 11:47:35 -04:00
|
|
|
}
|
2023-03-13 21:43:28 -04:00
|
|
|
|
|
|
|
// newFunnelCommand returns a new "funnel" subcommand using e as its environment.
|
|
|
|
// The funnel subcommand is used to turn on/off the Funnel service.
|
|
|
|
// Funnel is off by default.
|
|
|
|
// Funnel allows you to publish a 'tailscale serve' server publicly, open to the
|
|
|
|
// entire internet.
|
|
|
|
// newFunnelCommand shares the same serveEnv as the "serve" subcommand. See
|
|
|
|
// newServeCommand and serve.go for more details.
|
|
|
|
func newFunnelCommand(e *serveEnv) *ffcli.Command {
|
|
|
|
return &ffcli.Command{
|
|
|
|
Name: "funnel",
|
2023-04-03 10:09:04 -04:00
|
|
|
ShortHelp: "Turn on/off Funnel service",
|
2023-06-21 12:32:20 -04:00
|
|
|
ShortUsage: strings.Join([]string{
|
2024-04-04 17:06:09 +01:00
|
|
|
"tailscale funnel <serve-port> {on|off}",
|
|
|
|
"tailscale funnel status [--json]",
|
|
|
|
}, "\n"),
|
2023-03-13 21:43:28 -04:00
|
|
|
LongHelp: strings.Join([]string{
|
|
|
|
"Funnel allows you to publish a 'tailscale serve'",
|
|
|
|
"server publicly, open to the entire internet.",
|
|
|
|
"",
|
|
|
|
"Turning off Funnel only turns off serving to the internet.",
|
|
|
|
"It does not affect serving to your tailnet.",
|
|
|
|
}, "\n"),
|
2024-04-04 17:06:09 +01:00
|
|
|
Exec: e.runFunnel,
|
2023-03-13 21:43:28 -04:00
|
|
|
Subcommands: []*ffcli.Command{
|
|
|
|
{
|
2024-04-04 17:06:09 +01:00
|
|
|
Name: "status",
|
|
|
|
Exec: e.runServeStatus,
|
|
|
|
ShortUsage: "tailscale funnel status [--json]",
|
|
|
|
ShortHelp: "Show current serve/funnel status",
|
2023-03-13 21:43:28 -04:00
|
|
|
FlagSet: e.newFlags("funnel-status", func(fs *flag.FlagSet) {
|
|
|
|
fs.BoolVar(&e.json, "json", false, "output JSON")
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// runFunnel is the entry point for the "tailscale funnel" subcommand and
|
|
|
|
// manages turning on/off funnel. Funnel is off by default.
|
|
|
|
//
|
|
|
|
// Note: funnel is only supported on single DNS name for now. (2022-11-15)
|
|
|
|
func (e *serveEnv) runFunnel(ctx context.Context, args []string) error {
|
|
|
|
if len(args) != 2 {
|
|
|
|
return flag.ErrHelp
|
|
|
|
}
|
|
|
|
|
|
|
|
var on bool
|
|
|
|
switch args[1] {
|
|
|
|
case "on", "off":
|
|
|
|
on = args[1] == "on"
|
|
|
|
default:
|
|
|
|
return flag.ErrHelp
|
|
|
|
}
|
|
|
|
sc, err := e.lc.GetServeConfig(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if sc == nil {
|
|
|
|
sc = new(ipn.ServeConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
port64, err := strconv.ParseUint(args[0], 10, 16)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
port := uint16(port64)
|
|
|
|
|
2023-08-15 00:07:51 -04:00
|
|
|
if on {
|
|
|
|
// Don't block from turning off existing Funnel if
|
|
|
|
// network configuration/capabilities have changed.
|
|
|
|
// Only block from starting new Funnels.
|
2023-10-26 10:38:08 -07:00
|
|
|
if err := e.verifyFunnelEnabled(ctx, port); err != nil {
|
2023-08-15 00:07:51 -04:00
|
|
|
return err
|
|
|
|
}
|
2023-03-13 21:43:28 -04:00
|
|
|
}
|
2023-08-09 10:06:58 -04:00
|
|
|
|
2023-10-26 10:38:08 -07:00
|
|
|
st, err := e.getLocalClientStatusWithoutPeers(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("getting client status: %w", err)
|
|
|
|
}
|
2023-03-13 21:43:28 -04:00
|
|
|
dnsName := strings.TrimSuffix(st.Self.DNSName, ".")
|
|
|
|
hp := ipn.HostPort(dnsName + ":" + strconv.Itoa(int(port)))
|
|
|
|
if on == sc.AllowFunnel[hp] {
|
|
|
|
printFunnelWarning(sc)
|
|
|
|
// Nothing to do.
|
|
|
|
return nil
|
|
|
|
}
|
2024-03-05 18:46:42 -05:00
|
|
|
sc.SetFunnel(dnsName, port, on)
|
|
|
|
|
2023-03-13 21:43:28 -04:00
|
|
|
if err := e.lc.SetServeConfig(ctx, sc); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
printFunnelWarning(sc)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-09 10:06:58 -04:00
|
|
|
// verifyFunnelEnabled verifies that the self node is allowed to use Funnel.
|
|
|
|
//
|
|
|
|
// If Funnel is not yet enabled by the current node capabilities,
|
|
|
|
// the user is sent through an interactive flow to enable the feature.
|
|
|
|
// Once enabled, verifyFunnelEnabled checks that the given port is allowed
|
|
|
|
// with Funnel.
|
|
|
|
//
|
|
|
|
// If an error is reported, the CLI should stop execution and return the error.
|
|
|
|
//
|
|
|
|
// verifyFunnelEnabled may refresh the local state and modify the st input.
|
2023-10-26 10:38:08 -07:00
|
|
|
func (e *serveEnv) verifyFunnelEnabled(ctx context.Context, port uint16) error {
|
2023-09-18 09:36:26 -07:00
|
|
|
enableErr := e.enableFeatureInteractive(ctx, "funnel", tailcfg.CapabilityHTTPS, tailcfg.NodeAttrFunnel)
|
2023-08-17 14:09:58 -04:00
|
|
|
st, statusErr := e.getLocalClientStatusWithoutPeers(ctx) // get updated status; interactive flow may block
|
2023-08-09 10:06:58 -04:00
|
|
|
switch {
|
|
|
|
case statusErr != nil:
|
|
|
|
return fmt.Errorf("getting client status: %w", statusErr)
|
|
|
|
case enableErr != nil:
|
|
|
|
// enableFeatureInteractive is a new flow behind a control server
|
|
|
|
// feature flag. If anything caused it to error, fallback to using
|
|
|
|
// the old CheckFunnelAccess call. Likely this domain does not have
|
|
|
|
// the feature flag on.
|
|
|
|
// TODO(sonia,tailscale/corp#10577): Remove this fallback once the
|
|
|
|
// control flag is turned on for all domains.
|
2023-09-27 23:01:09 -07:00
|
|
|
if err := ipn.CheckFunnelAccess(port, st.Self); err != nil {
|
2023-08-09 10:06:58 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// Done with enablement, make sure the requested port is allowed.
|
2023-09-27 23:01:09 -07:00
|
|
|
if err := ipn.CheckFunnelPort(port, st.Self); err != nil {
|
2023-08-09 10:06:58 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-13 21:43:28 -04:00
|
|
|
// printFunnelWarning prints a warning if the Funnel is on but there is no serve
|
|
|
|
// config for its host:port.
|
|
|
|
func printFunnelWarning(sc *ipn.ServeConfig) {
|
|
|
|
var warn bool
|
|
|
|
for hp, a := range sc.AllowFunnel {
|
|
|
|
if !a {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
_, portStr, _ := net.SplitHostPort(string(hp))
|
|
|
|
p, _ := strconv.ParseUint(portStr, 10, 16)
|
|
|
|
if _, ok := sc.TCP[uint16(p)]; !ok {
|
|
|
|
warn = true
|
2024-04-07 18:17:25 -07:00
|
|
|
fmt.Fprintf(Stderr, "\nWarning: funnel=on for %s, but no serve config\n", hp)
|
2023-03-13 21:43:28 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if warn {
|
2024-04-07 18:17:25 -07:00
|
|
|
fmt.Fprintf(Stderr, " run: `tailscale serve --help` to see how to configure handlers\n")
|
2023-03-13 21:43:28 -04:00
|
|
|
}
|
|
|
|
}
|