cli_login.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. package user
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. "time"
  7. "github.com/porter-dev/porter/api/server/handlers"
  8. "github.com/porter-dev/porter/api/server/shared"
  9. "github.com/porter-dev/porter/api/server/shared/apierrors"
  10. "github.com/porter-dev/porter/api/server/shared/config"
  11. "github.com/porter-dev/porter/api/types"
  12. "github.com/porter-dev/porter/internal/auth/token"
  13. "github.com/porter-dev/porter/internal/encryption"
  14. "github.com/porter-dev/porter/internal/models"
  15. )
  16. type CLILoginHandler struct {
  17. handlers.PorterHandlerReader
  18. }
  19. func NewCLILoginHandler(
  20. config *config.Config,
  21. decoderValidator shared.RequestDecoderValidator,
  22. writer shared.ResultWriter,
  23. ) *CLILoginHandler {
  24. return &CLILoginHandler{
  25. PorterHandlerReader: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  26. }
  27. }
  28. func (c *CLILoginHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  29. request := &types.CLILoginUserRequest{}
  30. ok := c.DecodeAndValidate(w, r, request)
  31. if !ok {
  32. return
  33. }
  34. user, _ := r.Context().Value(types.UserScope).(*models.User)
  35. if err := checkUserRestrictions(c.Config().ServerConf, user.Email); err != nil {
  36. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusBadRequest))
  37. return
  38. }
  39. // generate the token
  40. jwt, err := token.GetTokenForUser(user.ID)
  41. if err != nil {
  42. err = fmt.Errorf("CLI token creation failed: %s", err.Error())
  43. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  44. return
  45. }
  46. encoded, err := jwt.EncodeToken(c.Config().TokenConf)
  47. if err != nil {
  48. err = fmt.Errorf("CLI token encoding failed: %s", err.Error())
  49. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  50. return
  51. }
  52. // generate 64 characters long authorization code
  53. code, err := encryption.GenerateRandomBytes(32)
  54. if err != nil {
  55. err = fmt.Errorf("CLI random code generation failed: %s", err.Error())
  56. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  57. return
  58. }
  59. expiry := time.Now().Add(30 * time.Second)
  60. // create auth code object and send back authorization code
  61. authCode := &models.AuthCode{
  62. Token: encoded,
  63. AuthorizationCode: code,
  64. Expiry: &expiry,
  65. }
  66. authCode, err = c.Repo().AuthCode().CreateAuthCode(authCode)
  67. if err != nil {
  68. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  69. return
  70. }
  71. http.Redirect(w, r, fmt.Sprintf("%s/?code=%s", request.Redirect, url.QueryEscape(authCode.AuthorizationCode)), 302)
  72. }
  73. type CLILoginExchangeHandler struct {
  74. handlers.PorterHandlerReadWriter
  75. }
  76. func NewCLILoginExchangeHandler(
  77. config *config.Config,
  78. decoderValidator shared.RequestDecoderValidator,
  79. writer shared.ResultWriter,
  80. ) *CLILoginExchangeHandler {
  81. return &CLILoginExchangeHandler{
  82. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  83. }
  84. }
  85. func (c *CLILoginExchangeHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  86. request := &types.CLILoginExchangeRequest{}
  87. ok := c.DecodeAndValidate(w, r, request)
  88. if !ok {
  89. return
  90. }
  91. // look up the auth code and exchange it for a token
  92. authCode, err := c.Repo().AuthCode().ReadAuthCode(request.AuthorizationCode)
  93. if err != nil || authCode.IsExpired() {
  94. http.Error(w, http.StatusText(http.StatusForbidden), http.StatusForbidden)
  95. return
  96. }
  97. res := &types.CLILoginExchangeResponse{
  98. Token: authCode.Token,
  99. }
  100. c.WriteResult(w, r, res)
  101. }