mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-07 07:45:28 +00:00
Merge branch 'main' into production
This commit is contained in:
commit
97f050c68e
9 changed files with 98 additions and 22 deletions
|
@ -251,6 +251,9 @@ def handle_delete_status(activity):
|
||||||
def handle_favorite(activity):
|
def handle_favorite(activity):
|
||||||
''' approval of your good good post '''
|
''' approval of your good good post '''
|
||||||
fav = activitypub.Like(**activity)
|
fav = activitypub.Like(**activity)
|
||||||
|
# we dont know this status, we don't care about this status
|
||||||
|
if not models.Status.objects.filter(remote_id=fav.object).exists():
|
||||||
|
return
|
||||||
|
|
||||||
fav = fav.to_model(models.Favorite)
|
fav = fav.to_model(models.Favorite)
|
||||||
if fav.user.local:
|
if fav.user.local:
|
||||||
|
|
|
@ -293,13 +293,17 @@ def find_mentions(content):
|
||||||
yield (match.group(), mention_user)
|
yield (match.group(), mention_user)
|
||||||
|
|
||||||
|
|
||||||
def to_markdown(content):
|
def format_links(content):
|
||||||
''' catch links and convert to markdown '''
|
''' detect and format links '''
|
||||||
content = re.sub(
|
return re.sub(
|
||||||
r'([^(href=")])(https?:\/\/([A-Za-z\.\-_\/]+' \
|
r'([^(href=")]|^)(https?:\/\/(%s([\w\.\-_\/+&\?=:;,])*))' % \
|
||||||
r'\.[A-Za-z]{2,}[A-Za-z\.\-_\/]+))',
|
regex.domain,
|
||||||
r'\g<1><a href="\g<2>">\g<3></a>',
|
r'\g<1><a href="\g<2>">\g<3></a>',
|
||||||
content)
|
content)
|
||||||
|
|
||||||
|
def to_markdown(content):
|
||||||
|
''' catch links and convert to markdown '''
|
||||||
|
content = format_links(content)
|
||||||
content = markdown(content)
|
content = markdown(content)
|
||||||
# sanitize resulting html
|
# sanitize resulting html
|
||||||
sanitizer = InputHtmlParser()
|
sanitizer = InputHtmlParser()
|
||||||
|
|
|
@ -3,7 +3,8 @@
|
||||||
|
|
||||||
<div class="columns">
|
<div class="columns">
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<div class="block login">
|
<div class="block">
|
||||||
|
{% if valid %}
|
||||||
<h1 class="title">Create an Account</h1>
|
<h1 class="title">Create an Account</h1>
|
||||||
<div>
|
<div>
|
||||||
<form name="register" method="post" action="/user-register">
|
<form name="register" method="post" action="/user-register">
|
||||||
|
@ -11,6 +12,12 @@
|
||||||
{% include 'snippets/register_form.html' %}
|
{% include 'snippets/register_form.html' %}
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="content">
|
||||||
|
<h1 class="title">Permission Denied</h1>
|
||||||
|
<p>Sorry! This invite code is no longer valid.</p>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="column">
|
<div class="column">
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
{% if status.status_type == 'Review' %}
|
{% if status.status_type == 'Review' %}
|
||||||
<div>
|
<div>
|
||||||
<h3 class="title is-5 has-subtitle">
|
<h3 class="title is-5 has-subtitle">
|
||||||
{% if status.name %}{{ status.name }}<br>{% endif %}
|
{% if status.name %}<span dir="auto">{{ status.name }}</span><br>{% endif %}
|
||||||
</h3>
|
</h3>
|
||||||
<p class="subtitle">{% include 'snippets/stars.html' with rating=status.rating %}</p>
|
<p class="subtitle">{% include 'snippets/stars.html' with rating=status.rating %}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -27,7 +27,7 @@
|
||||||
|
|
||||||
{% if status.quote %}
|
{% if status.quote %}
|
||||||
<div class="quote block">
|
<div class="quote block">
|
||||||
<blockquote>{{ status.quote | safe }}</blockquote>
|
<blockquote dir="auto">{{ status.quote | safe }}</blockquote>
|
||||||
|
|
||||||
<p> — {% include 'snippets/book_titleby.html' with book=status.book %}</p>
|
<p> — {% include 'snippets/book_titleby.html' with book=status.book %}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -6,18 +6,18 @@
|
||||||
{% if trimmed != full %}
|
{% if trimmed != full %}
|
||||||
<div>
|
<div>
|
||||||
<input type="radio" name="show-hide-{{ uuid }}" id="show-{{ uuid }}" class="toggle-control" checked>
|
<input type="radio" name="show-hide-{{ uuid }}" id="show-{{ uuid }}" class="toggle-control" checked>
|
||||||
<blockquote class="content toggle-content hidden">{{ trimmed | to_markdown | safe }}
|
<blockquote class="content toggle-content hidden"><span dir="auto">{{ trimmed | to_markdown | safe }}</span>
|
||||||
<label class="button is-small" for="hide-{{ uuid }}"><div role="button" tabindex="0">show more</div></label>
|
<label class="button is-small" for="hide-{{ uuid }}"><div role="button" tabindex="0">show more</div></label>
|
||||||
</blockquote>
|
</blockquote>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<input type="radio" name="show-hide-{{ uuid }}" id="hide-{{ uuid }}" class="toggle-control">
|
<input type="radio" name="show-hide-{{ uuid }}" id="hide-{{ uuid }}" class="toggle-control">
|
||||||
<blockquote class="content toggle-content hidden">{{ full | to_markdown | safe }}
|
<blockquote class="content toggle-content hidden"><span dir="auto">{{ full | to_markdown | safe }}</span>
|
||||||
<label class="button is-small" for="show-{{ uuid }}"><div role="button" tabindex="0">show less</div></label>
|
<label class="button is-small" for="show-{{ uuid }}"><div role="button" tabindex="0">show less</div></label>
|
||||||
</blockquote>
|
</blockquote>
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
<blockquote class="content">{{ full | to_markdown | safe }}</blockquote>
|
<blockquote class="content"><span dir="auto">{{ full | to_markdown | safe }}</span></blockquote>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endwith %}
|
{% endwith %}
|
||||||
|
|
||||||
|
|
|
@ -447,3 +447,65 @@ class Outgoing(TestCase):
|
||||||
self.assertEqual(reply.user, user)
|
self.assertEqual(reply.user, user)
|
||||||
self.assertTrue(self.remote_user in reply.mention_users.all())
|
self.assertTrue(self.remote_user in reply.mention_users.all())
|
||||||
self.assertTrue(self.local_user in reply.mention_users.all())
|
self.assertTrue(self.local_user in reply.mention_users.all())
|
||||||
|
|
||||||
|
def test_find_mentions(self):
|
||||||
|
''' detect and look up @ mentions of users '''
|
||||||
|
user = models.User.objects.create_user(
|
||||||
|
'nutria@%s' % DOMAIN, 'nutria@nutria.com', 'password',
|
||||||
|
local=True, localname='nutria')
|
||||||
|
self.assertEqual(user.username, 'nutria@%s' % DOMAIN)
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('@nutria'))[0],
|
||||||
|
('@nutria', user)
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('leading text @nutria'))[0],
|
||||||
|
('@nutria', user)
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('leading @nutria trailing text'))[0],
|
||||||
|
('@nutria', user)
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('@rat@example.com'))[0],
|
||||||
|
('@rat@example.com', self.remote_user)
|
||||||
|
)
|
||||||
|
|
||||||
|
multiple = list(outgoing.find_mentions('@nutria and @rat@example.com'))
|
||||||
|
self.assertEqual(multiple[0], ('@nutria', user))
|
||||||
|
self.assertEqual(multiple[1], ('@rat@example.com', self.remote_user))
|
||||||
|
|
||||||
|
with patch('bookwyrm.outgoing.handle_remote_webfinger') as rw:
|
||||||
|
rw.return_value = self.local_user
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('@beep@beep.com'))[0],
|
||||||
|
('@beep@beep.com', self.local_user)
|
||||||
|
)
|
||||||
|
with patch('bookwyrm.outgoing.handle_remote_webfinger') as rw:
|
||||||
|
rw.return_value = None
|
||||||
|
self.assertEqual(list(outgoing.find_mentions('@beep@beep.com')), [])
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
list(outgoing.find_mentions('@nutria@%s' % DOMAIN))[0],
|
||||||
|
('@nutria@%s' % DOMAIN, user)
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_format_links(self):
|
||||||
|
''' find and format urls into a tags '''
|
||||||
|
url = 'http://www.fish.com/'
|
||||||
|
self.assertEqual(
|
||||||
|
outgoing.format_links(url),
|
||||||
|
'<a href="%s">www.fish.com/</a>' % url)
|
||||||
|
url = 'https://archive.org/details/dli.granth.72113/page/n25/mode/2up'
|
||||||
|
self.assertEqual(
|
||||||
|
outgoing.format_links(url),
|
||||||
|
'<a href="%s">' \
|
||||||
|
'archive.org/details/dli.granth.72113/page/n25/mode/2up</a>' \
|
||||||
|
% url)
|
||||||
|
url = 'https://openlibrary.org/search' \
|
||||||
|
'?q=arkady+strugatsky&mode=everything'
|
||||||
|
self.assertEqual(
|
||||||
|
outgoing.format_links(url),
|
||||||
|
'<a href="%s">openlibrary.org/search' \
|
||||||
|
'?q=arkady+strugatsky&mode=everything</a>' % url)
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
''' defining regexes for regularly used concepts '''
|
''' defining regexes for regularly used concepts '''
|
||||||
|
|
||||||
domain = r'[a-z-A-Z0-9_\-]+\.[a-z]+'
|
domain = r'[\w_\-\.]+\.[a-z]{2,}'
|
||||||
localname = r'@?[a-zA-Z_\-\.0-9]+'
|
localname = r'@?[a-zA-Z_\-\.0-9]+\b'
|
||||||
strict_localname = r'@[a-zA-Z_\-\.0-9]+'
|
strict_localname = r'@[a-zA-Z_\-\.0-9]+'
|
||||||
username = r'%s(@%s)?' % (localname, domain)
|
username = r'%s(@%s)?' % (localname, domain)
|
||||||
strict_username = r'%s(@%s)?' % (strict_localname, domain)
|
strict_username = r'\B%s(@%s)?\b' % (strict_localname, domain)
|
||||||
full_username = r'%s@%s' % (localname, domain)
|
full_username = r'\B%s@%s\b' % (localname, domain)
|
||||||
# should match (BookWyrm/1.0.0; or (BookWyrm/99.1.2;
|
# should match (BookWyrm/1.0.0; or (BookWyrm/99.1.2;
|
||||||
bookwyrm_user_agent = r'\(BookWyrm/[0-9]+\.[0-9]+\.[0-9]+;'
|
bookwyrm_user_agent = r'\(BookWyrm/[0-9]+\.[0-9]+\.[0-9]+;'
|
||||||
|
|
|
@ -81,8 +81,12 @@ def register(request):
|
||||||
if errors:
|
if errors:
|
||||||
data = {
|
data = {
|
||||||
'login_form': forms.LoginForm(),
|
'login_form': forms.LoginForm(),
|
||||||
'register_form': form
|
'register_form': form,
|
||||||
|
'invite': invite,
|
||||||
|
'valid': invite.valid() if invite else True,
|
||||||
}
|
}
|
||||||
|
if invite:
|
||||||
|
return TemplateResponse(request, 'invite.html', data)
|
||||||
return TemplateResponse(request, 'login.html', data)
|
return TemplateResponse(request, 'login.html', data)
|
||||||
|
|
||||||
username = '%s@%s' % (localname, DOMAIN)
|
username = '%s@%s' % (localname, DOMAIN)
|
||||||
|
|
|
@ -377,17 +377,13 @@ def invite_page(request, code):
|
||||||
''' endpoint for sending invites '''
|
''' endpoint for sending invites '''
|
||||||
if request.user.is_authenticated:
|
if request.user.is_authenticated:
|
||||||
return redirect('/')
|
return redirect('/')
|
||||||
try:
|
invite = get_object_or_404(models.SiteInvite, code=code)
|
||||||
invite = models.SiteInvite.objects.get(code=code)
|
|
||||||
if not invite.valid():
|
|
||||||
raise PermissionDenied
|
|
||||||
except models.SiteInvite.DoesNotExist:
|
|
||||||
raise PermissionDenied
|
|
||||||
|
|
||||||
data = {
|
data = {
|
||||||
'title': 'Join',
|
'title': 'Join',
|
||||||
'register_form': forms.RegisterForm(),
|
'register_form': forms.RegisterForm(),
|
||||||
'invite': invite,
|
'invite': invite,
|
||||||
|
'valid': invite.valid() if invite else True,
|
||||||
}
|
}
|
||||||
return TemplateResponse(request, 'invite.html', data)
|
return TemplateResponse(request, 'invite.html', data)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue