neon.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. package oauth_callback
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. "github.com/porter-dev/porter/api/server/handlers"
  7. "github.com/porter-dev/porter/api/server/shared"
  8. "github.com/porter-dev/porter/api/server/shared/apierrors"
  9. "github.com/porter-dev/porter/api/server/shared/config"
  10. "github.com/porter-dev/porter/internal/models/integrations"
  11. "github.com/porter-dev/porter/internal/telemetry"
  12. )
  13. // OAuthCallbackNeonHandler is the handler responding to the neon oauth callback
  14. type OAuthCallbackNeonHandler struct {
  15. handlers.PorterHandlerReadWriter
  16. }
  17. // NewOAuthCallbackNeonHandler generates a new OAuthCallbackNeonHandler
  18. func NewOAuthCallbackNeonHandler(
  19. config *config.Config,
  20. decoderValidator shared.RequestDecoderValidator,
  21. writer shared.ResultWriter,
  22. ) *OAuthCallbackNeonHandler {
  23. return &OAuthCallbackNeonHandler{
  24. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  25. }
  26. }
  27. // ServeHTTP gets the neon oauth token from the callback code then creates a new neon integration
  28. func (p *OAuthCallbackNeonHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  29. ctx, span := telemetry.NewSpan(r.Context(), "serve-oauth-callback-neon")
  30. defer span.End()
  31. r = r.Clone(ctx)
  32. session, err := p.Config().Store.Get(r, p.Config().ServerConf.CookieName)
  33. if err != nil {
  34. err = telemetry.Error(ctx, span, err, "session could not be retrieved")
  35. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  36. return
  37. }
  38. if _, ok := session.Values["state"]; !ok {
  39. err = telemetry.Error(ctx, span, nil, "state not found in session")
  40. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  41. return
  42. }
  43. if r.URL.Query().Get("state") != session.Values["state"] {
  44. err = telemetry.Error(ctx, span, nil, "state does not match")
  45. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  46. return
  47. }
  48. projID, ok := session.Values["project_id"].(uint)
  49. if !ok {
  50. err = telemetry.Error(ctx, span, nil, "project id not found in session")
  51. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  52. return
  53. }
  54. telemetry.WithAttributes(span,
  55. telemetry.AttributeKV{Key: "project-id", Value: projID},
  56. )
  57. if projID == 0 {
  58. err = telemetry.Error(ctx, span, nil, "project id not found in session")
  59. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  60. return
  61. }
  62. code := r.URL.Query().Get("code")
  63. if code == "" {
  64. err = telemetry.Error(ctx, span, nil, "code not found in query params")
  65. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusForbidden))
  66. return
  67. }
  68. token, err := p.Config().NeonConf.Exchange(ctx, code)
  69. if err != nil {
  70. err = telemetry.Error(ctx, span, err, "exchange failed")
  71. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusForbidden))
  72. return
  73. }
  74. if !token.Valid() {
  75. err = telemetry.Error(ctx, span, nil, "invalid token")
  76. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusForbidden))
  77. return
  78. }
  79. oauthInt := integrations.NeonIntegration{
  80. SharedOAuthModel: integrations.SharedOAuthModel{
  81. AccessToken: []byte(token.AccessToken),
  82. RefreshToken: []byte(token.RefreshToken),
  83. Expiry: token.Expiry,
  84. },
  85. ProjectID: projID,
  86. }
  87. _, err = p.Repo().NeonIntegration().Insert(ctx, oauthInt)
  88. if err != nil {
  89. err = telemetry.Error(ctx, span, err, "error creating oauth integration")
  90. p.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  91. return
  92. }
  93. redirect := "/dashboard"
  94. if redirectStr, ok := session.Values["redirect_uri"].(string); ok && redirectStr != "" {
  95. redirectURI, err := url.Parse(redirectStr)
  96. if err == nil {
  97. redirect = fmt.Sprintf("%s?%s", redirectURI.Path, redirectURI.RawQuery)
  98. }
  99. }
  100. http.Redirect(w, r, redirect, http.StatusFound)
  101. }