ldp_rs.py 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  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. ResourceNotExistsError, ServerManagedTermError
  11. from lakesuperior.util.translator import Translator
  12. class LdpRs(Ldpr):
  13. '''LDP-RS (LDP RDF source).
  14. Definition: https://www.w3.org/TR/ldp/#ldprs
  15. '''
  16. base_types = {
  17. nsc['ldp'].RDFSource
  18. }
  19. std_headers = {
  20. 'Accept-Post' : {
  21. 'text/turtle',
  22. 'text/rdf+n3',
  23. 'text/n3',
  24. 'application/rdf+xml',
  25. 'application/n-triples',
  26. 'application/ld+json',
  27. 'multipart/form-data',
  28. 'application/sparql-update',
  29. },
  30. 'Accept-Patch' : {
  31. 'application/sparql-update',
  32. },
  33. }
  34. def head(self):
  35. '''
  36. Return values for the headers.
  37. '''
  38. headers = self.rdfly.headers
  39. for t in self.ldp_types:
  40. headers['Link'].append('{};rel="type"'.format(t.identifier.n3()))
  41. return headers
  42. def get(self, inbound=False, children=True, srv_mgd=True):
  43. '''
  44. https://www.w3.org/TR/ldp/#ldpr-HTTP_GET
  45. '''
  46. im_rsrc = self.rdfly.out_rsrc(inbound)
  47. if not len(im_rsrc.graph):
  48. raise ResourceNotExistsError()
  49. return Translator.globalize_rsrc(im_rsrc)
  50. @transactional
  51. def post(self, data, format='text/turtle'):
  52. '''
  53. https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
  54. Perform a POST action after a valid resource URI has been found.
  55. '''
  56. g = Graph()
  57. g.parse(data=data, format=format, publicID=self.urn)
  58. self._check_mgd_terms_rdf(g)
  59. for t in self.base_types:
  60. g.add((self.urn, RDF.type, t))
  61. self.rdfly.create_rsrc(g)
  62. self._set_containment_rel()
  63. @transactional
  64. def put(self, data, format='text/turtle'):
  65. '''
  66. https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
  67. '''
  68. g = Graph()
  69. g.parse(data=data, format=format, publicID=self.urn)
  70. self._check_mgd_terms_rdf(g)
  71. for t in self.base_types:
  72. g.add((self.urn, RDF.type, t))
  73. self.rdfly.create_or_replace_rsrc(g)
  74. self._set_containment_rel()
  75. @transactional
  76. @must_exist
  77. def patch(self, data):
  78. '''
  79. https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
  80. '''
  81. self._check_mgd_terms_sparql(data)
  82. self.rdfly.patch_rsrc(data)
  83. ## PROTECTED METHODS ##
  84. def _check_mgd_terms_rdf(self, g):
  85. '''
  86. Check whether server-managed terms are in a RDF payload.
  87. '''
  88. offending_subjects = set(g.subjects()) & srv_mgd_subjects
  89. if offending_subjects:
  90. raise ServerManagedTermError('Some subjects in RDF payload '
  91. 'are server managed and cannot be modified: {}'
  92. .format(' , '.join(offending_subjects)))
  93. offending_predicates = set(g.predicates()) & srv_mgd_predicates
  94. if offending_predicates:
  95. raise ServerManagedTermError('Some predicates in RDF payload '
  96. 'are server managed and cannot be modified: {}'
  97. .format(' , '.join(offending_predicates)))
  98. offending_types = set(g.objects(predicate=RDF.type)) & srv_mgd_types
  99. if offending_types:
  100. raise ServerManagedTermError('Some RDF types in RDF payload '
  101. 'are server managed and cannot be modified: {}'
  102. .format(' , '.join(offending_types)))
  103. def _check_mgd_terms_sparql(self, q):
  104. '''
  105. Parse tokens in update query and verify that none of the terms being
  106. modified is server-managed.
  107. The only reasonable way to do this is to perform the query on a copy
  108. and verify if any of the server managed terms is in the delta. If it
  109. is, it means that some server-managed term is being modified and
  110. an error should be raised.
  111. NOTE: This only checks if a server-managed term is effectively being
  112. modified. If a server-managed term is present in the query but does not
  113. cause any change in the updated resource, no error is raised.
  114. '''
  115. before_test = self.rdfly.extract_imr().graph
  116. after_test = deepcopy(before_test)
  117. after_test.update(q)
  118. delta = before_test ^ after_test
  119. self._logger.info('Delta: {}'.format(delta.serialize(format='turtle')
  120. .decode('utf8')))
  121. for s,p,o in delta:
  122. if s in srv_mgd_subjects:
  123. raise ServerManagedTermError(
  124. 'Subject {} is server managed and cannot be modified.'
  125. .format(s))
  126. if p in srv_mgd_predicates:
  127. raise ServerManagedTermError(
  128. 'Predicate {} is server managed and cannot be modified.'
  129. .format(p))
  130. if p == RDF.type and o in srv_mgd_types:
  131. raise ServerManagedTermError(
  132. 'RDF type {} is server managed and cannot be modified.'
  133. .format(o))
  134. class Ldpc(LdpRs):
  135. '''LDPC (LDP Container).'''
  136. def __init__(self, uuid):
  137. super().__init__(uuid)
  138. self.base_types.update({
  139. nsc['ldp'].Container,
  140. })
  141. class LdpBc(Ldpc):
  142. '''LDP-BC (LDP Basic Container).'''
  143. def __init__(self, uuid):
  144. super().__init__(uuid)
  145. self.base_types.update({
  146. nsc['ldp'].BasicContainer,
  147. })
  148. class LdpDc(Ldpc):
  149. '''LDP-DC (LDP Direct Container).'''
  150. def __init__(self, uuid):
  151. super().__init__(uuid)
  152. self.base_types.update({
  153. nsc['ldp'].DirectContainer,
  154. })
  155. class LdpIc(Ldpc):
  156. '''LDP-IC (LDP Indirect Container).'''
  157. def __init__(self, uuid):
  158. super().__init__(uuid)
  159. self.base_types.update({
  160. nsc['ldp'].IndirectContainer,
  161. })