diff --git a/cmd/cli.go b/cmd/cli.go index 41ee8629..a245d30e 100644 --- a/cmd/cli.go +++ b/cmd/cli.go @@ -15,6 +15,7 @@ import ( "github.com/equinix/metal-cli/internal/gateway" "github.com/equinix/metal-cli/internal/hardware" initPkg "github.com/equinix/metal-cli/internal/init" + "github.com/equinix/metal-cli/internal/interconnections" "github.com/equinix/metal-cli/internal/ips" "github.com/equinix/metal-cli/internal/metros" "github.com/equinix/metal-cli/internal/organizations" @@ -93,5 +94,6 @@ func (cli *Cli) RegisterCommands(client *root.Client) { twofa.NewClient(client, cli.Outputer).NewCommand(), gateway.NewClient(client, cli.Outputer).NewCommand(), ports.NewClient(client, cli.Outputer).NewCommand(), + interconnections.NewClient(client, cli.Outputer).NewCommand(), ) } diff --git a/docs/metal.md b/docs/metal.md index b20f223a..c3974cd6 100644 --- a/docs/metal.md +++ b/docs/metal.md @@ -36,6 +36,7 @@ Command line interface for Equinix Metal * [metal gateway](metal_gateway.md) - Metal Gateway operations: create, delete, and retrieve. * [metal hardware-reservation](metal_hardware-reservation.md) - Hardware reservation operations: get, move. * [metal init](metal_init.md) - Create a configuration file. +* [metal interconnections](metal_interconnections.md) - interconnections operations: create, get. * [metal ip](metal_ip.md) - IP address, reservations, and assignment operations: assign, unassign, remove, available, request and get. * [metal metros](metal_metros.md) - Metro operations: get. * [metal operating-systems](metal_operating-systems.md) - Operating system operations: get. diff --git a/docs/metal_interconnections.md b/docs/metal_interconnections.md new file mode 100644 index 00000000..874bb9b4 --- /dev/null +++ b/docs/metal_interconnections.md @@ -0,0 +1,35 @@ +## metal interconnections + +interconnections operations: create, get. + +### Synopsis + +Get information on Metro locations. For more information on https://deploy.equinix.com/developers/docs/metal/interconnections. + +### Options + +``` + -h, --help help for interconnections +``` + +### Options inherited from parent commands + +``` + --config string Path to JSON or YAML configuration file + --exclude strings Comma separated Href references to collapse in results, may be dotted three levels deep + --filter stringArray Filter 'get' actions with name value pairs. Filter is not supported by all resources and is implemented as request query parameters. + --http-header strings Headers to add to requests (in format key=value) + --include strings Comma separated Href references to expand in results, may be dotted three levels deep + -o, --output string Output format (*table, json, yaml). env output formats are (*sh, terraform, capp). + --search string Search keyword for use in 'get' actions. Search is not supported by all resources. + --sort-by string Sort fields for use in 'get' actions. Sort is not supported by all resources. + --sort-dir string Sort field direction for use in 'get' actions. Sort is not supported by all resources. + --token string Metal API Token (METAL_AUTH_TOKEN) +``` + +### SEE ALSO + +* [metal](metal.md) - Command line interface for Equinix Metal +* [metal interconnections create](metal_interconnections_create.md) - Creates an interconnection. +* [metal interconnections get](metal_interconnections_get.md) - Retrieves a list of interconnections available to the default OrganizationID OR by project ID + diff --git a/docs/metal_interconnections_create.md b/docs/metal_interconnections_create.md new file mode 100644 index 00000000..b031ddd3 --- /dev/null +++ b/docs/metal_interconnections_create.md @@ -0,0 +1,56 @@ +## metal interconnections create + +Creates an interconnection. + +### Synopsis + +Creates a new interconnection as per the organization ID or project ID + +``` +metal interconnections create -n [-m ] [-r ] [-t ] [-p ] | [-O ] [flags] +``` + +### Examples + +``` + # Creates a new interconnection named "it-interconnection": + metal interconnections create -n [-m ] [-r ] [-t "dedicated" ] [-p ] | [-O ] + + metal interconnections create -n [-m ] [-r ] [-t "shared" ] [-p ] | [-O ] -T + + metal interconnections create -n [-m ] [-r ] [-t "shared" ] [-p ] | [-O ] -T -v +``` + +### Options + +``` + -h, --help help for create + -m, --metro string metro in the interconnection + -n, --name string Name of the interconnection + -O, --organizationID string Org ID + -p, --projectID string project ID + -r, --redundancy string Website URL of the organization. + -T, --serviceTokentype string service token type for interconnection either fabric OR Metal builds + -t, --type string type of of interconnection. + -v, --vrfs strings Return only the specified vrfs. +``` + +### Options inherited from parent commands + +``` + --config string Path to JSON or YAML configuration file + --exclude strings Comma separated Href references to collapse in results, may be dotted three levels deep + --filter stringArray Filter 'get' actions with name value pairs. Filter is not supported by all resources and is implemented as request query parameters. + --http-header strings Headers to add to requests (in format key=value) + --include strings Comma separated Href references to expand in results, may be dotted three levels deep + -o, --output string Output format (*table, json, yaml). env output formats are (*sh, terraform, capp). + --search string Search keyword for use in 'get' actions. Search is not supported by all resources. + --sort-by string Sort fields for use in 'get' actions. Sort is not supported by all resources. + --sort-dir string Sort field direction for use in 'get' actions. Sort is not supported by all resources. + --token string Metal API Token (METAL_AUTH_TOKEN) +``` + +### SEE ALSO + +* [metal interconnections](metal_interconnections.md) - interconnections operations: create, get. + diff --git a/docs/metal_interconnections_get.md b/docs/metal_interconnections_get.md new file mode 100644 index 00000000..1aa5acd3 --- /dev/null +++ b/docs/metal_interconnections_get.md @@ -0,0 +1,44 @@ +## metal interconnections get + +Retrieves a list of interconnections available to the default OrganizationID OR by project ID + +### Synopsis + +Retrieves a list of interconnections available to the default Organizations and by project ID + +``` +metal interconnections get [flags] +``` + +### Examples + +``` + # Lists of interconnections available to the default OrganizationID OR by project ID: + metal interconnections get +``` + +### Options + +``` + -h, --help help for get +``` + +### Options inherited from parent commands + +``` + --config string Path to JSON or YAML configuration file + --exclude strings Comma separated Href references to collapse in results, may be dotted three levels deep + --filter stringArray Filter 'get' actions with name value pairs. Filter is not supported by all resources and is implemented as request query parameters. + --http-header strings Headers to add to requests (in format key=value) + --include strings Comma separated Href references to expand in results, may be dotted three levels deep + -o, --output string Output format (*table, json, yaml). env output formats are (*sh, terraform, capp). + --search string Search keyword for use in 'get' actions. Search is not supported by all resources. + --sort-by string Sort fields for use in 'get' actions. Sort is not supported by all resources. + --sort-dir string Sort field direction for use in 'get' actions. Sort is not supported by all resources. + --token string Metal API Token (METAL_AUTH_TOKEN) +``` + +### SEE ALSO + +* [metal interconnections](metal_interconnections.md) - interconnections operations: create, get. + diff --git a/internal/interconnections/create.go b/internal/interconnections/create.go new file mode 100644 index 00000000..4a3ce881 --- /dev/null +++ b/internal/interconnections/create.go @@ -0,0 +1,97 @@ +// Copyright © 2018 Jasmin Gacic +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in +// all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +// THE SOFTWARE. + +package interconnections + +import ( + "context" + "fmt" + + metal "github.com/equinix-labs/metal-go/metal/v1" + "github.com/spf13/cobra" +) + +func (c *Client) Create() *cobra.Command { + var name, metro, redundancy, connType, projectID, organizationID string + var serviceTokenType string + var vrfs []string + + createInterconnectionsCmd := &cobra.Command{ + Use: `create -n [-m ] [-r ] [-t ] [-p ] | [-O ]`, + Short: "Creates an interconnection.", + Long: "Creates a new interconnection as per the organization ID or project ID ", + Example: ` # Creates a new interconnection named "it-interconnection": + metal interconnections create -n [-m ] [-r ] [-t "dedicated" ] [-p ] | [-O ] + + metal interconnections create -n [-m ] [-r ] [-t "shared" ] [-p ] | [-O ] -T + + metal interconnections create -n [-m ] [-r ] [-t "shared" ] [-p ] | [-O ] -T -v `, + + RunE: func(cmd *cobra.Command, args []string) error { + cmd.SilenceUsage = true + var interconn *metal.Interconnection + var proErr error + var orgErr error + + interconnectionCreateInput := metal.NewInterconnectionCreateInput(metro, name, redundancy, connType) + if connType == "shared" { + interconnectionCreateInput.SetServiceTokenType(serviceTokenType) + } + + if len(vrfs) > 0 { + interconnectionCreateInput.SetVrfs(vrfs) + } + + if projectID != "" { + interconn, _, proErr = c.Service.CreateProjectInterconnection(context.Background(), projectID).InterconnectionCreateInput(*interconnectionCreateInput).Include(nil).Exclude(nil).Execute() + if proErr != nil { + return fmt.Errorf("could not create interconnections: %w", proErr) + } + } else { + interconn, _, orgErr = c.Service.CreateOrganizationInterconnection(context.Background(), organizationID).InterconnectionCreateInput(*interconnectionCreateInput).Include(nil).Exclude(nil).Execute() + if orgErr != nil { + return fmt.Errorf("could not create interconnections: %w", orgErr) + } + } + + data := make([][]string, 1) + + data[0] = []string{interconn.GetId(), interconn.GetName(), interconn.GetType(), interconn.GetCreatedAt().String()} + header := []string{"ID", "Name", "Type", "Created"} + + return c.Out.Output(interconn, header, &data) + }, + } + + createInterconnectionsCmd.Flags().StringVarP(&name, "name", "n", "", "Name of the interconnection") + createInterconnectionsCmd.Flags().StringVarP(&metro, "metro", "m", "", "metro in the interconnection") + createInterconnectionsCmd.Flags().StringVarP(&redundancy, "redundancy", "r", "", "Website URL of the organization.") + createInterconnectionsCmd.Flags().StringVarP(&connType, "type", "t", "", "type of of interconnection.") + createInterconnectionsCmd.Flags().StringVarP(&connType, "serviceTokentype", "T", "", "service token type for interconnection either fabric OR Metal builds") + createInterconnectionsCmd.Flags().StringSliceVarP(&vrfs, "vrfs", "v", []string{}, "Return only the specified vrfs.") + createInterconnectionsCmd.Flags().StringVarP(&connType, "projectID", "p", "", "project ID") + createInterconnectionsCmd.Flags().StringVarP(&connType, "organizationID", "O", "", "Org ID") + + _ = createInterconnectionsCmd.MarkFlagRequired("name") + _ = createInterconnectionsCmd.MarkFlagRequired("metro") + _ = createInterconnectionsCmd.MarkFlagRequired("redundancy") + _ = createInterconnectionsCmd.MarkFlagRequired("type") + return createInterconnectionsCmd +} diff --git a/internal/interconnections/interconnections.go b/internal/interconnections/interconnections.go new file mode 100644 index 00000000..1faaab29 --- /dev/null +++ b/internal/interconnections/interconnections.go @@ -0,0 +1,71 @@ +// Copyright © 2018 Jasmin Gacic +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in +// all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +// THE SOFTWARE. + +package interconnections + +import ( + metal "github.com/equinix-labs/metal-go/metal/v1" + "github.com/equinix/metal-cli/internal/outputs" + "github.com/spf13/cobra" +) + +type Client struct { + Servicer Servicer + Service metal.InterconnectionsApiService + Out outputs.Outputer +} + +func (c *Client) NewCommand() *cobra.Command { + cmd := &cobra.Command{ + Use: `interconnections`, + Aliases: []string{"conn"}, + Short: "interconnections operations: create, get.", + Long: "Get information on Metro locations. For more information on https://deploy.equinix.com/developers/docs/metal/interconnections.", + PersistentPreRun: func(cmd *cobra.Command, args []string) { + if root := cmd.Root(); root != nil { + if root.PersistentPreRun != nil { + root.PersistentPreRun(cmd, args) + } + } + c.Service = *c.Servicer.MetalAPI(cmd).InterconnectionsApi + }, + } + + cmd.AddCommand( + c.Retrieve(), + c.Create(), + ) + + return cmd +} + +type Servicer interface { + MetalAPI(*cobra.Command) *metal.APIClient + Filters() map[string]string + Includes(defaultIncludes []string) (incl []string) + Excludes(defaultExcludes []string) (excl []string) +} + +func NewClient(s Servicer, out outputs.Outputer) *Client { + return &Client{ + Servicer: s, + Out: out, + } +} diff --git a/internal/interconnections/retrieve.go b/internal/interconnections/retrieve.go new file mode 100644 index 00000000..0aa7a09d --- /dev/null +++ b/internal/interconnections/retrieve.go @@ -0,0 +1,73 @@ +// Copyright © 2018 Jasmin Gacic +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in +// all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +// THE SOFTWARE. + +package interconnections + +import ( + "context" + "fmt" + + metal "github.com/equinix-labs/metal-go/metal/v1" + pager "github.com/equinix/metal-cli/internal/pagination" + "github.com/spf13/cobra" +) + +func (c *Client) Retrieve() *cobra.Command { + // metrosCmd represents the metros command + var projectID, organizationID string + retrieveInterconnectionsCmd := &cobra.Command{ + Use: "get", + Aliases: []string{"list"}, + Short: "Retrieves a list of interconnections available to the default OrganizationID OR by project ID", + Long: "Retrieves a list of interconnections available to the default Organizations and by project ID", + Example: ` # Lists of interconnections available to the default OrganizationID OR by project ID: + metal interconnections get`, + + RunE: func(cmd *cobra.Command, args []string) error { + cmd.SilenceUsage = true + var interConnections []metal.Interconnection + var proErr error + inc := []string{} + exc := []string{} + if projectID != "" { + interConnections, proErr = pager.GetAllProjectInterconnections(c.Service, projectID, inc, exc) + if proErr != nil { + return fmt.Errorf("Could not list Project Interconnections: %w", proErr) + } + } else { + interConnectionsList, _, err := c.Service.OrganizationListInterconnections(context.Background(), organizationID).Include(inc).Exclude(exc).Execute() + if err != nil { + return fmt.Errorf("Could not list Organization Interconnections: %w", err) + } + interConnections = interConnectionsList.GetInterconnections() + } + + data := make([][]string, len(interConnections)) + + for i, interConn := range interConnections { + data[i] = []string{interConn.GetId(), interConn.GetName(), interConn.GetType(), interConn.GetCreatedAt().String()} + } + header := []string{"ID", "Name", "Code", "Type", "Created"} + + return c.Out.Output(interConnections, header, &data) + }, + } + return retrieveInterconnectionsCmd +} diff --git a/internal/pagination/pager.go b/internal/pagination/pager.go index fd8c728a..71bfbec9 100644 --- a/internal/pagination/pager.go +++ b/internal/pagination/pager.go @@ -127,3 +127,23 @@ func GetAllOrganizations(s metal.OrganizationsApiService, include, exclude []str return orgs, nil } } + +func GetAllProjectInterconnections(s metal.InterconnectionsApiService, projectId string, include []string, exclude []string) ([]metal.Interconnection, error) { + var interConn []metal.Interconnection + + page := int32(1) // int32 | Page to return (optional) (default to 1) + perPage := int32(20) // int32 | Items returned per page (optional) (default to 10) + + for { + interConnPage, _, err := s.ProjectListInterconnections(context.Background(), projectId).Include(include).Exclude(exclude).Page(page).PerPage(perPage).Execute() + if err != nil { + return nil, err + } + interConn = append(interConn, interConnPage.GetInterconnections()...) + if interConnPage.Meta.GetLastPage() > interConnPage.Meta.GetCurrentPage() { + page = page + 1 + continue + } + return interConn, nil + } +}