witchie/toot/api.py

361 lines
9.6 KiB
Python
Raw Normal View History

2017-04-16 12:14:33 +00:00
# -*- coding: utf-8 -*-
import re
import uuid
2017-04-16 12:14:33 +00:00
2018-06-12 08:40:36 +00:00
from urllib.parse import urlparse, urlencode, quote
2017-04-16 12:14:33 +00:00
2017-12-30 13:15:51 +00:00
from toot import http, CLIENT_NAME, CLIENT_WEBSITE
from toot.exceptions import AuthenticationError
2019-01-24 10:18:28 +00:00
from toot.utils import str_bool
2017-04-16 12:14:33 +00:00
SCOPES = 'read write follow'
2017-04-26 09:49:21 +00:00
def _account_action(app, user, account, action):
url = '/api/v1/accounts/{}/{}'.format(account, action)
2017-04-26 09:49:21 +00:00
2017-12-30 13:15:51 +00:00
return http.post(app, user, url).json()
2017-04-26 09:49:21 +00:00
def _status_action(app, user, status_id, action):
url = '/api/v1/statuses/{}/{}'.format(status_id, action)
return http.post(app, user, url).json()
2018-12-25 01:20:30 +00:00
def create_app(domain, scheme='https'):
url = '{}://{}/api/v1/apps'.format(scheme, domain)
2017-04-16 12:14:33 +00:00
json = {
2017-04-16 12:14:33 +00:00
'client_name': CLIENT_NAME,
'redirect_uris': 'urn:ietf:wg:oauth:2.0:oob',
'scopes': SCOPES,
'website': CLIENT_WEBSITE,
}
2017-04-16 12:14:33 +00:00
return http.anon_post(url, json=json).json()
2017-04-16 12:14:33 +00:00
def register_account(app, username, email, password, locale="en", agreement=True):
"""
Register an account
https://docs.joinmastodon.org/methods/accounts/#create
"""
token = fetch_app_token(app)["access_token"]
url = f"{app.base_url}/api/v1/accounts"
headers = {"Authorization": f"Bearer {token}"}
json = {
"username": username,
"email": email,
"password": password,
"agreement": agreement,
"locale": locale
}
return http.anon_post(url, json=json, headers=headers).json()
def fetch_app_token(app):
json = {
"client_id": app.client_id,
"client_secret": app.client_secret,
"grant_type": "client_credentials",
"redirect_uri": "urn:ietf:wg:oauth:2.0:oob",
"scope": "read write"
}
return http.anon_post(f"{app.base_url}/oauth/token", json=json).json()
2017-04-16 12:14:33 +00:00
def login(app, username, password):
url = app.base_url + '/oauth/token'
data = {
2017-04-16 12:14:33 +00:00
'grant_type': 'password',
'client_id': app.client_id,
'client_secret': app.client_secret,
'username': username,
'password': password,
'scope': SCOPES,
}
response = http.anon_post(url, data=data, allow_redirects=False)
2017-04-16 12:14:33 +00:00
# If auth fails, it redirects to the login page
if response.is_redirect:
raise AuthenticationError()
2017-04-16 12:14:33 +00:00
return response.json()
2017-04-16 12:14:33 +00:00
def get_browser_login_url(app):
"""Returns the URL for manual log in via browser"""
return "{}/oauth/authorize/?{}".format(app.base_url, urlencode({
"response_type": "code",
"redirect_uri": "urn:ietf:wg:oauth:2.0:oob",
2018-03-05 15:10:34 +00:00
"scope": SCOPES,
"client_id": app.client_id,
}))
def request_access_token(app, authorization_code):
url = app.base_url + '/oauth/token'
data = {
'grant_type': 'authorization_code',
'client_id': app.client_id,
'client_secret': app.client_secret,
'code': authorization_code,
'redirect_uri': 'urn:ietf:wg:oauth:2.0:oob',
}
return http.anon_post(url, data=data, allow_redirects=False).json()
2018-06-13 10:43:31 +00:00
def post_status(
app,
user,
status,
visibility='public',
media_ids=None,
sensitive=False,
spoiler_text=None,
in_reply_to_id=None,
language=None,
2021-01-17 11:42:08 +00:00
scheduled_at=None,
2021-08-28 19:08:44 +00:00
content_type=None,
2018-06-13 10:43:31 +00:00
):
"""
Publish a new status.
https://docs.joinmastodon.org/methods/statuses/#create
2018-06-13 10:43:31 +00:00
"""
# Idempotency key assures the same status is not posted multiple times
# if the request is retried.
headers = {"Idempotency-Key": uuid.uuid4().hex}
json = {
2017-04-16 12:14:33 +00:00
'status': status,
'media_ids': media_ids,
2017-04-16 12:14:33 +00:00
'visibility': visibility,
2019-01-24 10:18:28 +00:00
'sensitive': str_bool(sensitive),
'spoiler_text': spoiler_text,
2018-06-13 10:43:31 +00:00
'in_reply_to_id': in_reply_to_id,
'language': language,
2021-01-17 11:42:08 +00:00
'scheduled_at': scheduled_at
2021-08-28 19:08:44 +00:00
}
if content_type:
json['content_type'] = content_type
2021-08-28 19:08:44 +00:00
return http.post(app, user, '/api/v1/statuses', json=json, headers=headers).json()
2017-04-16 12:14:33 +00:00
def fetch_status(app, user, id):
"""
Fetch a single status
https://docs.joinmastodon.org/methods/statuses/#get
"""
return http.get(app, user, f"/api/v1/statuses/{id}").json()
2018-06-14 08:40:16 +00:00
def delete_status(app, user, status_id):
"""
Deletes a status with given ID.
https://github.com/tootsuite/documentation/blob/master/Using-the-API/API.md#deleting-a-status
"""
return http.delete(app, user, '/api/v1/statuses/{}'.format(status_id))
def favourite(app, user, status_id):
return _status_action(app, user, status_id, 'favourite')
def unfavourite(app, user, status_id):
return _status_action(app, user, status_id, 'unfavourite')
def reblog(app, user, status_id):
return _status_action(app, user, status_id, 'reblog')
def unreblog(app, user, status_id):
return _status_action(app, user, status_id, 'unreblog')
def pin(app, user, status_id):
return _status_action(app, user, status_id, 'pin')
def unpin(app, user, status_id):
return _status_action(app, user, status_id, 'unpin')
2022-11-17 05:28:41 +00:00
def bookmark(app, user, status_id):
return _status_action(app, user, status_id, 'bookmark')
def unbookmark(app, user, status_id):
return _status_action(app, user, status_id, 'unbookmark')
def context(app, user, status_id):
url = '/api/v1/statuses/{}/context'.format(status_id)
2019-01-24 08:36:25 +00:00
return http.get(app, user, url).json()
def reblogged_by(app, user, status_id):
url = '/api/v1/statuses/{}/reblogged_by'.format(status_id)
return http.get(app, user, url).json()
2019-02-13 13:19:27 +00:00
def _get_next_path(headers):
2018-01-06 10:25:05 +00:00
"""Given timeline response headers, returns the path to the next batch"""
links = headers.get('Link', '')
matches = re.match('<([^>]+)>; rel="next"', links)
if matches:
parsed = urlparse(matches.group(1))
return "?".join([parsed.path, parsed.query])
def _timeline_generator(app, user, path, params=None):
while path:
response = http.get(app, user, path, params)
yield response.json()
2019-02-13 13:19:27 +00:00
path = _get_next_path(response.headers)
2017-04-16 12:14:33 +00:00
2018-01-06 10:25:05 +00:00
def home_timeline_generator(app, user, limit=20):
path = '/api/v1/timelines/home?limit={}'.format(limit)
return _timeline_generator(app, user, path)
def public_timeline_generator(app, user, local=False, limit=20):
path = '/api/v1/timelines/public'
2019-01-24 10:18:28 +00:00
params = {'local': str_bool(local), 'limit': limit}
return _timeline_generator(app, user, path, params)
def tag_timeline_generator(app, user, hashtag, local=False, limit=20):
path = '/api/v1/timelines/tag/{}'.format(quote(hashtag))
2019-01-24 10:18:28 +00:00
params = {'local': str_bool(local), 'limit': limit}
return _timeline_generator(app, user, path, params)
2018-01-06 10:25:05 +00:00
def timeline_list_generator(app, user, list_id, limit=20):
path = '/api/v1/timelines/list/{}'.format(list_id)
return _timeline_generator(app, user, path, {'limit': limit})
def _anon_timeline_generator(instance, path, params=None):
while path:
url = "https://{}{}".format(instance, path)
response = http.anon_get(url, params)
yield response.json()
path = _get_next_path(response.headers)
def anon_public_timeline_generator(instance, local=False, limit=20):
path = '/api/v1/timelines/public'
params = {'local': str_bool(local), 'limit': limit}
return _anon_timeline_generator(instance, path, params)
def anon_tag_timeline_generator(instance, hashtag, local=False, limit=20):
path = '/api/v1/timelines/tag/{}'.format(quote(hashtag))
params = {'local': str_bool(local), 'limit': limit}
return _anon_timeline_generator(instance, path, params)
2021-08-26 15:54:31 +00:00
def upload_media(app, user, file, description=None):
return http.post(app, user, '/api/v1/media',
data={'description': description},
files={'file': file}
).json()
2017-04-16 13:07:27 +00:00
def search(app, user, query, resolve):
2019-09-22 11:04:58 +00:00
return http.get(app, user, '/api/v2/search', {
2017-04-16 13:07:27 +00:00
'q': query,
'resolve': resolve,
}).json()
2017-04-16 15:15:05 +00:00
2017-04-17 09:10:57 +00:00
def search_accounts(app, user, query):
2022-11-09 15:46:24 +00:00
return http.get(app, user, '/api/v2/search', {
2017-04-17 09:10:57 +00:00
'q': query,
2022-11-09 15:46:24 +00:00
'type': 'accounts',
'resolve': True,
}).json()['accounts']
2017-04-17 09:10:57 +00:00
2017-04-16 15:15:05 +00:00
def follow(app, user, account):
2017-04-26 09:49:21 +00:00
return _account_action(app, user, account, 'follow')
2017-04-16 15:15:05 +00:00
def unfollow(app, user, account):
2017-04-26 09:49:21 +00:00
return _account_action(app, user, account, 'unfollow')
2017-04-16 15:15:05 +00:00
2022-11-22 20:27:21 +00:00
def _get_account_list(app, user, path):
accounts = []
while path:
response = http.get(app, user, path)
accounts += response.json()
path = _get_next_path(response.headers)
return accounts
2022-11-22 20:27:21 +00:00
def following(app, user, account):
path = '/api/v1/accounts/{}/{}'.format(account, 'following')
return _get_account_list(app, user, path)
2022-11-22 20:27:21 +00:00
def followers(app, user, account):
path = '/api/v1/accounts/{}/{}'.format(account, 'followers')
return _get_account_list(app, user, path)
2017-04-26 09:49:21 +00:00
def mute(app, user, account):
return _account_action(app, user, account, 'mute')
def unmute(app, user, account):
return _account_action(app, user, account, 'unmute')
def block(app, user, account):
return _account_action(app, user, account, 'block')
def unblock(app, user, account):
return _account_action(app, user, account, 'unblock')
2017-04-16 15:52:54 +00:00
def verify_credentials(app, user):
2017-12-30 13:15:51 +00:00
return http.get(app, user, '/api/v1/accounts/verify_credentials').json()
def single_status(app, user, status_id):
url = '/api/v1/statuses/{}'.format(status_id)
return http.get(app, user, url).json()
def get_notifications(app, user, exclude_types=[], limit=20):
2022-11-22 20:27:21 +00:00
params = {"exclude_types[]": exclude_types, "limit": limit}
return http.get(app, user, '/api/v1/notifications', params).json()
2017-12-29 13:26:40 +00:00
def clear_notifications(app, user):
http.post(app, user, '/api/v1/notifications/clear')
def get_instance(domain, scheme="https"):
url = "{}://{}/api/v1/instance".format(scheme, domain)
return http.anon_get(url).json()