ldp_rs.py 4.4 KB

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