Ver código fonte

Make extract_imr compatible with bdb back end; add RDF types for resource graphs.

Stefano Cossu 7 anos atrás
pai
commit
dc50b0a51d

+ 4 - 2
data/bootstrap/rsrc_centric_layout.sparql

@@ -16,7 +16,9 @@ INSERT DATA {
   }
 
   GRAPH <info:fcsystem/meta> {
-    <info:fcsystem/graph/admin/> foaf:primaryTopic <info:fcres/> .
-    <info:fcsystem/graph/userdata/_main/> foaf:primaryTopic <info:fcres/> .
+    <info:fcsystem/graph/admin/> a <info:fcsystem/AdminGraph> ;
+      foaf:primaryTopic <info:fcres/> .
+    <info:fcsystem/graph/userdata/_main/> a <info:fcsystem/UserProvidedGraph> ;
+      foaf:primaryTopic <info:fcres/> .
   }
 }

+ 65 - 24
lakesuperior/store_layouts/ldp_rs/rsrc_centric_layout.py

@@ -106,6 +106,13 @@ class RsrcCentricLayout:
         },
     }
 
+    # RDF types of graphs by prefix.
+    graph_ns_types = {
+        nsc['fcadmin']: nsc['fcsystem'].AdminGraph,
+        nsc['fcmain']: nsc['fcsystem'].UserProvidedGraph,
+        nsc['fcstruct']: nsc['fcsystem'].StructureGraph,
+    }
+
 
     ## MAGIC METHODS ##
 
@@ -135,6 +142,23 @@ class RsrcCentricLayout:
         which is formatted for human readability and to avoid repetition.
         The attributes not mapped here (usually user-provided triples with no
         special meaning to the application) go to the `fcmain:` graph.
+
+        The output of this is a dict with a similar structure:
+
+        {
+            'p': {
+                <Predicate P1>: <destination graph G1>,
+                <Predicate P2>: <destination graph G1>,
+                <Predicate P3>: <destination graph G1>,
+                <Predicate P4>: <destination graph G2>,
+                [...]
+            },
+            't': {
+                <RDF Type T1>: <destination graph G1>,
+                <RDF Type T2>: <destination graph G3>,
+                [...]
+            }
+        }
         '''
         if not hasattr(self, '_attr_routes'):
             self._attr_routes = {'p': {}, 't': {}}
@@ -199,24 +223,28 @@ class RsrcCentricLayout:
         if ver_uid:
             uid = self.snapshot_uid(uid, ver_uid)
         if incl_children:
-            incl_child_qry = 'FROM {}'.format(nsc['fcstruct'][uid].n3())
+            incl_child_qry = ''
             if embed_children:
                 pass # Not implemented. May never be.
         else:
-            incl_child_qry = ''
+            incl_child_qry = (
+                    '\n FILTER NOT EXISTS { ?g a fcsystem:StructureGraph .}')
 
         qry = '''
-        CONSTRUCT
-        FROM {ag}
-        FROM {sg}
-        {chld}
-        WHERE {{ ?s ?p ?o . }}
-        '''.format(
-                ag=nsc['fcadmin'][uid].n3(),
-                sg=nsc['fcmain'][uid].n3(),
-                chld=incl_child_qry,
-            )
-        gr = self._parse_construct(qry)
+        CONSTRUCT {?s ?p ?o . }
+        WHERE {
+          GRAPH fcsystem:meta {
+            ?g foaf:primaryTopic ?rsrc .
+          }
+          GRAPH ?g { ?s ?p ?o . }
+        ''' + incl_child_qry + '\n}'
+
+        gr = self._parse_construct(qry, init_bindings={
+            'rsrc': nsc['fcres'][uid],
+            'ag': nsc['fcadmin'][uid],
+            'mg': nsc['fcmain'][uid],
+            'sg': nsc['fcstruct'][uid],
+        })
 
         if incl_inbound and len(gr):
             gr += self.get_inbound_rel(nsc['fcres'][uid])
@@ -396,16 +424,23 @@ class RsrcCentricLayout:
         add_routes = defaultdict(set)
         historic = VERS_CONT_LABEL in uid
 
+        graph_types = set() # Graphs that need RDF type metadata added.
         # Create add and remove sets for each graph.
         for t in remove_trp:
-            target_gr_uri = self._map_graph_uri(t, uid)
+            map_graph = self._map_graph_uri(t, uid)
+            target_gr_uri = map_graph[0]
             remove_routes[target_gr_uri].add(t)
+            graph_types.add(map_graph)
         for t in add_trp:
-            target_gr_uri = self._map_graph_uri(t, uid)
+            map_graph = self._map_graph_uri(t, uid)
+            target_gr_uri = map_graph[0]
             add_routes[target_gr_uri].add(t)
+            graph_types.add(map_graph)
 
         # Decide if metadata go into historic or current graph.
-        meta_uri = HIST_GR_URI if historic else META_GR_URI
+        meta_gr_uri = HIST_GR_URI if historic else META_GR_URI
+        meta_gr = self.ds.graph(meta_gr_uri)
+
         # Remove and add triple sets from each graph.
         for gr_uri, trp in remove_routes.items():
             gr = self.ds.graph(gr_uri)
@@ -414,17 +449,19 @@ class RsrcCentricLayout:
             gr = self.ds.graph(gr_uri)
             gr += trp
             # Add metadata.
-            self.ds.graph(meta_uri).set((
-                gr_uri, nsc['foaf'].primaryTopic, nsc['fcres'][uid]))
-            self.ds.graph(meta_uri).set((
-                gr_uri, nsc['fcrepo'].created, g.timestamp_term))
+            meta_gr.set((gr_uri, nsc['foaf'].primaryTopic, nsc['fcres'][uid]))
+            meta_gr.set((gr_uri, nsc['fcrepo'].created, g.timestamp_term))
             if historic:
                 # @FIXME Ugly reverse engineering.
                 ver_uid = uid.split(VERS_CONT_LABEL)[1].lstrip('/')
-                self.ds.graph(meta_uri).set((
+                meta_gr.set((
                     gr_uri, nsc['fcrepo'].hasVersionLabel, Literal(ver_uid)))
             # @TODO More provenance metadata can be added here.
 
+        # Add graph RDF types.
+        for gr_uri, gr_type in graph_types:
+            meta_gr.add((gr_uri, RDF.type, gr_type))
+
 
     def delete_rsrc_data(self, uid):
         for guid in self._graph_uids:
@@ -527,10 +564,14 @@ class RsrcCentricLayout:
     def _map_graph_uri(self, t, uid):
         '''
         Map a triple to a namespace prefix corresponding to a graph.
+
+        @return Tuple with a graph URI and an associated RDF type.
         '''
         if t[1] in self.attr_routes['p'].keys():
-            return self.attr_routes['p'][t[1]][uid]
+            pfx = self.attr_routes['p'][t[1]]
         elif t[1] == RDF.type and t[2] in self.attr_routes['t'].keys():
-            return self.attr_routes['t'][t[2]][uid]
+            pfx = self.attr_routes['t'][t[2]]
         else:
-            return nsc['fcmain'][uid]
+            pfx = nsc['fcmain']
+
+        return (pfx[uid], self.graph_ns_types[pfx])

+ 2 - 0
profiler.py

@@ -1,3 +1,5 @@
+#!/usr/bin/env python
+
 from werkzeug.contrib.profiler import ProfilerMiddleware
 
 from lakesuperior.app import create_app