From 4d5d74465cf1c1c5d69e4658bfec5cccf331d51f Mon Sep 17 00:00:00 2001 From: Bohdan Siryk Date: Wed, 11 Oct 2023 10:44:57 +0300 Subject: [PATCH] issue-572-2, handling of external deletion for clusterresources was implemented --- .../v1beta1/awsencryptionkey_types.go | 1 + .../awsendpointserviceprincipal_types.go | 7 ++ ...ces.instaclustr.com_awsencryptionkeys.yaml | 2 + ...ustr.com_awsendpointserviceprincipals.yaml | 3 + .../awsencryptionkey_controller.go | 43 +++++++++++- .../awsendpointserviceprincipal_controller.go | 69 +++++++++++++++++++ ...awssecuritygroupfirewallrule_controller.go | 42 +++++++++-- .../awsvpcpeering_controller.go | 37 ++++++++-- .../azurevnetpeering_controller.go | 42 ++++++++++- .../gcpvpcpeering_controller.go | 42 ++++++++++- controllers/clusterresources/helpers.go | 21 ++++++ pkg/instaclustr/client.go | 31 +++++++++ pkg/instaclustr/interfaces.go | 1 + pkg/instaclustr/mock/client.go | 4 ++ pkg/models/apiv2.go | 8 +++ pkg/models/operator.go | 4 ++ 16 files changed, 343 insertions(+), 14 deletions(-) diff --git a/apis/clusterresources/v1beta1/awsencryptionkey_types.go b/apis/clusterresources/v1beta1/awsencryptionkey_types.go index ec11cb02e..428aea855 100644 --- a/apis/clusterresources/v1beta1/awsencryptionkey_types.go +++ b/apis/clusterresources/v1beta1/awsencryptionkey_types.go @@ -34,6 +34,7 @@ type AWSEncryptionKeySpec struct { type AWSEncryptionKeyStatus struct { ID string `json:"id,omitempty"` InUse bool `json:"inUse,omitempty"` + State string `json:"state,omitempty"` } //+kubebuilder:object:root=true diff --git a/apis/clusterresources/v1beta1/awsendpointserviceprincipal_types.go b/apis/clusterresources/v1beta1/awsendpointserviceprincipal_types.go index 7f1f92309..4d08db9b3 100644 --- a/apis/clusterresources/v1beta1/awsendpointserviceprincipal_types.go +++ b/apis/clusterresources/v1beta1/awsendpointserviceprincipal_types.go @@ -40,6 +40,9 @@ type AWSEndpointServicePrincipalStatus struct { // The Instaclustr ID of the AWS endpoint service EndPointServiceID string `json:"endPointServiceId,omitempty"` + + // State describe current state of the resource + State string `json:"state,omitempty"` } //+kubebuilder:object:root=true @@ -58,6 +61,10 @@ func (r *AWSEndpointServicePrincipal) NewPatch() client.Patch { return client.MergeFrom(r.DeepCopy()) } +func (r *AWSEndpointServicePrincipal) GetJobID(job string) string { + return r.Namespace + "/" + r.Name + "/" + job +} + //+kubebuilder:object:root=true // AWSEndpointServicePrincipalList contains a list of AWSEndpointServicePrincipal diff --git a/config/crd/bases/clusterresources.instaclustr.com_awsencryptionkeys.yaml b/config/crd/bases/clusterresources.instaclustr.com_awsencryptionkeys.yaml index 781499026..5a578a20d 100644 --- a/config/crd/bases/clusterresources.instaclustr.com_awsencryptionkeys.yaml +++ b/config/crd/bases/clusterresources.instaclustr.com_awsencryptionkeys.yaml @@ -52,6 +52,8 @@ spec: type: string inUse: type: boolean + state: + type: string type: object type: object served: true diff --git a/config/crd/bases/clusterresources.instaclustr.com_awsendpointserviceprincipals.yaml b/config/crd/bases/clusterresources.instaclustr.com_awsendpointserviceprincipals.yaml index dc93be7ee..7c7ea2823 100644 --- a/config/crd/bases/clusterresources.instaclustr.com_awsendpointserviceprincipals.yaml +++ b/config/crd/bases/clusterresources.instaclustr.com_awsendpointserviceprincipals.yaml @@ -60,6 +60,9 @@ spec: id: description: The Instaclustr ID of the IAM Principal ARN type: string + state: + description: State describe current state of the resource + type: string type: object type: object served: true diff --git a/controllers/clusterresources/awsencryptionkey_controller.go b/controllers/clusterresources/awsencryptionkey_controller.go index 73d6d54a0..da4baba0f 100644 --- a/controllers/clusterresources/awsencryptionkey_controller.go +++ b/controllers/clusterresources/awsencryptionkey_controller.go @@ -129,6 +129,7 @@ func (r *AWSEncryptionKeyReconciler) handleCreate( ) encryptionKey.Status = *encryptionKeyStatus + encryptionKey.Status.State = models.CreatedStatus err = r.Status().Patch(ctx, encryptionKey, patch) if err != nil { l.Error(err, "Cannot patch AWS encryption key status ", "ID", encryptionKey.Status.ID) @@ -274,8 +275,30 @@ func (r *AWSEncryptionKeyReconciler) startEncryptionKeyStatusJob(encryptionKey * func (r *AWSEncryptionKeyReconciler) newWatchStatusJob(encryptionKey *v1beta1.AWSEncryptionKey) scheduler.Job { l := log.Log.WithValues("component", "EncryptionKeyStatusJob") return func() error { + ctx := context.Background() + + key := client.ObjectKeyFromObject(encryptionKey) + err := r.Get(ctx, key, encryptionKey) + if err != nil { + if k8serrors.IsNotFound(err) { + l.Info("Resource is not found in the k8s cluster. Closing Instaclustr status sync.", + "namespaced name", key, + ) + + r.Scheduler.RemoveJob(encryptionKey.GetJobID(scheduler.StatusChecker)) + + return nil + } + + return err + } + instaEncryptionKeyStatus, err := r.API.GetEncryptionKeyStatus(encryptionKey.Status.ID, instaclustr.AWSEncryptionKeyEndpoint) if err != nil { + if errors.Is(err, instaclustr.NotFound) { + return r.handleExternalDelete(ctx, encryptionKey) + } + l.Error(err, "Cannot get AWS encryption key status from Inst API", "encryption key ID", encryptionKey.Status.ID) return err } @@ -286,7 +309,7 @@ func (r *AWSEncryptionKeyReconciler) newWatchStatusJob(encryptionKey *v1beta1.AW "encryption key status", encryptionKey.Status) patch := encryptionKey.NewPatch() encryptionKey.Status = *instaEncryptionKeyStatus - err := r.Status().Patch(context.Background(), encryptionKey, patch) + err := r.Status().Patch(ctx, encryptionKey, patch) if err != nil { return err } @@ -296,6 +319,24 @@ func (r *AWSEncryptionKeyReconciler) newWatchStatusJob(encryptionKey *v1beta1.AW } } +func (r *AWSEncryptionKeyReconciler) handleExternalDelete(ctx context.Context, key *v1beta1.AWSEncryptionKey) error { + l := log.FromContext(ctx) + + patch := key.NewPatch() + key.Status.State = models.DeletedStatus + err := r.Status().Patch(ctx, key, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(key, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(key.GetJobID(scheduler.StatusChecker)) + + return nil +} + // SetupWithManager sets up the controller with the Manager. func (r *AWSEncryptionKeyReconciler) SetupWithManager(mgr ctrl.Manager) error { return ctrl.NewControllerManagedBy(mgr). diff --git a/controllers/clusterresources/awsendpointserviceprincipal_controller.go b/controllers/clusterresources/awsendpointserviceprincipal_controller.go index d7659ce11..fb81ddbb3 100644 --- a/controllers/clusterresources/awsendpointserviceprincipal_controller.go +++ b/controllers/clusterresources/awsendpointserviceprincipal_controller.go @@ -144,6 +144,19 @@ func (r *AWSEndpointServicePrincipalReconciler) handleCreate(ctx context.Context "AWS endpoint service principal resource has been created", ) + err = r.startWatchStatusJob(ctx, principal) + if err != nil { + l.Error(err, "failed to start status checker job") + r.EventRecorder.Eventf(principal, models.Warning, models.CreationFailed, + "Failed to start status checker job. Reason: %w", err, + ) + + return err + } + r.EventRecorder.Eventf(principal, models.Normal, models.Created, + "Status check job %s has been started", principal.GetJobID(scheduler.StatusChecker), + ) + return nil } @@ -175,6 +188,62 @@ func (r *AWSEndpointServicePrincipalReconciler) handleDelete(ctx context.Context return nil } +func (r *AWSEndpointServicePrincipalReconciler) startWatchStatusJob(ctx context.Context, resource *clusterresourcesv1beta1.AWSEndpointServicePrincipal) error { + job := r.newWatchStatusJob(ctx, resource) + return r.Scheduler.ScheduleJob(resource.GetJobID(scheduler.StatusChecker), scheduler.ClusterStatusInterval, job) +} + +func (r *AWSEndpointServicePrincipalReconciler) newWatchStatusJob(ctx context.Context, principal *clusterresourcesv1beta1.AWSEndpointServicePrincipal) scheduler.Job { + l := log.FromContext(ctx, "components", "WatchStatusJob") + + return func() error { + key := client.ObjectKeyFromObject(principal) + err := r.Get(ctx, key, principal) + if err != nil { + if k8serrors.IsNotFound(err) { + l.Info("Resource is not found in the k8s cluster. Closing Instaclustr status sync.", + "namespaced name", key, + ) + + r.Scheduler.RemoveJob(principal.GetJobID(scheduler.StatusChecker)) + + return nil + } + + return err + } + + _, err = r.API.GetAWSEndpointServicePrincipal(principal.Status.ID) + if err != nil { + if errors.Is(err, instaclustr.NotFound) { + return r.handleExternalDelete(ctx, principal) + } + + return err + } + + return nil + } +} + +func (r *AWSEndpointServicePrincipalReconciler) handleExternalDelete(ctx context.Context, principal *clusterresourcesv1beta1.AWSEndpointServicePrincipal) error { + l := log.FromContext(ctx) + + patch := principal.NewPatch() + principal.Status.State = models.DeletedStatus + err := r.Status().Patch(ctx, principal, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(principal, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(principal.GetJobID(scheduler.StatusChecker)) + + return nil +} + // SetupWithManager sets up the controller with the Manager. func (r *AWSEndpointServicePrincipalReconciler) SetupWithManager(mgr ctrl.Manager) error { return ctrl.NewControllerManagedBy(mgr). diff --git a/controllers/clusterresources/awssecuritygroupfirewallrule_controller.go b/controllers/clusterresources/awssecuritygroupfirewallrule_controller.go index 1c1f454d0..11da47237 100644 --- a/controllers/clusterresources/awssecuritygroupfirewallrule_controller.go +++ b/controllers/clusterresources/awssecuritygroupfirewallrule_controller.go @@ -288,11 +288,27 @@ func (r *AWSSecurityGroupFirewallRuleReconciler) newWatchStatusJob(firewallRule l := log.Log.WithValues("component", "FirewallRuleStatusJob") return func() error { ctx := context.Background() + + key := client.ObjectKeyFromObject(firewallRule) + err := r.Get(ctx, key, firewallRule) + if err != nil { + if k8serrors.IsNotFound(err) { + l.Info("Resource is not found in the k8s cluster. Closing Instaclustr status sync.", + "namespaced name", key, + ) + + r.Scheduler.RemoveJob(firewallRule.GetJobID(scheduler.StatusChecker)) + + return nil + } + + return err + } + instaFirewallRuleStatus, err := r.API.GetFirewallRuleStatus(firewallRule.Status.ID, instaclustr.AWSSecurityGroupFirewallRuleEndpoint) if err != nil { if errors.Is(err, instaclustr.NotFound) { - l.Info("The resource has been deleted on Instaclustr, deleting resource in k8s...") - return r.Delete(ctx, firewallRule) + return r.handleExternalDelete(ctx, firewallRule) } l.Error(err, "Cannot get AWS security group firewall rule status from Inst API", "firewall rule ID", firewallRule.Status.ID) @@ -309,16 +325,30 @@ func (r *AWSSecurityGroupFirewallRuleReconciler) newWatchStatusJob(firewallRule if err != nil { return err } - - if instaFirewallRuleStatus.Status == statusDELETED { - r.Scheduler.RemoveJob(firewallRule.GetJobID(scheduler.StatusChecker)) - } } return nil } } +func (r *AWSSecurityGroupFirewallRuleReconciler) handleExternalDelete(ctx context.Context, rule *v1beta1.AWSSecurityGroupFirewallRule) error { + l := log.FromContext(ctx) + + patch := rule.NewPatch() + rule.Status.Status = models.DeletedStatus + err := r.Status().Patch(ctx, rule, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(rule, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(rule.GetJobID(scheduler.StatusChecker)) + + return nil +} + // SetupWithManager sets up the controller with the Manager. func (r *AWSSecurityGroupFirewallRuleReconciler) SetupWithManager(mgr ctrl.Manager) error { return ctrl.NewControllerManagedBy(mgr). diff --git a/controllers/clusterresources/awsvpcpeering_controller.go b/controllers/clusterresources/awsvpcpeering_controller.go index a3cef3aaf..7aa4708fd 100644 --- a/controllers/clusterresources/awsvpcpeering_controller.go +++ b/controllers/clusterresources/awsvpcpeering_controller.go @@ -24,7 +24,6 @@ import ( k8serrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/runtime" "k8s.io/client-go/tools/record" - "k8s.io/utils/strings/slices" ctrl "sigs.k8s.io/controller-runtime" "sigs.k8s.io/controller-runtime/pkg/builder" "sigs.k8s.io/controller-runtime/pkg/client" @@ -208,7 +207,7 @@ func (r *AWSVPCPeeringReconciler) handleUpdatePeering( } if aws.Annotations[models.ExternalChangesAnnotation] == models.True { - if !slices.Equal(instaAWSPeering.PeerSubnets, aws.Spec.PeerSubnets) { + if !subnetsEqual(instaAWSPeering.PeerSubnets, aws.Spec.PeerSubnets) { l.Info("The resource specification still differs from the Instaclustr resource specification, please reconcile it manually", "AWS VPC ID", aws.Status.ID, "k8s peerSubnets", aws.Spec.PeerSubnets, @@ -404,8 +403,7 @@ func (r *AWSVPCPeeringReconciler) newWatchStatusJob(awsPeering *v1beta1.AWSVPCPe instaAWSPeering, err := r.API.GetAWSVPCPeering(awsPeering.Status.ID) if err != nil { if errors.Is(err, instaclustr.NotFound) { - l.Info("The resource has been deleted on Instaclustr, deleting resource in k8s...") - return r.Delete(ctx, awsPeering) + return r.handleExternalDelete(ctx, awsPeering) } l.Error(err, "cannot get AWS VPC Peering Status from Inst API", @@ -432,9 +430,20 @@ func (r *AWSVPCPeeringReconciler) newWatchStatusJob(awsPeering *v1beta1.AWSVPCPe } } + if awsPeering.Status.StatusCode == models.AWSVPCPeeringStatusCodeDeleted { + l.Info("The AWSPeering was deleted on AWS, stopping job...") + r.EventRecorder.Event(awsPeering, models.Warning, models.DeletedEvent, + "The AWSPeering was deleted on AWS, stopping job...", + ) + + r.Scheduler.RemoveJob(awsPeering.GetJobID(scheduler.StatusChecker)) + + return nil + } + if awsPeering.Annotations[models.ResourceStateAnnotation] != models.UpdatingEvent && awsPeering.Annotations[models.ExternalChangesAnnotation] != models.True && - !slices.Equal(instaAWSPeering.PeerSubnets, awsPeering.Spec.PeerSubnets) { + !subnetsEqual(instaAWSPeering.PeerSubnets, awsPeering.Spec.PeerSubnets) { l.Info("The k8s resource specification doesn't match the specification of Instaclustr, please change it manually", "k8s peerSubnets", instaAWSPeering.PeerSubnets, "instaclutr peerSubnets", awsPeering.Spec.PeerSubnets, @@ -498,3 +507,21 @@ func (r *AWSVPCPeeringReconciler) SetupWithManager(mgr ctrl.Manager) error { }, })).Complete(r) } + +func (r *AWSVPCPeeringReconciler) handleExternalDelete(ctx context.Context, key *v1beta1.AWSVPCPeering) error { + l := log.FromContext(ctx) + + patch := key.NewPatch() + key.Status.StatusCode = models.DeletedStatus + err := r.Status().Patch(ctx, key, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(key, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(key.GetJobID(scheduler.StatusChecker)) + + return nil +} diff --git a/controllers/clusterresources/azurevnetpeering_controller.go b/controllers/clusterresources/azurevnetpeering_controller.go index a8415eb6e..f3871baa9 100644 --- a/controllers/clusterresources/azurevnetpeering_controller.go +++ b/controllers/clusterresources/azurevnetpeering_controller.go @@ -304,8 +304,30 @@ func (r *AzureVNetPeeringReconciler) newWatchStatusJob(azureVNetPeering *v1beta1 ) scheduler.Job { l := log.Log.WithValues("component", "AzureVNetPeeringStatusJob") return func() error { + ctx := context.Background() + + key := client.ObjectKeyFromObject(azureVNetPeering) + err := r.Get(ctx, key, azureVNetPeering) + if err != nil { + if k8serrors.IsNotFound(err) { + l.Info("Resource is not found in the k8s cluster. Closing Instaclustr status sync.", + "namespaced name", key, + ) + + r.Scheduler.RemoveJob(azureVNetPeering.GetJobID(scheduler.StatusChecker)) + + return nil + } + + return err + } + instaPeeringStatus, err := r.API.GetPeeringStatus(azureVNetPeering.Status.ID, instaclustr.AzurePeeringEndpoint) if err != nil { + if errors.Is(err, instaclustr.NotFound) { + return r.handleExternalDelete(ctx, azureVNetPeering) + } + l.Error(err, "cannot get Azure VNet Peering Status from Inst API", "Azure VNet Peering ID", azureVNetPeering.Status.ID) return err } @@ -317,7 +339,7 @@ func (r *AzureVNetPeeringReconciler) newWatchStatusJob(azureVNetPeering *v1beta1 patch := azureVNetPeering.NewPatch() azureVNetPeering.Status.PeeringStatus = *instaPeeringStatus - err := r.Status().Patch(context.Background(), azureVNetPeering, patch) + err := r.Status().Patch(ctx, azureVNetPeering, patch) if err != nil { return err } @@ -327,6 +349,24 @@ func (r *AzureVNetPeeringReconciler) newWatchStatusJob(azureVNetPeering *v1beta1 } } +func (r *AzureVNetPeeringReconciler) handleExternalDelete(ctx context.Context, key *v1beta1.AzureVNetPeering) error { + l := log.FromContext(ctx) + + patch := key.NewPatch() + key.Status.StatusCode = models.DeletedStatus + err := r.Status().Patch(ctx, key, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(key, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(key.GetJobID(scheduler.StatusChecker)) + + return nil +} + // SetupWithManager sets up the controller with the Manager. func (r *AzureVNetPeeringReconciler) SetupWithManager(mgr ctrl.Manager) error { return ctrl.NewControllerManagedBy(mgr). diff --git a/controllers/clusterresources/gcpvpcpeering_controller.go b/controllers/clusterresources/gcpvpcpeering_controller.go index e11f67306..f2ce92a0d 100644 --- a/controllers/clusterresources/gcpvpcpeering_controller.go +++ b/controllers/clusterresources/gcpvpcpeering_controller.go @@ -287,8 +287,30 @@ func (r *GCPVPCPeeringReconciler) startGCPVPCPeeringStatusJob(gcpPeering *v1beta func (r *GCPVPCPeeringReconciler) newWatchStatusJob(gcpPeering *v1beta1.GCPVPCPeering) scheduler.Job { l := log.Log.WithValues("component", "GCPVPCPeeringStatusJob") return func() error { + ctx := context.Background() + + key := client.ObjectKeyFromObject(gcpPeering) + err := r.Get(ctx, key, gcpPeering) + if err != nil { + if k8serrors.IsNotFound(err) { + l.Info("Resource is not found in the k8s cluster. Closing Instaclustr status sync.", + "namespaced name", key, + ) + + r.Scheduler.RemoveJob(gcpPeering.GetJobID(scheduler.StatusChecker)) + + return nil + } + + return err + } + instaPeeringStatus, err := r.API.GetPeeringStatus(gcpPeering.Status.ID, instaclustr.GCPPeeringEndpoint) if err != nil { + if errors.Is(err, instaclustr.NotFound) { + return r.handleExternalDelete(ctx, gcpPeering) + } + l.Error(err, "Cannot get GCP VPC Peering Status from Inst API", "id", gcpPeering.Status.ID) return err } @@ -300,7 +322,7 @@ func (r *GCPVPCPeeringReconciler) newWatchStatusJob(gcpPeering *v1beta1.GCPVPCPe patch := gcpPeering.NewPatch() gcpPeering.Status.PeeringStatus = *instaPeeringStatus - err := r.Status().Patch(context.Background(), gcpPeering, patch) + err := r.Status().Patch(ctx, gcpPeering, patch) if err != nil { return err } @@ -310,6 +332,24 @@ func (r *GCPVPCPeeringReconciler) newWatchStatusJob(gcpPeering *v1beta1.GCPVPCPe } } +func (r *GCPVPCPeeringReconciler) handleExternalDelete(ctx context.Context, key *v1beta1.GCPVPCPeering) error { + l := log.FromContext(ctx) + + patch := key.NewPatch() + key.Status.StatusCode = models.DeletedStatus + err := r.Status().Patch(ctx, key, patch) + if err != nil { + return err + } + + l.Info(instaclustr.MsgInstaclustrResourceNotFound) + r.EventRecorder.Eventf(key, models.Warning, models.ExternalDeleted, instaclustr.MsgInstaclustrResourceNotFound) + + r.Scheduler.RemoveJob(key.GetJobID(scheduler.StatusChecker)) + + return nil +} + // SetupWithManager sets up the controller with the Manager. func (r *GCPVPCPeeringReconciler) SetupWithManager(mgr ctrl.Manager) error { return ctrl.NewControllerManagedBy(mgr). diff --git a/controllers/clusterresources/helpers.go b/controllers/clusterresources/helpers.go index 1735d8c59..799ae9bae 100644 --- a/controllers/clusterresources/helpers.go +++ b/controllers/clusterresources/helpers.go @@ -98,3 +98,24 @@ func getUserCreds(secret *k8sCore.Secret) (username, password string, err error) return username, password, nil } + +func subnetsEqual(subnets1, subnets2 []string) bool { + if len(subnets1) != len(subnets2) { + return false + } + + for _, s1 := range subnets1 { + var equal bool + for _, s2 := range subnets2 { + if s1 == s2 { + equal = true + } + } + + if !equal { + return false + } + } + + return true +} diff --git a/pkg/instaclustr/client.go b/pkg/instaclustr/client.go index fbfd87979..4bfa8119c 100644 --- a/pkg/instaclustr/client.go +++ b/pkg/instaclustr/client.go @@ -2181,6 +2181,37 @@ func (c *Client) UpdateClusterSettings(clusterID string, settings *models.Cluste return nil } +func (c *Client) GetAWSEndpointServicePrincipal(principalID string) (*models.AWSEndpointServicePrincipal, error) { + url := c.serverHostname + AWSEndpointServicePrincipalEndpoint + principalID + + resp, err := c.DoRequest(url, http.MethodGet, nil) + if err != nil { + return nil, err + } + + defer resp.Body.Close() + b, err := io.ReadAll(resp.Body) + if err != nil { + return nil, err + } + + if resp.StatusCode == http.StatusNotFound { + return nil, NotFound + } + + if resp.StatusCode != http.StatusOK { + return nil, fmt.Errorf("status code: %d, message: %s", resp.StatusCode, b) + } + + var principal models.AWSEndpointServicePrincipal + err = json.Unmarshal(b, &principal) + if err != nil { + return nil, err + } + + return &principal, nil +} + func (c *Client) CreateAWSEndpointServicePrincipal(spec any) ([]byte, error) { url := c.serverHostname + AWSEndpointServicePrincipalEndpoint diff --git a/pkg/instaclustr/interfaces.go b/pkg/instaclustr/interfaces.go index fca23f78d..6e19e5041 100644 --- a/pkg/instaclustr/interfaces.go +++ b/pkg/instaclustr/interfaces.go @@ -100,6 +100,7 @@ type API interface { ListAppVersions(app string) ([]*models.AppVersions, error) GetDefaultCredentialsV1(clusterID string) (string, string, error) UpdateClusterSettings(clusterID string, settings *models.ClusterSettings) error + GetAWSEndpointServicePrincipal(id string) (*models.AWSEndpointServicePrincipal, error) CreateAWSEndpointServicePrincipal(spec any) ([]byte, error) DeleteAWSEndpointServicePrincipal(principalID string) error GetResizeOperationsByClusterDataCentreID(cdcID string) ([]*v1beta1.ResizeOperation, error) diff --git a/pkg/instaclustr/mock/client.go b/pkg/instaclustr/mock/client.go index 92da8f4c6..36c8b1e32 100644 --- a/pkg/instaclustr/mock/client.go +++ b/pkg/instaclustr/mock/client.go @@ -375,3 +375,7 @@ func (c *mockClient) GetResizeOperationsByClusterDataCentreID(cdcID string) ([]* func (c *mockClient) GetAWSVPCPeering(peerID string) (*models.AWSVPCPeering, error) { panic("GetAWSVPCPeering: is not implemented") } + +func (c *mockClient) GetAWSEndpointServicePrincipal(id string) (*models.AWSEndpointServicePrincipal, error) { + panic("GetAWSEndpointServicePrincipal: is not implemented") +} diff --git a/pkg/models/apiv2.go b/pkg/models/apiv2.go index 080977872..1b78aca30 100644 --- a/pkg/models/apiv2.go +++ b/pkg/models/apiv2.go @@ -20,6 +20,7 @@ const ( NoOperation = "NO_OPERATION" OperationInProgress = "OPERATION_IN_PROGRESS" + CreatedStatus = "CREATED" DeletedStatus = "DELETED" DefaultAccountName = "INSTACLUSTR" @@ -205,3 +206,10 @@ type AWSVPCPeering struct { PeerVpcID string `json:"peerVpcId"` StatusCode string `json:"statusCode"` } + +type AWSEndpointServicePrincipal struct { + ID string `json:"id,omitempty"` + ClusterDataCenterID string `json:"clusterDataCenterId,omitempty"` + EndPointServiceID string `json:"endPointServiceId,omitempty"` + PrincipalARN string `json:"principalArn,omitempty"` +} diff --git a/pkg/models/operator.go b/pkg/models/operator.go index a5950fe77..435480682 100644 --- a/pkg/models/operator.go +++ b/pkg/models/operator.go @@ -157,6 +157,10 @@ const ( UpcomingME = "upcoming" ) +const ( + AWSVPCPeeringStatusCodeDeleted = "deleted" +) + const Requeue60 = time.Second * 60 var (