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 writeAsOauthClient 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 = writeAsOauthClient{}
  19. const (
  20. writeAsAuthLocation = "https://write.as/oauth/login"
  21. writeAsExchangeLocation = "https://write.as/oauth/token"
  22. writeAsIdentityLocation = "https://write.as/oauth/inspect"
  23. )
  24. func (c writeAsOauthClient) GetProvider() string {
  25. return "write.as"
  26. }
  27. func (c writeAsOauthClient) GetClientID() string {
  28. return c.ClientID
  29. }
  30. func (c writeAsOauthClient) GetCallbackLocation() string {
  31. return c.CallbackLocation
  32. }
  33. func (c writeAsOauthClient) buildLoginURL(state string) (string, error) {
  34. u, err := url.Parse(c.AuthLocation)
  35. if err != nil {
  36. return "", err
  37. }
  38. q := u.Query()
  39. q.Set("client_id", c.ClientID)
  40. q.Set("redirect_uri", c.CallbackLocation)
  41. q.Set("response_type", "code")
  42. q.Set("state", state)
  43. u.RawQuery = q.Encode()
  44. return u.String(), nil
  45. }
  46. func (c writeAsOauthClient) exchangeOauthCode(ctx context.Context, code string) (*TokenResponse, error) {
  47. form := url.Values{}
  48. form.Add("grant_type", "authorization_code")
  49. form.Add("redirect_uri", c.CallbackLocation)
  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 writeAsOauthClient) 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. }