Skip to content

Commit

Permalink
Add a lifecycle test for GCE
Browse files Browse the repository at this point in the history
- Move MockGCECloud to cloudmock/gce.
- Change Compute() and CloudDNS() of GCECloud to return interfaces
  for mocking
  • Loading branch information
kenji-cloudnatix committed Apr 22, 2021
1 parent 2649cbc commit bdb3375
Show file tree
Hide file tree
Showing 47 changed files with 3,185 additions and 611 deletions.
23 changes: 23 additions & 0 deletions cloudmock/gce/BUILD.bazel

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

174 changes: 174 additions & 0 deletions cloudmock/gce/mock_gce_cloud.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,174 @@
/*
Copyright 2017 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 gce

import (
"fmt"

compute "google.golang.org/api/compute/v1"
"google.golang.org/api/iam/v1"
"google.golang.org/api/storage/v1"
v1 "k8s.io/api/core/v1"
"k8s.io/klog/v2"
mockcompute "k8s.io/kops/cloudmock/gce/mockcompute"
"k8s.io/kops/cloudmock/gce/mockdns"
"k8s.io/kops/dnsprovider/pkg/dnsprovider"
dnsproviderclouddns "k8s.io/kops/dnsprovider/pkg/dnsprovider/providers/google/clouddns"
"k8s.io/kops/pkg/apis/kops"
"k8s.io/kops/pkg/cloudinstances"
"k8s.io/kops/upup/pkg/fi"
"k8s.io/kops/upup/pkg/fi/cloudup/gce"
)

// MockGCECloud is a mock implementation of GCECloud for testing
type MockGCECloud struct {
project string
region string
labels map[string]string

computeClient *mockcompute.MockClient
dnsClient *mockdns.MockClient
}

var _ gce.GCECloud = &MockGCECloud{}

// InstallMockGCECloud registers a MockGCECloud implementation for the specified region & project
func InstallMockGCECloud(region string, project string) *MockGCECloud {
c := &MockGCECloud{
project: project,
region: region,
computeClient: mockcompute.NewMockClient(project),
dnsClient: mockdns.NewMockClient(),
}
gce.CacheGCECloudInstance(region, project, c)
return c
}

func (c *MockGCECloud) AllResources() map[string]interface{} {
return c.computeClient.AllResources()
}

// GetCloudGroups is not implemented yet
func (c *MockGCECloud) GetCloudGroups(cluster *kops.Cluster, instancegroups []*kops.InstanceGroup, warnUnmatched bool, nodes []v1.Node) (map[string]*cloudinstances.CloudInstanceGroup, error) {
klog.V(8).Infof("MockGCECloud cloud provider GetCloudGroups not implemented yet")
return nil, fmt.Errorf("MockGCECloud cloud provider does not support getting cloud groups at this time")
}

// Zones is not implemented yet
func (c *MockGCECloud) Zones() ([]string, error) {
return nil, fmt.Errorf("not yet implemented")
}

// WithLabels returns a copy of the MockGCECloud bound to the specified labels
func (c *MockGCECloud) WithLabels(labels map[string]string) gce.GCECloud {
i := &MockGCECloud{}
*i = *c
i.labels = labels
return i
}

// ProviderID implements fi.Cloud::ProviderID
func (c *MockGCECloud) ProviderID() kops.CloudProviderID {
return kops.CloudProviderGCE
}

// FindVPCInfo implements fi.Cloud::FindVPCInfo
func (c *MockGCECloud) FindVPCInfo(id string) (*fi.VPCInfo, error) {
return nil, fmt.Errorf("MockGCECloud::MockGCECloud not implemented")
}

// DNS implements fi.Cloud::DNS
func (c *MockGCECloud) DNS() (dnsprovider.Interface, error) {
return dnsproviderclouddns.NewFakeInterface()
}

// Compute implements GCECloud::Compute
func (c *MockGCECloud) Compute() gce.ComputeClient {
return c.computeClient
}

// Storage implements GCECloud::Storage
func (c *MockGCECloud) Storage() *storage.Service {
klog.Fatalf("MockGCECloud::Storage not implemented")
return nil
}

// IAM returns the IAM client
func (c *MockGCECloud) IAM() *iam.Service {
klog.Fatalf("MockGCECloud::IAM not implemented")
return nil
}

// CloudDNS returns the DNS client
func (c *MockGCECloud) CloudDNS() gce.DNSClient {
return c.dnsClient
}

// WaitForOp implements GCECloud::WaitForOp
func (c *MockGCECloud) WaitForOp(op *compute.Operation) error {
if op.Status != "DONE" {
return fmt.Errorf("unexpected operation: %+v", op)
}
return nil
}

// FindClusterStatus implements GCECloud::FindClusterStatus
func (c *MockGCECloud) FindClusterStatus(cluster *kops.Cluster) (*kops.ClusterStatus, error) {
return nil, fmt.Errorf("MockGCECloud::FindClusterStatus not implemented")
}

// GetApiIngressStatus implements GCECloud::GetApiIngressStatus
func (c *MockGCECloud) GetApiIngressStatus(cluster *kops.Cluster) ([]kops.ApiIngressStatus, error) {
return nil, fmt.Errorf("MockGCECloud::GetApiIngressStatus not implemented")
}

// Region implements GCECloud::Region
func (c *MockGCECloud) Region() string {
return c.region
}

// Project implements GCECloud::Project
func (c *MockGCECloud) Project() string {
return c.project
}

// ServiceAccount implements GCECloud::ServiceAccount
func (c *MockGCECloud) ServiceAccount() (string, error) {
return "[email protected]", nil
}

// Labels implements GCECloud::Labels
func (c *MockGCECloud) Labels() map[string]string {
return c.labels
}

// DeleteGroup implements fi.Cloud::DeleteGroup
func (c *MockGCECloud) DeleteGroup(g *cloudinstances.CloudInstanceGroup) error {
return nil
// return deleteCloudInstanceGroup(c, g)
}

// DeleteInstance deletes a GCE instance
func (c *MockGCECloud) DeleteInstance(i *cloudinstances.CloudInstance) error {
return nil
// return recreateCloudInstance(c, i)
}

// DetachInstance is not implemented yet. It needs to cause a cloud instance to no longer be counted against the group's size limits.
func (c *MockGCECloud) DetachInstance(i *cloudinstances.CloudInstance) error {
return nil
}
27 changes: 27 additions & 0 deletions cloudmock/gce/mockcompute/BUILD.bazel

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

130 changes: 130 additions & 0 deletions cloudmock/gce/mockcompute/address.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
/*
Copyright 2021 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 mockcompute

import (
"context"
"fmt"
"sync"

compute "google.golang.org/api/compute/v1"
"k8s.io/kops/upup/pkg/fi/cloudup/gce"
)

type addressClient struct {
// addrs are addresses keyed by project, region, and address name.
addrs map[string]map[string]map[string]*compute.Address
sync.Mutex
}

var _ gce.AddressClient = &addressClient{}

func newAddressClient() *addressClient {
return &addressClient{
addrs: map[string]map[string]map[string]*compute.Address{},
}
}

func (c *addressClient) All() map[string]interface{} {
c.Lock()
defer c.Unlock()
m := map[string]interface{}{}
for _, regions := range c.addrs {
for _, addrs := range regions {
for n, a := range addrs {
m[n] = a
}
}
}
return m
}

func (c *addressClient) Insert(project, region string, addr *compute.Address) (*compute.Operation, error) {
c.Lock()
defer c.Unlock()
regions, ok := c.addrs[project]
if !ok {
regions = map[string]map[string]*compute.Address{}
c.addrs[project] = regions
}
addrs, ok := regions[region]
if !ok {
addrs = map[string]*compute.Address{}
regions[region] = addrs
}
addr.SelfLink = fmt.Sprintf("https://www.googleapis.com/compute/v1/projects/%s/regions/%s/addrs/%s", project, region, addr.Name)
addrs[addr.Name] = addr
return doneOperation(), nil
}

func (c *addressClient) Delete(project, region, name string) (*compute.Operation, error) {
c.Lock()
defer c.Unlock()
regions, ok := c.addrs[project]
if !ok {
return nil, notFoundError()
}
addrs, ok := regions[region]
if !ok {
return nil, notFoundError()
}
if _, ok := addrs[name]; !ok {
return nil, notFoundError()
}
delete(addrs, name)
return doneOperation(), nil
}

func (c *addressClient) Get(project, region, name string) (*compute.Address, error) {
c.Lock()
defer c.Unlock()
regions, ok := c.addrs[project]
if !ok {
return nil, notFoundError()
}
addrs, ok := regions[region]
if !ok {
return nil, notFoundError()
}
addr, ok := addrs[name]
if !ok {
return nil, notFoundError()
}
return addr, nil
}

func (c *addressClient) List(ctx context.Context, project, region string) ([]*compute.Address, error) {
c.Lock()
defer c.Unlock()
regions, ok := c.addrs[project]
if !ok {
return nil, nil
}
addrs, ok := regions[region]
if !ok {
return nil, nil
}
var l []*compute.Address
for _, a := range addrs {
l = append(l, a)
}
return l, nil
}

func (c *addressClient) ListWithFilter(project, region, filter string) ([]*compute.Address, error) {
return c.List(context.Background(), project, region)
}
Loading

0 comments on commit bdb3375

Please sign in to comment.