A clean, Markdown-based publishing platform made for writers. Write together, and build a community. https://writefreely.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

115 lines
2.9 KiB

  1. package writefreely
  2. import (
  3. "context"
  4. "errors"
  5. "net/http"
  6. "net/url"
  7. "strings"
  8. )
  9. type genericOauthClient struct {
  10. ClientID string
  11. ClientSecret string
  12. AuthLocation string
  13. ExchangeLocation string
  14. InspectLocation string
  15. CallbackLocation string
  16. HttpClient HttpClient
  17. }
  18. var _ oauthClient = genericOauthClient{}
  19. const (
  20. genericOauthDisplayName = "OAuth"
  21. )
  22. func (c genericOauthClient) GetProvider() string {
  23. return "generic"
  24. }
  25. func (c genericOauthClient) GetClientID() string {
  26. return c.ClientID
  27. }
  28. func (c genericOauthClient) GetCallbackLocation() string {
  29. return c.CallbackLocation
  30. }
  31. func (c genericOauthClient) buildLoginURL(state string) (string, error) {
  32. u, err := url.Parse(c.AuthLocation)
  33. if err != nil {
  34. return "", err
  35. }
  36. q := u.Query()
  37. q.Set("client_id", c.ClientID)
  38. q.Set("redirect_uri", c.CallbackLocation)
  39. q.Set("response_type", "code")
  40. q.Set("state", state)
  41. q.Set("scope", "read_user")
  42. u.RawQuery = q.Encode()
  43. return u.String(), nil
  44. }
  45. func (c genericOauthClient) exchangeOauthCode(ctx context.Context, code string) (*TokenResponse, error) {
  46. form := url.Values{}
  47. form.Add("grant_type", "authorization_code")
  48. form.Add("redirect_uri", c.CallbackLocation)
  49. form.Add("scope", "read_user")
  50. form.Add("code", code)
  51. req, err := http.NewRequest("POST", c.ExchangeLocation, strings.NewReader(form.Encode()))
  52. if err != nil {
  53. return nil, err
  54. }
  55. req.WithContext(ctx)
  56. req.Header.Set("User-Agent", ServerUserAgent(""))
  57. req.Header.Set("Accept", "application/json")
  58. req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
  59. req.SetBasicAuth(c.ClientID, c.ClientSecret)
  60. resp, err := c.HttpClient.Do(req)
  61. if err != nil {
  62. return nil, err
  63. }
  64. if resp.StatusCode != http.StatusOK {
  65. return nil, errors.New("unable to exchange code for access token")
  66. }
  67. var tokenResponse TokenResponse
  68. if err := limitedJsonUnmarshal(resp.Body, tokenRequestMaxLen, &tokenResponse); err != nil {
  69. return nil, err
  70. }
  71. if tokenResponse.Error != "" {
  72. return nil, errors.New(tokenResponse.Error)
  73. }
  74. return &tokenResponse, nil
  75. }
  76. func (c genericOauthClient) inspectOauthAccessToken(ctx context.Context, accessToken string) (*InspectResponse, error) {
  77. req, err := http.NewRequest("GET", c.InspectLocation, nil)
  78. if err != nil {
  79. return nil, err
  80. }
  81. req.WithContext(ctx)
  82. req.Header.Set("User-Agent", ServerUserAgent(""))
  83. req.Header.Set("Accept", "application/json")
  84. req.Header.Set("Authorization", "Bearer "+accessToken)
  85. resp, err := c.HttpClient.Do(req)
  86. if err != nil {
  87. return nil, err
  88. }
  89. if resp.StatusCode != http.StatusOK {
  90. return nil, errors.New("unable to inspect access token")
  91. }
  92. var inspectResponse InspectResponse
  93. if err := limitedJsonUnmarshal(resp.Body, infoRequestMaxLen, &inspectResponse); err != nil {
  94. return nil, err
  95. }
  96. if inspectResponse.Error != "" {
  97. return nil, errors.New(inspectResponse.Error)
  98. }
  99. return &inspectResponse, nil
  100. }