lib.itmens/common/setup.py
2024-01-14 13:59:39 -05:00

225 lines
7.5 KiB
Python

import django
from django.conf import settings
from django.core.checks import Error, Warning
from loguru import logger
from catalog.search.models import Indexer
from common.models import JobManager
from takahe.models import Config as TakaheConfig
from takahe.models import Domain as TakaheDomain
from takahe.models import Follow as TakaheFollow
from takahe.models import Identity as TakaheIdentity
from takahe.models import User as TakaheUser
from takahe.utils import Takahe
from users.models import User
class Setup:
"""
Post-Migration Setup
"""
def create_site(self, domain, service_domain):
TakaheDomain.objects.update_or_create(
domain=domain,
defaults={
"local": True,
"service_domain": service_domain,
"notes": "NeoDB",
"nodeinfo": {},
"state": "updated",
},
)
TakaheConfig.objects.update_or_create(
key="public_timeline",
user=None,
identity=None,
domain=None,
defaults={"json": False},
)
def sync_site_config(self):
domain = settings.SITE_INFO["site_domain"]
if not domain:
raise ValueError("Panic: site_domain is not set!")
icon = settings.SITE_INFO["site_logo"]
name = settings.SITE_INFO["site_name"]
service_domain = settings.SITE_INFO.get("site_service_domain")
if not TakaheDomain.objects.filter(domain=domain).exists():
logger.info(f"Domain {domain} not found, creating...")
self.create_site(domain, service_domain)
if (
TakaheIdentity.objects.filter(local=True)
.exclude(domain_id__isnull=True)
.exists()
):
logger.warning(
f"Local identities are found for other domains, there might be a configuration issue."
)
TakaheConfig.objects.update_or_create(
key="site_name",
user=None,
identity=None,
domain=None,
defaults={"json": name},
)
TakaheConfig.objects.update_or_create(
key="site_name",
user=None,
identity=None,
domain_id=domain,
defaults={"json": name},
)
TakaheConfig.objects.update_or_create(
key="site_icon",
user=None,
identity=None,
domain_id=None,
defaults={"json": icon},
)
TakaheConfig.objects.update_or_create(
key="site_icon",
user=None,
identity=None,
domain_id=domain,
defaults={"json": icon},
)
def sync_admin_user(self):
users = User.objects.filter(username__in=settings.SETUP_ADMIN_USERNAMES)
for user in users:
if user.is_superuser:
logger.debug(f"User {user.username} is already admin")
else:
user.is_superuser = True
user.save(update_fields=["is_superuser"])
TakaheUser.objects.filter(email=f"@{user.username}").update(admin=True)
logger.info(f"Updated user {user.username} as admin")
def sync_relay(self):
relay_follow = TakaheFollow.objects.filter(
source__username="__relay__",
source__local=True,
target__actor_uri=settings.DEFAULT_RELAY_SERVER,
).first()
if settings.DISABLE_DEFAULT_RELAY:
if relay_follow:
logger.info("Default relay is disabled, unsubscribing...")
Takahe.create_internal_message(
{
"type": "UnfollowRelay",
"actor_uri": settings.DEFAULT_RELAY_SERVER,
}
)
else:
logger.info(f"Default relay is disabled.")
else:
if relay_follow:
logger.debug(
f"Default relay is enabled and subscribed, state: {relay_follow.state}"
)
else:
logger.info("Default relay is enabled, subscribing...")
relay_actor = TakaheIdentity.objects.filter(
username="__relay__",
local=True,
).first()
if not relay_actor:
logger.warning(
f"Default relay is enabled but relay actor does not exist."
)
return
Takahe.create_internal_message(
{
"type": "AddFollow",
"source": relay_actor.pk,
"target_actor": settings.DEFAULT_RELAY_SERVER,
"boosts": False,
}
)
def run(self):
logger.info("Running post-migration setup...")
# Update site name if changed
self.sync_site_config()
# Create/update admin user if configured in env
self.sync_admin_user()
# Subscribe to default relay if enabled
self.sync_relay()
# Create basic emoji if not exists
# Create search index if not exists
Indexer.init()
# Register cron jobs if not yet
if settings.DISABLE_CRON_JOBS and "*" in settings.DISABLE_CRON_JOBS:
logger.info("Cron jobs are disabled.")
JobManager.cancel_all()
else:
JobManager.reschedule_all()
logger.info("Finished post-migration setup.")
def check(self):
from redis import Redis
errors = []
# check env
domain = settings.SITE_INFO.get("site_domain")
if not domain:
errors.append(
Error(
"SITE DOMAIN is not specified",
hint="Check NEODB_SITE_DOMAIN in .env",
id="neodb.E001",
)
)
# check redis
try:
redis = Redis.from_url(settings.REDIS_URL)
if not redis:
raise Exception("Redis unavailable")
redis.ping()
except Exception as e:
errors.append(
Error(
f"Error while connecting to redis: {e}",
hint="Check NEODB_REDIS_URL/TAKAHE_CACHES_DEFAULT in .env",
id="neodb.E002",
)
)
# check indexer
try:
Indexer.check()
except Exception as e:
errors.append(
Error(
f"Error while connecting to search index server: {e}",
hint='Check NEODB_SEARCH_URL in .env, and run "neodb-manage migration"',
id="neodb.E003",
)
)
# check takahe
try:
if not TakaheDomain.objects.filter(domain=domain).exists():
errors.append(
Warning(
f"Domain {domain} not found in takahe database",
hint="Run migration once to create the domain",
id="neodb.W001",
)
)
except Exception as e:
errors.append(
Error(
f"Error while querying takahe database: {e}",
hint='Check TAKAHE_DB_URL/TAKAHE_DATABASE_SERVER in .env, and run "takahe-manage migration"',
id="neodb.E004",
)
)
return errors