diff options
| author | David Crawshaw <crawshaw@tailscale.com> | 2021-03-12 09:19:19 -0800 |
|---|---|---|
| committer | David Crawshaw <crawshaw@tailscale.com> | 2021-03-12 14:27:04 -0800 |
| commit | b6d31436ecf393690baf6edb678d03851767a88f (patch) | |
| tree | aa742d80e66cbd4ff532bf1fa33675faf0e555cb | |
| parent | c6358f2247a5f76ce7536b3d5d457569cfe32bdb (diff) | |
| download | tailscale-crawshaw/upjson.tar.xz tailscale-crawshaw/upjson.zip | |
cmd/tailscale: add up -json flagcrawshaw/upjson
Signed-off-by: David Crawshaw <crawshaw@tailscale.com>
| -rw-r--r-- | cmd/tailscale/cli/up.go | 25 |
1 files changed, 25 insertions, 0 deletions
diff --git a/cmd/tailscale/cli/up.go b/cmd/tailscale/cli/up.go index 3f0081425..ac466dca5 100644 --- a/cmd/tailscale/cli/up.go +++ b/cmd/tailscale/cli/up.go @@ -48,6 +48,7 @@ specify any flags, options are reset to their default. upf.StringVar(&upArgs.exitNodeIP, "exit-node", "", "Tailscale IP of the exit node for internet traffic") upf.BoolVar(&upArgs.shieldsUp, "shields-up", false, "don't allow incoming connections") upf.BoolVar(&upArgs.forceReauth, "force-reauth", false, "force reauthentication") + upf.BoolVar(&upArgs.json, "json", false, "print output as JSON and exit when control server provides instructions") upf.StringVar(&upArgs.advertiseTags, "advertise-tags", "", "ACL tags to request (comma-separated, e.g. eng,montreal,ssh)") upf.StringVar(&upArgs.authKey, "authkey", "", "node authorization key") upf.StringVar(&upArgs.hostname, "hostname", "", "hostname to use instead of the one provided by the OS") @@ -79,6 +80,7 @@ var upArgs struct { exitNodeIP string shieldsUp bool forceReauth bool + json bool advertiseRoutes string advertiseDefaultRoute bool advertiseTags string @@ -294,6 +296,29 @@ func runUp(ctx context.Context, args []string) error { }, } + if upArgs.json { + opts.Notify = func(n ipn.Notify) { + if n.ErrMessage != nil { + fmt.Fprintf(os.Stdout, `{"error":%q}`, *n.ErrMessage) + os.Exit(1) + } + state := "" + if n.State != nil { + state = n.State.String() + } + if url := n.BrowseToURL; url != nil { + fmt.Fprintf(os.Stdout, `{"state":%q,"auth_url":%q}`, state, *url) + cancel() + return + } + if n.State != nil && *n.State == ipn.Running { + fmt.Fprintf(os.Stdout, `{"state":%q}`, state) + cancel() + return + } + } + } + // On Windows, we still run in mostly the "legacy" way that // predated the server's StateStore. That is, we send an empty // StateKey and send the prefs directly. Although the Windows |
