diff --git a/api/oidc.go b/api/oidc.go index d11e35726c..b28378894e 100644 --- a/api/oidc.go +++ b/api/oidc.go @@ -10,21 +10,24 @@ type OIDCToken struct { } type OIDCTokenRequest struct { - Job string - Audience string - Lifetime int - Claims []string + Job string + Audience string + Lifetime int + Claims []string + AWSSessionTags []string } func (c *Client) OIDCToken(ctx context.Context, methodReq *OIDCTokenRequest) (*OIDCToken, *Response, error) { m := &struct { - Audience string `json:"audience,omitempty"` - Lifetime int `json:"lifetime,omitempty"` - Claims []string `json:"claims,omitempty"` + Audience string `json:"audience,omitempty"` + Lifetime int `json:"lifetime,omitempty"` + Claims []string `json:"claims,omitempty"` + AWSSessionTags []string `json:"aws_session_tags,omitempty"` }{ - Audience: methodReq.Audience, - Lifetime: methodReq.Lifetime, - Claims: methodReq.Claims, + Audience: methodReq.Audience, + Lifetime: methodReq.Lifetime, + Claims: methodReq.Claims, + AWSSessionTags: methodReq.AWSSessionTags, } u := fmt.Sprintf("jobs/%s/oidc/tokens", railsPathEscape(methodReq.Job)) diff --git a/api/oidc_test.go b/api/oidc_test.go index be1227d196..4861cc9322 100644 --- a/api/oidc_test.go +++ b/api/oidc_test.go @@ -124,6 +124,24 @@ func TestOIDCToken(t *testing.T) { ExpectedBody: []byte(fmt.Sprintf(`{"lifetime":%d}`+"\n", lifetime)), OIDCToken: &api.OIDCToken{Token: oidcToken}, }, + { + AccessToken: accessToken, + OIDCTokenRequest: &api.OIDCTokenRequest{ + Job: jobID, + Claims: []string{"organization_id", "pipeline_id"}, + }, + ExpectedBody: []byte(`{"claims":["organization_id","pipeline_id"]}` + "\n"), + OIDCToken: &api.OIDCToken{Token: oidcToken}, + }, + { + AccessToken: accessToken, + OIDCTokenRequest: &api.OIDCTokenRequest{ + Job: jobID, + AWSSessionTags: []string{"organization_id", "pipeline_id"}, + }, + ExpectedBody: []byte(`{"aws_session_tags":["organization_id","pipeline_id"]}` + "\n"), + OIDCToken: &api.OIDCToken{Token: oidcToken}, + }, } for _, test := range tests { diff --git a/clicommand/oidc_request_token.go b/clicommand/oidc_request_token.go index 382df8816f..aa5e1ab106 100644 --- a/clicommand/oidc_request_token.go +++ b/clicommand/oidc_request_token.go @@ -16,7 +16,8 @@ type OIDCTokenConfig struct { Lifetime int `cli:"lifetime"` Job string `cli:"job" validate:"required"` // TODO: enumerate possible values, perhaps by adding a link to the documentation - Claims []string `cli:"claim" normalize:"list"` + Claims []string `cli:"claim" normalize:"list"` + AWSSessionTags []string `cli:"aws-session-tag" normalize:"list"` // Global flags Debug bool `cli:"debug"` @@ -79,6 +80,13 @@ var OIDCRequestTokenCommand = cli.Command{ EnvVar: "BUILDKITE_OIDC_TOKEN_CLAIMS", }, + cli.StringSliceFlag{ + Name: "aws-session-tag", + Value: &cli.StringSlice{}, + Usage: "Add claims as AWS Session Tags", + EnvVar: "BUILDKITE_OIDC_TOKEN_AWS_SESSION_TAGS", + }, + // API Flags AgentAccessTokenFlag, EndpointFlag, @@ -112,10 +120,11 @@ var OIDCRequestTokenCommand = cli.Command{ ) token, err := roko.DoFunc(ctx, r, func(r *roko.Retrier) (*api.OIDCToken, error) { req := &api.OIDCTokenRequest{ - Job: cfg.Job, - Audience: cfg.Audience, - Lifetime: cfg.Lifetime, - Claims: cfg.Claims, + Job: cfg.Job, + Audience: cfg.Audience, + Lifetime: cfg.Lifetime, + Claims: cfg.Claims, + AWSSessionTags: cfg.AWSSessionTags, } token, resp, err := client.OIDCToken(ctx, req)