]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.db.layer0/src/org/simantics/db/layer0/validation/ChangeSetValidator.java
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.db.layer0 / src / org / simantics / db / layer0 / validation / ChangeSetValidator.java
diff --git a/bundles/org.simantics.db.layer0/src/org/simantics/db/layer0/validation/ChangeSetValidator.java b/bundles/org.simantics.db.layer0/src/org/simantics/db/layer0/validation/ChangeSetValidator.java
new file mode 100644 (file)
index 0000000..0e646b7
--- /dev/null
@@ -0,0 +1,188 @@
+package org.simantics.db.layer0.validation;\r
+\r
+import java.util.Collection;\r
+import java.util.HashSet;\r
+import java.util.List;\r
+import java.util.Set;\r
+\r
+import org.simantics.db.ChangeSet;\r
+import org.simantics.db.ChangeSet.StatementChange;\r
+import org.simantics.db.Issue;\r
+import org.simantics.db.ReadGraph;\r
+import org.simantics.db.Resource;\r
+import org.simantics.db.common.utils.Functions;\r
+import org.simantics.db.common.utils.Logger;\r
+import org.simantics.db.common.utils.NameUtils;\r
+import org.simantics.db.event.ChangeEvent;\r
+import org.simantics.db.event.ChangeListener;\r
+import org.simantics.db.exception.DatabaseException;\r
+import org.simantics.layer0.Layer0;\r
+\r
+public class ChangeSetValidator implements ChangeListener {\r
+\r
+       void reportInconsistency(ReadGraph graph, Resource subject, String description) throws DatabaseException {\r
+               System.err.println("Change set validation reports the following issue: " + NameUtils.getSafeName(graph, subject, true) + ": " + description);\r
+       }\r
+       \r
+       @Override\r
+       public void graphChanged(ChangeEvent e) throws DatabaseException {    \r
+               ChangeSet changeSet = e.getChanges();\r
+               ReadGraph g = e.getGraph();\r
+//             Layer0 l0 = Layer0.getInstance(g);\r
+               \r
+               HashSet<Resource> allResources = new HashSet<Resource>();\r
+               \r
+               for(StatementChange change : changeSet.changedStatements()) {\r
+                       allResources.add(change.getSubject());\r
+                       allResources.add(change.getObject());\r
+//                     if(change.isClaim()) {\r
+//                             Resource subject = change.getSubject();                 \r
+//                             Resource predicate = change.getPredicate();\r
+//                             Resource object = change.getObject();\r
+//                             if(predicate == l0.InstanceOf) {\r
+//                                     if(!isType(g, l0, object)) {\r
+//                                             reportInconsistency(g, subject, "The range of InstanceOf relation is Type.");\r
+//                                     }\r
+//                                     else {\r
+//                                             // TODO\r
+//                                     }\r
+//                             }\r
+//                             else if(predicate == l0.SubrelationOf) {\r
+//                                     if(!isRelation(g, l0, object)) {\r
+//                                             reportInconsistency(g, subject, "The range of SubrelationOf relation is Relation.");\r
+//                                     }\r
+//                             }\r
+//                             else if(predicate == l0.Inherits) {\r
+//                                     if(!isType(g, l0, object)) {\r
+//                                             reportInconsistency(g, subject, "The range of Inherits relation is Type.");\r
+//                                     }\r
+//                             }\r
+//                             else {\r
+//                                     if(!isRelation(g, l0, predicate)) {\r
+//                                             reportInconsistency(g, subject, "The predicate of a statement must be a relation.");\r
+//                                     }\r
+//                                     else {\r
+//                                         if(g.isInstanceOf(predicate, l0.FunctionalRelation)) {\r
+//                                             if(g.getObjects(subject, predicate).size() > 1)\r
+//                                                 reportInconsistency(g, subject, \r
+//                                        "Relation " +\r
+//                                        NameUtils.getSafeName(g, predicate)\r
+//                                        + " is functional.");\r
+//                                         }\r
+//                                         if(g.isSubrelationOf(predicate, l0.HasProperty)) {\r
+//                                             String error = L0Validations.checkValueType(g, subject, predicate);\r
+//                                             if(error != null) reportInconsistency(g, subject, error);\r
+//                                         }\r
+//                                             {\r
+//                                                     Collection<Resource> domain = g.getObjects(predicate, l0.HasDomain);\r
+//                                                     if (!isInstanceOfAny(g, subject, domain, true)) {\r
+//                                                             StringBuilder sb = new StringBuilder()\r
+//                                                             .append("The domain of ")\r
+//                                                             .append(NameUtils.getSafeName(g, predicate))\r
+//                                                             .append(" relation is ");\r
+//                                                             orString(g, sb, domain).append(".");\r
+//                                                             reportInconsistency(g, subject, sb.toString());\r
+//                                                     }\r
+//                                             }\r
+//                                             {\r
+//                                                     Collection<Resource> range = g.getObjects(predicate, l0.HasRange);\r
+//                                                     if (!isInstanceOfAny(g, object, range, true)) {\r
+//                                                             StringBuilder sb = new StringBuilder()\r
+//                                                             .append("The range of ")\r
+//                                                             .append(NameUtils.getSafeName(g, predicate))\r
+//                                                             .append(" relation is ");\r
+//                                                             orString(g, sb, range).append(".");\r
+//                                                             reportInconsistency(g, object, sb.toString());\r
+//                                                     }\r
+//                                             }\r
+//                                     }\r
+//                             }\r
+//                     }\r
+//                     else {\r
+//                         // TODO\r
+//                     }\r
+               }\r
+               \r
+               for(Resource resource : changeSet.changedValues()) {\r
+                       allResources.add(resource);\r
+//                     if(g.hasValue(resource)) {\r
+//                             if(!g.isInstanceOf(resource, l0.Literal)) {\r
+//                                 reportInconsistency(g, resource, \r
+//                            "Resource has a value but it is not a literal.");\r
+//                             }\r
+//                             else {\r
+//                                     // TODO check that the value is valid for the data type\r
+//                             }\r
+//                     }\r
+//                     else {\r
+//                             if(g.isInstanceOf(resource, l0.Literal)) {\r
+//                                 reportInconsistency(g, resource, \r
+//                        "Resource is a literal but it does not have a value.");\r
+//                             }\r
+//                     }\r
+               }\r
+               \r
+               Layer0 L0 = Layer0.getInstance(g);\r
+\r
+               for(Resource r : allResources) {\r
+                       \r
+                       try {\r
+\r
+                               for(Resource constraint : g.sync(new GetConstraints(r))) {\r
+\r
+                                       try {\r
+\r
+                                               Resource function = g.getSingleObject(constraint, L0.Constraint_Validator);\r
+                                               @SuppressWarnings("unchecked")\r
+                                               Set<Issue> contexts = new HashSet<Issue>((List<Issue>)Functions.exec(g, function, g, r));\r
+                                               for(Issue i : contexts) {\r
+                                                       reportInconsistency(g, r, g.getURI(i.getType()));\r
+                                               }\r
+\r
+                                       } catch (Throwable t) {\r
+                                               Logger.defaultLogError(t);\r
+                                       }\r
+                                       \r
+                               }\r
+\r
+                       } catch (Throwable t) {\r
+                               Logger.defaultLogError(t);\r
+                       }\r
+                       \r
+               }\r
+               \r
+       }\r
+\r
+       private boolean isInstanceOfAny(ReadGraph graph, Resource r, Collection<Resource> types, boolean ifEmpty) throws DatabaseException {\r
+               if (types.isEmpty())\r
+                       return ifEmpty;\r
+               for (Resource type : types) {\r
+                       if (graph.isInstanceOf(r, type)) {\r
+                               return true;\r
+                       }\r
+               }\r
+               return false;\r
+       }\r
+\r
+       private StringBuilder orString(ReadGraph graph, StringBuilder sb, Collection<Resource> rs) throws DatabaseException {\r
+               sb.append("(");\r
+               boolean first = true;\r
+               for (Resource r : rs) {\r
+                       if (!first)\r
+                               sb.append(" | ");\r
+                       first = false;\r
+                       sb.append(NameUtils.getSafeName(graph, r));\r
+               }\r
+               sb.append(")");\r
+               return sb;\r
+       }\r
+\r
+    public static boolean isRelation(ReadGraph g, Layer0 l0, Resource relation) throws DatabaseException {\r
+               return g.hasStatement(relation, l0.SubrelationOf) || relation == l0.IsWeaklyRelatedTo;\r
+       }\r
+       \r
+       public static boolean isType(ReadGraph g, Layer0 l0, Resource type) throws DatabaseException {\r
+               return g.hasStatement(type, l0.Inherits) || type == l0.Entity;\r
+       }\r
+       \r
+}\r