Skip to content

Commit

Permalink
Enhancements to run Trident as an AKS extension
Browse files Browse the repository at this point in the history
*  Enhancement to run Trident as an AKS extension as part of offering Triden
t as an 1P service in Azure.

---------

Co-authored-by: VinayKumarHavanur <[email protected]>
Co-authored-by: Shubham Phadnis <[email protected]>
  • Loading branch information
3 people authored Apr 12, 2024
1 parent 833cd73 commit bd499b4
Show file tree
Hide file tree
Showing 86 changed files with 6,506 additions and 114 deletions.
12 changes: 8 additions & 4 deletions BUILD.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,13 @@

* Docker-compatible container cli, such as nerdctl or docker
* Make, if not building natively or with linker flags
* Go 1.19 or greater, to optionally build Trident natively
* Go 1.21 or greater, to optionally build Trident natively

### Multi-Platform

* Make
* Docker
* Go 1.19 or greater, to optionally build Trident binaries natively
* Go 1.21 or greater, to optionally build Trident binaries natively
* jq

## Makefile Parameters
Expand All @@ -32,7 +32,7 @@ Container registry used to tag images and manifests, and optionally to push imag

`GO_IMAGE`

Default: `golang:1.19`
Default: `golang:1.21`

Container image used by default `$GO_SHELL` to run binary build scripts.

Expand All @@ -52,11 +52,15 @@ Default: `alpine/helm:3.6.1`

Container image used by default `$HELM_CMD` to package helm chart.

`HELM_CHART_VERSION`

If defined, overrides the default chart version

`DOCKER_CLI`

Default: `docker`

docker-compatible cli, such as nerdctl or docker, used to run containers and tag single-platform images.
Docker-compatible cli, such as nerdctl or docker, used to run containers and tag single-platform images.

`BUILD_CLI`

Expand Down
17 changes: 11 additions & 6 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,9 @@ GOFLAGS ?=
# HELM_IMAGE helm image used in default HELM_CMD
HELM_IMAGE ?= alpine/helm:3.6.1

# HELM_CHART_VERSION overrides the default chart version
HELM_CHART_VERSION ?=

# DOCKER_CLI the docker-compatible cli used to run and tag images
DOCKER_CLI ?= docker

Expand Down Expand Up @@ -96,7 +99,7 @@ OPERATOR_CONFIG_PKG = github.com/netapp/trident/operator/config
TRIDENT_KUBERNETES_PKG = github.com/netapp/trident/persistent_store/crd
OPERATOR_CONFIG_PKG = github.com/netapp/trident/operator/config
OPERATOR_INSTALLER_CONFIG_PKG = github.com/netapp/trident/operator/controllers/orchestrator/installer
OPERATOR_KUBERNETES_PKG = github.com/netapp/trident/operator/controllers/orchestrator
OPERATOR_KUBERNETES_PKG = github.com/netapp/trident/operator/crd
VERSION_FILE = github.com/netapp/trident/hack/VERSION
BUILD_ROOT = /go/src/github.com/netapp/trident
TRIDENT_VOLUME = trident-build
Expand Down Expand Up @@ -363,7 +366,9 @@ operator_manifest: operator_images
# packages helm chart
chart:
@cp README.md ./helm/trident-operator/
@$(HELM_CMD) package ./helm/trident-operator $(if $(HELM_PGP_KEY),--sign --key "$(HELM_PGP_KEY)" --keyring "$(HELM_PGP_KEYRING)")
@$(HELM_CMD) package ./helm/trident-operator \
$(if $(HELM_PGP_KEY),--sign --key "$(HELM_PGP_KEY)" --keyring "$(HELM_PGP_KEYRING)") \
$(if $(HELM_CHART_VERSION),--version "$(HELM_CHART_VERSION)")
@rm -f ./helm/trident-operator/README.md

