ldp_rs.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. from copy import deepcopy
  2. from rdflib import Graph
  3. from rdflib.namespace import RDF, XSD
  4. from rdflib.plugins.sparql.parser import parseUpdate
  5. from rdflib.term import URIRef, Literal, Variable
  6. from lakesuperior.dictionaries.namespaces import ns_collection as nsc
  7. from lakesuperior.dictionaries.srv_mgd_terms import srv_mgd_subjects, \
  8. srv_mgd_predicates, srv_mgd_types
  9. from lakesuperior.model.ldpr import Ldpr, transactional, must_exist
  10. from lakesuperior.exceptions import ResourceNotExistsError, \
  11. ServerManagedTermError, SingleSubjectError
  12. from lakesuperior.util.translator import Translator
  13. class LdpRs(Ldpr):
  14. '''LDP-RS (LDP RDF source).
  15. Definition: https://www.w3.org/TR/ldp/#ldprs
  16. '''
  17. base_types = {
  18. nsc['ldp'].RDFSource
  19. }
  20. std_headers = {
  21. 'Accept-Post' : {
  22. 'text/turtle',
  23. 'text/rdf+n3',
  24. 'text/n3',
  25. 'application/rdf+xml',
  26. 'application/n-triples',
  27. 'application/ld+json',
  28. 'multipart/form-data',
  29. 'application/sparql-update',
  30. },
  31. 'Accept-Patch' : {
  32. 'application/sparql-update',
  33. },
  34. }
  35. def head(self):
  36. '''
  37. Return values for the headers.
  38. '''
  39. headers = self.rdfly.headers
  40. for t in self.ldp_types:
  41. headers['Link'].append('{};rel="type"'.format(t.identifier.n3()))
  42. return headers
  43. def get(self, inbound=False, children=True, srv_mgd=True):
  44. '''
  45. https://www.w3.org/TR/ldp/#ldpr-HTTP_GET
  46. '''
  47. im_rsrc = self.rdfly.out_rsrc(inbound)
  48. if not len(im_rsrc.graph):
  49. raise ResourceNotExistsError(im_rsrc.uuid)
  50. return Translator.globalize_rsrc(im_rsrc)
  51. @transactional
  52. def post(self, data, format='text/turtle'):
  53. '''
  54. https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
  55. Perform a POST action after a valid resource URI has been found.
  56. '''
  57. g = Graph().parse(data=data, format=format, publicID=self.urn)
  58. self._check_mgd_terms(g)
  59. self._ensure_single_subject_rdf(g)
  60. for t in self.base_types:
  61. g.add((self.urn, RDF.type, t))
  62. self.rdfly.create_rsrc(g)
  63. self._set_containment_rel()
  64. @transactional
  65. def put(self, data, format='text/turtle'):
  66. '''
  67. https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
  68. '''
  69. g = Graph().parse(data=data, format=format, publicID=self.urn)
  70. self._check_mgd_terms(g)
  71. self._ensure_single_subject_rdf(g)
  72. for t in self.base_types:
  73. g.add((self.urn, RDF.type, t))
  74. self.rdfly.create_or_replace_rsrc(g)
  75. self._set_containment_rel()
  76. @transactional
  77. @must_exist
  78. def patch(self, data):
  79. '''
  80. https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
  81. '''
  82. remove, add = self._sparql_delta(data)
  83. self.rdfly.modify_rsrc(remove, add)
  84. ## PROTECTED METHODS ##
  85. def _check_mgd_terms(self, g):
  86. '''
  87. Check whether server-managed terms are in a RDF payload.
  88. '''
  89. offending_subjects = set(g.subjects()) & srv_mgd_subjects
  90. if offending_subjects:
  91. raise ServerManagedTermError(offending_subjects, 's')
  92. offending_predicates = set(g.predicates()) & srv_mgd_predicates
  93. if offending_predicates:
  94. raise ServerManagedTermError(offending_predicates, 'p')
  95. offending_types = set(g.objects(predicate=RDF.type)) & srv_mgd_types
  96. if offending_types:
  97. raise ServerManagedTermError(offending_types, 't')
  98. def _sparql_delta(self, q):
  99. '''
  100. Calculate the delta obtained by a SPARQL Update operation.
  101. This does a couple of extra things:
  102. 1. It ensures that no resources outside of the subject of the request
  103. are modified (e.g. by variable subjects)
  104. 2. It verifies that none of the terms being modified is server-managed.
  105. This method extracts an in-memory copy of the resource and performs the
  106. query on that once it has checked if any of the server managed terms is
  107. in the delta. If it is, it raises an exception.
  108. NOTE: This only checks if a server-managed term is effectively being
  109. modified. If a server-managed term is present in the query but does not
  110. cause any change in the updated resource, no error is raised.
  111. @return tuple Remove and add triples. These can be used with
  112. `BaseStoreLayout.update_resource`.
  113. '''
  114. pre_g = self.rdfly.extract_imr().graph
  115. post_g = deepcopy(pre_g)
  116. post_g.update(q)
  117. remove = pre_g - post_g
  118. add = post_g - pre_g
  119. self._logger.info('Removing: {}'.format(
  120. remove.serialize(format='turtle').decode('utf8')))
  121. self._logger.info('Adding: {}'.format(
  122. add.serialize(format='turtle').decode('utf8')))
  123. self._check_mgd_terms(remove + add)
  124. return remove, add
  125. def _ensure_single_subject_rdf(self, g):
  126. '''
  127. Ensure that a RDF payload for a POST or PUT has a single resource.
  128. '''
  129. for s in set(g.subjects()):
  130. if not s == self.uri:
  131. return SingleSubjectError(s, self.uri)
  132. class Ldpc(LdpRs):
  133. '''LDPC (LDP Container).'''
  134. def __init__(self, uuid):
  135. super().__init__(uuid)
  136. self.base_types.update({
  137. nsc['ldp'].Container,
  138. })
  139. class LdpBc(Ldpc):
  140. '''LDP-BC (LDP Basic Container).'''
  141. def __init__(self, uuid):
  142. super().__init__(uuid)
  143. self.base_types.update({
  144. nsc['ldp'].BasicContainer,
  145. })
  146. class LdpDc(Ldpc):
  147. '''LDP-DC (LDP Direct Container).'''
  148. def __init__(self, uuid):
  149. super().__init__(uuid)
  150. self.base_types.update({
  151. nsc['ldp'].DirectContainer,
  152. })
  153. class LdpIc(Ldpc):
  154. '''LDP-IC (LDP Indirect Container).'''
  155. def __init__(self, uuid):
  156. super().__init__(uuid)
  157. self.base_types.update({
  158. nsc['ldp'].IndirectContainer,
  159. })