-
Notifications
You must be signed in to change notification settings - Fork 57
/
Copy pathcron.py
134 lines (104 loc) · 4.32 KB
/
cron.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
"""Cron jobs. Currently just minor cleanup tasks.
"""
from builtins import range
from datetime import datetime, timedelta, timezone
import itertools
import logging
from flask import g
from flask.views import View
from google.cloud import ndb
from oauth_dropins.webutil.models import StringIdModel
import requests
from blogger import Blogger
from bluesky import Bluesky
from flask_background import app
from flickr import Flickr
from mastodon import Mastodon
from reddit import Reddit
import models
from models import Source
import util
logger = logging.getLogger(__name__)
CIRCLECI_TOKEN = util.read('circleci_token')
PAGE_SIZE = 20
class LastUpdatedPicture(StringIdModel):
"""Stores the last user in a given silo that we updated profile picture for.
Key id is the silo's ``SHORT_NAME``.
"""
last = ndb.KeyProperty()
created = ndb.DateTimeProperty(auto_now_add=True, required=True, tzinfo=timezone.utc)
updated = ndb.DateTimeProperty(auto_now=True, tzinfo=timezone.utc)
@app.route('/cron/replace_poll_tasks')
def replace_poll_tasks():
"""Finds sources missing their poll tasks and adds new ones."""
queries = [cls.query(Source.features == 'listen', Source.status == 'enabled',
Source.last_poll_attempt < util.now() - timedelta(days=2))
for cls in models.sources.values() if cls.AUTO_POLL]
for source in itertools.chain(*queries):
age = util.now() - source.last_poll_attempt
logger.info(f'{source.bridgy_url()} last polled {age} ago. Adding new poll task.')
util.add_poll_task(source)
return ''
class UpdatePictures(View):
"""Finds sources with new profile pictures and updates them."""
SOURCE_CLS = None
@classmethod
def user_id(cls, source):
return source.key_id()
def dispatch_request(self):
g.TRANSIENT_ERROR_HTTP_CODES = (self.SOURCE_CLS.TRANSIENT_ERROR_HTTP_CODES +
self.SOURCE_CLS.RATE_LIMIT_HTTP_CODES)
query = self.SOURCE_CLS.query().order(self.SOURCE_CLS.key)
last = LastUpdatedPicture.get_by_id(self.SOURCE_CLS.SHORT_NAME)
if last and last.last:
query = query.filter(self.SOURCE_CLS.key > last.last)
results, _, more = query.fetch_page(PAGE_SIZE)
for source in results:
if source.features and source.status != 'disabled':
user_id = self.user_id(source)
logger.debug(f'checking for updated profile pictures for {source.bridgy_url()} {user_id}')
try:
actor = source.gr_source.get_actor(user_id)
except BaseException as e:
logger.debug('Failed', exc_info=True)
# Mastodon API returns HTTP 404 for deleted (etc) users, and
# often one or more users' Mastodon instances are down.
code, _ = util.interpret_http_exception(e)
if code:
continue
raise
if not actor:
logger.info(f"Couldn't fetch user")
continue
new_pic = actor.get('image', {}).get('url')
if not new_pic or source.picture == new_pic:
logger.info(f'No new picture found')
continue
@ndb.transactional()
def update():
src = source.key.get()
src.picture = new_pic
src.put()
logger.info(f'Updating profile picture from {source.picture} to {new_pic}')
update()
LastUpdatedPicture(id=self.SOURCE_CLS.SHORT_NAME,
last=source.key if more else None).put()
return 'OK'
class UpdateFlickrPictures(UpdatePictures):
"""Finds :class:`Flickr` sources with new profile pictures and updates them."""
SOURCE_CLS = Flickr
class UpdateMastodonPictures(UpdatePictures):
"""Finds :class:`Mastodon` sources with new profile pictures and updates them."""
SOURCE_CLS = Mastodon
@classmethod
def user_id(cls, source):
return source.auth_entity.get().user_id()
class UpdateRedditPictures(UpdatePictures):
"""Finds :class:`Reddit` sources with new profile pictures and updates them."""
SOURCE_CLS = Reddit
app.add_url_rule('/cron/update_flickr_pictures',
view_func=UpdateFlickrPictures.as_view('update_flickr_pictures'))
app.add_url_rule('/cron/update_mastodon_pictures',
view_func=UpdateMastodonPictures.as_view('update_mastodon_pictures'))
app.add_url_rule('/cron/update_reddit_pictures',
view_func=UpdateRedditPictures.as_view('update_reddit_pictures'))