ldp.py 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. import logging
  2. from collections import defaultdict
  3. from uuid import uuid4
  4. from flask import Blueprint, request, send_file
  5. from werkzeug.datastructures import FileStorage
  6. from lakesuperior.exceptions import InvalidResourceError, \
  7. ResourceExistsError, ResourceNotExistsError, ServerManagedTermError
  8. from lakesuperior.model.ldp_rs import Ldpr, Ldpc, LdpRs
  9. from lakesuperior.model.ldp_nr import LdpNr
  10. from lakesuperior.store_layouts.rdf.base_rdf_layout import BaseRdfLayout
  11. from lakesuperior.util.translator import Translator
  12. logger = logging.getLogger(__name__)
  13. # Blueprint for LDP REST API. This is what is usually found under `/rest/` in
  14. # standard fcrepo4. Here, it is under `/ldp` but initially `/rest` can be kept
  15. # for backward compatibility.
  16. ldp = Blueprint('ldp', __name__)
  17. accept_patch = (
  18. 'application/sparql-update',
  19. )
  20. accept_rdf = (
  21. 'application/ld+json',
  22. 'application/n-triples',
  23. 'application/rdf+xml',
  24. #'application/x-turtle',
  25. #'application/xhtml+xml',
  26. #'application/xml',
  27. #'text/html',
  28. 'text/n3',
  29. #'text/plain',
  30. 'text/rdf+n3',
  31. 'text/turtle',
  32. )
  33. #allow = (
  34. # 'COPY',
  35. # 'DELETE',
  36. # 'GET',
  37. # 'HEAD',
  38. # 'MOVE',
  39. # 'OPTIONS',
  40. # 'PATCH',
  41. # 'POST',
  42. # 'PUT',
  43. #)
  44. std_headers = {
  45. 'Accept-Patch' : ','.join(accept_patch),
  46. 'Accept-Post' : ','.join(accept_rdf),
  47. #'Allow' : ','.join(allow),
  48. }
  49. ## REST SERVICES ##
  50. @ldp.route('/<path:uuid>', methods=['GET'])
  51. @ldp.route('/', defaults={'uuid': None}, methods=['GET'],
  52. strict_slashes=False)
  53. def get_resource(uuid, force_rdf=False):
  54. '''
  55. Retrieve RDF or binary content.
  56. @param uuid (string) UUID of resource to retrieve.
  57. @param force_rdf (boolean) Whether to retrieve RDF even if the resource is
  58. a LDP-NR. This is not available in the API but is used e.g. by the
  59. `*/fcr:metadata` endpoint. The default is False.
  60. '''
  61. out_headers = std_headers
  62. repr_options = defaultdict(dict)
  63. if 'prefer' in request.headers:
  64. prefer = Translator.parse_rfc7240(request.headers['prefer'])
  65. logger.debug('Parsed Prefer header: {}'.format(prefer))
  66. if 'return' in prefer:
  67. repr_options = prefer['return']
  68. try:
  69. rsrc = Ldpr.readonly_inst(uuid, repr_options)
  70. except ResourceNotExistsError as e:
  71. return str(e), 404
  72. else:
  73. logger.debug('Resource URN in the route: {}'.format(rsrc._urn))
  74. x = rsrc.imr
  75. logger.debug('IMR options in the route: {}'.format(rsrc._imr_options))
  76. out_headers.update(rsrc.head())
  77. if isinstance(rsrc, LdpRs) \
  78. or request.headers['accept'] in accept_rdf \
  79. or force_rdf:
  80. return (rsrc.imr.graph.serialize(format='turtle'), out_headers)
  81. else:
  82. return send_file(rsrc.local_path, as_attachment=True,
  83. attachment_filename=rsrc.filename)
  84. @ldp.route('/<path:uuid>/fcr:metadata', methods=['GET'])
  85. def get_metadata(uuid):
  86. '''
  87. Retrieve RDF metadata of a LDP-NR.
  88. '''
  89. return get_resource(uuid, force_rdf=True)
  90. @ldp.route('/<path:parent>', methods=['POST'])
  91. @ldp.route('/', defaults={'parent': None}, methods=['POST'],
  92. strict_slashes=False)
  93. def post_resource(parent):
  94. '''
  95. Add a new resource in a new URI.
  96. '''
  97. out_headers = std_headers
  98. try:
  99. slug = request.headers['Slug']
  100. except KeyError:
  101. slug = None
  102. cls, data = class_from_req_body()
  103. try:
  104. rsrc = cls.inst_for_post(parent, slug)
  105. except ResourceNotExistsError as e:
  106. return str(e), 404
  107. except InvalidResourceError as e:
  108. return str(e), 409
  109. if cls == LdpNr:
  110. try:
  111. cont_disp = Translator.parse_rfc7240(
  112. request.headers['content-disposition'])
  113. except KeyError:
  114. cont_disp = None
  115. rsrc.post(data, mimetype=request.content_type, disposition=cont_disp)
  116. else:
  117. try:
  118. rsrc.post(data)
  119. except ServerManagedTermError as e:
  120. return str(e), 412
  121. out_headers.update({
  122. 'Location' : rsrc.uri,
  123. })
  124. return rsrc.uri, out_headers, 201
  125. @ldp.route('/<path:uuid>', methods=['PUT'])
  126. def put_resource(uuid):
  127. '''
  128. Add a new resource at a specified URI.
  129. '''
  130. logger.info('Request headers: {}'.format(request.headers))
  131. rsp_headers = std_headers
  132. cls, data = class_from_req_body()
  133. rsrc = cls(uuid)
  134. # Parse headers.
  135. pref_handling = None
  136. if cls == LdpNr:
  137. try:
  138. logger.debug('Headers: {}'.format(request.headers))
  139. cont_disp = Translator.parse_rfc7240(
  140. request.headers['content-disposition'])
  141. except KeyError:
  142. cont_disp = None
  143. try:
  144. ret = rsrc.put(data, disposition=cont_disp)
  145. except InvalidResourceError as e:
  146. return str(e), 409
  147. except ResourceExistsError as e:
  148. return str(e), 409
  149. else:
  150. if 'prefer' in request.headers:
  151. prefer = Translator.parse_rfc7240(request.headers['prefer'])
  152. logger.debug('Parsed Prefer header: {}'.format(prefer))
  153. if 'handling' in prefer:
  154. pref_handling = prefer['handling']['value']
  155. try:
  156. ret = rsrc.put(data, handling=pref_handling)
  157. except InvalidResourceError as e:
  158. return str(e), 409
  159. except ResourceExistsError as e:
  160. return str(e), 409
  161. except ServerManagedTermError as e:
  162. return str(e), 412
  163. res_code = 201 if ret == BaseRdfLayout.RES_CREATED else 204
  164. return '', res_code, rsp_headers
  165. @ldp.route('/<path:uuid>', methods=['PATCH'])
  166. def patch_resource(uuid):
  167. '''
  168. Update an existing resource with a SPARQL-UPDATE payload.
  169. '''
  170. headers = std_headers
  171. rsrc = Ldpc(uuid)
  172. try:
  173. rsrc.patch(request.get_data().decode('utf-8'))
  174. except ResourceNotExistsError:
  175. return 'Resource #{} not found.'.format(rsrc.uuid), 404
  176. except ServerManagedTermError as e:
  177. return str(e), 412
  178. return '', 204, headers
  179. @ldp.route('/<path:uuid>', methods=['DELETE'])
  180. def delete_resource(uuid):
  181. '''
  182. Delete a resource.
  183. '''
  184. headers = std_headers
  185. rsrc = Ldpc(uuid)
  186. try:
  187. rsrc.delete()
  188. except ResourceNotExistsError:
  189. return 'Resource #{} not found.'.format(rsrc.uuid), 404
  190. return '', 204, headers
  191. def class_from_req_body():
  192. logger.debug('Content type: {}'.format(request.mimetype))
  193. logger.debug('files: {}'.format(request.files))
  194. logger.debug('stream: {}'.format(request.stream))
  195. if not request.mimetype or request.mimetype in accept_rdf:
  196. cls = Ldpc
  197. # Parse out the RDF string.
  198. data = request.data.decode('utf-8')
  199. else:
  200. cls = LdpNr
  201. if request.mimetype == 'multipart/form-data':
  202. # This seems the "right" way to upload a binary file, with a
  203. # multipart/form-data MIME type and the file in the `file` field.
  204. # This however is not supported by FCREPO4.
  205. data = request.files.get('file').stream
  206. else:
  207. # This is a less clean way, with the file in the form body and the
  208. # request as application/x-www-form-urlencoded.
  209. # This is how FCREPO4 accepts binary uploads.
  210. data = request.stream
  211. logger.info('POSTing resource of type: {}'.format(cls.__name__))
  212. #logger.info('POST data: {}'.format(data))
  213. return cls, data
  214. def _get_bitstream(rsrc):
  215. out_headers = std_headers
  216. # @TODO This may change in favor of more low-level handling if the file
  217. # system is not local.
  218. return send_file(rsrc.local_path, as_attachment=True,
  219. attachment_filename=rsrc.filename)