deploy_handler.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. package api
  2. import (
  3. "encoding/json"
  4. "net/http"
  5. "net/url"
  6. "strconv"
  7. "strings"
  8. "github.com/go-chi/chi"
  9. "github.com/porter-dev/porter/internal/forms"
  10. "github.com/porter-dev/porter/internal/helm"
  11. "github.com/porter-dev/porter/internal/helm/loader"
  12. "github.com/porter-dev/porter/internal/integrations/ci/actions"
  13. "github.com/porter-dev/porter/internal/models"
  14. "github.com/porter-dev/porter/internal/repository"
  15. "gopkg.in/yaml.v2"
  16. )
  17. // HandleDeployTemplate triggers a chart deployment from a template
  18. func (app *App) HandleDeployTemplate(w http.ResponseWriter, r *http.Request) {
  19. projID, err := strconv.ParseUint(chi.URLParam(r, "project_id"), 0, 64)
  20. if err != nil || projID == 0 {
  21. app.handleErrorFormDecoding(err, ErrProjectDecode, w)
  22. return
  23. }
  24. name := chi.URLParam(r, "name")
  25. version := chi.URLParam(r, "version")
  26. // if version passed as latest, pass empty string to loader to get latest
  27. if version == "latest" {
  28. version = ""
  29. }
  30. getChartForm := &forms.ChartForm{
  31. Name: name,
  32. Version: version,
  33. RepoURL: app.ServerConf.DefaultApplicationHelmRepoURL,
  34. }
  35. // if a repo_url is passed as query param, it will be populated
  36. vals, err := url.ParseQuery(r.URL.RawQuery)
  37. if err != nil {
  38. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  39. return
  40. }
  41. getChartForm.PopulateRepoURLFromQueryParams(vals)
  42. chart, err := loader.LoadChartPublic(getChartForm.RepoURL, getChartForm.Name, getChartForm.Version)
  43. if err != nil {
  44. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  45. return
  46. }
  47. form := &forms.InstallChartTemplateForm{
  48. ReleaseForm: &forms.ReleaseForm{
  49. Form: &helm.Form{
  50. Repo: app.Repo,
  51. DigitalOceanOAuth: app.DOConf,
  52. },
  53. },
  54. ChartTemplateForm: &forms.ChartTemplateForm{},
  55. }
  56. form.ReleaseForm.PopulateHelmOptionsFromQueryParams(
  57. vals,
  58. app.Repo.Cluster,
  59. )
  60. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  61. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  62. return
  63. }
  64. agent, err := app.getAgentFromReleaseForm(
  65. w,
  66. r,
  67. form.ReleaseForm,
  68. )
  69. if err != nil {
  70. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  71. return
  72. }
  73. registries, err := app.Repo.Registry.ListRegistriesByProjectID(uint(projID))
  74. if err != nil {
  75. app.handleErrorDataRead(err, w)
  76. return
  77. }
  78. conf := &helm.InstallChartConfig{
  79. Chart: chart,
  80. Name: form.ChartTemplateForm.Name,
  81. Namespace: form.ReleaseForm.Form.Namespace,
  82. Values: form.ChartTemplateForm.FormValues,
  83. Cluster: form.ReleaseForm.Cluster,
  84. Repo: *app.Repo,
  85. Registries: registries,
  86. }
  87. _, err = agent.InstallChart(conf, app.DOConf)
  88. if err != nil {
  89. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  90. Code: ErrReleaseDeploy,
  91. Errors: []string{"error installing a new chart: " + err.Error()},
  92. }, w)
  93. return
  94. }
  95. token, err := repository.GenerateRandomBytes(16)
  96. if err != nil {
  97. app.handleErrorInternal(err, w)
  98. return
  99. }
  100. // create release with webhook token in db
  101. release := &models.Release{
  102. ClusterID: form.ReleaseForm.Form.Cluster.ID,
  103. ProjectID: form.ReleaseForm.Form.Cluster.ProjectID,
  104. Namespace: form.ReleaseForm.Form.Namespace,
  105. Name: form.ChartTemplateForm.Name,
  106. WebhookToken: token,
  107. }
  108. _, err = app.Repo.Release.CreateRelease(release)
  109. if err != nil {
  110. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  111. Code: ErrReleaseDeploy,
  112. Errors: []string{"error creating a webhook: " + err.Error()},
  113. }, w)
  114. }
  115. // if github action config is linked, call the github action config handler
  116. if form.GithubActionConfig != nil {
  117. gaForm := &forms.CreateGitAction{
  118. ReleaseID: release.ID,
  119. GitRepo: form.GithubActionConfig.GitRepo,
  120. ImageRepoURI: form.GithubActionConfig.ImageRepoURI,
  121. DockerfilePath: form.GithubActionConfig.DockerfilePath,
  122. GitRepoID: form.GithubActionConfig.GitRepoID,
  123. BuildEnv: form.GithubActionConfig.BuildEnv,
  124. RegistryID: form.GithubActionConfig.RegistryID,
  125. }
  126. // validate the form
  127. if err := app.validator.Struct(form); err != nil {
  128. app.handleErrorFormValidation(err, ErrProjectValidateFields, w)
  129. return
  130. }
  131. app.createGitActionFromForm(projID, release, name, gaForm, w, r)
  132. }
  133. w.WriteHeader(http.StatusOK)
  134. }
  135. // HandleUninstallTemplate triggers a chart deployment from a template
  136. func (app *App) HandleUninstallTemplate(w http.ResponseWriter, r *http.Request) {
  137. name := chi.URLParam(r, "name")
  138. vals, err := url.ParseQuery(r.URL.RawQuery)
  139. if err != nil {
  140. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  141. return
  142. }
  143. form := &forms.GetReleaseForm{
  144. ReleaseForm: &forms.ReleaseForm{
  145. Form: &helm.Form{
  146. Repo: app.Repo,
  147. },
  148. },
  149. Name: name,
  150. }
  151. agent, err := app.getAgentFromQueryParams(
  152. w,
  153. r,
  154. form.ReleaseForm,
  155. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  156. )
  157. // errors are handled in app.getAgentFromQueryParams
  158. if err != nil {
  159. return
  160. }
  161. resp, err := agent.UninstallChart(name)
  162. if err != nil {
  163. return
  164. }
  165. // update the github actions env if the release exists and is built from source
  166. if cName := resp.Release.Chart.Metadata.Name; cName == "job" || cName == "web" || cName == "worker" {
  167. clusterID, err := strconv.ParseUint(vals["cluster_id"][0], 10, 64)
  168. if err != nil {
  169. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  170. Code: ErrReleaseReadData,
  171. Errors: []string{"release not found"},
  172. }, w)
  173. }
  174. release, err := app.Repo.Release.ReadRelease(uint(clusterID), name, resp.Release.Namespace)
  175. if release != nil {
  176. gitAction := release.GitActionConfig
  177. if gitAction.ID != 0 {
  178. // parse env into build env
  179. cEnv := &ContainerEnvConfig{}
  180. rawValues, err := yaml.Marshal(resp.Release.Config)
  181. if err != nil {
  182. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  183. Code: ErrReleaseReadData,
  184. Errors: []string{"could not get values of previous revision"},
  185. }, w)
  186. }
  187. yaml.Unmarshal(rawValues, cEnv)
  188. gr, err := app.Repo.GitRepo.ReadGitRepo(gitAction.GitRepoID)
  189. if err != nil {
  190. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  191. Code: ErrReleaseReadData,
  192. Errors: []string{"github repo integration not found"},
  193. }, w)
  194. }
  195. repoSplit := strings.Split(gitAction.GitRepo, "/")
  196. projID, err := strconv.ParseUint(chi.URLParam(r, "project_id"), 0, 64)
  197. if err != nil || projID == 0 {
  198. app.handleErrorFormDecoding(err, ErrProjectDecode, w)
  199. return
  200. }
  201. gaRunner := &actions.GithubActions{
  202. GitIntegration: gr,
  203. GitRepoName: repoSplit[1],
  204. GitRepoOwner: repoSplit[0],
  205. Repo: *app.Repo,
  206. GithubConf: app.GithubProjectConf,
  207. WebhookToken: release.WebhookToken,
  208. ProjectID: uint(projID),
  209. ReleaseName: name,
  210. GitBranch: gitAction.GitBranch,
  211. DockerFilePath: gitAction.DockerfilePath,
  212. FolderPath: gitAction.FolderPath,
  213. ImageRepoURL: gitAction.ImageRepoURI,
  214. BuildEnv: cEnv.Container.Env.Normal,
  215. }
  216. err = gaRunner.Cleanup()
  217. if err != nil {
  218. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  219. Code: ErrReleaseReadData,
  220. Errors: []string{"could not remove github action"},
  221. }, w)
  222. }
  223. }
  224. }
  225. }
  226. w.WriteHeader(http.StatusOK)
  227. return
  228. }