mirror of https://github.com/knative/pkg.git
Add prow-cluster-operation package (#788)
* Add prow-cluster-operation package * Remove binary committed by accident
This commit is contained in:
parent
f0ffda4667
commit
0b19b4ad91
|
|
@ -0,0 +1,53 @@
|
||||||
|
## prow-cluster-operation
|
||||||
|
|
||||||
|
prow-cluster-operation is a tool for creating, deleting, getting a GKE
|
||||||
|
cluster
|
||||||
|
|
||||||
|
## Prerequisite
|
||||||
|
|
||||||
|
- `GOOGLE_APPLICATION_CREDENTIALS` set
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
This tool can be invoked from command line with following parameters:
|
||||||
|
|
||||||
|
- `--min-nodes`: minumum number of nodes, default 1
|
||||||
|
- `--max-nodes`: maximum number of nodes, default 3
|
||||||
|
- `--node-type`: GCE node type, default "n1-standard-4"
|
||||||
|
- `--region`: GKE region, default "us-central1"
|
||||||
|
- `--zone`: GKE zone, default empty
|
||||||
|
- `--project`: GCP project, default empty
|
||||||
|
- `--name`: cluster name, default empty
|
||||||
|
- `--backup-regions`: backup regions to be used if cluster creation in primary
|
||||||
|
region failed, comma separated list, default "us-west1,us-east1"
|
||||||
|
- `--addons`: GKE addons, comma separated list, default empty
|
||||||
|
|
||||||
|
## Flow
|
||||||
|
|
||||||
|
### Create
|
||||||
|
|
||||||
|
1. Acquiring cluster if kubeconfig already points to it
|
||||||
|
1. Get GCP project name if not provided as a parameter:
|
||||||
|
- [In Prow] Acquire from Boskos
|
||||||
|
- [Not in Prow] Read from gcloud config
|
||||||
|
|
||||||
|
Failed obtaining project name will fail the tool
|
||||||
|
1. Get default cluster name if not provided as a parameter
|
||||||
|
1. Delete cluster if cluster with same name and location already exists in GKE
|
||||||
|
1. Create cluster
|
||||||
|
1. Write cluster metadata to `${ARTIFACT}/metadata.json`
|
||||||
|
|
||||||
|
### Delete
|
||||||
|
|
||||||
|
1. Acquiring cluster if kubeconfig already points to it
|
||||||
|
1. If cluster name is defined then getting cluster by its name
|
||||||
|
1. If no cluster is found from previous step then it fails
|
||||||
|
1. Delete:
|
||||||
|
- [In Prow] Release Boskos project
|
||||||
|
- [Not in Prow] Delete cluster
|
||||||
|
|
||||||
|
### Get
|
||||||
|
|
||||||
|
1. Acquiring cluster if kubeconfig already points to it
|
||||||
|
1. If cluster name is defined then getting cluster by its name
|
||||||
|
1. If no cluster is found from previous step then it fails
|
||||||
|
|
@ -0,0 +1,108 @@
|
||||||
|
/*
|
||||||
|
Copyright 2019 The Knative 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 actions
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
container "google.golang.org/api/container/v1beta1"
|
||||||
|
"knative.dev/pkg/test/gke"
|
||||||
|
clm "knative.dev/pkg/testutils/clustermanager/e2e-tests"
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/e2e-tests/common"
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/prow-cluster-operation/options"
|
||||||
|
"knative.dev/pkg/testutils/metahelper/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Keys to be written into metadata.json
|
||||||
|
e2eRegionKey = "E2E:Region"
|
||||||
|
e2eZoneKey = "E2E:Zone"
|
||||||
|
clusterNameKey = "E2E:Machine"
|
||||||
|
clusterVersionKey = "E2E:Version"
|
||||||
|
minNodesKey = "E2E:MinNodes"
|
||||||
|
maxNodesKey = "E2E:MaxNodes"
|
||||||
|
projectKey = "E2E:Project"
|
||||||
|
)
|
||||||
|
|
||||||
|
func writeMetaData(cluster *container.Cluster, project string) {
|
||||||
|
// Set up metadata client for saving metadata
|
||||||
|
c, err := client.NewClient("")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Printf("Writing metadata to: %q", c.Path)
|
||||||
|
// Get minNodes and maxNodes counts from default-pool, this is
|
||||||
|
// usually the case in tests in Prow
|
||||||
|
var minNodes, maxNodes string
|
||||||
|
for _, np := range cluster.NodePools {
|
||||||
|
if np.Name == "default-pool" {
|
||||||
|
minNodes = strconv.FormatInt(np.InitialNodeCount, 10)
|
||||||
|
// maxNodes is equal to minNodes if autoscaling isn't on
|
||||||
|
maxNodes = minNodes
|
||||||
|
if np.Autoscaling != nil {
|
||||||
|
minNodes = strconv.FormatInt(np.Autoscaling.MinNodeCount, 10)
|
||||||
|
maxNodes = strconv.FormatInt(np.Autoscaling.MaxNodeCount, 10)
|
||||||
|
} else {
|
||||||
|
log.Printf("DEBUG: nodepool is default-pool but autoscaling is not on: '%+v'", np)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
e2eRegion, e2eZone := gke.RegionZoneFromLoc(cluster.Location)
|
||||||
|
for key, val := range map[string]string{
|
||||||
|
e2eRegionKey: e2eRegion,
|
||||||
|
e2eZoneKey: e2eZone,
|
||||||
|
clusterNameKey: cluster.Name,
|
||||||
|
clusterVersionKey: cluster.InitialClusterVersion,
|
||||||
|
minNodesKey: minNodes,
|
||||||
|
maxNodesKey: maxNodes,
|
||||||
|
projectKey: project,
|
||||||
|
} {
|
||||||
|
if err = c.Set(key, val); err != nil {
|
||||||
|
log.Fatalf("Failed saving metadata %q:%q: '%v'", key, val, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Println("Done writing metadata")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Create(o *options.RequestWrapper) {
|
||||||
|
o.Prep()
|
||||||
|
|
||||||
|
gkeClient := clm.GKEClient{}
|
||||||
|
clusterOps := gkeClient.Setup(o.Request)
|
||||||
|
gkeOps := clusterOps.(*clm.GKECluster)
|
||||||
|
if err := gkeOps.Acquire(); err != nil || gkeOps.Cluster == nil {
|
||||||
|
log.Fatalf("failed acquiring GKE cluster: '%v'", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// At this point we should have a cluster ready to run test. Need to save
|
||||||
|
// metadata so that following flow can understand the context of cluster, as
|
||||||
|
// well as for Prow usage later
|
||||||
|
writeMetaData(gkeOps.Cluster, gkeOps.Project)
|
||||||
|
|
||||||
|
// set up kube config points to cluster
|
||||||
|
// TODO(chaodaiG): this probably should also be part of clustermanager lib
|
||||||
|
if out, err := common.StandardExec("gcloud", "beta", "container", "clusters", "get-credentials",
|
||||||
|
gkeOps.Cluster.Name, "--region", gkeOps.Cluster.Location, "--project", gkeOps.Project); err != nil {
|
||||||
|
log.Fatalf("Failed connecting to cluster: %q, '%v'", out, err)
|
||||||
|
}
|
||||||
|
if out, err := common.StandardExec("gcloud", "config", "set", "project", gkeOps.Project); err != nil {
|
||||||
|
log.Fatalf("Failed setting gcloud: %q, '%v'", out, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
Copyright 2019 The Knative 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 actions
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
|
||||||
|
clm "knative.dev/pkg/testutils/clustermanager/e2e-tests"
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/prow-cluster-operation/options"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Delete(o *options.RequestWrapper) {
|
||||||
|
o.Request.NeedsCleanup = true
|
||||||
|
o.Request.SkipCreation = true
|
||||||
|
|
||||||
|
gkeClient := clm.GKEClient{}
|
||||||
|
clusterOps := gkeClient.Setup(o.Request)
|
||||||
|
gkeOps := clusterOps.(*clm.GKECluster)
|
||||||
|
if err := gkeOps.Acquire(); err != nil || gkeOps.Cluster == nil {
|
||||||
|
log.Fatalf("Failed identifying cluster for cleanup: '%v'", err)
|
||||||
|
}
|
||||||
|
log.Printf("Identified project %q and cluster %q for removal", gkeOps.Project, gkeOps.Cluster.Name)
|
||||||
|
var err error
|
||||||
|
if err = gkeOps.Delete(); err != nil {
|
||||||
|
log.Fatalf("Failed deleting cluster: '%v'", err)
|
||||||
|
}
|
||||||
|
// TODO: uncomment the lines below when previous Delete command becomes
|
||||||
|
// async operation
|
||||||
|
// // Unset context with best effort. The first command only unsets current
|
||||||
|
// // context, but doesn't delete the entry from kubeconfig, and should return it's
|
||||||
|
// // context if succeeded, which can be used by the second command to
|
||||||
|
// // delete it from kubeconfig
|
||||||
|
// if out, err := common.StandardExec("kubectl", "config", "unset", "current-context"); err != nil {
|
||||||
|
// common.StandardExec("kubectl", "config", "unset", "contexts."+string(out))
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
Copyright 2019 The Knative 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 actions
|
||||||
|
|
||||||
|
import (
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/prow-cluster-operation/options"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Get(o *options.RequestWrapper) {
|
||||||
|
o.Prep()
|
||||||
|
o.Request.SkipCreation = true
|
||||||
|
// Reuse `Create` for getting operation, so that we can reuse the same logic
|
||||||
|
// such as protected project/cluster etc.
|
||||||
|
Create(o)
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,53 @@
|
||||||
|
/*
|
||||||
|
Copyright 2019 The Knative 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 (
|
||||||
|
"flag"
|
||||||
|
"log"
|
||||||
|
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/prow-cluster-operation/actions"
|
||||||
|
"knative.dev/pkg/testutils/clustermanager/prow-cluster-operation/options"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
create bool
|
||||||
|
delete bool
|
||||||
|
get bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
flag.BoolVar(&create, "create", false, "Create cluster")
|
||||||
|
flag.BoolVar(&delete, "delete", false, "Delete cluster")
|
||||||
|
flag.BoolVar(&get, "get", false, "Get existing cluster from kubeconfig or gcloud")
|
||||||
|
o := options.NewRequestWrapper()
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
if (create && delete) || (create && get) || (delete && get) {
|
||||||
|
log.Fatal("--create, --delete, --get are mutually exclusive")
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case create:
|
||||||
|
actions.Create(o)
|
||||||
|
case delete:
|
||||||
|
actions.Delete(o)
|
||||||
|
case get:
|
||||||
|
actions.Get(o)
|
||||||
|
default:
|
||||||
|
log.Fatal("Must pass one of --create, --delete, --get")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
Copyright 2019 The Knative 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 options
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
clm "knative.dev/pkg/testutils/clustermanager/e2e-tests"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RequestWrapper struct {
|
||||||
|
Request clm.GKERequest
|
||||||
|
BackupRegionsStr string
|
||||||
|
AddonsStr string
|
||||||
|
NoWait bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRequestWrapper() *RequestWrapper {
|
||||||
|
rw := &RequestWrapper{
|
||||||
|
Request: clm.GKERequest{},
|
||||||
|
}
|
||||||
|
rw.addOptions()
|
||||||
|
return rw
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rw *RequestWrapper) Prep() {
|
||||||
|
if rw.BackupRegionsStr != "" {
|
||||||
|
rw.Request.BackupRegions = strings.Split(rw.BackupRegionsStr, ",")
|
||||||
|
}
|
||||||
|
if rw.AddonsStr != "" {
|
||||||
|
rw.Request.Addons = strings.Split(rw.AddonsStr, ",")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rw *RequestWrapper) addOptions() {
|
||||||
|
flag.Int64Var(&rw.Request.MinNodes, "min-nodes", 0, "minimal number of nodes")
|
||||||
|
flag.Int64Var(&rw.Request.MaxNodes, "max-nodes", 0, "maximal number of nodes")
|
||||||
|
flag.StringVar(&rw.Request.NodeType, "node-type", "", "node type")
|
||||||
|
flag.StringVar(&rw.Request.Region, "region", "", "GCP region")
|
||||||
|
flag.StringVar(&rw.Request.Zone, "zone", "", "GCP zone")
|
||||||
|
flag.StringVar(&rw.Request.Project, "project", "", "GCP project")
|
||||||
|
flag.StringVar(&rw.Request.ClusterName, "name", "", "cluster name")
|
||||||
|
flag.StringVar(&rw.BackupRegionsStr, "backup-regions", "", "GCP regions as backup, separated by comma")
|
||||||
|
flag.StringVar(&rw.AddonsStr, "addons", "", "addons to be added, separated by comma")
|
||||||
|
flag.BoolVar(&rw.Request.SkipCreation, "skip-creation", false, "should skip creation or not")
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue