Skip to content

Commit

Permalink
Add LongRunningOperationStates types and conditions
Browse files Browse the repository at this point in the history
  • Loading branch information
Cecile Robert-Michon committed Aug 31, 2021
1 parent feaeb16 commit 3bb54ef
Show file tree
Hide file tree
Showing 46 changed files with 1,163 additions and 349 deletions.
12 changes: 12 additions & 0 deletions api/v1alpha3/azurecluster_conversion.go
Original file line number Diff line number Diff line change
Expand Up @@ -304,3 +304,15 @@ func Convert_v1alpha3_VnetSpec_To_v1alpha4_VnetSpec(in *VnetSpec, out *infrav1al
func Convert_v1alpha4_LoadBalancerSpec_To_v1alpha3_LoadBalancerSpec(in *infrav1alpha4.LoadBalancerSpec, out *LoadBalancerSpec, s apiconversion.Scope) error {
return autoConvert_v1alpha4_LoadBalancerSpec_To_v1alpha3_LoadBalancerSpec(in, out, s)
}

// Convert_v1alpha3_Future_To_v1alpha4_Future is an autogenerated conversion function.
func Convert_v1alpha3_Future_To_v1alpha4_Future(in *Future, out *infrav1alpha4.Future, s apiconversion.Scope) error {
out.Data = in.FutureData
return autoConvert_v1alpha3_Future_To_v1alpha4_Future(in, out, s)
}

// Convert_v1alpha4_Future_To_v1alpha3_Future is an autogenerated conversion function.
func Convert_v1alpha4_Future_To_v1alpha3_Future(in *infrav1alpha4.Future, out *Future, s apiconversion.Scope) error {
out.FutureData = in.Data
return autoConvert_v1alpha4_Future_To_v1alpha3_Future(in, out, s)
}
17 changes: 5 additions & 12 deletions api/v1alpha3/zz_generated.conversion.go

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

15 changes: 15 additions & 0 deletions api/v1alpha4/azurecluster_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,11 @@ type AzureClusterStatus struct {
// Conditions defines current service state of the AzureCluster.
// +optional
Conditions clusterv1.Conditions `json:"conditions,omitempty"`

// LongRunningOperationStates saves the states for Azure long-running operations so they can be continued on the
// next reconciliation loop.
// +optional
LongRunningOperationStates Futures `json:"longRunningOperationStates,omitempty"`
}

// +kubebuilder:object:root=true
Expand Down Expand Up @@ -137,6 +142,16 @@ func (c *AzureCluster) SetConditions(conditions clusterv1.Conditions) {
c.Status.Conditions = conditions
}

// GetFutures returns the list of long running operation states for an AzureCluster API object.
func (c *AzureCluster) GetFutures() Futures {
return c.Status.LongRunningOperationStates
}

// SetFutures will set the given long running operation states on an AzureCluster object.
func (c *AzureCluster) SetFutures(futures Futures) {
c.Status.LongRunningOperationStates = futures
}

func init() {
SchemeBuilder.Register(&AzureCluster{}, &AzureClusterList{})
}
15 changes: 15 additions & 0 deletions api/v1alpha4/azuremachine_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -175,6 +175,11 @@ type AzureMachineStatus struct {
// Conditions defines current service state of the AzureMachine.
// +optional
Conditions clusterv1.Conditions `json:"conditions,omitempty"`

// LongRunningOperationStates saves the states for Azure long-running operations so they can be continued on the
// next reconciliation loop.
// +optional
LongRunningOperationStates Futures `json:"longRunningOperationStates,omitempty"`
}

// +kubebuilder:object:root=true
Expand Down Expand Up @@ -216,6 +221,16 @@ func (m *AzureMachine) SetConditions(conditions clusterv1.Conditions) {
m.Status.Conditions = conditions
}

// GetFutures returns the list of long running operation states for an AzureMachine API object.
func (m *AzureMachine) GetFutures() Futures {
return m.Status.LongRunningOperationStates
}

// SetFutures will set the given long running operation states on an AzureMachine object.
func (m *AzureMachine) SetFutures(futures Futures) {
m.Status.LongRunningOperationStates = futures
}

func init() {
SchemeBuilder.Register(&AzureMachine{}, &AzureMachineList{})
}
41 changes: 41 additions & 0 deletions api/v1alpha4/conditions_consts.go
Original file line number Diff line number Diff line change
Expand Up @@ -75,3 +75,44 @@ const (
// ScaleSetModelOutOfDateReason describes the machine pool model being out of date.
ScaleSetModelOutOfDateReason = "ScaleSetModelOutOfDate"
)

// Azure Services Conditions and Reasons.
const (
// ResourceGroupReadyCondition means the resource group exists and is ready to be used.
ResourceGroupReadyCondition clusterv1.ConditionType = "ResourceGroupReady"
// VNetReadyCondition means the virtual network exists and is ready to be used.
VNetReadyCondition clusterv1.ConditionType = "VNetReady"
// SecurityGroupsReadyCondition means the security groups exist and are ready to be used.
SecurityGroupsReadyCondition clusterv1.ConditionType = "SecurityGroupsReady"
// RouteTablesReadyCondition means the route tables exist and are ready to be used.
RouteTablesReadyCondition clusterv1.ConditionType = "RouteTablesReady"
// PublicIPsReadyCondition means the public IPs exist and are ready to be used.
PublicIPsReadyCondition clusterv1.ConditionType = "PublicIPsReady"
// NATGatewaysReadyCondition means the NAT gateways exist and are ready to be used.
NATGatewaysReadyCondition clusterv1.ConditionType = "NATGatewaysReady"
// SubnetsReadyCondition means the subnets exist and are ready to be used.
SubnetsReadyCondition clusterv1.ConditionType = "SubnetsReady"
// LoadBalancersReadyCondition means the load balancers exist and are ready to be used.
LoadBalancersReadyCondition clusterv1.ConditionType = "LoadBalancersReady"
// PrivateDNSReadyCondition means the private DNS exists and is ready to be used.
PrivateDNSReadyCondition clusterv1.ConditionType = "PrivateDNSReady"
// BastionHostReadyCondition means the bastion host exists and is ready to be used.
BastionHostReadyCondition clusterv1.ConditionType = "BastionHostReady"
// InboundNATRulesReadyCondition means the inbound NAT rules exist and are ready to be used.
InboundNATRulesReadyCondition clusterv1.ConditionType = "InboundNATRulesReady"
// AvailabilitySetReadyCondition means the availability set exists and is ready to be used.
AvailabilitySetReadyCondition clusterv1.ConditionType = "AvailabilitySetReady"
// RoleAssignmentReadyCondition means the role assignment exists and is ready to be used.
RoleAssignmentReadyCondition clusterv1.ConditionType = "RoleAssignmentReady"

// CreatingReason means the resource is being created.
CreatingReason = "Creating"
// FailedReason means the resource failed to be created.
FailedReason = "Failed"
// DeletingReason means the resource is being deleted.
DeletingReason = "Deleting"
// DeletedReason means the resource was deleted.
DeletedReason = "Deleted"
// DeletionFailedReason means the resource failed to be deleted.
DeletionFailedReason = "DeletionFailed"
)
20 changes: 13 additions & 7 deletions api/v1alpha4/types.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,21 +29,27 @@ const (
Node string = "node"
)

type Futures []Future

// Future contains the data needed for an Azure long-running operation to continue across reconcile loops.
type Future struct {
// Type describes the type of future, update, create, delete, etc
// Type describes the type of future, such update, create, delete, etc.
Type string `json:"type"`

// ResourceGroup is the Azure resource group for the resource
// ResourceGroup is the Azure resource group for the resource.
// +optional
ResourceGroup string `json:"resourceGroup,omitempty"`

// Name is the name of the Azure resource
// +optional
Name string `json:"name,omitempty"`
// ServiceName is the name of the Azure service.
// Together with the name of the resource, this forms the unique identifier for the future.
ServiceName string `json:"serviceName"`

// Name is the name of the Azure resource.
// Together with the service name, this forms the unique identifier for the future.
Name string `json:"name"`

// FutureData is the base64 url encoded json Azure AutoRest Future
FutureData string `json:"futureData,omitempty"`
// Data is the base64 url encoded json Azure AutoRest Future.
Data string `json:"data,omitempty"`
}

// NetworkSpec specifies what the Azure networking resources should look like.
Expand Down
29 changes: 29 additions & 0 deletions api/v1alpha4/zz_generated.deepcopy.go

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

62 changes: 62 additions & 0 deletions azure/converters/futures.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
/*
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 converters

import (
"encoding/base64"
"encoding/json"

azureautorest "github.com/Azure/go-autorest/autorest/azure"
"github.com/pkg/errors"
infrav1 "sigs.k8s.io/cluster-api-provider-azure/api/v1alpha4"
)

const (
// PatchFuture is a future that was derived from a PATCH request.
PatchFuture string = "PATCH"
// PutFuture is a future that was derived from a PUT request.
PutFuture string = "PUT"
// DeleteFuture is a future that was derived from a DELETE request.
DeleteFuture string = "DELETE"
)

func SDKToFuture(future azureautorest.FutureAPI, futureType, service, resourceName, rgName string) (*infrav1.Future, error) {
jsonData, err := future.MarshalJSON()
if err != nil {
return nil, errors.Wrap(err, "failed to marshal async future")
}

return &infrav1.Future{
Type: futureType,
ResourceGroup: rgName,
ServiceName: service,
Name: resourceName,
Data: base64.URLEncoding.EncodeToString(jsonData),
}, nil
}

func FutureToSDK(future *infrav1.Future) (azureautorest.FutureAPI, error) {
futureData, err := base64.URLEncoding.DecodeString(future.Data)
if err != nil {
return nil, errors.Wrap(err, "failed to base64 decode future data")
}
var genericFuture azureautorest.FutureAPI
if err := json.Unmarshal(futureData, &genericFuture); err != nil {
return nil, errors.Wrap(err, "failed to unmarshal future data")
}
return genericFuture, nil
}
7 changes: 7 additions & 0 deletions azure/defaults.go
Original file line number Diff line number Diff line change
Expand Up @@ -328,6 +328,13 @@ func SetAutoRestClientDefaults(c *autorest.Client, auth autorest.Authorizer) {
// The wrapped Sender should set the x-ms-correlation-request-id on the given
// request, then pass the new request to the underlying Sender.
c.Sender = autorest.DecorateSender(c.Sender, msCorrelationIDSendDecorator)
// The default number of retries is 3. This means the client will attempt to retry operation results like resource
// conflicts (HTTP 409). For a reconciling controller, this is undesirable behavior since if the controller runs
// into an error reconciling, the controller would be better off to end with an error and try again later.
//
// Unfortunately, the naming of this field is a bit misleading. This is not actually "retry attempts", it actually
// is attempts. Setting this to a value of 0 will cause a panic in Go AutoRest.
c.RetryAttempts = 1
AutoRestClientAppendUserAgent(c, UserAgent())
}

Expand Down
9 changes: 9 additions & 0 deletions azure/interfaces.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import (
"github.com/Azure/go-autorest/autorest"

infrav1 "sigs.k8s.io/cluster-api-provider-azure/api/v1alpha4"
clusterv1 "sigs.k8s.io/cluster-api/api/v1alpha4"
)

// Reconciler is a generic interface used by components offering a type of service.
Expand Down Expand Up @@ -80,6 +81,14 @@ type ClusterDescriber interface {
CloudProviderConfigOverrides() *infrav1.CloudProviderConfigOverrides
}

type AsyncStatusUpdater interface {
SetLongRunningOperationState(*infrav1.Future)
GetLongRunningOperationState(string, string) *infrav1.Future
DeleteLongRunningOperationState(string, string)
SetConditionTrue(clusterv1.ConditionType)
SetConditionFalse(clusterv1.ConditionType, string, clusterv1.ConditionSeverity)
}

// ClusterScoper combines the ClusterDescriber and NetworkDescriber interfaces.
type ClusterScoper interface {
ClusterDescriber
Expand Down
Loading

0 comments on commit 3bb54ef

Please sign in to comment.