mirror of https://github.com/kubernetes/kops.git
239 lines
6.6 KiB
Go
239 lines
6.6 KiB
Go
/*
|
|
Copyright 2019 The Kubernetes Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/kops/cmd/kops/util"
|
|
api "k8s.io/kops/pkg/apis/kops"
|
|
"k8s.io/kops/pkg/apis/kops/validation"
|
|
"k8s.io/kops/pkg/kopscodecs"
|
|
"k8s.io/kops/pkg/try"
|
|
"k8s.io/kops/upup/pkg/fi/cloudup"
|
|
"k8s.io/kubernetes/pkg/kubectl/cmd/util/editor"
|
|
"k8s.io/kubernetes/pkg/kubectl/util/i18n"
|
|
"k8s.io/kubernetes/pkg/kubectl/util/templates"
|
|
)
|
|
|
|
type CreateInstanceGroupOptions struct {
|
|
Role string
|
|
Subnets []string
|
|
// DryRun mode output an ig manifest of Output type.
|
|
DryRun bool
|
|
// Output type during a DryRun
|
|
Output string
|
|
// Edit will launch an editor when creating an instance group
|
|
Edit bool
|
|
}
|
|
|
|
var (
|
|
createIgLong = templates.LongDesc(i18n.T(`
|
|
Create an InstanceGroup configuration.
|
|
|
|
An InstanceGroup is a group of similar virtual machines.
|
|
On AWS, an InstanceGroup maps to an AutoScalingGroup.
|
|
|
|
The Role of an InstanceGroup defines whether machines will act as a Kubernetes master or node.`))
|
|
|
|
createIgExample = templates.Examples(i18n.T(`
|
|
|
|
# Create an instancegroup for the k8s-cluster.example.com cluster.
|
|
kops create ig --name=k8s-cluster.example.com node-example \
|
|
--role node --subnet my-subnet-name,my-other-subnet-name
|
|
|
|
# Create a YAML manifest for an instancegroup for the k8s-cluster.example.com cluster.
|
|
kops create ig --name=k8s-cluster.example.com node-example \
|
|
--role node --subnet my-subnet-name --dry-run -oyaml
|
|
`))
|
|
|
|
createIgShort = i18n.T(`Create an instancegroup.`)
|
|
)
|
|
|
|
// NewCmdCreateInstanceGroup create a new cobra command object for creating a instancegroup.
|
|
func NewCmdCreateInstanceGroup(f *util.Factory, out io.Writer) *cobra.Command {
|
|
options := &CreateInstanceGroupOptions{
|
|
Role: string(api.InstanceGroupRoleNode),
|
|
Edit: true,
|
|
}
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "instancegroup",
|
|
Aliases: []string{"instancegroups", "ig"},
|
|
Short: createIgShort,
|
|
Long: createIgLong,
|
|
Example: createIgExample,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
err := RunCreateInstanceGroup(f, cmd, args, os.Stdout, options)
|
|
if err != nil {
|
|
exitWithError(err)
|
|
}
|
|
},
|
|
}
|
|
|
|
// TODO: Create Enum helper - or is there one in k8s already?
|
|
var allRoles []string
|
|
for _, r := range api.AllInstanceGroupRoles {
|
|
allRoles = append(allRoles, string(r))
|
|
}
|
|
|
|
cmd.Flags().StringVar(&options.Role, "role", options.Role, "Type of instance group to create ("+strings.Join(allRoles, ",")+")")
|
|
cmd.Flags().StringSliceVar(&options.Subnets, "subnet", options.Subnets, "Subnet in which to create instance group. One of Availability Zone like eu-west-1a or a comma-separated list of multiple Availability Zones.")
|
|
// DryRun mode that will print YAML or JSON
|
|
cmd.Flags().BoolVar(&options.DryRun, "dry-run", options.DryRun, "If true, only print the object that would be sent, without sending it. This flag can be used to create a cluster YAML or JSON manifest.")
|
|
cmd.Flags().StringVarP(&options.Output, "output", "o", options.Output, "Output format. One of json|yaml")
|
|
cmd.Flags().BoolVar(&options.Edit, "edit", options.Edit, "If true, an editor will be opened to edit default values.")
|
|
|
|
return cmd
|
|
}
|
|
|
|
func RunCreateInstanceGroup(f *util.Factory, cmd *cobra.Command, args []string, out io.Writer, options *CreateInstanceGroupOptions) error {
|
|
if len(args) == 0 {
|
|
return fmt.Errorf("Specify name of instance group to create")
|
|
}
|
|
if len(args) != 1 {
|
|
return fmt.Errorf("Can only create one instance group at a time!")
|
|
}
|
|
groupName := args[0]
|
|
|
|
cluster, err := rootCommand.Cluster()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
clientset, err := rootCommand.Clientset()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
channel, err := cloudup.ChannelForCluster(cluster)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
existing, err := clientset.InstanceGroupsFor(cluster).Get(groupName, metav1.GetOptions{})
|
|
if err != nil {
|
|
// We expect a NotFound error when creating the instance group
|
|
if !errors.IsNotFound(err) {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if existing != nil {
|
|
return fmt.Errorf("instance group %q already exists", groupName)
|
|
}
|
|
|
|
// Populate some defaults
|
|
ig := &api.InstanceGroup{}
|
|
ig.ObjectMeta.Name = groupName
|
|
|
|
role, ok := api.ParseInstanceGroupRole(options.Role, true)
|
|
if !ok {
|
|
return fmt.Errorf("unknown role %q", options.Role)
|
|
}
|
|
ig.Spec.Role = role
|
|
|
|
ig.Spec.Subnets = options.Subnets
|
|
|
|
ig, err = cloudup.PopulateInstanceGroupSpec(cluster, ig, channel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ig.AddInstanceGroupNodeLabel()
|
|
|
|
if options.DryRun {
|
|
|
|
if options.Output == "" {
|
|
return fmt.Errorf("must set output flag; yaml or json")
|
|
}
|
|
|
|
// Cluster name is not populated, and we need it
|
|
ig.ObjectMeta.Labels = make(map[string]string)
|
|
ig.ObjectMeta.Labels[api.LabelClusterName] = cluster.ObjectMeta.Name
|
|
|
|
switch options.Output {
|
|
case OutputYaml:
|
|
if err := fullOutputYAML(out, ig); err != nil {
|
|
return fmt.Errorf("error writing cluster yaml to stdout: %v", err)
|
|
}
|
|
return nil
|
|
case OutputJSON:
|
|
if err := fullOutputJSON(out, ig); err != nil {
|
|
return fmt.Errorf("error writing cluster json to stdout: %v", err)
|
|
}
|
|
return nil
|
|
default:
|
|
return fmt.Errorf("unsupported output type %q", options.Output)
|
|
}
|
|
}
|
|
|
|
if options.Edit {
|
|
var (
|
|
edit = editor.NewDefaultEditor(editorEnvs)
|
|
)
|
|
|
|
raw, err := kopscodecs.ToVersionedYaml(ig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ext := "yaml"
|
|
|
|
// launch the editor
|
|
edited, file, err := edit.LaunchTempFile(fmt.Sprintf("%s-edit-", filepath.Base(os.Args[0])), ext, bytes.NewReader(raw))
|
|
defer func() {
|
|
if file != "" {
|
|
try.RemoveFile(file)
|
|
}
|
|
}()
|
|
if err != nil {
|
|
return fmt.Errorf("error launching editor: %v", err)
|
|
}
|
|
|
|
obj, _, err := kopscodecs.Decode(edited, nil)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing yaml: %v", err)
|
|
}
|
|
group, ok := obj.(*api.InstanceGroup)
|
|
if !ok {
|
|
return fmt.Errorf("unexpected object type: %T", obj)
|
|
}
|
|
|
|
err = validation.ValidateInstanceGroup(group)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ig = group
|
|
}
|
|
|
|
_, err = clientset.InstanceGroupsFor(cluster).Create(ig)
|
|
if err != nil {
|
|
return fmt.Errorf("error storing InstanceGroup: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|