ldp_rs.py 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. from copy import deepcopy
  2. from flask import g
  3. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  4. from lakesuperior.model.ldpr import Ldpr, atomic
  5. class LdpRs(Ldpr):
  6. '''LDP-RS (LDP RDF source).
  7. Definition: https://www.w3.org/TR/ldp/#ldprs
  8. '''
  9. def __init__(self, uuid, repr_opts={}, handling='lenient', **kwargs):
  10. '''
  11. Extends Ldpr.__init__ by adding LDP-RS specific parameters.
  12. @param handling (string) One of `strict`, `lenient` (the default) or
  13. `none`. `strict` raises an error if a server-managed term is in the
  14. graph. `lenient` removes all sever-managed triples encountered. `none`
  15. skips all server-managed checks. It is used for internal modifications.
  16. '''
  17. super().__init__(uuid, **kwargs)
  18. # provided_imr can be empty. If None, it is an outbound resource.
  19. if self.provided_imr is not None:
  20. self.workflow = self.WRKF_INBOUND
  21. else:
  22. self.workflow = self.WRKF_OUTBOUND
  23. self._imr_options = repr_opts
  24. self.handling = handling
  25. ## LDP METHODS ##
  26. @atomic
  27. def patch(self, update_str):
  28. '''
  29. https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
  30. Update an existing resource by applying a SPARQL-UPDATE query.
  31. @param update_str (string) SPARQL-Update staements.
  32. '''
  33. local_update_str = update_str.replace('<>', self.urn.n3()).replace(
  34. g.webroot + '/', nsc['fcres']).replace(
  35. g.webroot, nsc['fcres'])
  36. delta = self._sparql_delta(local_update_str)
  37. self._ensure_single_subject_rdf(delta[0] | delta[1])
  38. return self._modify_rsrc(self.RES_UPDATED, *delta)
  39. def _sparql_delta(self, q):
  40. '''
  41. Calculate the delta obtained by a SPARQL Update operation.
  42. This is a critical component of the SPARQL query prcess and does a
  43. couple of things:
  44. 1. It ensures that no resources outside of the subject of the request
  45. are modified (e.g. by variable subjects)
  46. 2. It verifies that none of the terms being modified is server managed.
  47. This method extracts an in-memory copy of the resource and performs the
  48. query on that once it has checked if any of the server managed terms is
  49. in the delta. If it is, it raises an exception.
  50. NOTE: This only checks if a server-managed term is effectively being
  51. modified. If a server-managed term is present in the query but does not
  52. cause any change in the updated resource, no error is raised.
  53. @return tuple(rdflib.Graph) Remove and add graphs. These can be used
  54. with `BaseStoreLayout.update_resource` and/or recorded as separate
  55. events in a provenance tracking system.
  56. '''
  57. self._logger.debug('Provided SPARQL query: {}'.format(q))
  58. pre_gr = self.imr.graph
  59. post_gr = deepcopy(pre_gr)
  60. post_gr.update(q)
  61. remove_gr, add_gr = self._dedup_deltas(pre_gr, post_gr)
  62. #self._logger.info('Removing: {}'.format(
  63. # remove_gr.serialize(format='turtle').decode('utf8')))
  64. #self._logger.info('Adding: {}'.format(
  65. # add_gr.serialize(format='turtle').decode('utf8')))
  66. remove_gr = self._check_mgd_terms(remove_gr)
  67. add_gr = self._check_mgd_terms(add_gr)
  68. return remove_gr, add_gr
  69. class Ldpc(LdpRs):
  70. '''LDPC (LDP Container).'''
  71. def __init__(self, uuid, *args, **kwargs):
  72. super().__init__(uuid, *args, **kwargs)
  73. self.base_types.update({
  74. nsc['fcrepo'].Container,
  75. nsc['ldp'].Container,
  76. })
  77. class LdpBc(Ldpc):
  78. '''LDP-BC (LDP Basic Container).'''
  79. def __init__(self, uuid, *args, **kwargs):
  80. super().__init__(uuid, *args, **kwargs)
  81. self.base_types.update({
  82. nsc['ldp'].BasicContainer,
  83. })
  84. class LdpDc(Ldpc):
  85. '''LDP-DC (LDP Direct Container).'''
  86. def __init__(self, uuid, *args, **kwargs):
  87. super().__init__(uuid, *args, **kwargs)
  88. self.base_types.update({
  89. nsc['ldp'].DirectContainer,
  90. })
  91. class LdpIc(Ldpc):
  92. '''LDP-IC (LDP Indirect Container).'''
  93. def __init__(self, uuid, *args, **kwargs):
  94. super().__init__(uuid, *args, **kwargs)
  95. self.base_types.update({
  96. nsc['ldp'].IndirectContainer,
  97. })