Skip to content

Commit

Permalink
Merge pull request #12 from krzysztofjeziorny/master
Browse files Browse the repository at this point in the history
1.32
  • Loading branch information
krzysztofjeziorny authored Sep 12, 2024
2 parents 77b5aeb + b736464 commit c2b0672
Show file tree
Hide file tree
Showing 8 changed files with 77 additions and 39 deletions.
12 changes: 11 additions & 1 deletion bookmarks/api/routes.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import logging

from rest_framework import viewsets, mixins, status
from rest_framework.decorators import action
from rest_framework.permissions import AllowAny
Expand All @@ -19,6 +21,8 @@
)
from bookmarks.services.website_loader import WebsiteMetadata

logger = logging.getLogger(__name__)


class BookmarkViewSet(
viewsets.GenericViewSet,
Expand Down Expand Up @@ -112,7 +116,13 @@ def check(self, request):
profile = request.user.profile
auto_tags = []
if profile.auto_tagging_rules:
auto_tags = auto_tagging.get_tags(profile.auto_tagging_rules, url)
try:
auto_tags = auto_tagging.get_tags(profile.auto_tagging_rules, url)
except Exception as e:
logger.error(
f"Failed to auto-tag bookmark. url={bookmark.url}",
exc_info=e,
)

return Response(
{
Expand Down
3 changes: 2 additions & 1 deletion bookmarks/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,8 @@ def resolved_description(self):

@property
def tag_names(self):
return [tag.name for tag in self.tags.all()]
names = [tag.name for tag in self.tags.all()]
return sorted(names)

def __str__(self):
return self.resolved_title + " (" + self.url[:30] + "...)"
Expand Down
26 changes: 10 additions & 16 deletions bookmarks/services/auto_tagging.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,27 +16,21 @@ def get_tags(script: str, url: str):
if len(parts) < 2:
continue

domain_pattern = re.sub("^https?://", "", parts[0])
path_pattern = None
qs_pattern = None
# to parse a host name from the pattern URL, ensure it has a scheme
pattern_url = "//" + re.sub("^https?://", "", parts[0])
parsed_pattern = urlparse(pattern_url)

if "/" in domain_pattern:
i = domain_pattern.index("/")
path_pattern = domain_pattern[i:]
domain_pattern = domain_pattern[:i]

if path_pattern and "?" in path_pattern:
i = path_pattern.index("?")
qs_pattern = path_pattern[i + 1 :]
path_pattern = path_pattern[:i]

if not _domains_matches(domain_pattern, parsed_url.netloc):
if not _domains_matches(parsed_pattern.hostname, parsed_url.hostname):
continue

if path_pattern and not _path_matches(path_pattern, parsed_url.path):
if parsed_pattern.path and not _path_matches(
parsed_pattern.path, parsed_url.path
):
continue

if qs_pattern and not _qs_matches(qs_pattern, parsed_url.query):
if parsed_pattern.query and not _qs_matches(
parsed_pattern.query, parsed_url.query
):
continue

for tag in parts[1:]:
Expand Down
18 changes: 12 additions & 6 deletions bookmarks/services/bookmarks.py
Original file line number Diff line number Diff line change
Expand Up @@ -245,12 +245,18 @@ def _update_bookmark_tags(bookmark: Bookmark, tag_string: str, user: User):
tag_names = parse_tag_string(tag_string)

if user.profile.auto_tagging_rules:
auto_tag_names = auto_tagging.get_tags(
user.profile.auto_tagging_rules, bookmark.url
)
for auto_tag_name in auto_tag_names:
if auto_tag_name not in tag_names:
tag_names.append(auto_tag_name)
try:
auto_tag_names = auto_tagging.get_tags(
user.profile.auto_tagging_rules, bookmark.url
)
for auto_tag_name in auto_tag_names:
if auto_tag_name not in tag_names:
tag_names.append(auto_tag_name)
except Exception as e:
logger.error(
f"Failed to auto-tag bookmark. url={bookmark.url}",
exc_info=e,
)

tags = get_or_create_tags(tag_names, user)
bookmark.tags.set(tags)
Expand Down
37 changes: 24 additions & 13 deletions bookmarks/tests/test_auto_tagging.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,18 @@ def test_auto_tag_by_domain(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["example"]))
self.assertEqual(tags, {"example"})

def test_auto_tag_by_domain_works_with_port(self):
script = """
example.com example
test.com test
"""
url = "https://example.com:8080/"

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, {"example"})

def test_auto_tag_by_domain_ignores_case(self):
script = """
Expand All @@ -22,7 +33,7 @@ def test_auto_tag_by_domain_ignores_case(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["example"]))
self.assertEqual(tags, {"example"})

def test_auto_tag_by_domain_should_add_all_tags(self):
script = """
Expand All @@ -32,7 +43,7 @@ def test_auto_tag_by_domain_should_add_all_tags(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["one", "two", "three"]))
self.assertEqual(tags, {"one", "two", "three"})

