forked from CircleCI-Public/circleci-cli
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add a new command, `context`: ``` $ circleci context Contexts provide a mechanism for securing and sharing environment variables across projects. The environment variables are defined as name/value pairs and are injected at runtime. Usage: circleci context [command] Available Commands: list List contexts remove Remove a secret from the named context show Show a context store Store an new secret in the named context. The value is read from stdin. ``` Currently, the commands enable users to list contexts, show contexts, and to add and remove variables from contexts. Not yet supported / things to do: - write tests - delete context - CRUD on context groups Co-Authored-By: Alex Engelberg <[email protected]>
- Loading branch information
1 parent
7f7e4e2
commit 2e3457d
Showing
11 changed files
with
766 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
package api_test | ||
|
||
import ( | ||
"testing" | ||
|
||
. "github.com/onsi/ginkgo" | ||
. "github.com/onsi/gomega" | ||
) | ||
|
||
func TestApi(t *testing.T) { | ||
RegisterFailHandler(Fail) | ||
RunSpecs(t, "Api Suite") | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,288 @@ | ||
// Go functions that expose the Context-related calls in the GraphQL API. | ||
package api | ||
|
||
import ( | ||
"fmt" | ||
"strings" | ||
|
||
"github.com/CircleCI-Public/circleci-cli/client" | ||
"github.com/pkg/errors" | ||
) | ||
|
||
type Resource struct { | ||
Variable string | ||
CreatedAt string | ||
TruncatedValue string | ||
} | ||
|
||
type CircleCIContext struct { | ||
ID string | ||
Name string | ||
CreatedAt string | ||
Groups struct { | ||
} | ||
Resources []Resource | ||
} | ||
|
||
type ContextsQueryResponse struct { | ||
Organization struct { | ||
Id string | ||
Contexts struct { | ||
Edges []struct { | ||
Node CircleCIContext | ||
} | ||
} | ||
} | ||
} | ||
|
||
func improveVcsTypeError(err error) error { | ||
if responseErrors, ok := err.(client.ResponseErrorsCollection); ok { | ||
if len(responseErrors) > 0 { | ||
details := responseErrors[0].Extensions | ||
if details.EnumType == "VCSType" { | ||
allowedValues := strings.ToLower(strings.Join(details.AllowedValues[:], ", ")) | ||
return fmt.Errorf("Invalid vcs-type '%s' provided, expected one of %s", strings.ToLower(details.Value), allowedValues) | ||
} | ||
} | ||
} | ||
return err | ||
} | ||
|
||
func CreateContext(cl *client.Client, vcsType, orgName, contextName string) error { | ||
|
||
org, err := getOrganization(cl, orgName, vcsType) | ||
|
||
if err != nil { | ||
return err | ||
} | ||
|
||
query := ` | ||
mutation CreateContext($input: CreateContextInput!) { | ||
createContext(input: $input) { | ||
...CreateButton | ||
} | ||
} | ||
fragment CreateButton on CreateContextPayload { | ||
error { | ||
type | ||
} | ||
} | ||
` | ||
|
||
var input struct { | ||
OwnerId string `json:"ownerId"` | ||
OwnerType string `json:"ownerType"` | ||
ContextName string `json:"contextName"` | ||
} | ||
|
||
input.OwnerId = org.Organization.ID | ||
input.OwnerType = "ORGANIZATION" | ||
input.ContextName = contextName | ||
|
||
request := client.NewRequest(query) | ||
request.SetToken(cl.Token) | ||
request.Var("input", input) | ||
|
||
var response struct { | ||
CreateContext struct { | ||
Error struct { | ||
Type string | ||
} | ||
} | ||
} | ||
|
||
if err = cl.Run(request, &response); err != nil { | ||
return improveVcsTypeError(err) | ||
} | ||
|
||
if response.CreateContext.Error.Type != "" { | ||
return fmt.Errorf("Error creating context: %s", response.CreateContext.Error.Type) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func ListContexts(cl *client.Client, orgName, vcsType string) (*ContextsQueryResponse, error) { | ||
|
||
query := ` | ||
query ContextsQuery($orgName: String!, $vcsType: VCSType!) { | ||
organization(name: $orgName, vcsType: $vcsType) { | ||
id | ||
contexts { | ||
edges { | ||
node { | ||
...Context | ||
} | ||
} | ||
} | ||
} | ||
} | ||
fragment Context on Context { | ||
id | ||
name | ||
createdAt | ||
groups { | ||
edges { | ||
node { | ||
...SecurityGroups | ||
} | ||
} | ||
} | ||
resources { | ||
...EnvVars | ||
} | ||
} | ||
fragment EnvVars on EnvironmentVariable { | ||
variable | ||
createdAt | ||
truncatedValue | ||
} | ||
fragment SecurityGroups on Group { | ||
id | ||
name | ||
} | ||
` | ||
|
||
request := client.NewRequest(query) | ||
request.SetToken(cl.Token) | ||
|
||
request.Var("orgName", orgName) | ||
request.Var("vcsType", strings.ToUpper(vcsType)) | ||
|
||
var response ContextsQueryResponse | ||
err := cl.Run(request, &response) | ||
return &response, errors.Wrapf(improveVcsTypeError(err), "failed to load context list") | ||
} | ||
|
||
func DeleteEnvironmentVariable(cl *client.Client, contextId, variableName string) error { | ||
query := ` | ||
mutation DeleteEnvVar($input: RemoveEnvironmentVariableInput!) { | ||
removeEnvironmentVariable(input: $input) { | ||
context { | ||
id | ||
resources { | ||
...EnvVars | ||
} | ||
} | ||
} | ||
} | ||
fragment EnvVars on EnvironmentVariable { | ||
variable | ||
createdAt | ||
truncatedValue | ||
}` | ||
|
||
var input struct { | ||
ContextId string `json:"contextId"` | ||
Variable string `json:"variable"` | ||
} | ||
|
||
input.ContextId = contextId | ||
input.Variable = variableName | ||
|
||
request := client.NewRequest(query) | ||
request.SetToken(cl.Token) | ||
request.Var("input", input) | ||
|
||
var response struct { | ||
RemoveEnvironmentVariable struct { | ||
Context CircleCIContext | ||
} | ||
} | ||
|
||
err := cl.Run(request, &response) | ||
return errors.Wrap(improveVcsTypeError(err), "failed to delete environment varaible") | ||
} | ||
|
||
func StoreEnvironmentVariable(cl *client.Client, contextId, variableName, secretValue string) error { | ||
query := ` | ||
mutation CreateEnvVar($input: StoreEnvironmentVariableInput!) { | ||
storeEnvironmentVariable(input: $input) { | ||
context { | ||
id | ||
resources { | ||
...EnvVars | ||
} | ||
} | ||
...CreateEnvVarButton | ||
} | ||
} | ||
fragment EnvVars on EnvironmentVariable { | ||
variable | ||
createdAt | ||
truncatedValue | ||
} | ||
fragment CreateEnvVarButton on StoreEnvironmentVariablePayload { | ||
error { | ||
type | ||
} | ||
}` | ||
|
||
request := client.NewRequest(query) | ||
request.SetToken(cl.Token) | ||
|
||
var input struct { | ||
ContextId string `json:"contextId"` | ||
Variable string `json:"variable"` | ||
Value string `json:"value"` | ||
} | ||
|
||
input.ContextId = contextId | ||
input.Variable = variableName | ||
input.Value = secretValue | ||
|
||
request.Var("input", input) | ||
|
||
var response struct { | ||
StoreEnvironmentVariable struct { | ||
Context CircleCIContext | ||
Error struct { | ||
Type string | ||
} | ||
} | ||
} | ||
|
||
if err := cl.Run(request, &response); err != nil { | ||
return errors.Wrap(improveVcsTypeError(err), "failed to store environment varaible in context") | ||
} | ||
|
||
if response.StoreEnvironmentVariable.Error.Type != "" { | ||
return fmt.Errorf("Error storing environment variable: %s", response.StoreEnvironmentVariable.Error.Type) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func DeleteContext(cl *client.Client, contextId string) error { | ||
query := ` | ||
mutation DeleteContext($input: DeleteContextInput!) { | ||
deleteContext(input: $input) { | ||
clientMutationId | ||
} | ||
}` | ||
|
||
request := client.NewRequest(query) | ||
request.SetToken(cl.Token) | ||
|
||
var input struct { | ||
ContextId string `json:"contextId"` | ||
} | ||
|
||
input.ContextId = contextId | ||
request.Var("input", input) | ||
|
||
var response struct { | ||
} | ||
|
||
err := cl.Run(request, &response) | ||
|
||
return errors.Wrap(improveVcsTypeError(err), "failed to delete context") | ||
} |
Oops, something went wrong.