accept.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. package invite
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. "strconv"
  7. "github.com/go-chi/chi"
  8. "github.com/porter-dev/porter/api/server/handlers"
  9. "github.com/porter-dev/porter/api/server/shared"
  10. "github.com/porter-dev/porter/api/server/shared/apierrors"
  11. "github.com/porter-dev/porter/api/server/shared/config"
  12. "github.com/porter-dev/porter/api/types"
  13. "github.com/porter-dev/porter/internal/models"
  14. )
  15. type InviteAcceptHandler struct {
  16. handlers.PorterHandlerReader
  17. }
  18. func NewInviteAcceptHandler(
  19. config *config.Config,
  20. decoderValidator shared.RequestDecoderValidator,
  21. ) *InviteAcceptHandler {
  22. return &InviteAcceptHandler{
  23. PorterHandlerReader: handlers.NewDefaultPorterHandler(config, decoderValidator, nil),
  24. }
  25. }
  26. func (c *InviteAcceptHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  27. request := &types.AcceptInviteRequest{}
  28. if ok := c.DecodeAndValidate(w, r, request); !ok {
  29. return
  30. }
  31. session, err := c.Config().Store.Get(r, c.Config().ServerConf.CookieName)
  32. if err != nil {
  33. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  34. }
  35. userID, _ := session.Values["user_id"].(uint)
  36. user, err := c.Repo().User().ReadUser(userID)
  37. if err != nil {
  38. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  39. return
  40. }
  41. projectID, err := strconv.ParseUint(chi.URLParam(r, "project_id"), 0, 64)
  42. if err != nil || projectID == 0 {
  43. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  44. return
  45. }
  46. invite, err := c.Repo().Invite().ReadInviteByToken(request.Token)
  47. if err != nil || invite.ProjectID != uint(projectID) {
  48. vals := url.Values{}
  49. vals.Add("error", "Invalid invite token")
  50. http.Redirect(w, r, fmt.Sprintf("/dashboard?%s", vals.Encode()), 302)
  51. return
  52. }
  53. // check that the invite has not expired and has not been accepted
  54. if invite.IsExpired() || invite.IsAccepted() {
  55. vals := url.Values{}
  56. vals.Add("error", "Invite has expired")
  57. http.Redirect(w, r, fmt.Sprintf("/dashboard?%s", vals.Encode()), 302)
  58. return
  59. }
  60. // check that the invite email matches the user's email
  61. if user.Email != invite.Email {
  62. vals := url.Values{}
  63. vals.Add("error", "Wrong email for invite")
  64. http.Redirect(w, r, fmt.Sprintf("/dashboard?%s", vals.Encode()), 302)
  65. return
  66. }
  67. kind := invite.Kind
  68. if kind == "" {
  69. kind = models.RoleDeveloper
  70. }
  71. project, err := c.Repo().Project().ReadProject(uint(projectID))
  72. if err != nil {
  73. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  74. return
  75. }
  76. if _, err = c.Repo().Project().CreateProjectRole(project, &models.Role{
  77. Role: types.Role{
  78. UserID: userID,
  79. ProjectID: project.ID,
  80. Kind: types.RoleKind(kind),
  81. },
  82. }); err != nil {
  83. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  84. return
  85. }
  86. // update the invite
  87. invite.UserID = userID
  88. if _, err = c.Repo().Invite().UpdateInvite(invite); err != nil {
  89. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  90. return
  91. }
  92. http.Redirect(w, r, "/dashboard", 302)
  93. }