@@ -19,6 +19,7 @@ package session
19
19
import (
20
20
"bytes"
21
21
"context"
22
+ "encoding/base64"
22
23
"errors"
23
24
"flag"
24
25
"fmt"
@@ -33,6 +34,7 @@ import (
33
34
"github.com/vmware/govmomi/cli/flags"
34
35
"github.com/vmware/govmomi/session"
35
36
"github.com/vmware/govmomi/sts"
37
+ "github.com/vmware/govmomi/vapi/authentication"
36
38
"github.com/vmware/govmomi/vapi/rest"
37
39
"github.com/vmware/govmomi/vim25"
38
40
"github.com/vmware/govmomi/vim25/methods"
@@ -45,6 +47,7 @@ type login struct {
45
47
46
48
clone bool
47
49
issue bool
50
+ jwt string
48
51
renew bool
49
52
long bool
50
53
vapi bool
@@ -69,6 +72,7 @@ func (cmd *login) Register(ctx context.Context, f *flag.FlagSet) {
69
72
70
73
f .BoolVar (& cmd .clone , "clone" , false , "Acquire clone ticket" )
71
74
f .BoolVar (& cmd .issue , "issue" , false , "Issue SAML token" )
75
+ f .StringVar (& cmd .jwt , "jwt" , "" , "Exchange SAML token for JWT audience" )
72
76
f .BoolVar (& cmd .renew , "renew" , false , "Renew SAML token" )
73
77
f .BoolVar (& cmd .vapi , "r" , false , "REST login" )
74
78
f .DurationVar (& cmd .life , "lifetime" , time .Minute * 10 , "SAML token lifetime" )
@@ -103,6 +107,7 @@ The session.login command can be used to:
103
107
- Login using a vCenter Extension certificate
104
108
- Issue a SAML token
105
109
- Renew a SAML token
110
+ - Exchange a SAML token for a JSON Web Token (JWT)
106
111
- Login using a SAML token
107
112
- Impersonate a user
108
113
- Avoid passing credentials to other govc commands
@@ -124,6 +129,7 @@ Examples:
124
129
token=$(govc session.login -u host -cert user.crt -key user.key -issue -token "$bearer")
125
130
govc session.login -u host -cert user.crt -key user.key -token "$token"
126
131
token=$(govc session.login -u host -cert user.crt -key user.key -renew -lifetime 24h -token "$token")
132
+ govc session.login -jwt vmware-tes:vc:nsxd-v2:nsx -token "$token"
127
133
# HTTP requests
128
134
govc session.login -r -X GET /api/vcenter/namespace-management/clusters | jq .
129
135
govc session.login -r -X POST /rest/vcenter/cluster/modules <<<'{"spec": {"cluster": "domain-c9"}}'`
@@ -210,6 +216,22 @@ func (cmd *login) issueToken(ctx context.Context, vc *vim25.Client) (string, err
210
216
return s .Token , nil
211
217
}
212
218
219
+ func (cmd * login ) exchangeTokenJWT (ctx context.Context , c * rest.Client ) (string , error ) {
220
+ spec := authentication.TokenIssueSpec {
221
+ Audience : cmd .jwt ,
222
+ GrantType : "urn:ietf:params:oauth:grant-type:token-exchange" ,
223
+ RequestedTokenType : "urn:ietf:params:oauth:token-type:id_token" ,
224
+ SubjectToken : base64 .StdEncoding .EncodeToString ([]byte (cmd .token )),
225
+ SubjectTokenType : "urn:ietf:params:oauth:token-type:saml2" ,
226
+ }
227
+
228
+ info , err := authentication .NewManager (c ).Issue (ctx , spec )
229
+ if err != nil {
230
+ return "" , err
231
+ }
232
+ return info .AccessToken , nil
233
+ }
234
+
213
235
func (cmd * login ) loginByToken (ctx context.Context , c * vim25.Client ) error {
214
236
header := soap.Header {
215
237
Security : & sts.Signer {
@@ -340,6 +362,8 @@ func (cmd *login) Run(ctx context.Context, f *flag.FlagSet) error {
340
362
case cmd .issue :
341
363
cmd .Session .LoginSOAP = nologinSOAP
342
364
cmd .Session .LoginREST = nologinREST
365
+ case cmd .jwt != "" :
366
+ cmd .Session .LoginSOAP = nologinSOAP
343
367
}
344
368
345
369
c , err := cmd .Client ()
@@ -349,6 +373,14 @@ func (cmd *login) Run(ctx context.Context, f *flag.FlagSet) error {
349
373
350
374
r := & ticketResult {cmd : cmd }
351
375
376
+ var rc * rest.Client
377
+ if cmd .vapi || cmd .jwt != "" {
378
+ rc , err = cmd .RestClient ()
379
+ if err != nil {
380
+ return err
381
+ }
382
+ }
383
+
352
384
switch {
353
385
case cmd .clone :
354
386
m := session .NewManager (c )
@@ -362,14 +394,8 @@ func (cmd *login) Run(ctx context.Context, f *flag.FlagSet) error {
362
394
return err
363
395
}
364
396
return cmd .WriteResult (r )
365
- }
366
-
367
- var rc * rest.Client
368
- if cmd .vapi {
369
- rc , err = cmd .RestClient ()
370
- if err != nil {
371
- return err
372
- }
397
+ case cmd .jwt != "" :
398
+ r .Token , err = cmd .exchangeTokenJWT (ctx , rc )
373
399
}
374
400
375
401
if f .NArg () == 1 {
0 commit comments