resources/compute-forwarding-rule.go
package resources
import (
"context"
"errors"
"fmt"
"github.com/sirupsen/logrus"
"google.golang.org/api/iterator"
compute "cloud.google.com/go/compute/apiv1"
"cloud.google.com/go/compute/apiv1/computepb"
liberror "github.com/ekristen/libnuke/pkg/errors"
"github.com/ekristen/libnuke/pkg/registry"
"github.com/ekristen/libnuke/pkg/resource"
"github.com/ekristen/libnuke/pkg/types"
"github.com/ekristen/gcp-nuke/pkg/nuke"
)
const ComputeForwardingRuleResource = "ComputeForwardingRule"
func init() {
registry.Register(®istry.Registration{
Name: ComputeForwardingRuleResource,
Scope: nuke.Project,
Lister: &ComputeForwardingRuleLister{},
})
}
type ComputeForwardingRuleLister struct {
svc *compute.ForwardingRulesClient
globalSvc *compute.GlobalForwardingRulesClient
}
func (l *ComputeForwardingRuleLister) List(ctx context.Context, o interface{}) ([]resource.Resource, error) {
var resources []resource.Resource
opts := o.(*nuke.ListerOpts)
if err := opts.BeforeList(nuke.Global, "compute.googleapis.com"); err == nil {
globalResources, err := l.listGlobal(ctx, opts)
if err != nil {
logrus.WithError(err).Error("unable to list global security policies")
} else {
resources = append(resources, globalResources...)
}
}
if err := opts.BeforeList(nuke.Regional, "compute.googleapis.com"); err == nil {
regionalResources, err := l.listRegional(ctx, opts)
if err != nil {
logrus.WithError(err).Error("unable to list regional security policies")
} else {
resources = append(resources, regionalResources...)
}
}
return resources, nil
}
func (l *ComputeForwardingRuleLister) listGlobal(ctx context.Context, opts *nuke.ListerOpts) ([]resource.Resource, error) {
var resources []resource.Resource
if l.globalSvc == nil {
var err error
l.globalSvc, err = compute.NewGlobalForwardingRulesRESTClient(ctx)
if err != nil {
return nil, err
}
}
req := &computepb.ListGlobalForwardingRulesRequest{
Project: *opts.Project,
}
it := l.globalSvc.List(ctx, req)
for {
resp, err := it.Next()
if errors.Is(err, iterator.Done) {
break
}
if err != nil {
logrus.WithError(err).Error("unable to iterate networks")
break
}
resources = append(resources, &ComputeForwardingRule{
globalSvc: l.globalSvc,
project: opts.Project,
Name: resp.Name,
CreatedAt: resp.CreationTimestamp,
Labels: resp.Labels,
})
}
return resources, nil
}
func (l *ComputeForwardingRuleLister) listRegional(ctx context.Context, opts *nuke.ListerOpts) ([]resource.Resource, error) {
var resources []resource.Resource
if l.svc == nil {
var err error
l.svc, err = compute.NewForwardingRulesRESTClient(ctx)
if err != nil {
return nil, err
}
}
req := &computepb.ListForwardingRulesRequest{
Project: *opts.Project,
Region: *opts.Region,
}
it := l.svc.List(ctx, req)
for {
resp, err := it.Next()
if errors.Is(err, iterator.Done) {
break
}
if err != nil {
logrus.WithError(err).Error("unable to iterate networks")
break
}
resources = append(resources, &ComputeForwardingRule{
svc: l.svc,
project: opts.Project,
region: opts.Region,
Name: resp.Name,
CreatedAt: resp.CreationTimestamp,
Labels: resp.Labels,
})
}
return resources, nil
}
type ComputeForwardingRule struct {
svc *compute.ForwardingRulesClient
globalSvc *compute.GlobalForwardingRulesClient
removeOp *compute.Operation
project *string
region *string
Name *string
CreatedAt *string
Labels map[string]string `property:"tagPrefix=label"`
}
func (r *ComputeForwardingRule) Remove(ctx context.Context) error {
if r.svc != nil {
return r.removeRegional(ctx)
} else if r.globalSvc != nil {
return r.removeGlobal(ctx)
}
return errors.New("unable to determine service")
}
func (r *ComputeForwardingRule) removeGlobal(ctx context.Context) (err error) {
r.removeOp, err = r.globalSvc.Delete(ctx, &computepb.DeleteGlobalForwardingRuleRequest{
Project: *r.project,
ForwardingRule: *r.Name,
})
return err
}
func (r *ComputeForwardingRule) removeRegional(ctx context.Context) (err error) {
r.removeOp, err = r.svc.Delete(ctx, &computepb.DeleteForwardingRuleRequest{
Project: *r.project,
Region: *r.region,
ForwardingRule: *r.Name,
})
return err
}
func (r *ComputeForwardingRule) Properties() types.Properties {
return types.NewPropertiesFromStruct(r)
}
func (r *ComputeForwardingRule) String() string {
return *r.Name
}
func (r *ComputeForwardingRule) HandleWait(ctx context.Context) error {
if r.removeOp == nil {
return nil
}
if err := r.removeOp.Poll(ctx); err != nil {
logrus.WithError(err).Trace("remove op polling encountered error")
return err
}
if !r.removeOp.Done() {
return liberror.ErrWaitResource("waiting for operation to complete")
}
if r.removeOp.Done() {
if r.removeOp.Proto().GetError() != nil {
removeErr := fmt.Errorf("delete error on '%s': %s", r.removeOp.Proto().GetTargetLink(), r.removeOp.Proto().GetHttpErrorMessage())
logrus.WithError(removeErr).WithField("status_code", r.removeOp.Proto().GetError()).Errorf("unable to delete %s", ComputeForwardingRuleResource)
return removeErr
}
}
return nil
}