ldp_rs.py 4.4 KB

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