2
0

get_controllers.go 4.0 KB

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