create_porter_app.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. package stacks
  2. import (
  3. "encoding/base64"
  4. "fmt"
  5. "net/http"
  6. "github.com/porter-dev/porter/api/server/authz"
  7. "github.com/porter-dev/porter/api/server/handlers"
  8. "github.com/porter-dev/porter/api/server/shared"
  9. "github.com/porter-dev/porter/api/server/shared/apierrors"
  10. "github.com/porter-dev/porter/api/server/shared/config"
  11. "github.com/porter-dev/porter/api/server/shared/requestutils"
  12. "github.com/porter-dev/porter/api/types"
  13. "github.com/porter-dev/porter/internal/helm"
  14. "github.com/porter-dev/porter/internal/models"
  15. "github.com/stefanmcshane/helm/pkg/chart"
  16. )
  17. type CreatePorterAppHandler struct {
  18. handlers.PorterHandlerReadWriter
  19. authz.KubernetesAgentGetter
  20. }
  21. func NewCreatePorterAppHandler(
  22. config *config.Config,
  23. decoderValidator shared.RequestDecoderValidator,
  24. writer shared.ResultWriter,
  25. ) *CreatePorterAppHandler {
  26. return &CreatePorterAppHandler{
  27. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  28. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  29. }
  30. }
  31. func (c *CreatePorterAppHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  32. ctx := r.Context()
  33. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  34. cluster, _ := ctx.Value(types.ClusterScope).(*models.Cluster)
  35. request := &types.CreatePorterAppRequest{}
  36. if ok := c.DecodeAndValidate(w, r, request); !ok {
  37. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error decoding request")))
  38. return
  39. }
  40. stackName, reqErr := requestutils.GetURLParamString(r, types.URLParamStackName)
  41. if reqErr != nil {
  42. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(reqErr, http.StatusBadRequest))
  43. return
  44. }
  45. namespace := fmt.Sprintf("porter-stack-%s", stackName)
  46. helmAgent, err := c.GetHelmAgent(r, cluster, namespace)
  47. if err != nil {
  48. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error getting helm agent: %w", err)))
  49. return
  50. }
  51. k8sAgent, err := c.GetAgent(r, cluster, namespace)
  52. if err != nil {
  53. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error getting k8s agent: %w", err)))
  54. return
  55. }
  56. helmRelease, err := helmAgent.GetRelease(stackName, 0, false)
  57. shouldCreate := err != nil
  58. porterYamlBase64 := request.PorterYAMLBase64
  59. porterYaml, err := base64.StdEncoding.DecodeString(porterYamlBase64)
  60. if err != nil {
  61. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error decoding porter.yaml: %w", err)))
  62. return
  63. }
  64. imageInfo := request.ImageInfo
  65. registries, err := c.Repo().Registry().ListRegistriesByProjectID(cluster.ProjectID)
  66. if err != nil {
  67. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error listing registries: %w", err)))
  68. return
  69. }
  70. var releaseValues map[string]interface{}
  71. var releaseDependencies []*chart.Dependency
  72. if shouldCreate || request.OverrideRelease {
  73. releaseValues = nil
  74. releaseDependencies = nil
  75. // this is required because when the front-end sends an update request with overrideRelease=true, it is unable to
  76. // get the image info from the release. unless it is explicitly provided in the request, we avoid overwriting it
  77. // by attempting to get the image info from the release
  78. if helmRelease != nil && (imageInfo.Repository == "" || imageInfo.Tag == "") {
  79. imageInfo = attemptToGetImageInfoFromRelease(helmRelease.Config)
  80. }
  81. } else {
  82. releaseValues = helmRelease.Config
  83. releaseDependencies = helmRelease.Chart.Metadata.Dependencies
  84. }
  85. chart, values, err := parse(
  86. porterYaml,
  87. imageInfo,
  88. c.Config(),
  89. cluster.ProjectID,
  90. releaseValues,
  91. releaseDependencies,
  92. SubdomainCreateOpts{
  93. k8sAgent: k8sAgent,
  94. dnsRepo: c.Repo().DNSRecord(),
  95. powerDnsClient: c.Config().PowerDNSClient,
  96. appRootDomain: c.Config().ServerConf.AppRootDomain,
  97. stackName: stackName,
  98. })
  99. if err != nil {
  100. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error parsing porter yaml into chart and values: %w", err)))
  101. return
  102. }
  103. if shouldCreate {
  104. // create the namespace if it does not exist already
  105. _, err = k8sAgent.CreateNamespace(namespace, nil)
  106. if err != nil {
  107. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error creating namespace: %w", err)))
  108. return
  109. }
  110. conf := &helm.InstallChartConfig{
  111. Chart: chart,
  112. Name: stackName,
  113. Namespace: namespace,
  114. Values: values,
  115. Cluster: cluster,
  116. Repo: c.Repo(),
  117. Registries: registries,
  118. }
  119. // create the chart
  120. _, err = helmAgent.InstallChart(conf, c.Config().DOConf, c.Config().ServerConf.DisablePullSecretsInjection)
  121. if err != nil {
  122. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error deploying app: %s", err.Error())))
  123. _, err = helmAgent.UninstallChart(stackName)
  124. if err != nil {
  125. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error uninstalling chart: %w", err)))
  126. }
  127. return
  128. }
  129. existing, err := c.Repo().PorterApp().ReadPorterAppByName(cluster.ID, stackName)
  130. if err != nil {
  131. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  132. return
  133. } else if existing.Name != "" {
  134. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(
  135. fmt.Errorf("porter app with name %s already exists in this environment", existing.Name), http.StatusForbidden))
  136. return
  137. }
  138. app := &models.PorterApp{
  139. Name: stackName,
  140. ClusterID: cluster.ID,
  141. ProjectID: project.ID,
  142. RepoName: request.RepoName,
  143. GitRepoID: request.GitRepoID,
  144. GitBranch: request.GitBranch,
  145. BuildContext: request.BuildContext,
  146. Builder: request.Builder,
  147. Buildpacks: request.Buildpacks,
  148. Dockerfile: request.Dockerfile,
  149. ImageRepoURI: request.ImageRepoURI,
  150. PullRequestURL: request.PullRequestURL,
  151. }
  152. // create the db entry
  153. porterApp, err := c.Repo().PorterApp().UpdatePorterApp(app)
  154. if err != nil {
  155. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error writing app to DB: %s", err.Error())))
  156. return
  157. }
  158. c.WriteResult(w, r, porterApp.ToPorterAppType())
  159. } else {
  160. conf := &helm.InstallChartConfig{
  161. Chart: chart,
  162. Name: stackName,
  163. Namespace: namespace,
  164. Values: values,
  165. Cluster: cluster,
  166. Repo: c.Repo(),
  167. Registries: registries,
  168. }
  169. // update the chart
  170. _, err = helmAgent.UpgradeInstallChart(conf, c.Config().DOConf, c.Config().ServerConf.DisablePullSecretsInjection)
  171. if err != nil {
  172. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error deploying app: %s", err.Error())))
  173. return
  174. }
  175. // update the DB entry
  176. app, err := c.Repo().PorterApp().ReadPorterAppByName(cluster.ID, stackName)
  177. if err != nil {
  178. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  179. return
  180. }
  181. if request.RepoName != "" {
  182. app.RepoName = request.RepoName
  183. }
  184. if request.GitBranch != "" {
  185. app.GitBranch = request.GitBranch
  186. }
  187. if request.BuildContext != "" {
  188. app.BuildContext = request.BuildContext
  189. }
  190. if request.Builder != "" {
  191. app.Builder = request.Builder
  192. }
  193. if request.Buildpacks != "" {
  194. if request.Buildpacks == "null" {
  195. app.Buildpacks = ""
  196. } else {
  197. app.Buildpacks = request.Buildpacks
  198. }
  199. }
  200. if request.Dockerfile != "" {
  201. if request.Dockerfile == "null" {
  202. app.Dockerfile = ""
  203. } else {
  204. app.Dockerfile = request.Dockerfile
  205. }
  206. }
  207. if request.ImageRepoURI != "" {
  208. app.ImageRepoURI = request.ImageRepoURI
  209. }
  210. if request.PullRequestURL != "" {
  211. app.PullRequestURL = request.PullRequestURL
  212. }
  213. updatedPorterApp, err := c.Repo().PorterApp().UpdatePorterApp(app)
  214. if err != nil {
  215. return
  216. }
  217. c.WriteResult(w, r, updatedPorterApp.ToPorterAppType())
  218. }
  219. }