Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ generate: deps
mockgen --destination ./internal/mocks/racks_service.go --package=mocks --source ./vendor/github.com/serverscom/serverscom-go-client/pkg/racks.go
mockgen --destination ./internal/mocks/invoices_service.go --package=mocks --source ./vendor/github.com/serverscom/serverscom-go-client/pkg/invoices.go
mockgen --destination ./internal/mocks/account_service.go --package=mocks --source ./vendor/github.com/serverscom/serverscom-go-client/pkg/accounts.go
mockgen --destination ./internal/mocks/locations_service.go --package=mocks --source ./vendor/github.com/serverscom/serverscom-go-client/pkg/locations.go
mockgen --destination ./internal/mocks/kubernetes_clusters_service.go --package=mocks --source ./vendor/github.com/serverscom/serverscom-go-client/pkg/kubernetes_clusters.go
sed -i '' 's|github.com/serverscom/srvctl/vendor/github.com/serverscom/serverscom-go-client/pkg|github.com/serverscom/serverscom-go-client/pkg|g' \
./internal/mocks/ssh_service.go \
./internal/mocks/hosts_service.go \
Expand All @@ -23,5 +25,7 @@ generate: deps
./internal/mocks/racks_service.go \
./internal/mocks/invoices_service.go \
./internal/mocks/account_service.go \
./internal/mocks/collection.go
./internal/mocks/collection.go \
./internal/mocks/locations_service.go \
./internal/mocks/kubernetes_clusters_service.go

1 change: 0 additions & 1 deletion cmd/base/flags.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@ func AddGlobalFlags(cmd *cobra.Command) {
cmd.PersistentFlags().StringP("output", "o", "text", "output format (text/json/yaml)")
// define help flag without shorthand before cobra adds it by default to avoid conflict with no-header flag shorthand
cmd.PersistentFlags().Bool("help", false, "Print usage")
cmd.PersistentFlags().Lookup("help").Hidden = true
cmd.PersistentFlags().BoolP("no-header", "h", false, "print output without headers")
}

Expand Down
211 changes: 184 additions & 27 deletions cmd/entities/hosts/add.go
Original file line number Diff line number Diff line change
@@ -1,53 +1,210 @@
package hosts

import (
"context"
"fmt"
"log"
"os"

"maps"

serverscom "github.com/serverscom/serverscom-go-client/pkg"
"github.com/serverscom/srvctl/cmd/base"
"github.com/spf13/cobra"
"github.com/spf13/pflag"
)

