ldp.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658
  1. import logging
  2. import pdb
  3. from base64 import b64encode
  4. from collections import defaultdict
  5. from io import BytesIO
  6. from pprint import pformat
  7. from uuid import uuid4
  8. import arrow
  9. from flask import (
  10. Blueprint, Response, g, make_response, render_template,
  11. request, send_file)
  12. from rdflib import Graph, plugin, parser#, serializer
  13. from lakesuperior.api import resource as rsrc_api
  14. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  15. from lakesuperior.dictionaries.namespaces import ns_mgr as nsm
  16. from lakesuperior.exceptions import (ResourceNotExistsError, TombstoneError,
  17. ServerManagedTermError, InvalidResourceError, SingleSubjectError,
  18. ResourceExistsError, IncompatibleLdpTypeError)
  19. from lakesuperior.globals import RES_CREATED
  20. from lakesuperior.model.ldp_factory import LdpFactory
  21. from lakesuperior.model.ldp_nr import LdpNr
  22. from lakesuperior.model.ldp_rs import LdpRs
  23. from lakesuperior.model.ldpr import Ldpr
  24. from lakesuperior.store.ldp_rs.lmdb_store import TxnManager
  25. from lakesuperior.toolbox import Toolbox
  26. DEFAULT_RDF_MIMETYPE = 'text/turtle'
  27. """
  28. Fallback serialization format used when no acceptable formats are specified.
  29. """
  30. logger = logging.getLogger(__name__)
  31. rdf_parsable_mimetypes = {
  32. mt.name for mt in plugin.plugins()
  33. if mt.kind is parser.Parser and '/' in mt.name
  34. }
  35. """MIMEtypes that can be parsed into RDF."""
  36. rdf_serializable_mimetypes = {
  37. #mt.name for mt in plugin.plugins()
  38. #if mt.kind is serializer.Serializer and '/' in mt.name
  39. 'application/ld+json',
  40. 'application/n-triples',
  41. 'application/rdf+xml',
  42. 'text/turtle',
  43. 'text/n3',
  44. }
  45. """
  46. MIMEtypes that RDF can be serialized into.
  47. These are not automatically derived from RDFLib because only triple
  48. (not quad) serializations are applicable.
  49. """
  50. accept_patch = (
  51. 'application/sparql-update',
  52. )
  53. std_headers = {
  54. 'Accept-Patch' : ','.join(accept_patch),
  55. 'Accept-Post' : ','.join(rdf_parsable_mimetypes),
  56. }
  57. """Predicates excluded by view."""
  58. vw_blacklist = {
  59. }
  60. ldp = Blueprint(
  61. 'ldp', __name__, template_folder='templates',
  62. static_url_path='/static', static_folder='templates/static')
  63. """
  64. Blueprint for LDP REST API. This is what is usually found under ``/rest/`` in
  65. standard fcrepo4. Here, it is under ``/ldp`` but initially ``/rest`` will be
  66. kept for backward compatibility.
  67. """
  68. ## ROUTE PRE- & POST-PROCESSING ##
  69. @ldp.url_defaults
  70. def bp_url_defaults(endpoint, values):
  71. url_prefix = getattr(g, 'url_prefix', None)
  72. if url_prefix is not None:
  73. values.setdefault('url_prefix', url_prefix)
  74. @ldp.url_value_preprocessor
  75. def bp_url_value_preprocessor(endpoint, values):
  76. g.url_prefix = values.pop('url_prefix')
  77. g.webroot = request.host_url + g.url_prefix
  78. # Normalize leading slashes for UID.
  79. if 'uid' in values:
  80. values['uid'] = '/' + values['uid'].lstrip('/')
  81. if 'parent_uid' in values:
  82. values['parent_uid'] = '/' + values['parent_uid'].lstrip('/')
  83. @ldp.before_request
  84. def log_request_start():
  85. logger.info('** Start {} {} **'.format(request.method, request.url))
  86. @ldp.before_request
  87. def instantiate_req_vars():
  88. g.tbox = Toolbox()
  89. @ldp.after_request
  90. def log_request_end(rsp):
  91. logger.info('** End {} {} **'.format(request.method, request.url))
  92. return rsp
  93. ## REST SERVICES ##
  94. @ldp.route('/<path:uid>', methods=['GET'], strict_slashes=False)
  95. @ldp.route('/', defaults={'uid': '/'}, methods=['GET'], strict_slashes=False)
  96. @ldp.route('/<path:uid>/fcr:metadata', defaults={'out_fmt' : 'rdf'},
  97. methods=['GET'])
  98. @ldp.route('/<path:uid>/fcr:content', defaults={'out_fmt' : 'non_rdf'},
  99. methods=['GET'])
  100. def get_resource(uid, out_fmt=None):
  101. r"""
  102. https://www.w3.org/TR/ldp/#ldpr-HTTP_GET
  103. Retrieve RDF or binary content.
  104. :param str uid: UID of resource to retrieve. The repository root has
  105. an empty string for UID.
  106. :param str out_fmt: Force output to RDF or non-RDF if the resource is
  107. a LDP-NR. This is not available in the API but is used e.g. by the
  108. ``\*/fcr:metadata`` and ``\*/fcr:content`` endpoints. The default is
  109. False.
  110. """
  111. logger.info('UID: {}'.format(uid))
  112. out_headers = std_headers
  113. repr_options = defaultdict(dict)
  114. if 'prefer' in request.headers:
  115. prefer = g.tbox.parse_rfc7240(request.headers['prefer'])
  116. logger.debug('Parsed Prefer header: {}'.format(pformat(prefer)))
  117. if 'return' in prefer:
  118. repr_options = parse_repr_options(prefer['return'])
  119. try:
  120. rsrc = rsrc_api.get(uid, repr_options)
  121. except ResourceNotExistsError as e:
  122. return str(e), 404
  123. except TombstoneError as e:
  124. return _tombstone_response(e, uid)
  125. else:
  126. if out_fmt is None:
  127. rdf_mimetype = _best_rdf_mimetype()
  128. out_fmt = (
  129. 'rdf'
  130. if isinstance(rsrc, LdpRs) or rdf_mimetype is not None
  131. else 'non_rdf')
  132. out_headers.update(_headers_from_metadata(rsrc, out_fmt))
  133. uri = g.tbox.uid_to_uri(uid)
  134. if out_fmt == 'rdf':
  135. if locals().get('rdf_mimetype', None) is None:
  136. rdf_mimetype = DEFAULT_RDF_MIMETYPE
  137. ggr = g.tbox.globalize_graph(rsrc.out_graph)
  138. ggr.namespace_manager = nsm
  139. return _negotiate_content(
  140. ggr, rdf_mimetype, out_headers, uid=uid, uri=uri)
  141. else:
  142. if not getattr(rsrc, 'local_path', False):
  143. return ('{} has no binary content.'.format(rsrc.uid), 404)
  144. logger.debug('Streaming out binary content.')
  145. rsp = make_response(send_file(
  146. rsrc.local_path, as_attachment=True,
  147. attachment_filename=rsrc.filename,
  148. mimetype=rsrc.mimetype), 200, out_headers)
  149. rsp.headers.add('Link',
  150. '<{}/fcr:metadata>; rel="describedby"'.format(uri))
  151. return rsp
  152. @ldp.route('/<path:uid>/fcr:versions', methods=['GET'])
  153. def get_version_info(uid):
  154. """
  155. Get version info (`fcr:versions`).
  156. :param str uid: UID of resource to retrieve versions for.
  157. """
  158. rdf_mimetype = _best_rdf_mimetype() or DEFAULT_RDF_MIMETYPE
  159. try:
  160. gr = rsrc_api.get_version_info(uid)
  161. except ResourceNotExistsError as e:
  162. return str(e), 404
  163. except InvalidResourceError as e:
  164. return str(e), 409
  165. except TombstoneError as e:
  166. return _tombstone_response(e, uid)
  167. else:
  168. return _negotiate_content(g.tbox.globalize_graph(gr), rdf_mimetype)
  169. @ldp.route('/<path:uid>/fcr:versions/<ver_uid>', methods=['GET'])
  170. def get_version(uid, ver_uid):
  171. """
  172. Get an individual resource version.
  173. :param str uid: Resource UID.
  174. :param str ver_uid: Version UID.
  175. """
  176. rdf_mimetype = _best_rdf_mimetype() or DEFAULT_RDF_MIMETYPE
  177. try:
  178. gr = rsrc_api.get_version(uid, ver_uid)
  179. except ResourceNotExistsError as e:
  180. return str(e), 404
  181. except InvalidResourceError as e:
  182. return str(e), 409
  183. except TombstoneError as e:
  184. return _tombstone_response(e, uid)
  185. else:
  186. return _negotiate_content(g.tbox.globalize_graph(gr), rdf_mimetype)
  187. @ldp.route('/<path:parent_uid>', methods=['POST'], strict_slashes=False)
  188. @ldp.route('/', defaults={'parent_uid': '/'}, methods=['POST'],
  189. strict_slashes=False)
  190. def post_resource(parent_uid):
  191. """
  192. https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
  193. Add a new resource in a new URI.
  194. """
  195. out_headers = std_headers
  196. try:
  197. slug = request.headers['Slug']
  198. logger.debug('Slug: {}'.format(slug))
  199. except KeyError:
  200. slug = None
  201. handling, disposition = set_post_put_params()
  202. stream, mimetype = _bistream_from_req()
  203. if mimetype in rdf_parsable_mimetypes:
  204. # If the content is RDF, localize in-repo URIs.
  205. global_rdf = stream.read()
  206. rdf_data = g.tbox.localize_payload(global_rdf)
  207. rdf_fmt = mimetype
  208. stream = mimetype = None
  209. else:
  210. rdf_data = rdf_fmt = None
  211. try:
  212. uid = rsrc_api.create(
  213. parent_uid, slug, stream=stream, mimetype=mimetype,
  214. rdf_data=rdf_data, rdf_fmt=rdf_fmt, handling=handling,
  215. disposition=disposition)
  216. except ResourceNotExistsError as e:
  217. return str(e), 404
  218. except InvalidResourceError as e:
  219. return str(e), 409
  220. except TombstoneError as e:
  221. return _tombstone_response(e, uid)
  222. except ServerManagedTermError as e:
  223. return str(e), 412
  224. uri = g.tbox.uid_to_uri(uid)
  225. hdr = {'Location' : uri}
  226. if mimetype and rdf_fmt is None:
  227. hdr['Link'] = '<{0}/fcr:metadata>; rel="describedby"; anchor="{0}"'\
  228. .format(uri)
  229. out_headers.update(hdr)
  230. return uri, 201, out_headers
  231. @ldp.route('/<path:uid>', methods=['PUT'], strict_slashes=False)
  232. @ldp.route('/<path:uid>/fcr:metadata', defaults={'force_rdf' : True},
  233. methods=['PUT'])
  234. def put_resource(uid):
  235. """
  236. https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
  237. Add or replace a new resource at a specified URI.
  238. """
  239. # Parse headers.
  240. logger.debug('Request headers: {}'.format(request.headers))
  241. rsp_headers = {'Content-Type' : 'text/plain; charset=utf-8'}
  242. handling, disposition = set_post_put_params()
  243. stream, mimetype = _bistream_from_req()
  244. if mimetype in rdf_parsable_mimetypes:
  245. # If the content is RDF, localize in-repo URIs.
  246. global_rdf = stream.read()
  247. rdf_data = g.tbox.localize_payload(global_rdf)
  248. rdf_fmt = mimetype
  249. stream = mimetype = None
  250. else:
  251. rdf_data = rdf_fmt = None
  252. try:
  253. evt = rsrc_api.create_or_replace(
  254. uid, stream=stream, mimetype=mimetype,
  255. rdf_data=rdf_data, rdf_fmt=rdf_fmt, handling=handling,
  256. disposition=disposition)
  257. except (InvalidResourceError, ResourceExistsError) as e:
  258. return str(e), 409
  259. except (ServerManagedTermError, SingleSubjectError) as e:
  260. return str(e), 412
  261. except IncompatibleLdpTypeError as e:
  262. return str(e), 415
  263. except TombstoneError as e:
  264. return _tombstone_response(e, uid)
  265. uri = g.tbox.uid_to_uri(uid)
  266. if evt == RES_CREATED:
  267. rsp_code = 201
  268. rsp_headers['Location'] = rsp_body = uri
  269. if mimetype and not rdf_data:
  270. rsp_headers['Link'] = (
  271. '<{0}/fcr:metadata>; rel="describedby"'.format(uri))
  272. else:
  273. rsp_code = 204
  274. rsp_body = ''
  275. return rsp_body, rsp_code, rsp_headers
  276. @ldp.route('/<path:uid>', methods=['PATCH'], strict_slashes=False)
  277. @ldp.route('/', defaults={'uid': '/'}, methods=['PATCH'],
  278. strict_slashes=False)
  279. def patch_resource(uid, is_metadata=False):
  280. """
  281. https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
  282. Update an existing resource with a SPARQL-UPDATE payload.
  283. """
  284. rsp_headers = {'Content-Type' : 'text/plain; charset=utf-8'}
  285. if request.mimetype != 'application/sparql-update':
  286. return 'Provided content type is not a valid parsable format: {}'\
  287. .format(request.mimetype), 415
  288. update_str = request.get_data().decode('utf-8')
  289. local_update_str = g.tbox.localize_ext_str(update_str, nsc['fcres'][uid])
  290. try:
  291. rsrc = rsrc_api.update(uid, local_update_str, is_metadata)
  292. except ResourceNotExistsError as e:
  293. return str(e), 404
  294. except TombstoneError as e:
  295. return _tombstone_response(e, uid)
  296. except (ServerManagedTermError, SingleSubjectError) as e:
  297. return str(e), 412
  298. except InvalidResourceError as e:
  299. return str(e), 415
  300. else:
  301. rsp_headers.update(_headers_from_metadata(rsrc))
  302. return '', 204, rsp_headers
  303. @ldp.route('/<path:uid>/fcr:metadata', methods=['PATCH'])
  304. def patch_resource_metadata(uid):
  305. return patch_resource(uid, True)
  306. @ldp.route('/<path:uid>', methods=['DELETE'])
  307. def delete_resource(uid):
  308. """
  309. Delete a resource and optionally leave a tombstone.
  310. This behaves differently from FCREPO. A tombstone indicated that the
  311. resource is no longer available at its current location, but its historic
  312. snapshots still are. Also, deleting a resource with a tombstone creates
  313. one more version snapshot of the resource prior to being deleted.
  314. In order to completely wipe out all traces of a resource, the tombstone
  315. must be deleted as well, or the ``Prefer:no-tombstone`` header can be used.
  316. The latter will forget (completely delete) the resource immediately.
  317. """
  318. headers = std_headers
  319. if 'prefer' in request.headers:
  320. prefer = g.tbox.parse_rfc7240(request.headers['prefer'])
  321. leave_tstone = 'no-tombstone' not in prefer
  322. else:
  323. leave_tstone = True
  324. try:
  325. rsrc_api.delete(uid, leave_tstone)
  326. except ResourceNotExistsError as e:
  327. return str(e), 404
  328. except TombstoneError as e:
  329. return _tombstone_response(e, uid)
  330. return '', 204, headers
  331. @ldp.route('/<path:uid>/fcr:tombstone', methods=['GET', 'POST', 'PUT',
  332. 'PATCH', 'DELETE'])
  333. def tombstone(uid):
  334. """
  335. Handle all tombstone operations.
  336. The only allowed methods are POST and DELETE; any other verb will return a
  337. 405.
  338. """
  339. try:
  340. rsrc = rsrc_api.get(uid)
  341. except TombstoneError as e:
  342. if request.method == 'DELETE':
  343. if e.uid == uid:
  344. rsrc_api.delete(uid, False)
  345. return '', 204
  346. else:
  347. return _tombstone_response(e, uid)
  348. elif request.method == 'POST':
  349. if e.uid == uid:
  350. rsrc_uri = rsrc_api.resurrect(uid)
  351. headers = {'Location' : rsrc_uri}
  352. return rsrc_uri, 201, headers
  353. else:
  354. return _tombstone_response(e, uid)
  355. else:
  356. return 'Method Not Allowed.', 405
  357. except ResourceNotExistsError as e:
  358. return str(e), 404
  359. else:
  360. return '', 404
  361. @ldp.route('/<path:uid>/fcr:versions', methods=['POST', 'PUT'])
  362. def post_version(uid):
  363. """
  364. Create a new resource version.
  365. """
  366. if request.method == 'PUT':
  367. return 'Method not allowed.', 405
  368. ver_uid = request.headers.get('slug', None)
  369. try:
  370. ver_uid = rsrc_api.create_version(uid, ver_uid)
  371. except ResourceNotExistsError as e:
  372. return str(e), 404
  373. except InvalidResourceError as e:
  374. return str(e), 409
  375. except TombstoneError as e:
  376. return _tombstone_response(e, uid)
  377. else:
  378. return '', 201, {'Location': g.tbox.uid_to_uri(ver_uid)}
  379. @ldp.route('/<path:uid>/fcr:versions/<ver_uid>', methods=['PATCH'])
  380. def patch_version(uid, ver_uid):
  381. """
  382. Revert to a previous version.
  383. NOTE: This creates a new version snapshot.
  384. :param str uid: Resource UID.
  385. :param str ver_uid: Version UID.
  386. """
  387. try:
  388. rsrc_api.revert_to_version(uid, rsrc_uid)
  389. except ResourceNotExistsError as e:
  390. return str(e), 404
  391. except InvalidResourceError as e:
  392. return str(e), 409
  393. except TombstoneError as e:
  394. return _tombstone_response(e, uid)
  395. else:
  396. return '', 204
  397. ## PRIVATE METHODS ##
  398. def _best_rdf_mimetype():
  399. """
  400. Check if any of the 'Accept' header values provided is a RDF parsable
  401. format.
  402. """
  403. for accept in request.accept_mimetypes:
  404. mimetype = accept[0]
  405. if mimetype in rdf_parsable_mimetypes:
  406. return mimetype
  407. return None
  408. def _negotiate_content(gr, rdf_mimetype, headers=None, **vw_kwargs):
  409. """
  410. Return HTML or serialized RDF depending on accept headers.
  411. """
  412. if request.accept_mimetypes.best == 'text/html':
  413. return render_template(
  414. 'resource.html', gr=gr, nsc=nsc, nsm=nsm,
  415. blacklist=vw_blacklist, arrow=arrow, **vw_kwargs)
  416. else:
  417. for p in vw_blacklist:
  418. gr.remove((None, p, None))
  419. return Response(
  420. gr.serialize(format=rdf_mimetype), 200, headers,
  421. mimetype=rdf_mimetype)
  422. def _bistream_from_req():
  423. """
  424. Find how a binary file and its MIMEtype were uploaded in the request.
  425. """
  426. #logger.debug('Content type: {}'.format(request.mimetype))
  427. #logger.debug('files: {}'.format(request.files))
  428. #logger.debug('stream: {}'.format(request.stream))
  429. if request.mimetype == 'multipart/form-data':
  430. # This seems the "right" way to upload a binary file, with a
  431. # multipart/form-data MIME type and the file in the `file`
  432. # field. This however is not supported by FCREPO4.
  433. stream = request.files.get('file').stream
  434. mimetype = request.files.get('file').content_type
  435. # @TODO This will turn out useful to provide metadata
  436. # with the binary.
  437. #metadata = request.files.get('metadata').stream
  438. else:
  439. # This is a less clean way, with the file in the form body and
  440. # the request as application/x-www-form-urlencoded.
  441. # This is how FCREPO4 accepts binary uploads.
  442. stream = request.stream
  443. # @FIXME Must decide what to do with this.
  444. mimetype = request.mimetype
  445. if mimetype == '' or mimetype == 'application/x-www-form-urlencoded':
  446. if getattr(stream, 'limit', 0) == 0:
  447. stream = mimetype = None
  448. else:
  449. mimetype = 'application/octet-stream'
  450. return stream, mimetype
  451. def _tombstone_response(e, uid):
  452. headers = {
  453. 'Link': '<{}/fcr:tombstone>; rel="hasTombstone"'.format(request.url),
  454. } if e.uid == uid else {}
  455. return str(e), 410, headers
  456. def set_post_put_params():
  457. """
  458. Sets handling and content disposition for POST and PUT by parsing headers.
  459. """
  460. handling = 'strict'
  461. if 'prefer' in request.headers:
  462. prefer = g.tbox.parse_rfc7240(request.headers['prefer'])
  463. logger.debug('Parsed Prefer header: {}'.format(prefer))
  464. if 'handling' in prefer:
  465. handling = prefer['handling']['value']
  466. try:
  467. disposition = g.tbox.parse_rfc7240(
  468. request.headers['content-disposition'])
  469. except KeyError:
  470. disposition = None
  471. return handling, disposition
  472. def parse_repr_options(retr_opts):
  473. """
  474. Set options to retrieve IMR.
  475. Ideally, IMR retrieval is done once per request, so all the options
  476. are set once in the `imr()` property.
  477. :param dict retr_opts:: Options parsed from `Prefer` header.
  478. """
  479. logger.debug('Parsing retrieval options: {}'.format(retr_opts))
  480. imr_options = {}
  481. if retr_opts.get('value') == 'minimal':
  482. imr_options = {
  483. 'embed_children' : False,
  484. 'incl_children' : False,
  485. 'incl_inbound' : False,
  486. 'incl_srv_mgd' : False,
  487. }
  488. else:
  489. # Default.
  490. imr_options = {
  491. 'embed_children' : False,
  492. 'incl_children' : True,
  493. 'incl_inbound' : False,
  494. 'incl_srv_mgd' : True,
  495. }
  496. # Override defaults.
  497. if 'parameters' in retr_opts:
  498. include = retr_opts['parameters']['include'].split(' ') \
  499. if 'include' in retr_opts['parameters'] else []
  500. omit = retr_opts['parameters']['omit'].split(' ') \
  501. if 'omit' in retr_opts['parameters'] else []
  502. logger.debug('Include: {}'.format(include))
  503. logger.debug('Omit: {}'.format(omit))
  504. if str(Ldpr.EMBED_CHILD_RES_URI) in include:
  505. imr_options['embed_children'] = True
  506. if str(Ldpr.RETURN_CHILD_RES_URI) in omit:
  507. imr_options['incl_children'] = False
  508. if str(Ldpr.RETURN_INBOUND_REF_URI) in include:
  509. imr_options['incl_inbound'] = True
  510. if str(Ldpr.RETURN_SRV_MGD_RES_URI) in omit:
  511. imr_options['incl_srv_mgd'] = False
  512. logger.debug('Retrieval options: {}'.format(pformat(imr_options)))
  513. return imr_options
  514. def _headers_from_metadata(rsrc, out_fmt='text/turtle'):
  515. """
  516. Create a dict of headers from a metadata graph.
  517. :param lakesuperior.model.ldpr.Ldpr rsrc: Resource to extract metadata
  518. from.
  519. """
  520. out_headers = defaultdict(list)
  521. digest = rsrc.metadata.value(rsrc.uri, nsc['premis'].hasMessageDigest)
  522. # Only add ETag and digest if output is not RDF.
  523. if digest and out_fmt == 'non_rdf':
  524. digest_components = digest.split(':')
  525. cksum_hex = digest_components[-1]
  526. cksum = bytearray.fromhex(cksum_hex)
  527. digest_algo = digest_components[-2]
  528. etag_str = (
  529. 'W/"{}"'.format(cksum_hex)
  530. if nsc['ldp'].RDFSource in rsrc.ldp_types
  531. else cksum_hex)
  532. out_headers['ETag'] = etag_str,
  533. out_headers['Digest'] = '{}={}'.format(
  534. digest_algo.upper(), b64encode(cksum).decode('ascii'))
  535. last_updated_term = rsrc.metadata.value(nsc['fcrepo'].lastModified)
  536. if last_updated_term:
  537. out_headers['Last-Modified'] = arrow.get(last_updated_term)\
  538. .format('ddd, D MMM YYYY HH:mm:ss Z')
  539. for t in rsrc.ldp_types:
  540. out_headers['Link'].append('{};rel="type"'.format(t.n3()))
  541. mimetype = rsrc.metadata.value(nsc['ebucore'].hasMimeType)
  542. if mimetype:
  543. out_headers['Content-Type'] = mimetype
  544. return out_headers