-
Notifications
You must be signed in to change notification settings - Fork 57
/
bluesky.py
179 lines (146 loc) · 6.16 KB
/
bluesky.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
import logging
from urllib.parse import quote
from flask import flash, render_template, request
from google.cloud import ndb
from granary import bluesky as gr_bluesky
import lexrpc.client
from oauth_dropins import bluesky as oauth_bluesky
from oauth_dropins.webutil.flask_util import error
from oauth_dropins.webutil.util import json_loads
import requests
from flask_app import app
import models
import util
logger = logging.getLogger(__name__)
class Bluesky(models.Source):
"""A Bluesky account. Key id is DID."""
SHORT_NAME = 'bluesky'
GR_CLASS = gr_bluesky.Bluesky
CAN_PUBLISH = True
OAUTH_START = oauth_bluesky.Start
AUTH_MODEL = oauth_bluesky.BlueskyAuth
MICROPUB_TOKEN_PROPERTY = 'password'
URL_CANONICALIZER = util.UrlCanonicalizer(
domain=GR_CLASS.DOMAIN,
# Bluesky does not support HEAD requests.
redirects=False)
@staticmethod
def new(auth_entity, **kwargs):
"""Creates and returns a :class:`Bluesky` entity.
Args:
auth_entity: :class:`oauth_bluesky.BlueskyAuth`
kwargs: property values
"""
assert 'username' not in kwargs
assert 'id' not in kwargs
user = json_loads(auth_entity.user_json)
handle = user.get('handle')
return Bluesky(id=auth_entity.key_id(),
username=handle,
auth_entity=auth_entity.key,
name=user.get('displayName'),
picture=user.get('avatar'),
url=gr_bluesky.Bluesky.user_url(handle),
**kwargs)
def silo_url(self):
"""Returns the Bluesky account URL, e.g. ``https://bsky.app/profile/foo.com``."""
return self.gr_source.user_url(self.username)
def label_name(self):
"""Returns the Bluesky handle."""
return self.username
def format_for_source_url(self, id):
"""
Bluesky keys (AT URIs) contain slashes, so must be double-encoded.
This is due to a particular behaviour in WSGI:
https://github.com/pallets/flask/issues/900
They do not need to be decoded correspondingly.
"""
return quote(quote(id, safe=''))
def post_id(self, url):
if url.startswith('at://'):
# Bluesky can't currently resolve AT URIs containing handles,
# even though they are technically valid. Replace it with DID.
return url.replace(f'at://{self.username}', f'at://{self.key_id()}')
return gr_bluesky.web_url_to_at_uri(url, did=self.key_id(), handle=self.username)
@classmethod
def button_html(cls, feature, **kwargs):
"""Override oauth-dropins's button_html() to send a GET."""
return super().button_html(feature, form_method='get', **kwargs)
def canonicalize_url(self, url, **kwargs):
"""Canonicalizes a post or object URL.
Overrides :class:`Source`'s to convert ``staging.bsky.app`` to ``bsky.app``,
and to convert at:// URIs to bsky.app URLs.
"""
if url.startswith('at://'):
url = gr_bluesky.at_uri_to_web_url(url, handle=self.username)
url = url.replace('https://staging.bsky.app/', 'https://bsky.app/')
return super().canonicalize_url(url)
def get_activities_response(self, *args, **kwargs):
"""Raise DisableSource if we get a 400 with a token error.
https://atproto.com/specs/xrpc#summary-of-http-status-codes ...but not
explicitly mentioned there, or in https://atproto.com/specs/lexicon , or
anywhere else similar. Some discussion in
https://github.com/bluesky-social/atproto/discussions/1711#discussioncomment-7203160
"""
try:
return super().get_activities_response(*args, **kwargs)
except requests.HTTPError as e:
util.interpret_http_exception(e)
content_type = e.response.headers.get('Content-Type', '').split(';')[0]
if content_type == 'application/json':
resp = e.response.json()
msg = resp.get('message') or ''
if (resp.get('error') in lexrpc.client.TOKEN_ERRORS
or msg == 'Profile not found'
or msg.startswith('Could not find user info ')):
raise models.DisableSource()
raise
class Callback(oauth_bluesky.Callback):
def finish(self, auth_entity, state=None):
if not auth_entity:
flash("Failed to log in to Bluesky. Are your credentials correct?")
return util.redirect("/bluesky/start")
state = {
'operation': request.form['operation'],
'feature': request.form['feature'],
}
if request.form['operation'] == 'delete':
state['source'] = Bluesky(id=auth_entity.key.id()).key.urlsafe().decode()
util.maybe_add_or_delete_source(Bluesky, auth_entity,
util.encode_oauth_state(state))
@app.get('/bluesky/start')
def bluesky_start():
"""Serves the Bluesky login form page to sign up."""
request_values = request.values.to_dict()
feature = request_values.pop('feature', 'listen')
if 'username' in request_values:
error(f'username parameter is not supported')
username = ''
if id := request_values.get('id'):
if source := Bluesky.get_by_id(id):
username = source.username
return render_template('provide_app_password.html',
post_url='/bluesky/callback',
operation='add',
feature=feature,
username=username,
**request_values)
@app.get('/bluesky/delete/start')
def bluesky_delete():
"""Serves the Bluesky login form page to delete an existing account."""
return render_template('provide_app_password.html',
post_url='/bluesky/callback',
operation='delete',
**request.values)
@app.post('/bluesky/publish/start', endpoint='bluesky_publish_start')
def bluesky_publish_start():
username = ''
state = util.decode_oauth_state(request.values.get('state') or '')
if source_key := state.get('source_key'):
if source := ndb.Key(urlsafe=source_key).get():
username = source.username
return render_template('provide_app_password.html',
post_url='/publish/bluesky/finish',
username=username,
**request.values)
app.add_url_rule('/bluesky/callback', view_func=Callback.as_view('bluesky_callback', 'unused'), methods=['POST'])