bookwyrm/bookwyrm/models/site.py

142 lines
4.9 KiB
Python
Raw Normal View History

2021-03-08 16:49:10 +00:00
""" the particulars for this instance of BookWyrm """
import base64
2020-10-02 20:32:19 +00:00
import datetime
from Crypto import Random
2021-03-21 01:23:59 +00:00
from django.db import models, IntegrityError
2021-05-26 08:19:39 +00:00
from django.dispatch import receiver
2020-06-03 16:38:30 +00:00
from django.utils import timezone
from model_utils import FieldTracker
2021-05-26 08:19:39 +00:00
from bookwyrm.preview_images import generate_site_preview_image_task
2021-06-18 22:24:10 +00:00
from bookwyrm.settings import DOMAIN, ENABLE_PREVIEW_IMAGES
2021-03-21 01:23:59 +00:00
from .base_model import BookWyrmModel
2020-06-03 16:38:30 +00:00
from .user import User
2021-03-08 16:49:10 +00:00
class SiteSettings(models.Model):
2021-04-26 16:15:42 +00:00
"""customized settings for this instance"""
2021-03-08 16:49:10 +00:00
name = models.CharField(default="BookWyrm", max_length=100)
instance_tagline = models.CharField(
2021-03-08 16:49:10 +00:00
max_length=150, default="Social Reading and Reviewing"
2020-10-05 23:07:37 +00:00
)
2021-03-08 16:49:10 +00:00
instance_description = models.TextField(default="This instance has no description.")
2021-04-30 17:42:27 +00:00
# about page
2021-03-08 16:49:10 +00:00
registration_closed_text = models.TextField(
default="Contact an administrator to get an invite"
2020-10-05 23:07:37 +00:00
)
2021-03-08 16:49:10 +00:00
code_of_conduct = models.TextField(default="Add a code of conduct here.")
privacy_policy = models.TextField(default="Add a privacy policy here.")
2021-04-30 17:42:27 +00:00
# registration
2021-03-08 16:49:10 +00:00
allow_registration = models.BooleanField(default=True)
2021-03-21 01:23:59 +00:00
allow_invite_requests = models.BooleanField(default=True)
2021-04-30 17:42:27 +00:00
# images
2021-03-08 16:49:10 +00:00
logo = models.ImageField(upload_to="logos/", null=True, blank=True)
logo_small = models.ImageField(upload_to="logos/", null=True, blank=True)
favicon = models.ImageField(upload_to="logos/", null=True, blank=True)
2021-05-26 12:46:34 +00:00
preview_image = models.ImageField(
upload_to="previews/logos/", null=True, blank=True
)
2021-04-30 17:42:27 +00:00
# footer
2020-12-11 20:31:02 +00:00
support_link = models.CharField(max_length=255, null=True, blank=True)
support_title = models.CharField(max_length=100, null=True, blank=True)
admin_email = models.EmailField(max_length=255, null=True, blank=True)
2021-04-30 17:42:27 +00:00
footer_item = models.TextField(null=True, blank=True)
2021-05-27 19:40:23 +00:00
field_tracker = FieldTracker(fields=["name", "instance_tagline", "logo"])
@classmethod
def get(cls):
2021-04-26 16:15:42 +00:00
"""gets the site settings db entry or defaults"""
try:
return cls.objects.get(id=1)
except cls.DoesNotExist:
default_settings = SiteSettings(id=1)
default_settings.save()
return default_settings
2021-03-08 16:49:10 +00:00
2020-10-02 20:32:19 +00:00
def new_access_code():
2021-04-26 16:15:42 +00:00
"""the identifier for a user invite"""
2021-03-08 16:49:10 +00:00
return base64.b32encode(Random.get_random_bytes(5)).decode("ascii")
class SiteInvite(models.Model):
2021-04-26 16:15:42 +00:00
"""gives someone access to create an account on the instance"""
2021-03-08 16:49:10 +00:00
2021-01-05 19:37:48 +00:00
created_date = models.DateTimeField(auto_now_add=True)
2020-10-02 20:32:19 +00:00
code = models.CharField(max_length=32, default=new_access_code)
expiry = models.DateTimeField(blank=True, null=True)
use_limit = models.IntegerField(blank=True, null=True)
times_used = models.IntegerField(default=0)
2020-06-03 16:38:30 +00:00
user = models.ForeignKey(User, on_delete=models.CASCADE)
2021-04-02 00:19:29 +00:00
invitees = models.ManyToManyField(User, related_name="invitees")
def valid(self):
2021-04-26 16:15:42 +00:00
"""make sure it hasn't expired or been used"""
2021-03-08 16:49:10 +00:00
return (self.expiry is None or self.expiry > timezone.now()) and (
self.use_limit is None or self.times_used < self.use_limit
)
2020-06-03 16:38:30 +00:00
@property
def link(self):
2021-04-26 16:15:42 +00:00
"""formats the invite link"""
2021-03-08 16:49:10 +00:00
return "https://{}/invite/{}".format(DOMAIN, self.code)
2020-10-02 20:32:19 +00:00
2021-03-21 01:23:59 +00:00
class InviteRequest(BookWyrmModel):
2021-04-26 16:15:42 +00:00
"""prospective users can request an invite"""
2021-03-21 01:23:59 +00:00
email = models.EmailField(max_length=255, unique=True)
invite = models.ForeignKey(
SiteInvite, on_delete=models.SET_NULL, null=True, blank=True
)
invite_sent = models.BooleanField(default=False)
ignored = models.BooleanField(default=False)
2021-03-21 01:23:59 +00:00
def save(self, *args, **kwargs):
2021-04-26 16:15:42 +00:00
"""don't create a request for a registered email"""
if not self.id and User.objects.filter(email=self.email).exists():
2021-03-21 01:23:59 +00:00
raise IntegrityError()
super().save(*args, **kwargs)
2020-10-02 20:32:19 +00:00
def get_passowrd_reset_expiry():
2021-04-26 16:15:42 +00:00
"""give people a limited time to use the link"""
now = timezone.now()
2020-10-02 20:32:19 +00:00
return now + datetime.timedelta(days=1)
class PasswordReset(models.Model):
2021-04-26 16:15:42 +00:00
"""gives someone access to create an account on the instance"""
2021-03-08 16:49:10 +00:00
2020-10-02 20:32:19 +00:00
code = models.CharField(max_length=32, default=new_access_code)
expiry = models.DateTimeField(default=get_passowrd_reset_expiry)
user = models.OneToOneField(User, on_delete=models.CASCADE)
def valid(self):
2021-04-26 16:15:42 +00:00
"""make sure it hasn't expired or been used"""
2020-10-02 20:32:19 +00:00
return self.expiry > timezone.now()
@property
def link(self):
2021-04-26 16:15:42 +00:00
"""formats the invite link"""
2021-03-08 16:49:10 +00:00
return "https://{}/password-reset/{}".format(DOMAIN, self.code)
2021-05-26 08:19:39 +00:00
# pylint: disable=unused-argument
2021-06-18 22:24:10 +00:00
@receiver(models.signals.post_save, sender=SiteSettings)
2021-05-26 08:19:39 +00:00
def preview_image(instance, *args, **kwargs):
2021-06-18 22:24:10 +00:00
"""Update image preview for the default site image"""
if not ENABLE_PREVIEW_IMAGES:
return
changed_fields = instance.field_tracker.changed()
2021-05-26 08:19:39 +00:00
if len(changed_fields) > 0:
2021-05-26 08:19:39 +00:00
generate_site_preview_image_task.delay()