def test_auto_tag_by_domain_work_with_idn_domains(self):
script = """
Expand All @@ -42,7 +53,7 @@ def test_auto_tag_by_domain_work_with_idn_domains(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["tag1"]))
self.assertEqual(tags, {"tag1"})

script = """
xn--81bg3cc2b2bk5hb.xn--h2brj9c tag1
Expand All @@ -51,7 +62,7 @@ def test_auto_tag_by_domain_work_with_idn_domains(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["tag1"]))
self.assertEqual(tags, {"tag1"})

def test_auto_tag_by_domain_and_path(self):
script = """
Expand All @@ -63,7 +74,7 @@ def test_auto_tag_by_domain_and_path(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["one"]))
self.assertEqual(tags, {"one"})

def test_auto_tag_by_domain_and_path_ignores_case(self):
script = """
Expand All @@ -73,7 +84,7 @@ def test_auto_tag_by_domain_and_path_ignores_case(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["one"]))
self.assertEqual(tags, {"one"})

def test_auto_tag_by_domain_and_path_matches_path_ltr(self):
script = """
Expand All @@ -85,7 +96,7 @@ def test_auto_tag_by_domain_and_path_matches_path_ltr(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["one"]))
self.assertEqual(tags, {"one"})

def test_auto_tag_by_domain_ignores_domain_in_path(self):
script = """
Expand All @@ -107,7 +118,7 @@ def test_auto_tag_by_domain_includes_subdomains(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["example", "test"]))
self.assertEqual(tags, {"example", "test"})

def test_auto_tag_by_domain_matches_domain_rtl(self):
script = """
Expand All @@ -128,7 +139,7 @@ def test_auto_tag_by_domain_ignores_schema(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["https", "http"]))
self.assertEqual(tags, {"https", "http"})

def test_auto_tag_by_domain_ignores_lines_with_no_tags(self):
script = """
Expand All @@ -154,7 +165,7 @@ def test_auto_tag_by_domain_path_and_qs(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["tag1", "tag2", "tag5", "tag6", "tag7"]))
self.assertEqual(tags, {"tag1", "tag2", "tag5", "tag6", "tag7"})

def test_auto_tag_by_domain_path_and_qs_with_empty_value(self):
script = """
Expand All @@ -165,7 +176,7 @@ def test_auto_tag_by_domain_path_and_qs_with_empty_value(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["tag1"]))
self.assertEqual(tags, {"tag1"})

def test_auto_tag_by_domain_path_and_qs_works_with_encoded_url(self):
script = """
Expand All @@ -176,4 +187,4 @@ def test_auto_tag_by_domain_path_and_qs_works_with_encoded_url(self):

tags = auto_tagging.get_tags(script, url)

self.assertEqual(tags, set(["tag1", "tag2"]))
self.assertEqual(tags, {"tag1", "tag2"})
16 changes: 16 additions & 0 deletions bookmarks/tests/test_bookmarks_list_template.py
Original file line number Diff line number Diff line change
Expand Up @@ -457,6 +457,22 @@ def test_bookmark_description_max_lines(self):
style = bookmark_list["style"]
self.assertIn("--ld-bookmark-description-max-lines:3;", style)

def test_bookmark_tag_ordering(self):
bookmark = self.setup_bookmark()
tag3 = self.setup_tag(name="tag3")
tag1 = self.setup_tag(name="tag1")
tag2 = self.setup_tag(name="tag2")
bookmark.tags.add(tag3, tag1, tag2)

html = self.render_template()
soup = self.make_soup(html)
tags = soup.select_one(".tags")
tag_links = tags.find_all("a")
self.assertEqual(len(tag_links), 3)
self.assertEqual(tag_links[0].text, "#tag1")
self.assertEqual(tag_links[1].text, "#tag2")
self.assertEqual(tag_links[2].text, "#tag3")

def test_should_render_web_archive_link_with_absolute_date_setting(self):
bookmark = self.setup_date_format_test(
UserProfile.BOOKMARK_DATE_DISPLAY_ABSOLUTE,
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "linkding",
"version": "1.31.1",
"version": "1.32.0",
"description": "",
"main": "index.js",
"scripts": {
Expand Down
2 changes: 1 addition & 1 deletion version.txt
Original file line number Diff line number Diff line change
@@ -1 +1 @@
1.31.1
1.32.0

0 comments on commit c2b0672

Please sign in to comment.