aboutsummaryrefslogtreecommitdiffstats
path: root/mediagoblin/api/views.py
diff options
context:
space:
mode:
authorJessica Tallon <jessica@megworld.co.uk>2015-01-12 13:42:02 +0000
committerJessica Tallon <jessica@megworld.co.uk>2015-02-15 17:17:01 +0100
commit4fd520364f7d2405d09ac0d6bf561285356e4adb (patch)
tree147a144115c0ac858170e5280ff69c97bff4761e /mediagoblin/api/views.py
parent4aaa7fac14ade38fa12fe4ee1d4af3b57e1d993f (diff)
downloadmediagoblin-4fd520364f7d2405d09ac0d6bf561285356e4adb.tar.lz
mediagoblin-4fd520364f7d2405d09ac0d6bf561285356e4adb.tar.xz
mediagoblin-4fd520364f7d2405d09ac0d6bf561285356e4adb.zip
Change 'federation' name to 'api' which is more suitable
Diffstat (limited to 'mediagoblin/api/views.py')
-rw-r--r--mediagoblin/api/views.py796
1 files changed, 796 insertions, 0 deletions
diff --git a/mediagoblin/api/views.py b/mediagoblin/api/views.py
new file mode 100644
index 00000000..6095a721
--- /dev/null
+++ b/mediagoblin/api/views.py
@@ -0,0 +1,796 @@
+# GNU MediaGoblin -- federated, autonomous media hosting
+# Copyright (C) 2011, 2012 MediaGoblin contributors. See AUTHORS.
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU Affero General Public License for more details.
+#
+# You should have received a copy of the GNU Affero General Public License
+# along with this program. If not, see <http://www.gnu.org/licenses/>.
+
+import json
+import io
+import mimetypes
+
+from werkzeug.datastructures import FileStorage
+
+from mediagoblin.decorators import oauth_required, require_active_login
+from mediagoblin.api.decorators import user_has_privilege
+from mediagoblin.db.models import User, MediaEntry, MediaComment, Activity
+from mediagoblin.tools.federation import create_activity, create_generator
+from mediagoblin.tools.routing import extract_url_arguments
+from mediagoblin.tools.response import redirect, json_response, json_error, \
+ render_404, render_to_response
+from mediagoblin.meddleware.csrf import csrf_exempt
+from mediagoblin.submit.lib import new_upload_entry, api_upload_request, \
+ api_add_to_feed
+
+# MediaTypes
+from mediagoblin.media_types.image import MEDIA_TYPE as IMAGE_MEDIA_TYPE
+
+# Getters
+def get_profile(request):
+ """
+ Gets the user's profile for the endpoint requested.
+
+ For example an endpoint which is /api/{username}/feed
+ as /api/cwebber/feed would get cwebber's profile. This
+ will return a tuple (username, user_profile). If no user
+ can be found then this function returns a (None, None).
+ """
+ username = request.matchdict["username"]
+ user = User.query.filter_by(username=username).first()
+
+ if user is None:
+ return None, None
+
+ return user, user.serialize(request)
+
+
+# Endpoints
+@oauth_required
+def profile_endpoint(request):
+ """ This is /api/user/<username>/profile - This will give profile info """
+ user, user_profile = get_profile(request)
+
+ if user is None:
+ username = request.matchdict["username"]
+ return json_error(
+ "No such 'user' with username '{0}'".format(username),
+ status=404
+ )
+
+ # user profiles are public so return information
+ return json_response(user_profile)
+
+@oauth_required
+def user_endpoint(request):
+ """ This is /api/user/<username> - This will get the user """
+ user, user_profile = get_profile(request)
+
+ if user is None:
+ username = request.matchdict["username"]
+ return json_error(
+ "No such 'user' with username '{0}'".format(username),
+ status=404
+ )
+
+ return json_response({
+ "nickname": user.username,
+ "updated": user.created.isoformat(),
+ "published": user.created.isoformat(),
+ "profile": user_profile,
+ })
+
+@oauth_required
+@csrf_exempt
+@user_has_privilege(u'uploader')
+def uploads_endpoint(request):
+ """ Endpoint for file uploads """
+ username = request.matchdict["username"]
+ requested_user = User.query.filter_by(username=username).first()
+
+ if requested_user is None:
+ return json_error("No such 'user' with id '{0}'".format(username), 404)
+
+ if request.method == "POST":
+ # Ensure that the user is only able to upload to their own
+ # upload endpoint.
+ if requested_user.id != request.user.id:
+ return json_error(
+ "Not able to post to another users feed.",
+ status=403
+ )
+
+ # Wrap the data in the werkzeug file wrapper
+ if "Content-Type" not in request.headers:
+ return json_error(
+ "Must supply 'Content-Type' header to upload media."
+ )
+
+ mimetype = request.headers["Content-Type"]
+ filename = mimetypes.guess_all_extensions(mimetype)
+ filename = 'unknown' + filename[0] if filename else filename
+ file_data = FileStorage(
+ stream=io.BytesIO(request.data),
+ filename=filename,
+ content_type=mimetype
+ )
+
+ # Find media manager
+ entry = new_upload_entry(request.user)
+ entry.media_type = IMAGE_MEDIA_TYPE
+ return api_upload_request(request, file_data, entry)
+
+ return json_error("Not yet implemented", 501)
+
+@oauth_required
+@csrf_exempt
+def inbox_endpoint(request, inbox=None):
+ """ This is the user's inbox
+
+ Currently because we don't have the ability to represent the inbox in the
+ database this is not a "real" inbox in the pump.io/Activity streams 1.0
+ sense but instead just gives back all the data on the website
+
+ inbox: allows you to pass a query in to limit inbox scope
+ """
+ username = request.matchdict["username"]
+ user = User.query.filter_by(username=username).first()
+
+ if user is None:
+ return json_error("No such 'user' with id '{0}'".format(username), 404)
+
+
+ # Only the user who's authorized should be able to read their inbox
+ if user.id != request.user.id:
+ return json_error(
+ "Only '{0}' can read this inbox.".format(user.username),
+ 403
+ )
+
+ if inbox is None:
+ inbox = Activity.query
+
+ # Count how many items for the "totalItems" field
+ total_items = inbox.count()
+
+ # We want to make a query for all media on the site and then apply GET
+ # limits where we can.
+ inbox = inbox.order_by(Activity.published.desc())
+
+ # Limit by the "count" (default: 20)
+ try:
+ limit = int(request.args.get("count", 20))
+ except ValueError:
+ limit = 20
+
+ # Prevent the count being too big (pump uses 200 so we shall)
+ limit = limit if limit <= 200 else 200
+
+ # Apply the limit
+ inbox = inbox.limit(limit)
+
+ # Offset (default: no offset - first <count> results)
+ inbox = inbox.offset(request.args.get("offset", 0))
+
+ # build the inbox feed
+ feed = {
+ "displayName": "Activities for {0}".format(user.username),
+ "author": user.serialize(request),
+ "objectTypes": ["activity"],
+ "url": request.base_url,
+ "links": {"self": {"href": request.url}},
+ "items": [],
+ "totalItems": total_items,
+ }
+
+ for activity in inbox:
+ try:
+ feed["items"].append(activity.serialize(request))
+ except AttributeError:
+ # As with the feed endpint this occurs because of how we our
+ # hard-deletion method. Some activites might exist where the
+ # Activity object and/or target no longer exist, for this case we
+ # should just skip them.
+ pass
+
+ return json_response(feed)
+
+@oauth_required
+@csrf_exempt
+def inbox_minor_endpoint(request):
+ """ Inbox subset for less important Activities """
+ inbox = Activity.query.filter(
+ (Activity.verb == "update") | (Activity.verb == "delete")
+ )
+
+ return inbox_endpoint(request=request, inbox=inbox)
+
+@oauth_required
+@csrf_exempt
+def inbox_major_endpoint(request):
+ """ Inbox subset for most important Activities """
+ inbox = Activity.query.filter_by(verb="post")
+ return inbox_endpoint(request=request, inbox=inbox)
+
+@oauth_required
+@csrf_exempt
+def feed_endpoint(request, outbox=None):
+ """ Handles the user's outbox - /api/user/<username>/feed """
+ username = request.matchdict["username"]
+ requested_user = User.query.filter_by(username=username).first()
+
+ # check if the user exists
+ if requested_user is None:
+ return json_error("No such 'user' with id '{0}'".format(username), 404)
+
+ if request.data:
+ data = json.loads(request.data.decode())
+ else:
+ data = {"verb": None, "object": {}}
+
+
+ if request.method in ["POST", "PUT"]:
+ # Validate that the activity is valid
+ if "verb" not in data or "object" not in data:
+ return json_error("Invalid activity provided.")
+
+ # Check that the verb is valid
+ if data["verb"] not in ["post", "update", "delete"]:
+ return json_error("Verb not yet implemented", 501)
+
+ # We need to check that the user they're posting to is
+ # the person that they are.
+ if requested_user.id != request.user.id:
+ return json_error(
+ "Not able to post to another users feed.",
+ status=403
+ )
+
+ # Handle new posts
+ if data["verb"] == "post":
+ obj = data.get("object", None)
+ if obj is None:
+ return json_error("Could not find 'object' element.")
+
+ if obj.get("objectType", None) == "comment":
+ # post a comment
+ if not request.user.has_privilege(u'commenter'):
+ return json_error(
+ "Privilege 'commenter' required to comment.",
+ status=403
+ )
+
+ comment = MediaComment(author=request.user.id)
+ comment.unserialize(data["object"], request)
+ comment.save()
+
+ # Create activity for comment
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="post",
+ actor=request.user,
+ obj=comment,
+ target=comment.get_entry,
+ generator=generator
+ )
+
+ return json_response(activity.serialize(request))
+
+ elif obj.get("objectType", None) == "image":
+ # Posting an image to the feed
+ media_id = int(extract_url_arguments(
+ url=data["object"]["id"],
+ urlmap=request.app.url_map
+ )["id"])
+
+ media = MediaEntry.query.filter_by(id=media_id).first()
+
+ if media is None:
+ return json_response(
+ "No such 'image' with id '{0}'".format(media_id),
+ status=404
+ )
+
+ if media.uploader != request.user.id:
+ return json_error(
+ "Privilege 'commenter' required to comment.",
+ status=403
+ )
+
+
+ if not media.unserialize(data["object"]):
+ return json_error(
+ "Invalid 'image' with id '{0}'".format(media_id)
+ )
+
+
+ # Add location if one exists
+ if "location" in data:
+ Location.create(data["location"], self)
+
+ media.save()
+ activity = api_add_to_feed(request, media)
+
+ return json_response(activity.serialize(request))
+
+ elif obj.get("objectType", None) is None:
+ # They need to tell us what type of object they're giving us.
+ return json_error("No objectType specified.")
+ else:
+ # Oh no! We don't know about this type of object (yet)
+ object_type = obj.get("objectType", None)
+ return json_error(
+ "Unknown object type '{0}'.".format(object_type)
+ )
+
+ # Updating existing objects
+ if data["verb"] == "update":
+ # Check we've got a valid object
+ obj = data.get("object", None)
+
+ if obj is None:
+ return json_error("Could not find 'object' element.")
+
+ if "objectType" not in obj:
+ return json_error("No objectType specified.")
+
+ if "id" not in obj:
+ return json_error("Object ID has not been specified.")
+
+ obj_id = int(extract_url_arguments(
+ url=obj["id"],
+ urlmap=request.app.url_map
+ )["id"])
+
+ # Now try and find object
+ if obj["objectType"] == "comment":
+ if not request.user.has_privilege(u'commenter'):
+ return json_error(
+ "Privilege 'commenter' required to comment.",
+ status=403
+ )
+
+ comment = MediaComment.query.filter_by(id=obj_id).first()
+ if comment is None:
+ return json_error(
+ "No such 'comment' with id '{0}'.".format(obj_id)
+ )
+
+ # Check that the person trying to update the comment is
+ # the author of the comment.
+ if comment.author != request.user.id:
+ return json_error(
+ "Only author of comment is able to update comment.",
+ status=403
+ )
+
+ if not comment.unserialize(data["object"], request):
+ return json_error(
+ "Invalid 'comment' with id '{0}'".format(obj["id"])
+ )
+
+ comment.save()
+
+ # Create an update activity
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="update",
+ actor=request.user,
+ obj=comment,
+ generator=generator
+ )
+
+ return json_response(activity.serialize(request))
+
+ elif obj["objectType"] == "image":
+ image = MediaEntry.query.filter_by(id=obj_id).first()
+ if image is None:
+ return json_error(
+ "No such 'image' with the id '{0}'.".format(obj["id"])
+ )
+
+ # Check that the person trying to update the comment is
+ # the author of the comment.
+ if image.uploader != request.user.id:
+ return json_error(
+ "Only uploader of image is able to update image.",
+ status=403
+ )
+
+ if not image.unserialize(obj):
+ return json_error(
+ "Invalid 'image' with id '{0}'".format(obj_id)
+ )
+ image.generate_slug()
+ image.save()
+
+ # Create an update activity
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="update",
+ actor=request.user,
+ obj=image,
+ generator=generator
+ )
+
+ return json_response(activity.serialize(request))
+ elif obj["objectType"] == "person":
+ # check this is the same user
+ if "id" not in obj or obj["id"] != requested_user.id:
+ return json_error(
+ "Incorrect user id, unable to update"
+ )
+
+ requested_user.unserialize(obj)
+ requested_user.save()
+
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="update",
+ actor=request.user,
+ obj=requested_user,
+ generator=generator
+ )
+
+ return json_response(activity.serialize(request))
+
+ elif data["verb"] == "delete":
+ obj = data.get("object", None)
+ if obj is None:
+ return json_error("Could not find 'object' element.")
+
+ if "objectType" not in obj:
+ return json_error("No objectType specified.")
+
+ if "id" not in obj:
+ return json_error("Object ID has not been specified.")
+
+ # Parse out the object ID
+ obj_id = int(extract_url_arguments(
+ url=obj["id"],
+ urlmap=request.app.url_map
+ )["id"])
+
+ if obj.get("objectType", None) == "comment":
+ # Find the comment asked for
+ comment = MediaComment.query.filter_by(
+ id=obj_id,
+ author=request.user.id
+ ).first()
+
+ if comment is None:
+ return json_error(
+ "No such 'comment' with id '{0}'.".format(obj_id)
+ )
+
+ # Make a delete activity
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="delete",
+ actor=request.user,
+ obj=comment,
+ generator=generator
+ )
+
+ # Unfortunately this has to be done while hard deletion exists
+ context = activity.serialize(request)
+
+ # now we can delete the comment
+ comment.delete()
+
+ return json_response(context)
+
+ if obj.get("objectType", None) == "image":
+ # Find the image
+ entry = MediaEntry.query.filter_by(
+ id=obj_id,
+ uploader=request.user.id
+ ).first()
+
+ if entry is None:
+ return json_error(
+ "No such 'image' with id '{0}'.".format(obj_id)
+ )
+
+ # Make the delete activity
+ generator = create_generator(request)
+ activity = create_activity(
+ verb="delete",
+ actor=request.user,
+ obj=entry,
+ generator=generator
+ )
+
+ # This is because we have hard deletion
+ context = activity.serialize(request)
+
+ # Now we can delete the image
+ entry.delete()
+
+ return json_response(context)
+
+ elif request.method != "GET":
+ return json_error(
+ "Unsupported HTTP method {0}".format(request.method),
+ status=501
+ )
+
+ feed = {
+ "displayName": "Activities by {user}@{host}".format(
+ user=request.user.username,
+ host=request.host
+ ),
+ "objectTypes": ["activity"],
+ "url": request.base_url,
+ "links": {"self": {"href": request.url}},
+ "author": request.user.serialize(request),
+ "items": [],
+ }
+
+ # Create outbox
+ if outbox is None:
+ outbox = Activity.query.filter_by(actor=request.user.id)
+ else:
+ outbox = outbox.filter_by(actor=request.user.id)
+
+ # We want the newest things at the top (issue: #1055)
+ outbox = outbox.order_by(Activity.published.desc())
+
+ # Limit by the "count" (default: 20)
+ limit = request.args.get("count", 20)
+
+ try:
+ limit = int(limit)
+ except ValueError:
+ limit = 20
+
+ # The upper most limit should be 200
+ limit = limit if limit < 200 else 200
+
+ # apply the limit
+ outbox = outbox.limit(limit)
+
+ # Offset (default: no offset - first <count> result)
+ outbox = outbox.offset(request.args.get("offset", 0))
+
+ # Build feed.
+ for activity in outbox:
+ try:
+ feed["items"].append(activity.serialize(request))
+ except AttributeError:
+ # This occurs because of how we hard-deletion and the object
+ # no longer existing anymore. We want to keep the Activity
+ # in case someone wishes to look it up but we shouldn't display
+ # it in the feed.
+ pass
+ feed["totalItems"] = len(feed["items"])
+
+ return json_response(feed)
+
+@oauth_required
+def feed_minor_endpoint(request):
+ """ Outbox for minor activities such as updates """
+ # If it's anything but GET pass it along
+ if request.method != "GET":
+ return feed_endpoint(request)
+
+ outbox = Activity.query.filter(
+ (Activity.verb == "update") | (Activity.verb == "delete")
+ )
+ return feed_endpoint(request, outbox=outbox)
+
+@oauth_required
+def feed_major_endpoint(request):
+ """ Outbox for all major activities """
+ # If it's anything but a GET pass it along
+ if request.method != "GET":
+ return feed_endpoint(request)
+
+ outbox = Activity.query.filter_by(verb="post")
+ return feed_endpoint(request, outbox=outbox)
+
+@oauth_required
+def object_endpoint(request):
+ """ Lookup for a object type """
+ object_type = request.matchdict["object_type"]
+ try:
+ object_id = request.matchdict["id"]
+ except ValueError:
+ error = "Invalid object ID '{0}' for '{1}'".format(
+ request.matchdict["id"],
+ object_type
+ )
+ return json_error(error)
+
+ if object_type not in ["image"]:
+ # not sure why this is 404, maybe ask evan. Maybe 400?
+ return json_error(
+ "Unknown type: {0}".format(object_type),
+ status=404
+ )
+
+ media = MediaEntry.query.filter_by(id=object_id).first()
+ if media is None:
+ return json_error(
+ "Can't find '{0}' with ID '{1}'".format(object_type, object_id),
+ status=404
+ )
+
+ return json_response(media.serialize(request))
+
+@oauth_required
+def object_comments(request):
+ """ Looks up for the comments on a object """
+ media = MediaEntry.query.filter_by(id=request.matchdict["id"]).first()
+ if media is None:
+ return json_error("Can't find '{0}' with ID '{1}'".format(
+ request.matchdict["object_type"],
+ request.matchdict["id"]
+ ), 404)
+
+ comments = media.serialize(request)
+ comments = comments.get("replies", {
+ "totalItems": 0,
+ "items": [],
+ "url": request.urlgen(
+ "mediagoblin.api.object.comments",
+ object_type=media.object_type,
+ id=media.id,
+ qualified=True
+ )
+ })
+
+ comments["displayName"] = "Replies to {0}".format(comments["url"])
+ comments["links"] = {
+ "first": comments["url"],
+ "self": comments["url"],
+ }
+ return json_response(comments)
+
+##
+# RFC6415 - Web Host Metadata
+##
+def host_meta(request):
+ """
+ This provides the host-meta URL information that is outlined
+ in RFC6415. By default this should provide XRD+XML however
+ if the client accepts JSON we will provide that over XRD+XML.
+ The 'Accept' header is used to decude this.
+
+ A client should use this endpoint to determine what URLs to
+ use for OAuth endpoints.
+ """
+
+ links = [
+ {
+ "rel": "lrdd",
+ "type": "application/json",
+ "href": request.urlgen(
+ "mediagoblin.webfinger.well-known.webfinger",
+ qualified=True
+ )
+ },
+ {
+ "rel": "registration_endpoint",
+ "href": request.urlgen(
+ "mediagoblin.oauth.client_register",
+ qualified=True
+ ),
+ },
+ {
+ "rel": "http://apinamespace.org/oauth/request_token",
+ "href": request.urlgen(
+ "mediagoblin.oauth.request_token",
+ qualified=True
+ ),
+ },
+ {
+ "rel": "http://apinamespace.org/oauth/authorize",
+ "href": request.urlgen(
+ "mediagoblin.oauth.authorize",
+ qualified=True
+ ),
+ },
+ {
+ "rel": "http://apinamespace.org/oauth/access_token",
+ "href": request.urlgen(
+ "mediagoblin.oauth.access_token",
+ qualified=True
+ ),
+ },
+ {
+ "rel": "http://apinamespace.org/activitypub/whoami",
+ "href": request.urlgen(
+ "mediagoblin.webfinger.whoami",
+ qualified=True
+ ),
+ },
+ ]
+
+ if "application/json" in request.accept_mimetypes:
+ return json_response({"links": links})
+
+ # provide XML+XRD
+ return render_to_response(
+ request,
+ "mediagoblin/api/host-meta.xml",
+ {"links": links},
+ mimetype="application/xrd+xml"
+ )
+
+def lrdd_lookup(request):
+ """
+ This is the lrdd endpoint which can lookup a user (or
+ other things such as activities). This is as specified by
+ RFC6415.
+
+ The cleint must provide a 'resource' as a GET parameter which
+ should be the query to be looked up.
+ """
+
+ if "resource" not in request.args:
+ return json_error("No resource parameter", status=400)
+
+ resource = request.args["resource"]
+
+ if "@" in resource:
+ # Lets pull out the username
+ resource = resource[5:] if resource.startswith("acct:") else resource
+ username, host = resource.split("@", 1)
+
+ # Now lookup the user
+ user = User.query.filter_by(username=username).first()
+
+ if user is None:
+ return json_error(
+ "Can't find 'user' with username '{0}'".format(username))
+
+ return json_response([
+ {
+ "rel": "http://webfinger.net/rel/profile-page",
+ "href": user.url_for_self(request.urlgen),
+ "type": "text/html"
+ },
+ {
+ "rel": "self",
+ "href": request.urlgen(
+ "mediagoblin.api.user",
+ username=user.username,
+ qualified=True
+ )
+ },
+ {
+ "rel": "activity-outbox",
+ "href": request.urlgen(
+ "mediagoblin.api.feed",
+ username=user.username,
+ qualified=True
+ )
+ }
+ ])
+ else:
+ return json_error("Unrecognized resource parameter", status=404)
+
+
+def whoami(request):
+ """ /api/whoami - HTTP redirect to API profile """
+ if request.user is None:
+ return json_error("Not logged in.", status=401)
+
+ profile = request.urlgen(
+ "mediagoblin.api.user.profile",
+ username=request.user.username,
+ qualified=True
+ )
+
+ return redirect(request, location=profile)
+
+