Skip to content

Commit 00b3f72

Browse files
authored
Reorganize tctl commands to not require an auth client by default (#48894) (#50966)
* Reorganize tctl commands to have commands not required auth client * Replace auth client with lazy loading approach * Fix linter warning * Replace camel case in import alias Replace logrus to use slog * Rename close function * Refactor plugin commands to use interface of auth client and plugin client Code review changes * Refactor workload identity commands * Add access to global config for the commands * Add test checking all tctl commands match process * Fix golangci-lint warnings
1 parent 4e0af56 commit 00b3f72

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

49 files changed

+1122
-577
lines changed

integration/tctl_terraform_env_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@ func TestTCTLTerraformCommand_ProxyJoin(t *testing.T) {
105105
tctlCommand := common.TerraformCommand{}
106106

107107
app := kingpin.New("test", "test")
108-
tctlCommand.Initialize(app, tctlCfg)
108+
tctlCommand.Initialize(app, nil, tctlCfg)
109109
_, err = app.Parse([]string{"terraform", "env"})
110110
require.NoError(t, err)
111111
// Create io buffer writer
@@ -179,7 +179,7 @@ func TestTCTLTerraformCommand_AuthJoin(t *testing.T) {
179179
tctlCommand := common.TerraformCommand{}
180180

181181
app := kingpin.New("test", "test")
182-
tctlCommand.Initialize(app, tctlCfg)
182+
tctlCommand.Initialize(app, nil, tctlCfg)
183183
_, err = app.Parse([]string{"terraform", "env"})
184184
require.NoError(t, err)
185185
// Create io buffer writer

tool/tctl/common/access_request_command.go

+21-10
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,8 @@ import (
3939
"github.com/gravitational/teleport/lib/service/servicecfg"
4040
"github.com/gravitational/teleport/lib/services"
4141
"github.com/gravitational/teleport/lib/tlsca"
42+
commonclient "github.com/gravitational/teleport/tool/tctl/common/client"
43+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
4244
)
4345

4446
// AccessRequestCommand implements `tctl users` set of commands
@@ -76,7 +78,7 @@ type AccessRequestCommand struct {
7678
}
7779

7880
// Initialize allows AccessRequestCommand to plug itself into the CLI parser
79-
func (c *AccessRequestCommand) Initialize(app *kingpin.Application, config *servicecfg.Config) {
81+
func (c *AccessRequestCommand) Initialize(app *kingpin.Application, _ *tctlcfg.GlobalCLIFlags, config *servicecfg.Config) {
8082
c.config = config
8183
requests := app.Command("requests", "Manage access requests.").Alias("request")
8284

@@ -125,27 +127,36 @@ func (c *AccessRequestCommand) Initialize(app *kingpin.Application, config *serv
125127
}
126128

127129
// TryRun takes the CLI command as an argument (like "access-request list") and executes it.
128-
func (c *AccessRequestCommand) TryRun(ctx context.Context, cmd string, client *authclient.Client) (match bool, err error) {
130+
func (c *AccessRequestCommand) TryRun(ctx context.Context, cmd string, clientFunc commonclient.InitFunc) (match bool, err error) {
131+
var commandFunc func(ctx context.Context, client *authclient.Client) error
129132
switch cmd {
130133
case c.requestList.FullCommand():
131-
err = c.List(ctx, client)
134+
commandFunc = c.List
132135
case c.requestGet.FullCommand():
133-
err = c.Get(ctx, client)
136+
commandFunc = c.Get
134137
case c.requestApprove.FullCommand():
135-
err = c.Approve(ctx, client)
138+
commandFunc = c.Approve
136139
case c.requestDeny.FullCommand():
137-
err = c.Deny(ctx, client)
140+
commandFunc = c.Deny
138141
case c.requestCreate.FullCommand():
139-
err = c.Create(ctx, client)
142+
commandFunc = c.Create
140143
case c.requestDelete.FullCommand():
141-
err = c.Delete(ctx, client)
144+
commandFunc = c.Delete
142145
case c.requestCaps.FullCommand():
143-
err = c.Caps(ctx, client)
146+
commandFunc = c.Caps
144147
case c.requestReview.FullCommand():
145-
err = c.Review(ctx, client)
148+
commandFunc = c.Review
146149
default:
147150
return false, nil
148151
}
152+
153+
client, closeFn, err := clientFunc(ctx)
154+
if err != nil {
155+
return false, trace.Wrap(err)
156+
}
157+
err = commandFunc(ctx, client)
158+
closeFn(ctx)
159+
149160
return true, trace.Wrap(err)
150161
}
151162

tool/tctl/common/accessmonitoring/command.go

+12-4
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,8 @@ import (
3535
"github.com/gravitational/teleport/lib/auth/authclient"
3636
"github.com/gravitational/teleport/lib/service/servicecfg"
3737
"github.com/gravitational/teleport/lib/utils"
38+
commonclient "github.com/gravitational/teleport/tool/tctl/common/client"
39+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
3840
)
3941

4042
// Command implements `tctl audit` group of commands.
@@ -44,7 +46,7 @@ type Command struct {
4446
}
4547

4648
// Initialize allows to implement Command interface.
47-
func (c *Command) Initialize(app *kingpin.Application, cfg *servicecfg.Config) {
49+
func (c *Command) Initialize(app *kingpin.Application, _ *tctlcfg.GlobalCLIFlags, cfg *servicecfg.Config) {
4850
c.innerCmdMap = map[string]runFunc{}
4951

5052
auditCmd := app.Command("audit", "Audit command.")
@@ -114,13 +116,19 @@ func (c *Command) initAuditReportsCommands(auditCmd *kingpin.CmdClause, cfg *ser
114116

115117
type runFunc func(context.Context, *authclient.Client) error
116118

117-
func (c *Command) TryRun(ctx context.Context, selectedCommand string, authClient *authclient.Client) (match bool, err error) {
118-
handler, ok := c.innerCmdMap[selectedCommand]
119+
func (c *Command) TryRun(ctx context.Context, cmd string, clientFunc commonclient.InitFunc) (match bool, err error) {
120+
handler, ok := c.innerCmdMap[cmd]
119121
if !ok {
120122
return false, nil
121123
}
122124

123-
switch err := trail.FromGRPC(handler(ctx, authClient)); {
125+
client, closeFn, err := clientFunc(ctx)
126+
if err != nil {
127+
return false, trace.Wrap(err)
128+
}
129+
defer closeFn(ctx)
130+
131+
switch err := trail.FromGRPC(handler(ctx, client)); {
124132
case trace.IsNotImplemented(err):
125133
return true, trace.AccessDenied("Access Monitoring requires a Teleport Enterprise Auth Server.")
126134
default:

tool/tctl/common/acl_command.go

+17-7
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,8 @@ import (
3535
"github.com/gravitational/teleport/lib/auth/authclient"
3636
"github.com/gravitational/teleport/lib/service/servicecfg"
3737
"github.com/gravitational/teleport/lib/utils"
38+
commonclient "github.com/gravitational/teleport/tool/tctl/common/client"
39+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
3840
)
3941

4042
// ACLCommand implements the `tctl acl` family of commands.
@@ -64,7 +66,7 @@ const (
6466
)
6567

6668
// Initialize allows ACLCommand to plug itself into the CLI parser
67-
func (c *ACLCommand) Initialize(app *kingpin.Application, _ *servicecfg.Config) {
69+
func (c *ACLCommand) Initialize(app *kingpin.Application, _ *tctlcfg.GlobalCLIFlags, _ *servicecfg.Config) {
6870
acl := app.Command("acl", "Manage access lists.").Alias("access-lists")
6971

7072
c.ls = acl.Command("ls", "List cluster access lists.")
@@ -93,21 +95,29 @@ func (c *ACLCommand) Initialize(app *kingpin.Application, _ *servicecfg.Config)
9395
}
9496

9597
// TryRun takes the CLI command as an argument (like "acl ls") and executes it.
96-
func (c *ACLCommand) TryRun(ctx context.Context, cmd string, client *authclient.Client) (match bool, err error) {
98+
func (c *ACLCommand) TryRun(ctx context.Context, cmd string, clientFunc commonclient.InitFunc) (match bool, err error) {
99+
var commandFunc func(ctx context.Context, client *authclient.Client) error
97100
switch cmd {
98101
case c.ls.FullCommand():
99-
err = c.List(ctx, client)
102+
commandFunc = c.List
100103
case c.get.FullCommand():
101-
err = c.Get(ctx, client)
104+
commandFunc = c.Get
102105
case c.usersAdd.FullCommand():
103-
err = c.UsersAdd(ctx, client)
106+
commandFunc = c.UsersAdd
104107
case c.usersRemove.FullCommand():
105-
err = c.UsersRemove(ctx, client)
108+
commandFunc = c.UsersRemove
106109
case c.usersList.FullCommand():
107-
err = c.UsersList(ctx, client)
110+
commandFunc = c.UsersList
108111
default:
109112
return false, nil
110113
}
114+
client, closeFn, err := clientFunc(ctx)
115+
if err != nil {
116+
return false, trace.Wrap(err)
117+
}
118+
err = commandFunc(ctx, client)
119+
closeFn(ctx)
120+
111121
return true, trace.Wrap(err)
112122
}
113123

tool/tctl/common/admin_action_test.go

+5-2
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ import (
5858
"github.com/gravitational/teleport/lib/utils"
5959
"github.com/gravitational/teleport/lib/utils/hostid"
6060
tctl "github.com/gravitational/teleport/tool/tctl/common"
61+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
6162
testserver "github.com/gravitational/teleport/tool/teleport/testenv"
6263
tsh "github.com/gravitational/teleport/tool/tsh/common"
6364
)
@@ -1156,13 +1157,15 @@ func runTestCase(t *testing.T, ctx context.Context, client *authclient.Client, t
11561157

11571158
app := utils.InitCLIParser("tctl", tctl.GlobalHelpString)
11581159
cfg := servicecfg.MakeDefaultConfig()
1159-
tc.cliCommand.Initialize(app, cfg)
1160+
tc.cliCommand.Initialize(app, &tctlcfg.GlobalCLIFlags{}, cfg)
11601161

11611162
args := strings.Split(tc.command, " ")
11621163
commandName, err := app.Parse(args)
11631164
require.NoError(t, err)
11641165

1165-
match, err := tc.cliCommand.TryRun(ctx, commandName, client)
1166+
match, err := tc.cliCommand.TryRun(ctx, commandName, func(context.Context) (*authclient.Client, func(context.Context), error) {
1167+
return client, func(context.Context) {}, nil
1168+
})
11661169
require.True(t, match)
11671170
return err
11681171
}

tool/tctl/common/alert_command.go

+15-5
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,8 @@ import (
3737
"github.com/gravitational/teleport/lib/auth/authclient"
3838
libclient "github.com/gravitational/teleport/lib/client"
3939
"github.com/gravitational/teleport/lib/service/servicecfg"
40+
commonclient "github.com/gravitational/teleport/tool/tctl/common/client"
41+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
4042
)
4143

4244
// AlertCommand implements the `tctl alerts` family of commands.
@@ -62,7 +64,7 @@ type AlertCommand struct {
6264
}
6365

6466
// Initialize allows AlertCommand to plug itself into the CLI parser
65-
func (c *AlertCommand) Initialize(app *kingpin.Application, config *servicecfg.Config) {
67+
func (c *AlertCommand) Initialize(app *kingpin.Application, _ *tctlcfg.GlobalCLIFlags, config *servicecfg.Config) {
6668
c.config = config
6769
alert := app.Command("alerts", "Manage cluster alerts.").Alias("alert")
6870

@@ -93,17 +95,25 @@ func (c *AlertCommand) Initialize(app *kingpin.Application, config *servicecfg.C
9395
}
9496

9597
// TryRun takes the CLI command as an argument (like "alerts ls") and executes it.
96-
func (c *AlertCommand) TryRun(ctx context.Context, cmd string, client *authclient.Client) (match bool, err error) {
98+
func (c *AlertCommand) TryRun(ctx context.Context, cmd string, clientFunc commonclient.InitFunc) (match bool, err error) {
99+
var commandFunc func(ctx context.Context, client *authclient.Client) error
97100
switch cmd {
98101
case c.alertList.FullCommand():
99-
err = c.List(ctx, client)
102+
commandFunc = c.List
100103
case c.alertCreate.FullCommand():
101-
err = c.Create(ctx, client)
104+
commandFunc = c.Create
102105
case c.alertAck.FullCommand():
103-
err = c.Ack(ctx, client)
106+
commandFunc = c.Ack
104107
default:
105108
return false, nil
106109
}
110+
client, closeFn, err := clientFunc(ctx)
111+
if err != nil {
112+
return false, trace.Wrap(err)
113+
}
114+
err = commandFunc(ctx, client)
115+
closeFn(ctx)
116+
107117
return true, trace.Wrap(err)
108118
}
109119

tool/tctl/common/app_command.go

+13-3
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,8 @@ import (
3434
libclient "github.com/gravitational/teleport/lib/client"
3535
"github.com/gravitational/teleport/lib/service/servicecfg"
3636
"github.com/gravitational/teleport/lib/utils"
37+
commonclient "github.com/gravitational/teleport/tool/tctl/common/client"
38+
tctlcfg "github.com/gravitational/teleport/tool/tctl/common/config"
3739
)
3840

3941
// AppsCommand implements "tctl apps" group of commands.
@@ -55,7 +57,7 @@ type AppsCommand struct {
5557
}
5658

5759
// Initialize allows AppsCommand to plug itself into the CLI parser
58-
func (c *AppsCommand) Initialize(app *kingpin.Application, config *servicecfg.Config) {
60+
func (c *AppsCommand) Initialize(app *kingpin.Application, _ *tctlcfg.GlobalCLIFlags, config *servicecfg.Config) {
5961
c.config = config
6062

6163
apps := app.Command("apps", "Operate on applications registered with the cluster.")
@@ -68,13 +70,21 @@ func (c *AppsCommand) Initialize(app *kingpin.Application, config *servicecfg.Co
6870
}
6971

7072
// TryRun attempts to run subcommands like "apps ls".
71-
func (c *AppsCommand) TryRun(ctx context.Context, cmd string, client *authclient.Client) (match bool, err error) {
73+
func (c *AppsCommand) TryRun(ctx context.Context, cmd string, clientFunc commonclient.InitFunc) (match bool, err error) {
74+
var commandFunc func(ctx context.Context, client *authclient.Client) error
7275
switch cmd {
7376
case c.appsList.FullCommand():
74-
err = c.ListApps(ctx, client)
77+
commandFunc = c.ListApps
7578
default:
7679
return false, nil
7780
}
81+
client, closeFn, err := clientFunc(ctx)
82+
if err != nil {
83+
return false, trace.Wrap(err)
84+
}
85+
err = commandFunc(ctx, client)
86+
closeFn(ctx)
87+
7888
return true, trace.Wrap(err)
7989
}
8090

0 commit comments

Comments
 (0)