diff --git a/fedireads/activity.py b/fedireads/activity.py index 1099ec7c..2e4a80c1 100644 --- a/fedireads/activity.py +++ b/fedireads/activity.py @@ -8,7 +8,6 @@ from uuid import uuid4 from fedireads import models from fedireads.openlibrary import get_or_create_book from fedireads.sanitize_html import InputHtmlParser -from fedireads.settings import DOMAIN def create_review(user, possible_book, name, content, rating): @@ -54,6 +53,16 @@ def create_status(user, content, reply_parent=None, mention_books=None): return status +def get_review_json(review): + ''' fedireads json for book reviews ''' + status = get_status_json(review) + status['inReplyTo'] = review.book.absolute_id + status['fedireadsType'] = review.status_type, + status['name'] = review.name + status['rating'] = review.rating + return status + + def get_status_json(status): ''' create activitypub json for a status ''' user = status.user @@ -67,11 +76,10 @@ def get_status_json(status): 'attributedTo': user.actor, # TODO: assuming all posts are public -- should check privacy db field 'to': ['https://www.w3.org/ns/activitystreams#Public'], - 'cc': ['https://%s/user/%s/followers' % (DOMAIN, user.localname)], + 'cc': ['%s/followers' % user.absolute_id], 'sensitive': status.sensitive, 'content': status.content, 'type': status.activity_type, - 'fedireadsType': status.status_type, 'attachment': [], # TODO: the book cover 'replies': { 'id': '%s/replies' % uri, @@ -85,11 +93,6 @@ def get_status_json(status): } } - if status.status_type == 'Review': - status_json['name'] = status.name, - status_json['rating'] = status.rating - status_json['fedireadsType'] = status.status_type - return status_json @@ -112,7 +115,7 @@ def get_create_json(user, status_json): 'object': status_json, 'signature': { 'type': 'RsaSignature2017', - 'creator': 'https://%s/user/%s#main-key' % (DOMAIN, user.localname), + 'creator': '%s#main-key' % user.absolute_id, 'created': status_json['published'], 'signatureValue': b64encode(signed_message).decode('utf8'), } @@ -146,8 +149,7 @@ def get_add_remove_json(user, book, shelf, action='Add'): 'target': { 'type': 'Collection', 'name': shelf.name, - 'id': 'https://%s/user/%s/shelf/%s' % \ - (DOMAIN, user.localname, shelf.identifier) + 'id': shelf.absolute_id, } } diff --git a/fedireads/incoming.py b/fedireads/incoming.py index 50029e1f..5d442c8e 100644 --- a/fedireads/incoming.py +++ b/fedireads/incoming.py @@ -11,7 +11,7 @@ import requests from fedireads import models from fedireads import outgoing -from fedireads.activity import create_review, create_status +from fedireads.activity import create_review, create_status, get_status_json from fedireads.remote_user import get_or_create_remote_user @@ -151,7 +151,7 @@ def get_status(request, username, status_id): if user != status.user: return HttpResponseNotFound() - return JsonResponse(status.activity) + return JsonResponse(get_status_json(status)) @csrf_exempt diff --git a/fedireads/models/book.py b/fedireads/models/book.py index 8beb382b..c067db07 100644 --- a/fedireads/models/book.py +++ b/fedireads/models/book.py @@ -18,6 +18,13 @@ class Shelf(FedireadsModel): through_fields=('shelf', 'book') ) + @property + def absolute_id(self): + ''' use shelf identifier as absolute id ''' + base_path = self.user.absolute_id + model_name = type(self).__name__.lower() + return '%s/%s/%s' % (base_path, model_name, self.identifier) + class Meta: unique_together = ('user', 'identifier') @@ -61,8 +68,8 @@ class Book(FedireadsModel): def absolute_id(self): ''' constructs the absolute reference to any db object ''' base_path = 'https://%s' % DOMAIN - model_name = self.__name__.lower() - return '%s/%s/%d' % (base_path, model_name, self.openlibrary_key) + model_name = type(self).__name__.lower() + return '%s/%s/%s' % (base_path, model_name, self.openlibrary_key) class Author(FedireadsModel): diff --git a/fedireads/models/user.py b/fedireads/models/user.py index a7b0ca11..977c1553 100644 --- a/fedireads/models/user.py +++ b/fedireads/models/user.py @@ -38,7 +38,7 @@ class User(AbstractUser): @property def absolute_id(self): ''' users are identified by their username, so overriding this prop ''' - model_name = type(self).__name__ + model_name = type(self).__name__.lower() return 'https://%s/%s/%s' % (DOMAIN, model_name, self.localname) @@ -61,10 +61,10 @@ def execute_before_save(sender, instance, *args, **kwargs): # populate fields for local users instance.localname = instance.username instance.username = '%s@%s' % (instance.username, DOMAIN) - instance.actor = 'https://%s/user/%s' % (DOMAIN, instance.localname) - instance.inbox = 'https://%s/user/%s/inbox' % (DOMAIN, instance.localname) + instance.actor = instance.absolute_id + instance.inbox = '%s/inbox' % instance.absolute_id instance.shared_inbox = 'https://%s/inbox' % DOMAIN - instance.outbox = 'https://%s/user/%s/outbox' % (DOMAIN, instance.localname) + instance.outbox = '%s/outbox' % instance.absolute_id if not instance.private_key: random_generator = Random.new().read key = RSA.generate(1024, random_generator) diff --git a/fedireads/outgoing.py b/fedireads/outgoing.py index b8587ad7..9fe68ee0 100644 --- a/fedireads/outgoing.py +++ b/fedireads/outgoing.py @@ -6,7 +6,8 @@ from urllib.parse import urlencode from uuid import uuid4 from fedireads import models -from fedireads.activity import create_review, create_status, get_status_json +from fedireads.activity import create_review, create_status +from fedireads.activity import get_status_json, get_review_json from fedireads.activity import get_add_json, get_remove_json, get_create_json from fedireads.remote_user import get_or_create_remote_user from fedireads.broadcast import get_recipients, broadcast @@ -112,7 +113,7 @@ def handle_outgoing_accept(user, to_follow, activity): to_follow.followers.add(user) activity = { '@context': 'https://www.w3.org/ns/activitystreams', - 'id': 'https://%s/%s#accepts/follows/' % (DOMAIN, to_follow.localname), + 'id': '%s#accepts/follows/' % to_follow.absolute_id, 'type': 'Accept', 'actor': to_follow.actor, 'object': activity, @@ -169,9 +170,7 @@ def handle_review(user, book, name, content, rating): # validated and saves the review in the database so it has an id review = create_review(user, book, name, content, rating) - #book_path = 'https://%s/book/%s' % (DOMAIN, review.book.openlibrary_key) - - review_activity = get_status_json(review) + review_activity = get_review_json(review) create_activity = get_create_json(user, review_activity) recipients = get_recipients(user, 'public') diff --git a/fedireads/templates/user.html b/fedireads/templates/user.html index f81f60f7..6093fe03 100644 --- a/fedireads/templates/user.html +++ b/fedireads/templates/user.html @@ -92,7 +92,7 @@ - {{ book.data.title }} + {{ book.data.title }} {{ book.authors.first.data.name }} diff --git a/fedireads/urls.py b/fedireads/urls.py index b7312d0d..a601b644 100644 --- a/fedireads/urls.py +++ b/fedireads/urls.py @@ -17,14 +17,14 @@ urlpatterns = [ re_path(r'^user/(?P\w+)/followers/?$', incoming.get_followers), re_path(r'^user/(?P\w+)/following/?$', incoming.get_following), re_path( - r'^user/(?P\w+)/status/(?P\d+)/?$', + r'^user/(?P\w+)/(status|review)/(?P\d+)/?$', incoming.get_status ), re_path( - r'^user/(?P\w+)/status/(?P\d+)/activity/?$', + r'^user/(?P\w+)/(status|review)/(?P\d+)/activity/?$', incoming.get_status ), - re_path(r'^user/(?P\w+)/status/?$', incoming.get_following), + re_path(r'^user/(?P\w+)/(status|review)/?$', incoming.get_following), # TODO: shelves need pages in the UI and for their activitypub Collection # .well-known endpoints diff --git a/fedireads/utils/models.py b/fedireads/utils/models.py index 5b80c02c..eb002986 100644 --- a/fedireads/utils/models.py +++ b/fedireads/utils/models.py @@ -13,7 +13,7 @@ class FedireadsModel(models.Model): base_path = 'https://%s' % DOMAIN if self.user: base_path = self.user.absolute_id - model_name = type(self).__name__ + model_name = type(self).__name__.lower() return '%s/%s/%d' % (base_path, model_name, self.id) class Meta: