create.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. package stacks
  2. import (
  3. "fmt"
  4. "net/http"
  5. "github.com/porter-dev/porter/api/server/authz"
  6. "github.com/porter-dev/porter/api/server/handlers"
  7. "github.com/porter-dev/porter/api/server/shared"
  8. "github.com/porter-dev/porter/api/server/shared/apierrors"
  9. "github.com/porter-dev/porter/api/server/shared/config"
  10. "github.com/porter-dev/porter/api/types"
  11. "github.com/porter-dev/porter/internal/helm"
  12. "github.com/porter-dev/porter/internal/models"
  13. )
  14. type CreateStackHandler struct {
  15. handlers.PorterHandlerReadWriter
  16. authz.KubernetesAgentGetter
  17. }
  18. func NewCreateStackHandler(
  19. config *config.Config,
  20. decoderValidator shared.RequestDecoderValidator,
  21. writer shared.ResultWriter,
  22. ) *CreateStackHandler {
  23. return &CreateStackHandler{
  24. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  25. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  26. }
  27. }
  28. func (c *CreateStackHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  29. ctx := r.Context()
  30. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  31. request := &types.CreateStackReleaseRequest{}
  32. if ok := c.DecodeAndValidate(w, r, request); !ok {
  33. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error decoding request")))
  34. return
  35. }
  36. stackName := request.StackName
  37. namespace := fmt.Sprintf("porter-stack-%s", stackName)
  38. helmAgent, err := c.GetHelmAgent(r, cluster, namespace)
  39. if err != nil {
  40. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error getting helm agent: %w", err)))
  41. return
  42. }
  43. k8sAgent, err := c.GetAgent(r, cluster, namespace)
  44. if err != nil {
  45. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error getting k8s agent: %w", err)))
  46. return
  47. }
  48. // create the namespace if it does not exist already
  49. _, err = k8sAgent.CreateNamespace(namespace, nil)
  50. if err != nil {
  51. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error creating namespace: %w", err)))
  52. return
  53. }
  54. chart, err := createChartFromDependencies(request.Dependencies)
  55. if err != nil {
  56. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error creating chart: %w", err)))
  57. return
  58. }
  59. registries, err := c.Repo().Registry().ListRegistriesByProjectID(cluster.ProjectID)
  60. if err != nil {
  61. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error listing registries: %w", err)))
  62. return
  63. }
  64. conf := &helm.InstallChartConfig{
  65. Chart: chart,
  66. Name: stackName,
  67. Namespace: namespace,
  68. Values: request.Values,
  69. Cluster: cluster,
  70. Repo: c.Repo(),
  71. Registries: registries,
  72. }
  73. _, err = helmAgent.InstallChart(conf, c.Config().DOConf, c.Config().ServerConf.DisablePullSecretsInjection)
  74. if err != nil {
  75. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(
  76. fmt.Errorf("error installing a new chart: %s", err.Error()),
  77. http.StatusBadRequest,
  78. ))
  79. return
  80. }
  81. w.WriteHeader(http.StatusCreated)
  82. }