get_controllers.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. package release
  2. import (
  3. "errors"
  4. "fmt"
  5. "net/http"
  6. "strings"
  7. "github.com/porter-dev/porter/api/server/authz"
  8. "github.com/porter-dev/porter/api/server/handlers"
  9. "github.com/porter-dev/porter/api/server/shared"
  10. "github.com/porter-dev/porter/api/server/shared/apierrors"
  11. "github.com/porter-dev/porter/api/server/shared/config"
  12. "github.com/porter-dev/porter/api/types"
  13. "github.com/porter-dev/porter/internal/helm/grapher"
  14. "github.com/porter-dev/porter/internal/kubernetes"
  15. "github.com/porter-dev/porter/internal/models"
  16. "github.com/stefanmcshane/helm/pkg/release"
  17. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  18. )
  19. type GetControllersHandler struct {
  20. handlers.PorterHandlerReadWriter
  21. authz.KubernetesAgentGetter
  22. }
  23. func NewGetControllersHandler(
  24. config *config.Config,
  25. writer shared.ResultWriter,
  26. ) *GetControllersHandler {
  27. return &GetControllersHandler{
  28. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, nil, writer),
  29. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  30. }
  31. }
  32. func (c *GetControllersHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  33. helmRelease, _ := r.Context().Value(types.ReleaseScope).(*release.Release)
  34. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  35. agent, err := c.GetAgent(r, cluster, "")
  36. if err != nil {
  37. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  38. return
  39. }
  40. yamlArr := grapher.ImportMultiDocYAML([]byte(helmRelease.Manifest))
  41. controllers := grapher.ParseControllers(yamlArr)
  42. retrievedControllers := []interface{}{}
  43. // get current status of each controller
  44. for _, controller := range controllers {
  45. controller.Namespace = helmRelease.Namespace
  46. rc, _, err := getController(controller, agent)
  47. if targetErr := kubernetes.IsNotFoundError; errors.Is(err, targetErr) {
  48. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(
  49. fmt.Errorf("%s/%s of kind %s was not found", controller.Namespace, controller.Name, controller.Kind),
  50. http.StatusNotFound,
  51. ))
  52. return
  53. } else if err != nil {
  54. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  55. return
  56. }
  57. retrievedControllers = append(retrievedControllers, rc)
  58. }
  59. c.WriteResult(w, r, retrievedControllers)
  60. }
  61. func getController(controller grapher.Object, agent *kubernetes.Agent) (rc interface{}, selector *metav1.LabelSelector, err error) {
  62. switch strings.ToLower(controller.Kind) {
  63. case "deployment":
  64. obj, err := agent.GetDeployment(controller)
  65. if err != nil {
  66. controller.Namespace = "default"
  67. obj, err = agent.GetDeployment(controller)
  68. if err != nil {
  69. err = fmt.Errorf("error getting deployment: %w", err)
  70. return nil, nil, err
  71. }
  72. }
  73. return obj, obj.Spec.Selector, nil
  74. case "statefulset":
  75. obj, err := agent.GetStatefulSet(controller)
  76. if err != nil {
  77. controller.Namespace = "default"
  78. obj, err = agent.GetStatefulSet(controller)
  79. if err != nil {
  80. err = fmt.Errorf("error getting stateful set: %w", err)
  81. return nil, nil, err
  82. }
  83. }
  84. return obj, obj.Spec.Selector, nil
  85. case "daemonset":
  86. obj, err := agent.GetDaemonSet(controller)
  87. if err != nil {
  88. controller.Namespace = "default"
  89. obj, err = agent.GetDaemonSet(controller)
  90. if err != nil {
  91. err = fmt.Errorf("error getting daemon set: %w", err)
  92. return nil, nil, err
  93. }
  94. }
  95. return obj, obj.Spec.Selector, nil
  96. case "replicaset":
  97. obj, err := agent.GetReplicaSet(controller)
  98. if err != nil {
  99. controller.Namespace = "default"
  100. obj, err = agent.GetReplicaSet(controller)
  101. if err != nil {
  102. err = fmt.Errorf("error getting replica set: %w", err)
  103. return nil, nil, err
  104. }
  105. }
  106. return obj, obj.Spec.Selector, nil
  107. case "cronjob":
  108. obj, err := agent.GetCronJob(controller)
  109. if err != nil {
  110. controller.Namespace = "default"
  111. obj, err = agent.GetCronJob(controller)
  112. if err != nil {
  113. err = fmt.Errorf("error getting cron job %w", err)
  114. return nil, nil, err
  115. }
  116. }
  117. res := &metav1.LabelSelector{
  118. MatchLabels: make(map[string]string),
  119. }
  120. for key, val := range obj.Spec.JobTemplate.Labels {
  121. res.MatchLabels[key] = val
  122. }
  123. return obj, res, nil
  124. case "job":
  125. obj, err := agent.GetJob(controller)
  126. if err != nil {
  127. controller.Namespace = "default"
  128. obj, err = agent.GetJob(controller)
  129. if err != nil {
  130. err = fmt.Errorf("error getting job: %w", err)
  131. return nil, nil, err
  132. }
  133. }
  134. return obj, obj.Spec.Selector, nil
  135. }
  136. return nil, nil, fmt.Errorf("not a valid controller")
  137. }