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.
162 lines
6.6 KiB
162 lines
6.6 KiB
import requests
|
|
import random
|
|
import gevent
|
|
|
|
from holster.enum import Enum
|
|
|
|
from disco.util.logging import LoggingClass
|
|
from disco.api.ratelimit import RateLimiter
|
|
|
|
HTTPMethod = Enum(
|
|
GET='GET',
|
|
POST='POST',
|
|
PUT='PUT',
|
|
PATCH='PATCH',
|
|
DELETE='DELETE',
|
|
)
|
|
|
|
|
|
class Routes(object):
|
|
# Gateway
|
|
GATEWAY_GET = (HTTPMethod.GET, '/gateway')
|
|
|
|
# Channels
|
|
CHANNELS = '/channels/{channel}'
|
|
CHANNELS_GET = (HTTPMethod.GET, CHANNELS)
|
|
CHANNELS_MODIFY = (HTTPMethod.PATCH, CHANNELS)
|
|
CHANNELS_DELETE = (HTTPMethod.DELETE, CHANNELS)
|
|
|
|
CHANNELS_MESSAGES_LIST = (HTTPMethod.GET, CHANNELS + '/messages')
|
|
CHANNELS_MESSAGES_GET = (HTTPMethod.GET, CHANNELS + '/messages/{message}')
|
|
CHANNELS_MESSAGES_CREATE = (HTTPMethod.POST, CHANNELS + '/messages')
|
|
CHANNELS_MESSAGES_MODIFY = (HTTPMethod.PATCH, CHANNELS + '/messages/{message}')
|
|
CHANNELS_MESSAGES_DELETE = (HTTPMethod.DELETE, CHANNELS + '/messages/{message}')
|
|
CHANNELS_MESSAGES_DELETE_BULK = (HTTPMethod.POST, CHANNELS + '/messages/bulk_delete')
|
|
|
|
CHANNELS_PERMISSIONS_MODIFY = (HTTPMethod.PUT, CHANNELS + '/permissions/{permission}')
|
|
CHANNELS_PERMISSIONS_DELETE = (HTTPMethod.DELETE, CHANNELS + '/permissions/{permission}')
|
|
CHANNELS_INVITES_LIST = (HTTPMethod.GET, CHANNELS + '/invites')
|
|
CHANNELS_INVITES_CREATE = (HTTPMethod.POST, CHANNELS + '/invites')
|
|
|
|
CHANNELS_PINS_LIST = (HTTPMethod.GET, CHANNELS + '/pins')
|
|
CHANNELS_PINS_CREATE = (HTTPMethod.PUT, CHANNELS + '/pins/{pin}')
|
|
CHANNELS_PINS_DELETE = (HTTPMethod.DELETE, CHANNELS + '/pins/{pin}')
|
|
|
|
# Guilds
|
|
GUILDS = '/guilds/{guild}'
|
|
GUILDS_GET = (HTTPMethod.GET, GUILDS)
|
|
GUILDS_MODIFY = (HTTPMethod.PATCH, GUILDS)
|
|
GUILDS_DELETE = (HTTPMethod.DELETE, GUILDS)
|
|
GUILDS_CHANNELS_LIST = (HTTPMethod.GET, GUILDS + '/channels')
|
|
GUILDS_CHANNELS_CREATE = (HTTPMethod.POST, GUILDS + '/channels')
|
|
GUILDS_CHANNELS_MODIFY = (HTTPMethod.PATCH, GUILDS + '/channels')
|
|
GUILDS_MEMBERS_LIST = (HTTPMethod.GET, GUILDS + '/members')
|
|
GUILDS_MEMBERS_GET = (HTTPMethod.GET, GUILDS + '/members/{member}')
|
|
GUILDS_MEMBERS_MODIFY = (HTTPMethod.PATCH, GUILDS + '/members/{member}')
|
|
GUILDS_MEMBERS_KICK = (HTTPMethod.DELETE, GUILDS + '/members/{member}')
|
|
GUILDS_BANS_LIST = (HTTPMethod.GET, GUILDS + '/bans')
|
|
GUILDS_BANS_CREATE = (HTTPMethod.PUT, GUILDS + '/bans/{user}')
|
|
GUILDS_BANS_DELETE = (HTTPMethod.DELETE, GUILDS + '/bans/{user}')
|
|
GUILDS_ROLES_LIST = (HTTPMethod.GET, GUILDS + '/roles')
|
|
GUILDS_ROLES_CREATE = (HTTPMethod.GET, GUILDS + '/roles')
|
|
GUILDS_ROLES_MODIFY_BATCH = (HTTPMethod.PATCH, GUILDS + '/roles')
|
|
GUILDS_ROLES_MODIFY = (HTTPMethod.PATCH, GUILDS + '/roles/{role}')
|
|
GUILDS_ROLES_DELETE = (HTTPMethod.DELETE, GUILDS + '/roles/{role}')
|
|
GUILDS_PRUNE_COUNT = (HTTPMethod.GET, GUILDS + '/prune')
|
|
GUILDS_PRUNE_BEGIN = (HTTPMethod.POST, GUILDS + '/prune')
|
|
GUILDS_VOICE_REGIONS_LIST = (HTTPMethod.GET, GUILDS + '/regions')
|
|
GUILDS_INVITES_LIST = (HTTPMethod.GET, GUILDS + '/invites')
|
|
GUILDS_INTEGRATIONS_LIST = (HTTPMethod.GET, GUILDS + '/integrations')
|
|
GUILDS_INTEGRATIONS_CREATE = (HTTPMethod.POST, GUILDS + '/integrations')
|
|
GUILDS_INTEGRATIONS_MODIFY = (HTTPMethod.PATCH, GUILDS + '/integrations/{integration}')
|
|
GUILDS_INTEGRATIONS_DELETE = (HTTPMethod.DELETE, GUILDS + '/integrations/{integration}')
|
|
GUILDS_INTEGRATIONS_SYNC = (HTTPMethod.POST, GUILDS + '/integrations/{integration}/sync')
|
|
GUILDS_EMBED_GET = (HTTPMethod.GET, GUILDS + '/embed')
|
|
GUILDS_EMBED_MODIFY = (HTTPMethod.PATCH, GUILDS + '/embed')
|
|
|
|
# Users
|
|
USERS = '/users'
|
|
USERS_ME_GET = (HTTPMethod.GET, USERS + '/@me')
|
|
USERS_ME_PATCH = (HTTPMethod.PATCH, USERS + '/@me')
|
|
USERS_ME_GUILDS_LIST = (HTTPMethod.GET, USERS + '/@me/guilds')
|
|
USERS_ME_GUILDS_LEAVE = (HTTPMethod.DELETE, USERS + '/@me/guilds/{guild}')
|
|
USERS_ME_DMS_LIST = (HTTPMethod.GET, USERS + '/@me/channels')
|
|
USERS_ME_DMS_CREATE = (HTTPMethod.POST, USERS + '/@me/channels')
|
|
USERS_ME_CONNECTIONS_LIST = (HTTPMethod.GET, USERS + '/@me/connections')
|
|
USERS_GET = (HTTPMethod.GET, USERS + '/{user}')
|
|
|
|
|
|
class APIException(Exception):
|
|
def __init__(self, msg, status_code=0, content=None):
|
|
self.status_code = status_code
|
|
self.content = content
|
|
self.msg = msg
|
|
|
|
if self.status_code:
|
|
self.msg += ' code: {}'.format(status_code)
|
|
|
|
super(APIException, self).__init__(self.msg)
|
|
|
|
|
|
class HTTPClient(LoggingClass):
|
|
BASE_URL = 'https://discordapp.com/api/v6'
|
|
MAX_RETRIES = 5
|
|
|
|
def __init__(self, token):
|
|
super(HTTPClient, self).__init__()
|
|
|
|
self.limiter = RateLimiter()
|
|
self.headers = {
|
|
'Authorization': 'Bot ' + token,
|
|
}
|
|
|
|
def __call__(self, route, args=None, **kwargs):
|
|
args = args or {}
|
|
retry = kwargs.pop('retry_number', 0)
|
|
|
|
# Merge or set headers
|
|
if 'headers' in kwargs:
|
|
kwargs['headers'].update(self.headers)
|
|
else:
|
|
kwargs['headers'] = self.headers
|
|
|
|
# Build the bucket URL
|
|
filtered = {k: (v if v in ('guild', 'channel') else '') for k, v in args.items()}
|
|
bucket = (route[0].value, route[1].format(**filtered))
|
|
|
|
# Possibly wait if we're rate limited
|
|
self.limiter.check(bucket)
|
|
|
|
# Make the actual request
|
|
url = self.BASE_URL + route[1].format(**args)
|
|
r = requests.request(route[0].value, url, **kwargs)
|
|
|
|
# Update rate limiter
|
|
self.limiter.update(bucket, r)
|
|
|
|
# If we got a success status code, just return the data
|
|
if r.status_code < 400:
|
|
return r
|
|
elif r.status_code != 429 and 400 < r.status_code < 500:
|
|
raise APIException('Request failed', r.status_code, r.content)
|
|
else:
|
|
if r.status_code == 429:
|
|
self.log.warning('Request responded w/ 429, retrying (but this should not happen, check your clock sync')
|
|
|
|
# If we hit the max retries, throw an error
|
|
retry += 1
|
|
if retry > self.MAX_RETRIES:
|
|
self.log.error('Failing request, hit max retries')
|
|
raise APIException('Request failed after {} attempts'.format(self.MAX_RETRIES), r.status_code, r.content)
|
|
|
|
backoff = self.random_backoff()
|
|
self.log.warning('Request to `{}` failed with code {}, retrying after {}s'.format(url, r.status_code, backoff))
|
|
gevent.sleep(backoff)
|
|
|
|
# Otherwise just recurse and try again
|
|
return self(route, args, retry_number=retry, **kwargs)
|
|
|
|
@staticmethod
|
|
def random_backoff():
|
|
# 500 milliseconds to 5 seconds)
|
|
return random.randint(500, 5000) / 1000.0
|
|
|