# builds installer bundle. Skips binaries that have not been built.
Expand Down Expand Up @@ -429,10 +434,10 @@ k8s_codegen_operator:
@$(K8S_CODE_GENERATOR)/generate-groups.sh all $(OPERATOR_KUBERNETES_PKG)/client \
$(OPERATOR_KUBERNETES_PKG)/apis "netapp:v1" -h ./hack/boilerplate.go.txt
@rm -rf $(K8S_CODE_GENERATOR)
@rm -rf ./operator/controllers/orchestrator/client/*
@mv $(OPERATOR_KUBERNETES_PKG)/client/* ./operator/controllers/orchestrator/client/
@rm -rf ./operator/controllers/orchestrator/apis/netapp/v1/zz_generated.deepcopy.go
@mv $(OPERATOR_KUBERNETES_PKG)/apis/netapp/v1/zz_generated.deepcopy.go ./operator/controllers/orchestrator/apis/netapp/v1/
@rm -rf ./operator/crd/client/*
@mv $(OPERATOR_KUBERNETES_PKG)/client/* ./operator/crd/client/
@rm -rf ./operator/crd/apis/netapp/v1/zz_generated.deepcopy.go
@mv $(OPERATOR_KUBERNETES_PKG)/apis/netapp/v1/zz_generated.deepcopy.go ./operator/crd/apis/netapp/v1/

mocks:
@go install github.com/golang/mock/[email protected]
Expand Down
22 changes: 22 additions & 0 deletions cli/api/types.go
Original file line number Diff line number Diff line change
Expand Up @@ -86,3 +86,25 @@ type KubernetesNamespace struct {
Kind string `json:"kind"`
Metadata Metadata `json:"metadata"`
}

type CRStatus struct {
Status string `json:"status"`
Message string `json:"message"`
}

type OperatorPhaseStatus string

const (
OperatorPhaseDone OperatorPhaseStatus = "Done"
OperatorPhaseProcessing OperatorPhaseStatus = "Processing"
OperatorPhaseUnknown OperatorPhaseStatus = "Unknown"
OperatorPhaseFailed OperatorPhaseStatus = "Failed"
OperatorPhaseError OperatorPhaseStatus = "Error"
)

type OperatorStatus struct {
ErrorMessage string `json:"errorMessage"`
Status string `json:"operatorStatus"`
TorcStatus map[string]CRStatus `json:"torcStatus"`
TconfStatus map[string]CRStatus `json:"tconfStatus"`
}
20 changes: 20 additions & 0 deletions cli/cmd/check.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
// Copyright 2024 NetApp, Inc. All Rights Reserved.

package cmd

import "github.com/spf13/cobra"

func init() {
RootCmd.AddCommand(checkCmd)
}

var checkCmd = &cobra.Command{
Use: "check",
Short: "check status of a Trident pod",
Hidden: true,
PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
initCmdLogging()
err := discoverOperatingMode(cmd)
return err
},
}
158 changes: 158 additions & 0 deletions cli/cmd/check_operator.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,158 @@
// Copyright 2024 NetApp, Inc. All Rights Reserved.

package cmd

import (
"encoding/json"
"fmt"
"net/http"
"os"
"time"

"github.com/cenkalti/backoff/v4"
"github.com/olekukonko/tablewriter"
"github.com/spf13/cobra"

cliapi "github.com/netapp/trident/cli/api"
"github.com/netapp/trident/utils/errors"
)

var checkTimeout int32

const (
RequestTimeout = 5 * time.Second
backoffMaxInterval = 5 * time.Second
operatorService = "trident-operator"
operatorServicePort = "8000"
operatorServicePath = "/operator/status"
)

func init() {
checkCmd.AddCommand(checkOperatorCmd)
checkOperatorCmd.Flags().Int32VarP(&checkTimeout, "timeout", "t", 1, "timeout in seconds for the operator check")
if err := checkOperatorCmd.Flags().MarkHidden("timeout"); err != nil {
fmt.Fprintln(os.Stderr, err)
}
}

var checkOperatorCmd = &cobra.Command{
Use: "operator",
Short: "check operator pod status",
Aliases: []string{"o"},
Hidden: true,
RunE: func(cmd *cobra.Command, args []string) error {
if OperatingMode == ModeTunnel {
command := []string{
"check",
"operator",
"--timeout",
fmt.Sprintf("%d", checkTimeout),
}
out, err := TunnelCommand(append(command, args...))
printOutput(cmd, out, err)
return err
} else {
return checkOperatorStatus(checkTimeout)
}
},
}

func checkOperatorStatus(checkTimeout int32) error {
var err error
status := cliapi.OperatorStatus{}
var response *http.Response
var url string

getOperatorStatus := func() error {
if namespace := os.Getenv("POD_NAMESPACE"); namespace == "" {
return errors.New("error in getting trident operator pod")
} else {
url = "http://" + operatorService + "." + namespace + "." + "svc.cluster.local" + ":" + operatorServicePort + operatorServicePath
}
request, err := http.NewRequest("GET", url, nil)
request.Header.Set("Content-Type", "application/json")

client := &http.Client{Timeout: RequestTimeout}
if response, err = client.Do(request); err != nil {
return err
}

defer response.Body.Close()

err = json.NewDecoder(response.Body).Decode(&status)
if status.Status != string(cliapi.OperatorPhaseDone) {
return errors.New("operator pod status is not set to done")
}

return nil
}

checkOperatorNotify := func(err error, duration time.Duration) {}

checkOperatorBackoff := backoff.NewExponentialBackOff()
checkOperatorBackoff.MaxInterval = backoffMaxInterval
checkOperatorBackoff.MaxElapsedTime = time.Duration(checkTimeout) * time.Second

if err = backoff.RetryNotify(getOperatorStatus, checkOperatorBackoff, checkOperatorNotify); err != nil {
if status.Status != "" {
writeStatus(status)
}
return err
}

writeStatus(status)
return nil
}

func writeStatus(status cliapi.OperatorStatus) {
switch OutputFormat {
case FormatJSON:
WriteJSON(status)
case FormatYAML:
WriteYAML(status)
case FormatWide:
writeStatusTableWide(status)
default:
writeStatusTable(status)
}
}

func writeStatusTable(status cliapi.OperatorStatus) {
table := tablewriter.NewWriter(os.Stdout)
table.SetHeader([]string{"Name", "Type", "Status"})
if status.ErrorMessage != "" {
table.SetFooter([]string{"Overall Status", status.Status, status.ErrorMessage})
} else {
table.SetFooter([]string{"Overall Status", status.Status, ""})
}

for torcName, torcStatus := range status.TorcStatus {
table.Append([]string{torcName, "TridentOrchestratorCR", torcStatus.Status})
}

for tconfName, tconfStatus := range status.TconfStatus {
table.Append([]string{tconfName, "TridentConfiguratorCR", tconfStatus.Status})
}

table.Render()
}

func writeStatusTableWide(status cliapi.OperatorStatus) {
table := tablewriter.NewWriter(os.Stdout)
table.SetHeader([]string{"Name", "Type", "Status", "Message"})
if status.ErrorMessage != "" {
table.SetFooter([]string{"Overall Status", status.Status, status.ErrorMessage, ""})
} else {
table.SetFooter([]string{"Overall Status", status.Status, "", ""})
}

for torcName, torcStatus := range status.TorcStatus {
table.Append([]string{torcName, "TridentOrchestratorCR", torcStatus.Status, torcStatus.Message})
}

for tconfName, tconfStatus := range status.TconfStatus {
table.Append([]string{tconfName, "TridentConfiguratorCR", tconfStatus.Status, tconfStatus.Message})
}

table.Render()
}
2 changes: 2 additions & 0 deletions cli/cmd/install.go
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ const (
VolumeCRDName = "tridentvolumes.trident.netapp.io"
VolumePublicationCRDName = "tridentvolumepublications.trident.netapp.io"
VolumeReferenceCRDName = "tridentvolumereferences.trident.netapp.io"
ConfiguratorCRDName = "tridentconfigurators.trident.netapp.io"

ControllerRoleFilename = "trident-controller-role.yaml"
ControllerClusterRoleFilename = "trident-controller-clusterrole.yaml"
Expand Down Expand Up @@ -175,6 +176,7 @@ var (
VolumeCRDName,
VolumePublicationCRDName,
ActionSnapshotRestoreCRDName,
ConfiguratorCRDName,
}
)

Expand Down
2 changes: 1 addition & 1 deletion cli/k8s_client/client_factory.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ import (

"github.com/netapp/trident/config"
. "github.com/netapp/trident/logging"
torc "github.com/netapp/trident/operator/controllers/orchestrator/client/clientset/versioned"
torc "github.com/netapp/trident/operator/crd/client/clientset/versioned"
tridentv1clientset "github.com/netapp/trident/persistent_store/crd/client/clientset/versioned"
"github.com/netapp/trident/utils/errors"
)
Expand Down
Loading

0 comments on commit bd499b4

Please sign in to comment.