ldp_rs.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  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_rdf(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_rdf(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. self._check_mgd_terms_sparql(data)
  83. self._ensure_single_subject_sparql_update(data)
  84. self.rdfly.patch_rsrc(data)
  85. ## PROTECTED METHODS ##
  86. def _check_mgd_terms_rdf(self, g):
  87. '''
  88. Check whether server-managed terms are in a RDF payload.
  89. '''
  90. offending_subjects = set(g.subjects()) & srv_mgd_subjects
  91. if offending_subjects:
  92. raise ServerManagedTermError(offending_subjects, 's')
  93. offending_predicates = set(g.predicates()) & srv_mgd_predicates
  94. if offending_predicates:
  95. raise ServerManagedTermError(offending_predicates, 'p')
  96. offending_types = set(g.objects(predicate=RDF.type)) & srv_mgd_types
  97. if offending_types:
  98. raise ServerManagedTermError(offending_types, 't')
  99. def _check_mgd_terms_sparql(self, q):
  100. '''
  101. Parse tokens in update query and verify that none of the terms being
  102. modified is server-managed.
  103. The only reasonable way to do this is to perform the query on a copy
  104. and verify if any of the server managed terms is in the delta. If it
  105. is, it means that some server-managed term is being modified and
  106. an error should be raised.
  107. NOTE: This only checks if a server-managed term is effectively being
  108. modified. If a server-managed term is present in the query but does not
  109. cause any change in the updated resource, no error is raised.
  110. '''
  111. before_test = self.rdfly.extract_imr().graph
  112. after_test = deepcopy(before_test)
  113. after_test.update(q)
  114. delta = before_test ^ after_test
  115. self._logger.info('Delta: {}'.format(delta.serialize(format='turtle')
  116. .decode('utf8')))
  117. for s,p,o in delta:
  118. if s in srv_mgd_subjects:
  119. raise ServerManagedTermError(s, 's')
  120. if p in srv_mgd_predicates:
  121. raise ServerManagedTermError(p, 'p')
  122. if p == RDF.type and o in srv_mgd_types:
  123. raise ServerManagedTermError(o, 't')
  124. def _ensure_single_subject_sparql_update(self, qs):
  125. '''
  126. Ensure that a SPARQL update query only affects the current resource.
  127. This prevents a query such as
  128. DELETE {
  129. ?s a ns:Class .
  130. }
  131. INSERT {
  132. ?s a ns:OtherClass .
  133. }
  134. WHERE {
  135. ?s a ns:Class .
  136. }
  137. from affecting multiple resources.
  138. '''
  139. # @TODO This requires some quirky algebra parsing and manipulation.
  140. # Will need to investigate.
  141. pass
  142. def _ensure_single_subject_rdf(self, g):
  143. '''
  144. Ensure that a RDF payload for a POST or PUT has a single resource.
  145. '''
  146. if not all(s == self.uri for s in set(g.subjects())):
  147. return SingleSubjectError(self.uri)
  148. class Ldpc(LdpRs):
  149. '''LDPC (LDP Container).'''
  150. def __init__(self, uuid):
  151. super().__init__(uuid)
  152. self.base_types.update({
  153. nsc['ldp'].Container,
  154. })
  155. class LdpBc(Ldpc):
  156. '''LDP-BC (LDP Basic Container).'''
  157. def __init__(self, uuid):
  158. super().__init__(uuid)
  159. self.base_types.update({
  160. nsc['ldp'].BasicContainer,
  161. })
  162. class LdpDc(Ldpc):
  163. '''LDP-DC (LDP Direct Container).'''
  164. def __init__(self, uuid):
  165. super().__init__(uuid)
  166. self.base_types.update({
  167. nsc['ldp'].DirectContainer,
  168. })
  169. class LdpIc(Ldpc):
  170. '''LDP-IC (LDP Indirect Container).'''
  171. def __init__(self, uuid):
  172. super().__init__(uuid)
  173. self.base_types.update({
  174. nsc['ldp'].IndirectContainer,
  175. })