release_handler.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  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. )
  11. // Enumeration of release API error codes, represented as int64
  12. const (
  13. ErrReleaseDecode ErrorCode = iota + 600
  14. ErrReleaseValidateFields
  15. ErrReleaseReadData
  16. )
  17. // HandleListReleases retrieves a list of releases for a cluster
  18. // with various filter options
  19. func (app *App) HandleListReleases(w http.ResponseWriter, r *http.Request) {
  20. form := &forms.ListReleaseForm{
  21. ReleaseForm: &forms.ReleaseForm{
  22. Form: &helm.Form{},
  23. },
  24. ListFilter: &helm.ListFilter{},
  25. }
  26. agent, err := app.getAgentFromQueryParams(
  27. w,
  28. r,
  29. form.ReleaseForm,
  30. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  31. form.PopulateListFromQueryParams,
  32. )
  33. // errors are handled in app.getAgentFromQueryParams
  34. if err != nil {
  35. return
  36. }
  37. releases, err := agent.ListReleases(form.Namespace, form.ListFilter)
  38. if err != nil {
  39. app.handleErrorRead(err, ErrReleaseReadData, w)
  40. return
  41. }
  42. if err := json.NewEncoder(w).Encode(releases); err != nil {
  43. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  44. return
  45. }
  46. }
  47. // HandleGetRelease retrieves a single release based on a name and revision
  48. func (app *App) HandleGetRelease(w http.ResponseWriter, r *http.Request) {
  49. name := chi.URLParam(r, "name")
  50. revision, err := strconv.ParseUint(chi.URLParam(r, "revision"), 0, 64)
  51. form := &forms.GetReleaseForm{
  52. ReleaseForm: &forms.ReleaseForm{
  53. Form: &helm.Form{},
  54. },
  55. Name: name,
  56. Revision: int(revision),
  57. }
  58. agent, err := app.getAgentFromQueryParams(
  59. w,
  60. r,
  61. form.ReleaseForm,
  62. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  63. )
  64. // errors are handled in app.getAgentFromQueryParams
  65. if err != nil {
  66. return
  67. }
  68. release, err := agent.GetRelease(form.Name, form.Revision)
  69. if err != nil {
  70. app.handleErrorRead(err, ErrReleaseReadData, w)
  71. return
  72. }
  73. if err := json.NewEncoder(w).Encode(release); err != nil {
  74. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  75. return
  76. }
  77. }
  78. // HandleListReleaseHistory retrieves a history of releases based on a release name
  79. func (app *App) HandleListReleaseHistory(w http.ResponseWriter, r *http.Request) {
  80. name := chi.URLParam(r, "name")
  81. form := &forms.ListReleaseHistoryForm{
  82. ReleaseForm: &forms.ReleaseForm{
  83. Form: &helm.Form{},
  84. },
  85. Name: name,
  86. }
  87. agent, err := app.getAgentFromQueryParams(
  88. w,
  89. r,
  90. form.ReleaseForm,
  91. form.ReleaseForm.PopulateHelmOptionsFromQueryParams,
  92. )
  93. // errors are handled in app.getAgentFromQueryParams
  94. if err != nil {
  95. return
  96. }
  97. release, err := agent.GetReleaseHistory(form.Name)
  98. if err != nil {
  99. app.handleErrorFormValidation(err, ErrReleaseValidateFields, w)
  100. return
  101. }
  102. if err := json.NewEncoder(w).Encode(release); err != nil {
  103. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  104. return
  105. }
  106. }
  107. // HandleUpgradeRelease upgrades a release with new values.yaml
  108. func (app *App) HandleUpgradeRelease(w http.ResponseWriter, r *http.Request) {
  109. name := chi.URLParam(r, "name")
  110. form := &forms.UpgradeReleaseForm{
  111. ReleaseForm: &forms.ReleaseForm{
  112. Form: &helm.Form{},
  113. },
  114. Name: name,
  115. }
  116. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  117. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  118. return
  119. }
  120. agent, err := app.getAgentFromReleaseForm(
  121. w,
  122. r,
  123. form.ReleaseForm,
  124. )
  125. // errors are handled in app.getAgentFromBodyParams
  126. if err != nil {
  127. return
  128. }
  129. _, err = agent.UpgradeRelease(form.Name, form.Values)
  130. if err != nil {
  131. app.handleErrorInternal(err, w)
  132. return
  133. }
  134. w.WriteHeader(http.StatusOK)
  135. }
  136. // HandleRollbackRelease rolls a release back to a specified revision
  137. func (app *App) HandleRollbackRelease(w http.ResponseWriter, r *http.Request) {
  138. name := chi.URLParam(r, "name")
  139. form := &forms.RollbackReleaseForm{
  140. ReleaseForm: &forms.ReleaseForm{
  141. Form: &helm.Form{},
  142. },
  143. Name: name,
  144. }
  145. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  146. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  147. return
  148. }
  149. agent, err := app.getAgentFromReleaseForm(
  150. w,
  151. r,
  152. form.ReleaseForm,
  153. )
  154. // errors are handled in app.getAgentFromBodyParams
  155. if err != nil {
  156. return
  157. }
  158. err = agent.RollbackRelease(form.Name, form.Revision)
  159. if err != nil {
  160. app.handleErrorInternal(err, w)
  161. return
  162. }
  163. w.WriteHeader(http.StatusOK)
  164. }
  165. // ------------------------ Release handler helper functions ------------------------ //
  166. // getAgentFromQueryParams uses the query params to populate a form, and then
  167. // passes that form to the underlying app.getAgentFromReleaseForm to create a new
  168. // Helm agent.
  169. func (app *App) getAgentFromQueryParams(
  170. w http.ResponseWriter,
  171. r *http.Request,
  172. form *forms.ReleaseForm,
  173. // populate uses the query params to populate a form
  174. populate ...func(vals url.Values),
  175. ) (*helm.Agent, error) {
  176. vals, err := url.ParseQuery(r.URL.RawQuery)
  177. if err != nil {
  178. app.handleErrorFormDecoding(err, ErrReleaseDecode, w)
  179. return nil, err
  180. }
  181. for _, f := range populate {
  182. f(vals)
  183. }
  184. return app.getAgentFromReleaseForm(w, r, form)
  185. }
  186. // getAgentFromReleaseForm uses a non-validated form to construct a new Helm agent based on
  187. // the userID found in the session and the options required by the Helm agent.
  188. func (app *App) getAgentFromReleaseForm(
  189. w http.ResponseWriter,
  190. r *http.Request,
  191. form *forms.ReleaseForm,
  192. ) (*helm.Agent, error) {
  193. // read the session in order to generate the Helm agent
  194. session, err := app.store.Get(r, app.cookieName)
  195. // since we have already authenticated the user, throw a data read error if the session
  196. // cannot be found
  197. if err != nil {
  198. app.handleErrorDataRead(err, w)
  199. return nil, err
  200. }
  201. if userID, ok := session.Values["user_id"].(uint); ok {
  202. form.PopulateHelmOptionsFromUserID(userID, app.repo.User)
  203. }
  204. // validate the form
  205. if err := app.validator.Struct(form); err != nil {
  206. app.handleErrorFormValidation(err, ErrReleaseValidateFields, w)
  207. return nil, err
  208. }
  209. // create a new agent
  210. var agent *helm.Agent
  211. if app.testing {
  212. agent = app.TestAgents.HelmAgent
  213. } else {
  214. agent, err = helm.GetAgentOutOfClusterConfig(form.Form, app.logger)
  215. }
  216. return agent, err
  217. }