release_handler.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. package api
  2. import (
  3. "encoding/json"
  4. "net/http"
  5. "net/url"
  6. "strconv"
  7. "github.com/go-chi/chi"
  8. "github.com/porter-dev/porter/internal/forms"
  9. "github.com/porter-dev/porter/internal/helm"
  10. "github.com/porter-dev/porter/internal/helm/grapher"
  11. )
  12. // Enumeration of release API error codes, represented as int64
  13. const (
  14. ErrReleaseDecode ErrorCode = iota + 600
  15. ErrReleaseValidateFields
  16. ErrReleaseReadData
  17. ErrReleaseDeploy
  18. )
  19. // HandleListReleases retrieves a list of releases for a cluster
  20. // with various filter options
  21. func (app *App) HandleListReleases(w http.ResponseWriter, r *http.Request) {
  22. form := &forms.ListReleaseForm{
  23. ReleaseForm: &forms.ReleaseForm{
  24. Form: &helm.Form{},
  25. },
  26. ListFilter: &helm.ListFilter{},
  27. }
  28. agent, err := app.getAgentFromQueryParams(
  29. w,
  30. r,
  31. form.ReleaseForm,
  32. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  33. form.PopulateListFromQueryParams,
  34. )
  35. // errors are handled in app.getAgentFromQueryParams
  36. if err != nil {
  37. return
  38. }
  39. releases, err := agent.ListReleases(form.Namespace, form.ListFilter)
  40. if err != nil {
  41. app.handleErrorRead(err, ErrReleaseReadData, w)
  42. return
  43. }
  44. if err := json.NewEncoder(w).Encode(releases); err != nil {
  45. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  46. return
  47. }
  48. }
  49. // HandleGetRelease retrieves a single release based on a name and revision
  50. func (app *App) HandleGetRelease(w http.ResponseWriter, r *http.Request) {
  51. name := chi.URLParam(r, "name")
  52. revision, err := strconv.ParseUint(chi.URLParam(r, "revision"), 0, 64)
  53. form := &forms.GetReleaseForm{
  54. ReleaseForm: &forms.ReleaseForm{
  55. Form: &helm.Form{},
  56. },
  57. Name: name,
  58. Revision: int(revision),
  59. }
  60. agent, err := app.getAgentFromQueryParams(
  61. w,
  62. r,
  63. form.ReleaseForm,
  64. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  65. )
  66. // errors are handled in app.getAgentFromQueryParams
  67. if err != nil {
  68. return
  69. }
  70. release, err := agent.GetRelease(form.Name, form.Revision)
  71. if err != nil {
  72. app.sendExternalError(err, http.StatusNotFound, HTTPError{
  73. Code: ErrReleaseReadData,
  74. Errors: []string{"release not found"},
  75. }, w)
  76. return
  77. }
  78. if err := json.NewEncoder(w).Encode(release); err != nil {
  79. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  80. return
  81. }
  82. }
  83. // HandleGetReleaseComponents retrieves a single release based on a name and revision
  84. func (app *App) HandleGetReleaseComponents(w http.ResponseWriter, r *http.Request) {
  85. name := chi.URLParam(r, "name")
  86. revision, err := strconv.ParseUint(chi.URLParam(r, "revision"), 0, 64)
  87. form := &forms.GetReleaseForm{
  88. ReleaseForm: &forms.ReleaseForm{
  89. Form: &helm.Form{},
  90. },
  91. Name: name,
  92. Revision: int(revision),
  93. }
  94. agent, err := app.getAgentFromQueryParams(
  95. w,
  96. r,
  97. form.ReleaseForm,
  98. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  99. )
  100. // errors are handled in app.getAgentFromQueryParams
  101. if err != nil {
  102. return
  103. }
  104. release, err := agent.GetRelease(form.Name, form.Revision)
  105. if err != nil {
  106. app.sendExternalError(err, http.StatusNotFound, HTTPError{
  107. Code: ErrReleaseReadData,
  108. Errors: []string{"release not found"},
  109. }, w)
  110. return
  111. }
  112. yamlArr := grapher.ImportMultiDocYAML([]byte(release.Manifest))
  113. objects := grapher.ParseObjs(yamlArr)
  114. parsed := grapher.ParsedObjs{
  115. Objects: objects,
  116. }
  117. parsed.GetControlRel()
  118. parsed.GetLabelRel()
  119. if err := json.NewEncoder(w).Encode(parsed.Objects); err != nil {
  120. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  121. return
  122. }
  123. }
  124. // HandleListReleaseHistory retrieves a history of releases based on a release name
  125. func (app *App) HandleListReleaseHistory(w http.ResponseWriter, r *http.Request) {
  126. name := chi.URLParam(r, "name")
  127. form := &forms.ListReleaseHistoryForm{
  128. ReleaseForm: &forms.ReleaseForm{
  129. Form: &helm.Form{},
  130. },
  131. Name: name,
  132. }
  133. agent, err := app.getAgentFromQueryParams(
  134. w,
  135. r,
  136. form.ReleaseForm,
  137. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  138. )
  139. // errors are handled in app.getAgentFromQueryParams
  140. if err != nil {
  141. return
  142. }
  143. release, err := agent.GetReleaseHistory(form.Name)
  144. if err != nil {
  145. app.sendExternalError(err, http.StatusNotFound, HTTPError{
  146. Code: ErrReleaseReadData,
  147. Errors: []string{"release not found"},
  148. }, w)
  149. return
  150. }
  151. if err := json.NewEncoder(w).Encode(release); err != nil {
  152. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  153. return
  154. }
  155. }
  156. // HandleUpgradeRelease upgrades a release with new values.yaml
  157. func (app *App) HandleUpgradeRelease(w http.ResponseWriter, r *http.Request) {
  158. name := chi.URLParam(r, "name")
  159. form := &forms.UpgradeReleaseForm{
  160. ReleaseForm: &forms.ReleaseForm{
  161. Form: &helm.Form{},
  162. },
  163. Name: name,
  164. }
  165. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  166. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  167. return
  168. }
  169. agent, err := app.getAgentFromReleaseForm(
  170. w,
  171. r,
  172. form.ReleaseForm,
  173. )
  174. // errors are handled in app.getAgentFromBodyParams
  175. if err != nil {
  176. return
  177. }
  178. _, err = agent.UpgradeRelease(form.Name, form.Values)
  179. if err != nil {
  180. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  181. Code: ErrReleaseDeploy,
  182. Errors: []string{"error upgrading release " + err.Error()},
  183. }, w)
  184. return
  185. }
  186. w.WriteHeader(http.StatusOK)
  187. }
  188. // HandleRollbackRelease rolls a release back to a specified revision
  189. func (app *App) HandleRollbackRelease(w http.ResponseWriter, r *http.Request) {
  190. name := chi.URLParam(r, "name")
  191. form := &forms.RollbackReleaseForm{
  192. ReleaseForm: &forms.ReleaseForm{
  193. Form: &helm.Form{},
  194. },
  195. Name: name,
  196. }
  197. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  198. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  199. return
  200. }
  201. agent, err := app.getAgentFromReleaseForm(
  202. w,
  203. r,
  204. form.ReleaseForm,
  205. )
  206. // errors are handled in app.getAgentFromBodyParams
  207. if err != nil {
  208. return
  209. }
  210. err = agent.RollbackRelease(form.Name, form.Revision)
  211. if err != nil {
  212. app.sendExternalError(err, http.StatusInternalServerError, HTTPError{
  213. Code: ErrReleaseDeploy,
  214. Errors: []string{"error rolling back release " + err.Error()},
  215. }, w)
  216. return
  217. }
  218. w.WriteHeader(http.StatusOK)
  219. }
  220. // ------------------------ Release handler helper functions ------------------------ //
  221. // getAgentFromQueryParams uses the query params to populate a form, and then
  222. // passes that form to the underlying app.getAgentFromReleaseForm to create a new
  223. // Helm agent.
  224. func (app *App) getAgentFromQueryParams(
  225. w http.ResponseWriter,
  226. r *http.Request,
  227. form *forms.ReleaseForm,
  228. // populate uses the query params to populate a form
  229. populate ...func(vals url.Values),
  230. ) (*helm.Agent, error) {
  231. vals, err := url.ParseQuery(r.URL.RawQuery)
  232. if err != nil {
  233. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  234. return nil, err
  235. }
  236. for _, f := range populate {
  237. f(vals)
  238. }
  239. return app.getAgentFromReleaseForm(w, r, form)
  240. }
  241. // getAgentFromReleaseForm uses a non-validated form to construct a new Helm agent based on
  242. // the userID found in the session and the options required by the Helm agent.
  243. func (app *App) getAgentFromReleaseForm(
  244. w http.ResponseWriter,
  245. r *http.Request,
  246. form *forms.ReleaseForm,
  247. ) (*helm.Agent, error) {
  248. // read the session in order to generate the Helm agent
  249. session, err := app.store.Get(r, app.cookieName)
  250. // since we have already authenticated the user, throw a data read error if the session
  251. // cannot be found
  252. if err != nil {
  253. app.handleErrorDataRead(err, w)
  254. return nil, err
  255. }
  256. if userID, ok := session.Values["user_id"].(uint); ok {
  257. form.PopulateHelmOptionsFromUserID(userID, app.repo.User)
  258. }
  259. // validate the form
  260. if err := app.validator.Struct(form); err != nil {
  261. app.handleErrorFormValidation(err, ErrReleaseValidateFields, w)
  262. return nil, err
  263. }
  264. // create a new agent
  265. var agent *helm.Agent
  266. if app.testing {
  267. agent = app.TestAgents.HelmAgent
  268. } else {
  269. agent, err = helm.GetAgentOutOfClusterConfig(form.Form, app.logger)
  270. }
  271. return agent, err
  272. }