type HostCreator interface {
Create(ctx context.Context, client *serverscom.Client, input any) (any, error)
NewCreateInput() any
type AddDSFlags struct {
InputPath string
LocationID int
ServerModelID int
OperatingSystemID int
Features []string
RAMSize int
PublicUplinkID int
PublicBandwidthID int
PrivateUplinkID int
DriveSlots map[string]int
Layout []string
Partitions []string
IPv6 bool
UserDataFile string
UserData string
Labels map[string]string
}

type DSCreateMgr struct{}
func applyFlagsToInput(
input *serverscom.DedicatedServerCreateInput,
flags *AddDSFlags,
pflags *pflag.FlagSet,
) error {
if pflags.Changed("location-id") {
input.LocationID = int64(flags.LocationID)
}
if pflags.Changed("server-model-id") {
input.ServerModelID = int64(flags.ServerModelID)
}
if pflags.Changed("operating-system-id") {
val := int64(flags.OperatingSystemID)
input.OperatingSystemID = &val
}
if pflags.Changed("feature") {
input.Features = flags.Features
}
if pflags.Changed("ram-size") {
input.RAMSize = flags.RAMSize
}
if pflags.Changed("public-uplink-id") {
if input.UplinkModels.Public == nil {
input.UplinkModels.Public = &serverscom.DedicatedServerPublicUplinkInput{}
}
input.UplinkModels.Public.ID = int64(flags.PublicUplinkID)
}
if pflags.Changed("public-bandwidth-id") {
if input.UplinkModels.Public == nil {
input.UplinkModels.Public = &serverscom.DedicatedServerPublicUplinkInput{}
}
input.UplinkModels.Public.BandwidthModelID = int64(flags.PublicBandwidthID)
}
if pflags.Changed("private-uplink-id") {
input.UplinkModels.Private.ID = int64(flags.PrivateUplinkID)
}

func (c *DSCreateMgr) Create(ctx context.Context, client *serverscom.Client, input any) (any, error) {
dsInput, ok := input.(*serverscom.DedicatedServerCreateInput)
if !ok {
return nil, fmt.Errorf("invalid input type for dedicated server")
if pflags.Changed("drive-slots") {
slots, err := parseDriveSlots(flags.DriveSlots)
if err != nil {
return err
}
input.Drives.Slots = slots
}
return client.Hosts.CreateDedicatedServers(ctx, *dsInput)
}

func (c *DSCreateMgr) NewCreateInput() any {
return &serverscom.DedicatedServerCreateInput{}
}
if pflags.Changed("layout") {
layouts, err := parseLayout(flags.Layout)
if err != nil {
return err
}
input.Drives.Layout = mergeLayouts(input.Drives.Layout, layouts)
}

type SBMCreateMgr struct{}
if pflags.Changed("partition") {
if len(input.Drives.Layout) == 0 {
return fmt.Errorf("partition given but layout is empty")
}
partitions, err := parsePartitions(flags.Partitions)
if err != nil {
return err
}
err = applyPartitions(input.Drives.Layout, partitions)
if err != nil {
return err
}
}
if pflags.Changed("ipv6") {
input.IPv6 = flags.IPv6
}
if pflags.Changed("user-data") && pflags.Changed("user-data-file") {
return fmt.Errorf("'user-data' and 'user-data-file' can't be used together")
}
if pflags.Changed("user-data-file") {
data, err := os.ReadFile(flags.UserDataFile)
if err != nil {
return fmt.Errorf("can't read user-data-file: %v", err)
}
dataStr := string(data)
input.UserData = &dataStr
}
if pflags.Changed("user-data") {
input.UserData = &flags.UserData
}

func (c *SBMCreateMgr) Create(ctx context.Context, client *serverscom.Client, input any) (any, error) {
sbmInput, ok := input.(*serverscom.SBMServerCreateInput)
if !ok {
return nil, fmt.Errorf("invalid input type for SBM server")
if pflags.Changed("labels") {
for i := range input.Hosts {
maps.Copy(input.Hosts[i].Labels, flags.Labels)
}
}
return client.Hosts.CreateSBMServers(ctx, *sbmInput)
return nil
}

func (c *SBMCreateMgr) NewCreateInput() any {
return &serverscom.SBMServerCreateInput{}
func newAddDSCmd(cmdContext *base.CmdContext) *cobra.Command {
flags := &AddDSFlags{}

cmd := &cobra.Command{
Use: "add",
Short: "Create a dedicated server",
Args: cobra.ArbitraryArgs,
RunE: func(cmd *cobra.Command, args []string) error {

manager := cmdContext.GetManager()
ctx, cancel := base.SetupContext(cmd, manager)
defer cancel()

base.SetupProxy(cmd, manager)

input := serverscom.DedicatedServerCreateInput{}

if flags.InputPath != "" {
if err := base.ReadInputJSON(flags.InputPath, cmd.InOrStdin(), &input); err != nil {
return err
}
}

if len(input.Hosts) == 0 && len(args) == 0 {
return fmt.Errorf("no hosts found from positional args and no hosts found from input, can't continue")
}

for _, hostname := range args {
input.Hosts = append(input.Hosts, serverscom.DedicatedServerHostInput{
Hostname: hostname,
Labels: make(map[string]string),
})
}

err := applyFlagsToInput(&input, flags, cmd.Flags())
if err != nil {
return err
}

scClient := cmdContext.GetClient().SetVerbose(manager.GetVerbose(cmd)).GetScClient()

server, err := scClient.Hosts.CreateDedicatedServers(ctx, input)
if err != nil {
return err
}

if server != nil {
formatter := cmdContext.GetOrCreateFormatter(cmd)
return formatter.Format(server)
}

return nil
},
}

cmd.Flags().StringVarP(&flags.InputPath, "input", "i", "", "path to input file or '-' to read from stdin")

cmd.Flags().IntVar(&flags.LocationID, "location-id", 0, "Create the server(s) in the specific location ID")
cmd.Flags().IntVar(&flags.ServerModelID, "server-model-id", 0, "Use specific server model ID to create the server")
cmd.Flags().IntVar(&flags.OperatingSystemID, "operating-system-id", 0, "Install the specific operating system")
cmd.Flags().StringSliceVar(&flags.Features, "feature", nil, "Set of features")
cmd.Flags().IntVar(&flags.RAMSize, "ram-size", 0, "Desired amount of RAM in GB")
cmd.Flags().IntVar(&flags.PublicUplinkID, "public-uplink-id", 0, "The public uplink ID, can be omitted if do not want public uplink")
cmd.Flags().IntVar(&flags.PrivateUplinkID, "private-uplink-id", 0, "The private uplink ID")
cmd.Flags().IntVar(&flags.PublicBandwidthID, "public-bandwidth-id", 0, "The public bandwidth ID, MUST be omitted if public uplink id is not passed")
cmd.Flags().StringToIntVar(&flags.DriveSlots, "drive-slots", nil, "mapping of the specific slot to the specific drive model")
cmd.Flags().StringArrayVar(&flags.Layout, "layout", nil, "Configuration of drives layout")
cmd.Flags().StringArrayVar(&flags.Partitions, "partition", nil, "Configuration of the specific partitions")
cmd.Flags().BoolVar(&flags.IPv6, "ipv6", false, "Enable IPv6")
cmd.Flags().StringVar(&flags.UserDataFile, "user-data-file", "", "Path to user data which should be readed")
cmd.Flags().StringVar(&flags.UserData, "user-data", "", "Content of user data")
cmd.Flags().StringToStringVar(&flags.Labels, "labels", nil, "The set of labels which will be applied to the all hosts of this operation")

return cmd
}

func newAddCmd(cmdContext *base.CmdContext, hostType *HostTypeCmd) *cobra.Command {
func newAddSBMCmd(cmdContext *base.CmdContext) *cobra.Command {
var path string
cmd := &cobra.Command{
Use: "add --input <path>",
Short: fmt.Sprintf("Create a %s", hostType.entityName),
Short: "Create an SBM server",
Args: cobra.ExactArgs(0),
RunE: func(cmd *cobra.Command, args []string) error {
manager := cmdContext.GetManager()
Expand All @@ -56,15 +213,15 @@ func newAddCmd(cmdContext *base.CmdContext, hostType *HostTypeCmd) *cobra.Comman

base.SetupProxy(cmd, manager)

input := hostType.managers.createMgr.NewCreateInput()
input := serverscom.SBMServerCreateInput{}

if err := base.ReadInputJSON(path, cmd.InOrStdin(), input); err != nil {
if err := base.ReadInputJSON(path, cmd.InOrStdin(), &input); err != nil {
return err
}

scClient := cmdContext.GetClient().SetVerbose(manager.GetVerbose(cmd)).GetScClient()

server, err := hostType.managers.createMgr.Create(ctx, scClient, input)
server, err := scClient.Hosts.CreateSBMServers(ctx, input)
if err != nil {
return err
}
Expand Down
10 changes: 3 additions & 7 deletions cmd/entities/hosts/hosts.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,7 @@ type HostTypeCmd struct {
}

type HostManagers struct {
getMgr HostGetter
createMgr HostCreator
getMgr HostGetter
// for update we use simple commands in sake of simplicity
powerMgr HostPowerer
reinstallMgr HostReinstaller
Expand All @@ -40,11 +39,11 @@ func NewCmd(cmdContext *base.CmdContext) *cobra.Command {
typeFlag: "dedicated_server",
managers: HostManagers{
getMgr: &DSGetMgr{},
createMgr: &DSCreateMgr{},
powerMgr: &DSPowerMgr{},
reinstallMgr: &DSReinstallMgr{},
},
extraCmds: []func(*base.CmdContext) *cobra.Command{
newAddDSCmd,
newUpdateDSCmd,
newListDSDriveSlotsCmd,
newListDSConnectionsCmd,
Expand Down Expand Up @@ -73,11 +72,11 @@ func NewCmd(cmdContext *base.CmdContext) *cobra.Command {
typeFlag: "sbm_server",
managers: HostManagers{
getMgr: &SBMGetMgr{},
createMgr: &SBMCreateMgr{},
powerMgr: &SBMPowerMgr{},
reinstallMgr: &SBMReinstallMgr{},
},
extraCmds: []func(*base.CmdContext) *cobra.Command{
newAddSBMCmd,
newUpdateSBMCmd,
newSBMReleaseCmd,
},
Expand Down Expand Up @@ -119,9 +118,6 @@ func newHostTypeCmd(cmdContext *base.CmdContext, hostTypeCmd HostTypeCmd) *cobra
hostCmd.AddCommand(newListCmd(cmdContext, &hostTypeCmd))
hostCmd.AddCommand(newGetCmd(cmdContext, &hostTypeCmd))

if hostTypeCmd.managers.createMgr != nil {
hostCmd.AddCommand(newAddCmd(cmdContext, &hostTypeCmd))
}
if hostTypeCmd.managers.powerMgr != nil {
hostCmd.AddCommand(newPowerCmd(cmdContext, &hostTypeCmd))
hostCmd.AddCommand(newListPowerFeedsCmd(cmdContext, &hostTypeCmd))
Expand Down
Loading
Loading