linkding/bookmarks/models.py

424 lines
14 KiB
Python
Raw Normal View History

import binascii
import os
2019-07-01 20:05:38 +00:00
from typing import List
2019-06-28 23:08:22 +00:00
from django import forms
2019-06-27 06:09:51 +00:00
from django.contrib.auth import get_user_model
2021-03-28 10:11:56 +00:00
from django.contrib.auth.models import User
2019-06-27 06:09:51 +00:00
from django.db import models
2021-03-28 10:11:56 +00:00
from django.db.models.signals import post_save
from django.dispatch import receiver
from django.http import QueryDict
2019-06-27 06:09:51 +00:00
from bookmarks.utils import unique
from bookmarks.validators import BookmarkURLValidator
2019-06-27 06:09:51 +00:00
2019-06-30 05:15:46 +00:00
class Tag(models.Model):
name = models.CharField(max_length=64)
date_added = models.DateTimeField()
owner = models.ForeignKey(get_user_model(), on_delete=models.CASCADE)
2019-06-30 06:24:21 +00:00
def __str__(self):
return self.name
2019-06-30 05:15:46 +00:00
def sanitize_tag_name(tag_name: str):
# strip leading/trailing spaces
# replace inner spaces with replacement char
2024-01-27 10:29:16 +00:00
return tag_name.strip().replace(" ", "-")
2024-01-27 10:29:16 +00:00
def parse_tag_string(tag_string: str, delimiter: str = ","):
2019-07-01 20:05:38 +00:00
if not tag_string:
return []
names = tag_string.strip().split(delimiter)
# remove empty names, sanitize remaining names
names = [sanitize_tag_name(name) for name in names if name]
# remove duplicates
names = unique(names, str.lower)
2019-07-01 20:05:38 +00:00
names.sort(key=str.lower)
return names
2024-01-27 10:29:16 +00:00
def build_tag_string(tag_names: List[str], delimiter: str = ","):
2019-07-01 20:05:38 +00:00
return delimiter.join(tag_names)
2019-06-27 06:09:51 +00:00
class Bookmark(models.Model):
url = models.CharField(max_length=2048, validators=[BookmarkURLValidator()])
title = models.CharField(max_length=512, blank=True)
description = models.TextField(blank=True)
notes = models.TextField(blank=True)
2019-06-29 00:01:26 +00:00
website_title = models.CharField(max_length=512, blank=True, null=True)
website_description = models.TextField(blank=True, null=True)
web_archive_snapshot_url = models.CharField(max_length=2048, blank=True)
favicon_file = models.CharField(max_length=512, blank=True)
unread = models.BooleanField(default=False)
is_archived = models.BooleanField(default=False)
shared = models.BooleanField(default=False)
2019-06-27 06:09:51 +00:00
date_added = models.DateTimeField()
2019-06-28 22:27:20 +00:00
date_modified = models.DateTimeField()
date_accessed = models.DateTimeField(blank=True, null=True)
2019-06-27 06:09:51 +00:00
owner = models.ForeignKey(get_user_model(), on_delete=models.CASCADE)
2019-06-30 05:15:46 +00:00
tags = models.ManyToManyField(Tag)
2019-06-27 06:09:51 +00:00
2019-06-28 17:37:41 +00:00
@property
def resolved_title(self):
if self.title:
return self.title
elif self.website_title:
return self.website_title
else:
return self.url
2019-06-28 17:37:41 +00:00
@property
def resolved_description(self):
return self.website_description if not self.description else self.description
2019-06-30 06:24:21 +00:00
@property
def tag_names(self):
return [tag.name for tag in self.tags.all()]
2019-06-30 06:24:21 +00:00
2019-06-27 06:09:51 +00:00
def __str__(self):
2024-01-27 10:29:16 +00:00
return self.resolved_title + " (" + self.url[:30] + "...)"
2019-06-28 23:08:22 +00:00
class BookmarkForm(forms.ModelForm):
# Use URLField for URL
url = forms.CharField(validators=[BookmarkURLValidator()])
2019-07-01 20:05:38 +00:00
tag_string = forms.CharField(required=False)
2019-06-28 23:08:22 +00:00
# Do not require title and description in form as we fill these automatically if they are empty
2024-01-27 10:29:16 +00:00
title = forms.CharField(max_length=512, required=False)
description = forms.CharField(required=False, widget=forms.Textarea())
# Include website title and description as hidden field as they only provide info when editing bookmarks
2024-01-27 10:29:16 +00:00
website_title = forms.CharField(
max_length=512, required=False, widget=forms.HiddenInput()
)
website_description = forms.CharField(required=False, widget=forms.HiddenInput())
unread = forms.BooleanField(required=False)
shared = forms.BooleanField(required=False)
2019-07-05 20:29:21 +00:00
# Hidden field that determines whether to close window/tab after saving the bookmark
auto_close = forms.CharField(required=False)
2019-06-28 23:08:22 +00:00
class Meta:
model = Bookmark
fields = [
2024-01-27 10:29:16 +00:00
"url",
"tag_string",
"title",
"description",
"notes",
"website_title",
"website_description",
"unread",
"shared",
"auto_close",
]
@property
def has_notes(self):
return self.instance and self.instance.notes
class BookmarkSearch:
2024-01-27 10:29:16 +00:00
SORT_ADDED_ASC = "added_asc"
SORT_ADDED_DESC = "added_desc"
SORT_TITLE_ASC = "title_asc"
SORT_TITLE_DESC = "title_desc"
2024-01-27 10:29:16 +00:00
FILTER_SHARED_OFF = "off"
FILTER_SHARED_SHARED = "yes"
FILTER_SHARED_UNSHARED = "no"
2024-01-27 10:29:16 +00:00
FILTER_UNREAD_OFF = "off"
FILTER_UNREAD_YES = "yes"
FILTER_UNREAD_NO = "no"
2023-09-16 08:39:27 +00:00
2024-01-27 10:29:16 +00:00
params = ["q", "user", "sort", "shared", "unread"]
preferences = ["sort", "shared", "unread"]
defaults = {
2024-01-27 10:29:16 +00:00
"q": "",
"user": "",
"sort": SORT_ADDED_DESC,
"shared": FILTER_SHARED_OFF,
"unread": FILTER_UNREAD_OFF,
}
2024-01-27 10:29:16 +00:00
def __init__(
self,
q: str = None,
user: str = None,
sort: str = None,
shared: str = None,
unread: str = None,
preferences: dict = None,
):
if not preferences:
preferences = {}
self.defaults = {**BookmarkSearch.defaults, **preferences}
2024-01-27 10:29:16 +00:00
self.q = q or self.defaults["q"]
self.user = user or self.defaults["user"]
self.sort = sort or self.defaults["sort"]
self.shared = shared or self.defaults["shared"]
self.unread = unread or self.defaults["unread"]
def is_modified(self, param):
value = self.__dict__[param]
return value != self.defaults[param]
@property
def modified_params(self):
return [field for field in self.params if self.is_modified(field)]
@property
def modified_preferences(self):
2024-01-27 10:29:16 +00:00
return [
preference
for preference in self.preferences
if self.is_modified(preference)
]
@property
def has_modifications(self):
return len(self.modified_params) > 0
@property
def has_modified_preferences(self):
return len(self.modified_preferences) > 0
@property
def query_params(self):
return {param: self.__dict__[param] for param in self.modified_params}
@property
def preferences_dict(self):
2024-01-27 10:29:16 +00:00
return {
preference: self.__dict__[preference] for preference in self.preferences
}
@staticmethod
def from_request(query_dict: QueryDict, preferences: dict = None):
initial_values = {}
for param in BookmarkSearch.params:
value = query_dict.get(param)
if value:
initial_values[param] = value
return BookmarkSearch(**initial_values, preferences=preferences)
class BookmarkSearchForm(forms.Form):
SORT_CHOICES = [
2024-01-27 10:29:16 +00:00
(BookmarkSearch.SORT_ADDED_ASC, "Added ↑"),
(BookmarkSearch.SORT_ADDED_DESC, "Added ↓"),
(BookmarkSearch.SORT_TITLE_ASC, "Title ↑"),
(BookmarkSearch.SORT_TITLE_DESC, "Title ↓"),
]
FILTER_SHARED_CHOICES = [
2024-01-27 10:29:16 +00:00
(BookmarkSearch.FILTER_SHARED_OFF, "Off"),
(BookmarkSearch.FILTER_SHARED_SHARED, "Shared"),
(BookmarkSearch.FILTER_SHARED_UNSHARED, "Unshared"),
]
2023-09-16 08:39:27 +00:00
FILTER_UNREAD_CHOICES = [
2024-01-27 10:29:16 +00:00
(BookmarkSearch.FILTER_UNREAD_OFF, "Off"),
(BookmarkSearch.FILTER_UNREAD_YES, "Unread"),
(BookmarkSearch.FILTER_UNREAD_NO, "Read"),
2023-09-16 08:39:27 +00:00
]
q = forms.CharField()
user = forms.ChoiceField()
sort = forms.ChoiceField(choices=SORT_CHOICES)
shared = forms.ChoiceField(choices=FILTER_SHARED_CHOICES, widget=forms.RadioSelect)
2023-09-16 08:39:27 +00:00
unread = forms.ChoiceField(choices=FILTER_UNREAD_CHOICES, widget=forms.RadioSelect)
2024-01-27 10:29:16 +00:00
def __init__(
self,
search: BookmarkSearch,
editable_fields: List[str] = None,
users: List[User] = None,
):
super().__init__()
editable_fields = editable_fields or []
self.editable_fields = editable_fields
# set choices for user field if users are provided
if users:
user_choices = [(user.username, user.username) for user in users]
2024-01-27 10:29:16 +00:00
user_choices.insert(0, ("", "Everyone"))
self.fields["user"].choices = user_choices
for param in search.params:
# set initial values for modified params
self.fields[param].initial = search.__dict__[param]
# Mark non-editable modified fields as hidden. That way, templates
# rendering a form can just loop over hidden_fields to ensure that
# all necessary search options are kept when submitting the form.
if search.is_modified(param) and param not in editable_fields:
self.fields[param].widget = forms.HiddenInput()
2021-03-28 10:11:56 +00:00
class UserProfile(models.Model):
2024-01-27 10:29:16 +00:00
THEME_AUTO = "auto"
THEME_LIGHT = "light"
THEME_DARK = "dark"
2021-03-28 10:11:56 +00:00
THEME_CHOICES = [
2024-01-27 10:29:16 +00:00
(THEME_AUTO, "Auto"),
(THEME_LIGHT, "Light"),
(THEME_DARK, "Dark"),
2021-03-28 10:11:56 +00:00
]
2024-01-27 10:29:16 +00:00
BOOKMARK_DATE_DISPLAY_RELATIVE = "relative"
BOOKMARK_DATE_DISPLAY_ABSOLUTE = "absolute"
BOOKMARK_DATE_DISPLAY_HIDDEN = "hidden"
BOOKMARK_DATE_DISPLAY_CHOICES = [
2024-01-27 10:29:16 +00:00
(BOOKMARK_DATE_DISPLAY_RELATIVE, "Relative"),
(BOOKMARK_DATE_DISPLAY_ABSOLUTE, "Absolute"),
(BOOKMARK_DATE_DISPLAY_HIDDEN, "Hidden"),
]
BOOKMARK_DESCRIPTION_DISPLAY_INLINE = "inline"
BOOKMARK_DESCRIPTION_DISPLAY_SEPARATE = "separate"
BOOKMARK_DESCRIPTION_DISPLAY_CHOICES = [
(BOOKMARK_DESCRIPTION_DISPLAY_INLINE, "Inline"),
(BOOKMARK_DESCRIPTION_DISPLAY_SEPARATE, "Separate"),
]
2024-01-27 10:29:16 +00:00
BOOKMARK_LINK_TARGET_BLANK = "_blank"
BOOKMARK_LINK_TARGET_SELF = "_self"
BOOKMARK_LINK_TARGET_CHOICES = [
2024-01-27 10:29:16 +00:00
(BOOKMARK_LINK_TARGET_BLANK, "New page"),
(BOOKMARK_LINK_TARGET_SELF, "Same page"),
]
2024-01-27 10:29:16 +00:00
WEB_ARCHIVE_INTEGRATION_DISABLED = "disabled"
WEB_ARCHIVE_INTEGRATION_ENABLED = "enabled"
WEB_ARCHIVE_INTEGRATION_CHOICES = [
2024-01-27 10:29:16 +00:00
(WEB_ARCHIVE_INTEGRATION_DISABLED, "Disabled"),
(WEB_ARCHIVE_INTEGRATION_ENABLED, "Enabled"),
]
2024-01-27 10:29:16 +00:00
TAG_SEARCH_STRICT = "strict"
TAG_SEARCH_LAX = "lax"
TAG_SEARCH_CHOICES = [
2024-01-27 10:29:16 +00:00
(TAG_SEARCH_STRICT, "Strict"),
(TAG_SEARCH_LAX, "Lax"),
]
2024-01-27 10:29:16 +00:00
user = models.OneToOneField(
get_user_model(), related_name="profile", on_delete=models.CASCADE
)
theme = models.CharField(
max_length=10, choices=THEME_CHOICES, blank=False, default=THEME_AUTO
)
bookmark_date_display = models.CharField(
max_length=10,
choices=BOOKMARK_DATE_DISPLAY_CHOICES,
blank=False,
default=BOOKMARK_DATE_DISPLAY_RELATIVE,
)
bookmark_description_display = models.CharField(
max_length=10,
choices=BOOKMARK_DESCRIPTION_DISPLAY_CHOICES,
blank=False,
default=BOOKMARK_DESCRIPTION_DISPLAY_INLINE,
)
bookmark_description_max_lines = models.IntegerField(
null=False,
default=1,
)
2024-01-27 10:29:16 +00:00
bookmark_link_target = models.CharField(
max_length=10,
choices=BOOKMARK_LINK_TARGET_CHOICES,
blank=False,
default=BOOKMARK_LINK_TARGET_BLANK,
)
web_archive_integration = models.CharField(
max_length=10,
choices=WEB_ARCHIVE_INTEGRATION_CHOICES,
blank=False,
default=WEB_ARCHIVE_INTEGRATION_DISABLED,
)
tag_search = models.CharField(
max_length=10,
choices=TAG_SEARCH_CHOICES,
blank=False,
default=TAG_SEARCH_STRICT,
)
enable_sharing = models.BooleanField(default=False, null=False)
enable_public_sharing = models.BooleanField(default=False, null=False)
enable_favicons = models.BooleanField(default=False, null=False)
display_url = models.BooleanField(default=False, null=False)
display_view_bookmark_action = models.BooleanField(default=True, null=False)
display_edit_bookmark_action = models.BooleanField(default=True, null=False)
display_archive_bookmark_action = models.BooleanField(default=True, null=False)
display_remove_bookmark_action = models.BooleanField(default=True, null=False)
permanent_notes = models.BooleanField(default=False, null=False)
custom_css = models.TextField(blank=True, null=False)
search_preferences = models.JSONField(default=dict, null=False)
2021-03-28 10:11:56 +00:00
class UserProfileForm(forms.ModelForm):
class Meta:
model = UserProfile
2024-01-27 10:29:16 +00:00
fields = [
"theme",
"bookmark_date_display",
"bookmark_description_display",
"bookmark_description_max_lines",
2024-01-27 10:29:16 +00:00
"bookmark_link_target",
"web_archive_integration",
"tag_search",
"enable_sharing",
"enable_public_sharing",
"enable_favicons",
"display_url",
"display_view_bookmark_action",
"display_edit_bookmark_action",
"display_archive_bookmark_action",
"display_remove_bookmark_action",
2024-01-27 10:29:16 +00:00
"permanent_notes",
"custom_css",
2024-01-27 10:29:16 +00:00
]
2021-03-28 10:11:56 +00:00
@receiver(post_save, sender=get_user_model())
def create_user_profile(sender, instance, created, **kwargs):
if created:
UserProfile.objects.create(user=instance)
@receiver(post_save, sender=get_user_model())
def save_user_profile(sender, instance, **kwargs):
instance.profile.save()
class Toast(models.Model):
key = models.CharField(max_length=50)
message = models.TextField()
acknowledged = models.BooleanField(default=False)
owner = models.ForeignKey(get_user_model(), on_delete=models.CASCADE)
class FeedToken(models.Model):
"""
Adapted from authtoken.models.Token
"""
2024-01-27 10:29:16 +00:00
key = models.CharField(max_length=40, primary_key=True)
2024-01-27 10:29:16 +00:00
user = models.OneToOneField(
get_user_model(),
related_name="feed_token",
on_delete=models.CASCADE,
)
created = models.DateTimeField(auto_now_add=True)
def save(self, *args, **kwargs):
if not self.key:
self.key = self.generate_key()
return super().save(*args, **kwargs)
@classmethod
def generate_key(cls):
return binascii.hexlify(os.urandom(20)).decode()
def __str__(self):
return self.key