mirror of
https://codeberg.org/rimu/pyfedi
synced 2025-01-23 19:36:56 -08:00
daily maintenance: handle exceptions
This commit is contained in:
parent
156c328c6f
commit
8a1aa648ae
3 changed files with 103 additions and 84 deletions
102
app/cli.py
102
app/cli.py
|
@ -26,7 +26,8 @@ from app.models import Settings, BannedInstances, Interest, Role, User, RolePerm
|
|||
Tag, InstanceRole, Community
|
||||
from app.post.routes import post_delete_post
|
||||
from app.utils import file_get_contents, retrieve_block_list, blocked_domains, retrieve_peertube_block_list, \
|
||||
shorten_string, get_request, html_to_text, blocked_communities, ap_datetime, gibberish
|
||||
shorten_string, get_request, html_to_text, blocked_communities, ap_datetime, gibberish, get_request_instance, \
|
||||
instance_banned
|
||||
|
||||
|
||||
def register(app):
|
||||
|
@ -208,98 +209,78 @@ def register(app):
|
|||
db.session.execute(text('DELETE FROM "post_reply_vote" WHERE created_at < :cutoff'), {'cutoff': utcnow() - timedelta(days=28 * 6)})
|
||||
db.session.commit()
|
||||
|
||||
# Check for dormant or dead instances
|
||||
try:
|
||||
# Check for dormant or dead instances
|
||||
instances = Instance.query.filter(Instance.gone_forever == False, Instance.id != 1).all()
|
||||
HEADERS = {'User-Agent': 'PieFed/1.0', 'Accept': 'application/activity+json'}
|
||||
|
||||
for instance in instances:
|
||||
if instance_banned(instance.domain) or instance.domain == 'flipboard.com':
|
||||
continue
|
||||
nodeinfo_href = instance.nodeinfo_href
|
||||
if instance.software == 'lemmy' and instance.version >= '0.19.4' and instance.nodeinfo_href and instance.nodeinfo_href.endswith('nodeinfo/2.0.json'):
|
||||
nodeinfo_href = None # Lemmy v0.19.4 no longer provides .well-known/nodeinfo response for 2.0, and
|
||||
# 'solves' this by redirecting calls for nodeinfo/2.0.json to nodeinfo/2.1
|
||||
if instance.software == 'lemmy' and instance.version >= '0.19.4' and instance.nodeinfo_href and instance.nodeinfo_href.endswith(
|
||||
'nodeinfo/2.0.json'):
|
||||
nodeinfo_href = None
|
||||
|
||||
if not nodeinfo_href:
|
||||
try:
|
||||
nodeinfo = get_request(f"https://{instance.domain}/.well-known/nodeinfo", headers=HEADERS)
|
||||
nodeinfo = get_request_instance(f"https://{instance.domain}/.well-known/nodeinfo",
|
||||
headers=HEADERS, instance=instance)
|
||||
|
||||
if nodeinfo.status_code == 200:
|
||||
try:
|
||||
nodeinfo_json = nodeinfo.json()
|
||||
except Exception as e:
|
||||
nodeinfo_json = {}
|
||||
finally:
|
||||
nodeinfo.close()
|
||||
for links in nodeinfo_json['links']:
|
||||
if isinstance(links, dict) and 'rel' in links and (
|
||||
links['rel'] == 'http://nodeinfo.diaspora.software/ns/schema/2.0' or
|
||||
links['rel'] == 'https://nodeinfo.diaspora.software/ns/schema/2.0' or
|
||||
links['rel'] == 'http://nodeinfo.diaspora.software/ns/schema/2.1'):
|
||||
if isinstance(links, dict) and 'rel' in links and links['rel'] in [
|
||||
'http://nodeinfo.diaspora.software/ns/schema/2.0',
|
||||
'https://nodeinfo.diaspora.software/ns/schema/2.0',
|
||||
'http://nodeinfo.diaspora.software/ns/schema/2.1']:
|
||||
instance.nodeinfo_href = links['href']
|
||||
instance.failures = 0
|
||||
instance.dormant = False
|
||||
db.session.commit()
|
||||
sleep(0.1)
|
||||
break
|
||||
else:
|
||||
instance.failures += 1
|
||||
elif nodeinfo.status_code >= 400:
|
||||
current_app.logger.info(f"{instance.domain} has no well-known/nodeinfo response")
|
||||
except httpx.HTTPError:
|
||||
instance.failures += 1
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.error(f"Error processing instance {instance.domain}: {e}")
|
||||
finally:
|
||||
nodeinfo.close()
|
||||
|
||||
if instance.nodeinfo_href:
|
||||
try:
|
||||
node = get_request(instance.nodeinfo_href, headers=HEADERS)
|
||||
node = get_request_instance(instance.nodeinfo_href, headers=HEADERS, instance=instance)
|
||||
if node.status_code == 200:
|
||||
try:
|
||||
node_json = node.json()
|
||||
except Exception as e:
|
||||
node_json = {}
|
||||
finally:
|
||||
node.close()
|
||||
if 'software' in node_json:
|
||||
instance.software = node_json['software']['name'].lower()
|
||||
instance.version = node_json['software']['version']
|
||||
instance.software = node_json['software']['name'].lower()[:50]
|
||||
instance.version = node_json['software']['version'][:50]
|
||||
instance.failures = 0
|
||||
instance.dormant = False
|
||||
elif node.status_code >= 400:
|
||||
instance.failures += 1
|
||||
instance.nodeinfo_href = None
|
||||
except httpx.HTTPError:
|
||||
instance.failures += 1
|
||||
instance.nodeinfo_href = None
|
||||
if instance.failures > 7 and instance.dormant == True:
|
||||
instance.gone_forever = True
|
||||
elif instance.failures > 2 and instance.dormant == False:
|
||||
instance.dormant = True
|
||||
db.session.commit()
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.error(f"Error processing nodeinfo for {instance.domain}: {e}")
|
||||
finally:
|
||||
node.close()
|
||||
|
||||
# retrieve list of Admins from /api/v3/site, update InstanceRole
|
||||
# Handle admin roles
|
||||
if instance.online() and (instance.software == 'lemmy' or instance.software == 'piefed'):
|
||||
try:
|
||||
response = get_request(f'https://{instance.domain}/api/v3/site')
|
||||
except:
|
||||
response = None
|
||||
|
||||
if response and response.status_code == 200:
|
||||
try:
|
||||
instance_data = response.json()
|
||||
except:
|
||||
instance_data = None
|
||||
finally:
|
||||
response.close()
|
||||
|
||||
if instance_data:
|
||||
if 'admins' in instance_data:
|
||||
admin_profile_ids = []
|
||||
for admin in instance_data['admins']:
|
||||
if admin['person']['actor_id'].startswith('http://'):
|
||||
continue # suppo.fi has rogue entry in its v3/site
|
||||
admin_profile_ids.append(admin['person']['actor_id'].lower())
|
||||
user = find_actor_or_create(admin['person']['actor_id'])
|
||||
if user and not instance.user_is_admin(user.id):
|
||||
new_instance_role = InstanceRole(instance_id=instance.id, user_id=user.id,
|
||||
role='admin')
|
||||
db.session.add(new_instance_role)
|
||||
db.session.commit()
|
||||
# remove any InstanceRoles that are no longer part of instance-data['admins']
|
||||
for instance_admin in InstanceRole.query.filter_by(instance_id=instance.id):
|
||||
if instance_admin.user.profile_id() not in admin_profile_ids:
|
||||
|
@ -307,8 +288,29 @@ def register(app):
|
|||
InstanceRole.user_id == instance_admin.user.id,
|
||||
InstanceRole.instance_id == instance.id,
|
||||
InstanceRole.role == 'admin').delete()
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.error(f"Error updating admins for {instance.domain}: {e}")
|
||||
finally:
|
||||
if response:
|
||||
response.close()
|
||||
|
||||
# Commit all changes at once
|
||||
db.session.commit()
|
||||
|
||||
except Exception as e:
|
||||
db.session.rollback()
|
||||
current_app.logger.error(f"Error in daily maintenance: {e}")
|
||||
|
||||
# remove any InstanceRoles that are no longer part of instance-data['admins']
|
||||
#for instance_admin in InstanceRole.query.filter_by(instance_id=instance.id):
|
||||
# if instance_admin.user.profile_id() not in admin_profile_ids:
|
||||
# db.session.query(InstanceRole).filter(
|
||||
# InstanceRole.user_id == instance_admin.user.id,
|
||||
# InstanceRole.instance_id == instance.id,
|
||||
# InstanceRole.role == 'admin').delete()
|
||||
# db.session.commit()
|
||||
|
||||
@app.cli.command("spaceusage")
|
||||
def spaceusage():
|
||||
with app.app_context():
|
||||
|
|
|
@ -104,6 +104,12 @@ class Instance(db.Model):
|
|||
return db.session.execute(text('SELECT COUNT(id) as c FROM "user" WHERE instance_id = :instance_id'),
|
||||
{'instance_id': self.id}).scalar()
|
||||
|
||||
def update_dormant_gone(self):
|
||||
if self.failures > 7 and self.dormant == True:
|
||||
self.gone_forever = True
|
||||
elif self.failures > 2 and self.dormant == False:
|
||||
self.dormant = True
|
||||
|
||||
@classmethod
|
||||
def weight(cls, domain: str):
|
||||
if domain:
|
||||
|
|
11
app/utils.py
11
app/utils.py
|
@ -114,6 +114,17 @@ def get_request(uri, params=None, headers=None) -> httpx.Response:
|
|||
return response
|
||||
|
||||
|
||||
# Same as get_request except updates instance on failure and does not raise any exceptions
|
||||
def get_request_instance(uri, instance: Instance, params=None, headers=None) -> httpx.Response:
|
||||
try:
|
||||
return get_request(uri, params, headers)
|
||||
except:
|
||||
instance.failures += 1
|
||||
instance.update_dormant_gone()
|
||||
db.session.commit()
|
||||
return httpx.Response(status_code=500)
|
||||
|
||||
|
||||
# do a HEAD request to a uri, return the result
|
||||
def head_request(uri, params=None, headers=None) -> httpx.Response:
|
||||
if headers is None:
|
||||
|
|
Loading…
Add table
Reference in a new issue