Browse Source

Move POST and PUT methods from LDPR to LDP-RS; add server-managed term
check for RDF payload in PUT and POST.

Stefano Cossu 7 years ago
parent
commit
951fe1a949
3 changed files with 72 additions and 37 deletions
  1. 8 2
      lakesuperior/endpoints/ldp.py
  2. 64 3
      lakesuperior/model/ldp_rs.py
  3. 0 32
      lakesuperior/model/ldpr.py

+ 8 - 2
lakesuperior/endpoints/ldp.py

@@ -94,7 +94,10 @@ def post_resource(parent):
     except InvalidResourceError as e:
     except InvalidResourceError as e:
         return str(e), 409
         return str(e), 409
 
 
-    rsrc.post(request.get_data().decode('utf-8'))
+    try:
+        rsrc.post(request.get_data().decode('utf-8'))
+    except ServerManagedTermError as e:
+        return str(e), 412
 
 
     headers.update({
     headers.update({
         'Location' : rsrc.uri,
         'Location' : rsrc.uri,
@@ -111,7 +114,10 @@ def put_resource(uuid):
     headers = std_headers
     headers = std_headers
     rsrc = Ldpc(uuid)
     rsrc = Ldpc(uuid)
 
 
-    rsrc.put(request.get_data().decode('utf-8'))
+    try:
+        rsrc.put(request.get_data().decode('utf-8'))
+    except ServerManagedTermError as e:
+        return str(e), 412
     return '', 204, headers
     return '', 204, headers
 
 
 
 

+ 64 - 3
lakesuperior/model/ldp_rs.py

@@ -1,5 +1,6 @@
 from copy import deepcopy
 from copy import deepcopy
 
 
+from rdflib import Graph
 from rdflib.namespace import RDF, XSD
 from rdflib.namespace import RDF, XSD
 from rdflib.plugins.sparql.parser import parseUpdate
 from rdflib.plugins.sparql.parser import parseUpdate
 from rdflib.term import URIRef, Literal, Variable
 from rdflib.term import URIRef, Literal, Variable
@@ -60,21 +61,81 @@ class LdpRs(Ldpr):
         return Translator.globalize_rsrc(im_rsrc)
         return Translator.globalize_rsrc(im_rsrc)
 
 
 
 
+    @transactional
+    def post(self, data, format='text/turtle'):
+        '''
+        https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
+
+        Perform a POST action after a valid resource URI has been found.
+        '''
+        g = Graph()
+        g.parse(data=data, format=format, publicID=self.urn)
+        self._check_mgd_terms_rdf(g)
+
+        for t in self.base_types:
+            g.add((self.urn, RDF.type, t))
+
+        self.rdfly.create_rsrc(g)
+
+        self._set_containment_rel()
+
+
+    @transactional
+    def put(self, data, format='text/turtle'):
+        '''
+        https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
+        '''
+        g = Graph()
+        g.parse(data=data, format=format, publicID=self.urn)
+        self._check_mgd_terms_rdf(g)
+
+        for t in self.base_types:
+            g.add((self.urn, RDF.type, t))
+
+        self.rdfly.create_or_replace_rsrc(g)
+
+        self._set_containment_rel()
+
+
     @transactional
     @transactional
     @must_exist
     @must_exist
     def patch(self, data):
     def patch(self, data):
         '''
         '''
         https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
         https://www.w3.org/TR/ldp/#ldpr-HTTP_PATCH
         '''
         '''
-        self._check_mgd_terms(data)
+        self._check_mgd_terms_sparql(data)
 
 
         self.rdfly.patch_rsrc(data)
         self.rdfly.patch_rsrc(data)
 
 
 
 
     ## PROTECTED METHODS ##
     ## PROTECTED METHODS ##
 
 
-    def _check_mgd_terms(self, q):
-        '''Parse tokens in update query and verify that none of the terms being
+    def _check_mgd_terms_rdf(self, g):
+        '''
+        Check whether server-managed terms are in a RDF payload.
+        '''
+        offending_subjects = set(g.subjects()) & srv_mgd_subjects
+        if offending_subjects:
+            raise ServerManagedTermError('Some subjects in RDF payload '
+                    'are server managed and cannot be modified: {}'
+                    .format(' , '.join(offending_subjects)))
+
+        offending_predicates = set(g.predicates()) & srv_mgd_predicates
+        if offending_predicates:
+            raise ServerManagedTermError('Some predicates in RDF payload '
+                    'are server managed and cannot be modified: {}'
+                    .format(' , '.join(offending_predicates)))
+
+        offending_types = set(g.objects(predicate=RDF.type)) & srv_mgd_types
+        if offending_types:
+            raise ServerManagedTermError('Some RDF types in RDF payload '
+                    'are server managed and cannot be modified: {}'
+                    .format(' , '.join(offending_types)))
+
+
+    def _check_mgd_terms_sparql(self, q):
+        '''
+        Parse tokens in update query and verify that none of the terms being
         modified is server-managed.
         modified is server-managed.
 
 
         The only reasonable way to do this is to perform the query on a copy
         The only reasonable way to do this is to perform the query on a copy

+ 0 - 32
lakesuperior/model/ldpr.py

@@ -371,38 +371,6 @@ class Ldpr(metaclass=ABCMeta):
 
 
     ## LDP METHODS ##
     ## LDP METHODS ##
 
 
-    @transactional
-    def post(self, data, format='text/turtle'):
-        '''
-        https://www.w3.org/TR/ldp/#ldpr-HTTP_POST
-
-        Perform a POST action after a valid resource URI has been found.
-        '''
-        g = Graph()
-        g.parse(data=data, format=format, publicID=self.urn)
-        for t in self.base_types:
-            g.add((self.urn, RDF.type, t))
-
-        self.rdfly.create_rsrc(g)
-
-        self._set_containment_rel()
-
-
-    @transactional
-    def put(self, data, format='text/turtle'):
-        '''
-        https://www.w3.org/TR/ldp/#ldpr-HTTP_PUT
-        '''
-        g = Graph()
-        g.parse(data=data, format=format, publicID=self.urn)
-        for t in self.base_types:
-            g.add((self.urn, RDF.type, t))
-
-        self.rdfly.create_or_replace_rsrc(g)
-
-        self._set_containment_rel()
-
-
     @transactional
     @transactional
     @must_exist
     @must_exist
     def delete(self):
     def delete(self):