2020-01-15 04:56:50 +00:00
|
|
|
// Package api provides an interface to interact with the Discord REST API. It
|
|
|
|
// handles rate limiting, as well as authorizing and more.
|
2020-01-02 05:39:52 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2020-05-03 21:02:03 +00:00
|
|
|
"context"
|
2020-01-02 05:39:52 +00:00
|
|
|
"net/http"
|
|
|
|
|
2021-06-02 02:53:19 +00:00
|
|
|
"github.com/diamondburned/arikawa/v3/api/rate"
|
|
|
|
"github.com/diamondburned/arikawa/v3/utils/httputil"
|
|
|
|
"github.com/diamondburned/arikawa/v3/utils/httputil/httpdriver"
|
2020-01-02 05:39:52 +00:00
|
|
|
)
|
|
|
|
|
2020-04-19 16:30:12 +00:00
|
|
|
var (
|
2020-05-04 19:08:55 +00:00
|
|
|
BaseEndpoint = "https://discord.com"
|
2021-05-29 09:03:24 +00:00
|
|
|
Version = "9"
|
2020-07-29 23:29:01 +00:00
|
|
|
Path = "/api/v" + Version
|
2020-01-02 05:39:52 +00:00
|
|
|
|
2020-07-29 23:29:01 +00:00
|
|
|
Endpoint = BaseEndpoint + Path + "/"
|
2020-01-02 05:39:52 +00:00
|
|
|
EndpointGateway = Endpoint + "gateway"
|
|
|
|
EndpointGatewayBot = EndpointGateway + "/bot"
|
|
|
|
)
|
|
|
|
|
2021-05-29 09:03:52 +00:00
|
|
|
var UserAgent = "DiscordBot (https://github.com/diamondburned/arikawa/v3)"
|
2020-01-02 05:39:52 +00:00
|
|
|
|
|
|
|
type Client struct {
|
2020-04-19 21:53:53 +00:00
|
|
|
*httputil.Client
|
2021-03-06 04:01:33 +00:00
|
|
|
*Session
|
2021-05-29 20:28:45 +00:00
|
|
|
AcquireOptions rate.AcquireOptions
|
2020-01-02 05:39:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewClient(token string) *Client {
|
2020-04-19 21:53:53 +00:00
|
|
|
return NewCustomClient(token, httputil.NewClient())
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCustomClient(token string, httpClient *httputil.Client) *Client {
|
2021-05-29 20:28:45 +00:00
|
|
|
c := &Client{
|
|
|
|
Session: &Session{
|
|
|
|
Limiter: rate.NewLimiter(Path),
|
|
|
|
Token: token,
|
|
|
|
UserAgent: UserAgent,
|
|
|
|
},
|
|
|
|
Client: httpClient.Copy(),
|
2020-01-02 05:39:52 +00:00
|
|
|
}
|
|
|
|
|
2021-05-29 20:28:45 +00:00
|
|
|
c.Client.OnRequest = append(c.Client.OnRequest, c.InjectRequest)
|
|
|
|
c.Client.OnResponse = append(c.Client.OnResponse, c.OnResponse)
|
2020-05-03 21:02:03 +00:00
|
|
|
|
2021-05-29 20:28:45 +00:00
|
|
|
return c
|
2020-05-03 21:02:03 +00:00
|
|
|
}
|
|
|
|
|
2020-05-03 21:04:09 +00:00
|
|
|
// WithContext returns a shallow copy of Client with the given context. It's
|
|
|
|
// used for method timeouts and such. This method is thread-safe.
|
2020-05-03 21:02:03 +00:00
|
|
|
func (c *Client) WithContext(ctx context.Context) *Client {
|
|
|
|
return &Client{
|
2021-05-29 20:28:45 +00:00
|
|
|
Client: c.Client.WithContext(ctx),
|
|
|
|
Session: c.Session,
|
|
|
|
AcquireOptions: c.AcquireOptions,
|
2020-01-02 05:39:52 +00:00
|
|
|
}
|
2020-05-03 21:02:03 +00:00
|
|
|
}
|
|
|
|
|
2021-05-29 20:28:45 +00:00
|
|
|
func (c *Client) InjectRequest(r httpdriver.Request) error {
|
|
|
|
r.AddHeader(http.Header{
|
|
|
|
"Authorization": {c.Session.Token},
|
|
|
|
"User-Agent": {c.Session.UserAgent},
|
|
|
|
})
|
|
|
|
|
|
|
|
ctx := c.AcquireOptions.Context(r.GetContext())
|
|
|
|
return c.Session.Limiter.Acquire(ctx, r.GetPath())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) OnResponse(r httpdriver.Request, resp httpdriver.Response) error {
|
|
|
|
return c.Session.Limiter.Release(r.GetPath(), httpdriver.OptHeader(resp))
|
|
|
|
}
|
|
|
|
|
2020-05-03 21:02:03 +00:00
|
|
|
// Session keeps a single session. This is typically wrapped around Client.
|
|
|
|
type Session struct {
|
|
|
|
Limiter *rate.Limiter
|
|
|
|
|
|
|
|
Token string
|
|
|
|
UserAgent string
|
|
|
|
}
|