user_handler.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. package api
  2. import (
  3. "encoding/json"
  4. "net/http"
  5. "strconv"
  6. "github.com/go-chi/chi"
  7. "github.com/porter-dev/porter/internal/forms"
  8. "github.com/porter-dev/porter/internal/models"
  9. "github.com/porter-dev/porter/internal/queries"
  10. "gorm.io/gorm"
  11. )
  12. // Enumeration of user API error codes, represented as int64
  13. const (
  14. ErrUserDecode ErrorCode = iota
  15. ErrUserValidateFields
  16. ErrUserDataWrite
  17. ErrUserDataRead
  18. )
  19. // HandleCreateUser validates a user form entry, converts the user to a gorm
  20. // model, and saves the user to the database
  21. func (app *App) HandleCreateUser(w http.ResponseWriter, r *http.Request) {
  22. form := &forms.CreateUserForm{}
  23. user, err := app.writeUser(form, queries.CreateUser, w, r)
  24. if err == nil {
  25. app.logger.Info().Msgf("New user created: %d", user.ID)
  26. w.WriteHeader(http.StatusCreated)
  27. }
  28. }
  29. // HandleReadUser returns an externalized User (models.UserExternal)
  30. // based on an ID
  31. func (app *App) HandleReadUser(w http.ResponseWriter, r *http.Request) {
  32. id, err := strconv.ParseUint(chi.URLParam(r, "id"), 0, 64)
  33. if err != nil || id == 0 {
  34. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  35. return
  36. }
  37. user, err := queries.ReadUser(app.db, uint(id))
  38. if err != nil {
  39. app.handleErrorRead(err, ErrUserDataRead, w)
  40. return
  41. }
  42. extUser := user.Externalize()
  43. if err := json.NewEncoder(w).Encode(extUser); err != nil {
  44. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  45. return
  46. }
  47. w.WriteHeader(http.StatusOK)
  48. }
  49. // HandleUpdateUser validates an update user form entry, updates the user
  50. // in the database, and writes status accepted
  51. func (app *App) HandleUpdateUser(w http.ResponseWriter, r *http.Request) {
  52. id, err := strconv.ParseUint(chi.URLParam(r, "id"), 0, 64)
  53. if err != nil || id == 0 {
  54. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  55. return
  56. }
  57. form := &forms.UpdateUserForm{
  58. ID: uint(id),
  59. }
  60. user, err := app.writeUser(form, queries.UpdateUser, w, r)
  61. if err == nil {
  62. app.logger.Info().Msgf("User updated: %d", user.ID)
  63. w.WriteHeader(http.StatusAccepted)
  64. }
  65. }
  66. // HandleDeleteUser is majestic
  67. func (app *App) HandleDeleteUser(w http.ResponseWriter, r *http.Request) {
  68. id, err := strconv.ParseUint(chi.URLParam(r, "id"), 0, 64)
  69. if err != nil || id == 0 {
  70. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  71. return
  72. }
  73. // TODO -- HASH AND VERIFY PASSWORD BEFORE USER DELETION
  74. form := &forms.DeleteUserForm{
  75. ID: uint(id),
  76. Password: "testing",
  77. }
  78. user, err := app.writeUser(form, queries.DeleteUser, w, r)
  79. if err == nil {
  80. app.logger.Info().Msgf("User deleted: %d", user.ID)
  81. w.WriteHeader(http.StatusAccepted)
  82. }
  83. }
  84. // ------------------------ User handler helper functions ------------------------ //
  85. // writeUser will take a POST or PUT request to the /api/users endpoint and decode
  86. // the request into a forms.WriteUserForm model, convert it to a models.User, and
  87. // write to the database.
  88. func (app *App) writeUser(
  89. form forms.WriteUserForm,
  90. dbWrite func(db *gorm.DB, user *models.User) (*models.User, error),
  91. w http.ResponseWriter,
  92. r *http.Request,
  93. ) (*models.User, error) {
  94. // decode from JSON to form value
  95. if err := json.NewDecoder(r.Body).Decode(form); err != nil {
  96. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  97. return nil, err
  98. }
  99. // validate the form
  100. if err := app.validator.Struct(form); err != nil {
  101. app.handleErrorFormValidation(err, ErrUserValidateFields, w)
  102. return nil, err
  103. }
  104. // convert the form to a user model -- WriteUserForm must implement ToUser
  105. userModel, err := form.ToUser()
  106. if err != nil {
  107. app.handleErrorFormDecoding(err, ErrUserDecode, w)
  108. return nil, err
  109. }
  110. // handle write to the database
  111. user, err := dbWrite(app.db, userModel)
  112. if err != nil {
  113. app.handleErrorDataWrite(err, ErrUserDataWrite, w)
  114. return nil, err
  115. }
  116. return user, nil
  117. }