Explorar o código

POST version (not complete).

Stefano Cossu %!s(int64=7) %!d(string=hai) anos
pai
achega
8564390f5d
Modificáronse 3 ficheiros con 95 adicións e 8 borrados
  1. 6 3
      lakesuperior/endpoints/ldp.py
  2. 72 5
      lakesuperior/model/ldpr.py
  3. 17 0
      lakesuperior/toolbox.py

+ 6 - 3
lakesuperior/endpoints/ldp.py

@@ -194,13 +194,16 @@ def post_version(uuid):
     if not label:
         return 'Specify label for version.', 400
 
-    parent_uuid = uuid + '/fcr:versions'
     try:
-        parent_rsrc = Ldpr.create_version(parent_uuid)
+        ver_uri = Ldpr.outbound_inst(uuid).create_version(label)
+    except ResourceNotExistsError as e:
+        return str(e), 404
     except InvalidResourceError as e:
         return str(e), 409
+    except TombstoneError as e:
+        return _tombstone_response(e, uuid)
     else:
-        return '', 201, {'Location': parent_rsrc.uri}
+        return '', 201, {'Location': ver_uri}
 
 
 @ldp.route('/<path:uuid>', methods=['PUT'], strict_slashes=False)

+ 72 - 5
lakesuperior/model/ldpr.py

@@ -102,6 +102,8 @@ class Ldpr(metaclass=ABCMeta):
     RES_DELETED = '_delete_'
     RES_UPDATED = '_update_'
 
+    RES_VER_CONT_LABEL = 'fcr:versions'
+
     protected_pred = (
         nsc['fcrepo'].created,
         nsc['fcrepo'].createdBy,
@@ -114,7 +116,7 @@ class Ldpr(metaclass=ABCMeta):
     ## STATIC & CLASS METHODS ##
 
     @classmethod
-    def outbound_inst(cls, uuid, repr_opts=None, **kwargs):
+    def outbound_inst(cls, uuid, repr_opts={}, **kwargs):
         '''
         Create an instance for retrieval purposes.
 
@@ -367,8 +369,9 @@ class Ldpr(metaclass=ABCMeta):
 
         Internal URNs are replaced by global URIs using the endpoint webroot.
         '''
-        # Remove digest hash.
+        # Remove digest hash and version information.
         self.imr.remove(nsc['premis'].hasMessageDigest)
+        self.imr.remove(nsc['fcrepo'].hasVersion)
 
         if not self._imr_options.get('incl_srv_mgd', True):
             for p in srv_mgd_predicates:
@@ -394,6 +397,16 @@ class Ldpr(metaclass=ABCMeta):
             return self.rdfly.ask_rsrc_exists(self.urn)
 
 
+    @property
+    def has_versions(self):
+        '''
+        Whether if a current resource has versions.
+
+        @return boolean
+        '''
+        return bool(self.imr.value(nsc['fcrepo'].hasVersions, any=False))
+
+
     @property
     def types(self):
         '''All RDF types.
@@ -523,6 +536,49 @@ class Ldpr(metaclass=ABCMeta):
         self.rdfly.modify_dataset(remove_trp)
 
 
+    @atomic
+    def create_version(self, label):
+        '''
+        Create a new version of the resource.
+
+        NOTE: This creates an event only for the resource being updated (due
+        to the added `hasVersion` triple and possibly to the `hasVersions` one)
+        but not for the version being created.
+
+        @param label Version label. If already existing, an exception is
+        raised.
+        '''
+        add_gr = Graph()
+        vers_uuid = '{}/{}'.format(self.uuid, self.RES_VER_CONT_LABEL)
+        ver_uuid = '{}/{}'.format(vers_uuid, label)
+        ver_urn = nsc['fcres'][ver_uuid]
+        add_gr.add((ver_urn, RDF.type, nsc['fcrepo'].Version))
+        for t in self.imr.graph:
+            if t[1] == RDF.type and t[2] in {
+                nsc['fcrepo'].Resource,
+                nsc['fcrepo'].Container,
+                nsc['fcrepo'].Binary,
+            }:
+                pass
+            else:
+                add_gr.add((
+                        g.tbox.replace_term_domain(t[0], self.urn, ver_urn),
+                        t[1], t[2]))
+
+        self.rdfly.modify_dataset(add_trp=add_gr)
+
+        add_gr = Graph()
+        add_gr.add((
+            self.urn, nsc['fcrepo'].hasVersion, ver_urn))
+        if not self.has_versions:
+            add_gr.add((
+                self.urn, nsc['fcrepo'].hasVersions, nsc['fcres'][vers_uuid]))
+
+        self._modify_rsrc(self.RES_UPDATED, add_trp=add_gr)
+
+        return g.tbox.uuid_to_uri(vers_uuid)
+
+
     ## PROTECTED METHODS ##
 
     def _create_or_replace_rsrc(self, create_only=False):
@@ -615,6 +671,18 @@ class Ldpr(metaclass=ABCMeta):
         return self.RES_DELETED
 
 
+    def _create_version_container(self):
+        '''
+        Create the relationship with `fcr:versions` the first time a version is
+        created.
+        '''
+        add_gr = Graph()
+        add_gr.add((self.urn, nsc['fcrepo'].hasVersions,
+                URIRef(str(self.urn) + '/fcr:versions')))
+
+        self._modify_rsrc(self.RES_UPDATED, add_trp=add_gr)
+
+
     def _modify_rsrc(self, ev_type, remove_trp=Graph(), add_trp=Graph()):
         '''
         Low-level method to modify a graph for a single resource.
@@ -637,9 +705,8 @@ class Ldpr(metaclass=ABCMeta):
                     if trp[1] == nsc['fcrepo'].createdBy }
         else:
             merge_gr = remove_trp | add_trp
-            type = merge_gr[ self.urn : RDF.type ]
-            actor = merge_gr[ self.urn : nsc['fcrepo'].createdBy ]
-
+            type = merge_gr[self.urn : RDF.type]
+            actor = merge_gr[self.urn : nsc['fcrepo'].createdBy]
 
         return self.rdfly.modify_dataset(remove_trp, add_trp, metadata={
             'ev_type' : ev_type,

+ 17 - 0
lakesuperior/toolbox.py

@@ -19,6 +19,23 @@ class Toolbox:
 
     ROOT_NODE_URN = nsc['fcsystem'].root
 
+    def replace_term_domain(self, term, search, replace):
+        '''
+        Replace the domain of a term.
+
+        @param term (URIRef) The term (URI) to change.
+        @param search (string) Domain string to replace.
+        @param replace (string) Domain string to use for replacement.
+
+        @return URIRef
+        '''
+        s = str(term)
+        if s.startswith(search):
+            s = s.replace(search, replace)
+
+        return URIRef(s)
+
+
     def uuid_to_uri(self, uuid):
         '''Convert a UUID to a URI.