]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.db.services/src/org/simantics/db/services/adaption/AdapterRegistry2.java
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.db.services / src / org / simantics / db / services / adaption / AdapterRegistry2.java
diff --git a/bundles/org.simantics.db.services/src/org/simantics/db/services/adaption/AdapterRegistry2.java b/bundles/org.simantics.db.services/src/org/simantics/db/services/adaption/AdapterRegistry2.java
new file mode 100644 (file)
index 0000000..43d6ab7
--- /dev/null
@@ -0,0 +1,519 @@
+/*******************************************************************************\r
+ * Copyright (c) 2007, 2010 Association for Decentralized Information Management\r
+ * in Industry THTH ry.\r
+ * All rights reserved. This program and the accompanying materials\r
+ * are made available under the terms of the Eclipse Public License v1.0\r
+ * which accompanies this distribution, and is available at\r
+ * http://www.eclipse.org/legal/epl-v10.html\r
+ *\r
+ * Contributors:\r
+ *     VTT Technical Research Centre of Finland - initial API and implementation\r
+ *******************************************************************************/\r
+package org.simantics.db.services.adaption;\r
+\r
+import java.io.File;\r
+import java.io.StringReader;\r
+import java.net.URL;\r
+import java.util.ArrayList;\r
+import java.util.Collection;\r
+import java.util.HashMap;\r
+import java.util.List;\r
+import java.util.Map;\r
+\r
+import javax.xml.parsers.DocumentBuilder;\r
+import javax.xml.parsers.DocumentBuilderFactory;\r
+\r
+import org.eclipse.core.runtime.Path;\r
+import org.eclipse.core.runtime.Platform;\r
+import org.osgi.framework.Bundle;\r
+import org.osgi.framework.BundleContext;\r
+import org.simantics.db.ReadGraph;\r
+import org.simantics.db.Resource;\r
+import org.simantics.db.Session;\r
+import org.simantics.db.adaption.Adapter;\r
+import org.simantics.db.adaption.AdapterInstaller;\r
+import org.simantics.db.adaption.AdaptionService;\r
+import org.simantics.db.common.request.ReadRequest;\r
+import org.simantics.db.exception.DatabaseException;\r
+import org.simantics.db.services.adaption.reflection.AdaptingDynamicAdapter2;\r
+import org.simantics.db.services.adaption.reflection.AtMostOneRelatedResource2;\r
+import org.simantics.db.services.adaption.reflection.ConstantAdapter;\r
+import org.simantics.db.services.adaption.reflection.GraphObject2;\r
+import org.simantics.db.services.adaption.reflection.IDynamicAdapter2;\r
+import org.simantics.db.services.adaption.reflection.OrderedSetResources2;\r
+import org.simantics.db.services.adaption.reflection.ReflectionAdapter2;\r
+import org.simantics.db.services.adaption.reflection.RelatedResources2;\r
+import org.simantics.db.services.adaption.reflection.SingleRelatedResource2;\r
+import org.simantics.db.services.adaption.reflection.StaticMethodAdapter;\r
+import org.simantics.db.services.adaption.reflection.ThisResource2;\r
+import org.simantics.scl.reflection.OntologyVersions;\r
+import org.simantics.utils.FileUtils;\r
+import org.w3c.dom.DOMException;\r
+import org.w3c.dom.Document;\r
+import org.w3c.dom.NamedNodeMap;\r
+import org.w3c.dom.Node;\r
+import org.w3c.dom.NodeList;\r
+import org.xml.sax.ErrorHandler;\r
+import org.xml.sax.InputSource;\r
+import org.xml.sax.SAXException;\r
+import org.xml.sax.SAXParseException;\r
+\r
+public class AdapterRegistry2 {\r
+\r
+    public static final String ADAPTERS_FILE = "adapters.xml";\r
+\r
+    public static final String ADAPTERS = "adapters";\r
+    public static final String ADAPTER = "adapter";\r
+    public static final String TARGET = "target";\r
+    public static final String BASE_TYPE = "baseType";\r
+    public static final String TYPE = "type";\r
+    public static final String RESOURCE = "resource";\r
+    public static final String URI = "uri";\r
+    public static final String INTERFACE = "interface";\r
+    public static final String CLASS = "class";\r
+    public static final String ADAPTER_CLASS = "adapterClass";\r
+    public static final String CONTEXT_CLASS = "contextClass";\r
+    public static final String INSTALLER = "installer";\r
+    public static final String CONSTRUCTOR = "constructor";\r
+\r
+    static private AdapterRegistry2 instance = new AdapterRegistry2();\r
+    Collection<AdapterInstaller> installers = new ArrayList<AdapterInstaller>();\r
+    Map<AdapterInstaller, String> installerSources = new HashMap<AdapterInstaller, String>();\r
+    Collection<Exception> exceptions = new ArrayList<Exception>();\r
+\r
+    public static AdapterRegistry2 getInstance() {\r
+        return instance;\r
+    }\r
+\r
+    private void addInstaller(AdapterInstaller installer, String sourceDesc) {\r
+        installers.add(installer);\r
+        installerSources.put(installer, sourceDesc);\r
+    }\r
+\r
+    private void handleException(Exception e, String fileName) {\r
+        System.err.println("At " + fileName);\r
+        e.printStackTrace();\r
+    }\r
+\r
+    private void handleException(Exception e, AdapterInstaller installer) {\r
+        String desc = installerSources.get(installer);\r
+        if (desc != null)\r
+            System.err.println("At " + desc);\r
+        e.printStackTrace();\r
+    }\r
+\r
+    private void handleAdaptersDocument(Loader b, Document doc, String fileName) {\r
+        try {\r
+            Node node = doc.getDocumentElement();\r
+            if(node.getNodeName().equals(ADAPTERS)) {\r
+                NodeList nodeList = node.getChildNodes();\r
+                for(int i=0;i<nodeList.getLength();++i) {\r
+                    Node n = nodeList.item(i);\r
+                    if(n.getNodeName().equals(TARGET))\r
+                        handleTarget(b, n, fileName);\r
+                    else if(n.getNodeName().equals(INSTALLER))\r
+                        handleInstaller(b, n, fileName);\r
+                }\r
+            }\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private void handleTarget(Loader b, Node node, String fileName) {\r
+        try {\r
+            Class<?> interface_ =\r
+                b.loadClass(node.getAttributes().getNamedItem("interface")\r
+                        .getNodeValue());\r
+            NodeList nodeList = node.getChildNodes();\r
+            for(int i=0;i<nodeList.getLength();++i) {\r
+                Node n = nodeList.item(i);\r
+                String nodeName = n.getNodeName();\r
+                if(nodeName.equals(BASE_TYPE))\r
+                    handleBaseType(b, interface_, n, fileName);\r
+                else if(nodeName.equals(TYPE))\r
+                    handleType(b, interface_, n, fileName);\r
+                else if(nodeName.equals(ADAPTER))\r
+                    handleAdapter(b, interface_, n, fileName);\r
+                else if(nodeName.equals(RESOURCE))\r
+                    handleResource(b, interface_, n, fileName);\r
+            }\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private void handleInstaller(Loader b, Node node, String fileName) {\r
+        try {\r
+            AdapterInstaller installer =\r
+                ((Class<?>)b.loadClass(node.getAttributes().getNamedItem("class").getNodeValue()))\r
+                .asSubclass(AdapterInstaller.class).newInstance();\r
+            addInstaller(installer, fileName);\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private <T> void handleResource(final Loader b, final Class<T> interface_, final Node node, String fileName) {\r
+        try {\r
+            NamedNodeMap attr = node.getAttributes();\r
+            final String uri = attr.getNamedItem(URI).getNodeValue();\r
+            final String className = attr.getNamedItem(CLASS).getNodeValue();\r
+            Node constructorNode = attr.getNamedItem(CONSTRUCTOR);\r
+            final String constructor = constructorNode == null ? null : constructorNode.getNodeValue();\r
+//            System.out.println("AdapterRegistry2.handleResource: " + b + " " + uri + " " + interface_);\r
+            addInstaller(\r
+\r
+                    new AdapterInstaller() {\r
+\r
+                        @Override\r
+                        public void install(ReadGraph g, AdaptionService service) throws Exception {\r
+                            Class<? extends T> clazz = b.loadClass(className).asSubclass(interface_);\r
+                            List<IDynamicAdapter2> parameters = readParameters(g, node, b);\r
+                            IDynamicAdapter2[] parameterArray = \r
+                                parameters.toArray(new IDynamicAdapter2[parameters.size()]);\r
+                            Resource r = g.getResource(uri);\r
+                            service.addInstanceAdapter(\r
+                                    r,\r
+                                    interface_,\r
+                                    constructor == null \r
+                                    ? new ReflectionAdapter2<T>(clazz, parameterArray)\r
+                                    : new StaticMethodAdapter<T>(clazz, constructor, parameterArray));\r
+                        }\r
+\r
+                    }, fileName);\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private <T> void handleType(final Loader b, final Class<T> interface_, final Node node, String fileName) {\r
+        try {\r
+            final NamedNodeMap attr = node.getAttributes();\r
+            final String uri = attr.getNamedItem(URI).getNodeValue();\r
+            Node constructorNode = attr.getNamedItem(CONSTRUCTOR);\r
+            final String constructor = constructorNode == null ? null : constructorNode.getNodeValue();\r
+            //System.out.println("AdapterRegistry2.handleType: " + b + " " + uri + " " + interface_);\r
+            addInstaller(\r
+                    new AdapterInstaller() {\r
+\r
+                        @Override\r
+                        public void install(ReadGraph g, AdaptionService service) throws Exception {\r
+                               try {\r
+                            Class<? extends T> clazz =\r
+                                ((Class<?>)b.loadClass(attr.getNamedItem(CLASS).getNodeValue()))\r
+                                .asSubclass(interface_);\r
+                            List<IDynamicAdapter2> parameters = readParameters(g, node, b);\r
+                            IDynamicAdapter2[] parameterArray = \r
+                                parameters.toArray(new IDynamicAdapter2[parameters.size()]);\r
+                            service.addAdapter(\r
+                                    g.getResource(uri),\r
+                                    interface_,\r
+                                    constructor == null \r
+                                    ? new ReflectionAdapter2<T>(clazz, parameterArray)\r
+                                    : new StaticMethodAdapter<T>(clazz, constructor, parameterArray));\r
+                               } catch(Error t) {\r
+                                       System.err.println("Failed to adapt "+interface_.getName());\r
+                                       throw t;\r
+                               } catch(RuntimeException t) {\r
+                                       System.err.println("Failed to adapt "+interface_.getName());\r
+                                       throw t;\r
+                               }\r
+                        }\r
+\r
+                    }, fileName);\r
+        } catch (Exception e) {\r
+            e.printStackTrace();\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private List<IDynamicAdapter2> readParameters(ReadGraph g, Node node, Loader b) throws DatabaseException, DOMException, ClassNotFoundException {\r
+        NodeList nodeList = node.getChildNodes();\r
+        ArrayList<IDynamicAdapter2> parameters = new ArrayList<IDynamicAdapter2>();\r
+        for(int i=0;i<nodeList.getLength();++i) {\r
+            Node n = nodeList.item(i);\r
+            if(n.getNodeType() == Node.ELEMENT_NODE) {\r
+                NamedNodeMap attr = n.getAttributes();\r
+                IDynamicAdapter2 da = null;\r
+                if(n.getNodeName().equals("this"))\r
+                    da = new ThisResource2();\r
+                else if(n.getNodeName().equals("graph"))\r
+                    da = new GraphObject2();\r
+                else if(n.getNodeName().equals("bundle")) {\r
+                    String bundleId = null;\r
+                    Node fc = n.getFirstChild();\r
+                    if (fc != null)\r
+                        bundleId = fc.getNodeValue();\r
+                    if (bundleId == null) {\r
+                        da = new ConstantAdapter(Bundle.class, b.getBundle());\r
+                    } else {\r
+                        Bundle ob = Platform.getBundle(bundleId);\r
+                        if (ob != null) {\r
+                            da = new ConstantAdapter(Bundle.class, ob);\r
+                        } else {\r
+                            throw new DOMException(DOMException.NOT_FOUND_ERR, "bundle '" + bundleId + "' not found");\r
+                        }\r
+                    }\r
+                } else if(n.getNodeName().equals("related"))\r
+                    da = new RelatedResources2(\r
+                            g.getResource(attr.getNamedItem("uri").getNodeValue()));\r
+                else if(n.getNodeName().equals("orderedSet"))\r
+                    da = new OrderedSetResources2(\r
+                            g.getResource(attr.getNamedItem("uri").getNodeValue()));\r
+                else if(n.getNodeName().equals("single"))\r
+                    da = new SingleRelatedResource2(\r
+                            g.getResource(attr.getNamedItem("uri").getNodeValue()));\r
+                else if(n.getNodeName().equals("atMostOne"))\r
+                    da = new AtMostOneRelatedResource2(\r
+                            g.getResource(attr.getNamedItem("uri").getNodeValue()));\r
+                else if(n.getNodeName().equals("string"))\r
+                    da = new ConstantAdapter(String.class, n.getFirstChild().getNodeValue());\r
+                {\r
+                    Node toNode = attr.getNamedItem("to");\r
+                    if(toNode != null) {\r
+                        String to = toNode.getNodeValue();\r
+                        da = new AdaptingDynamicAdapter2(da, b.loadClass(to));\r
+                    }\r
+                }\r
+                parameters.add(da);\r
+            }\r
+        }\r
+        return parameters;\r
+    }\r
+\r
+    private <T> void handleAdapter(final Loader b, final Class<T> interface_, Node node, String fileName) {\r
+        try {\r
+            NamedNodeMap attr = node.getAttributes();\r
+            final String uri = attr.getNamedItem(URI).getNodeValue();\r
+            final String clazz = attr.getNamedItem(ADAPTER_CLASS).getNodeValue();\r
+\r
+            Node contextNode = attr.getNamedItem(CONTEXT_CLASS);\r
+            final Class<?> contextClass = contextNode != null ? b.loadClass(contextNode.getNodeValue()) : Resource.class;\r
+            \r
+            //System.out.println("AdapterRegistry2.handleAdapter: " + b + " " + uri + " " + interface_ + ", class=" + clazz);\r
+            addInstaller(\r
+                    new AdapterInstaller() {\r
+\r
+                        @SuppressWarnings("unchecked")\r
+                        @Override\r
+                        public void install(ReadGraph g, AdaptionService service) throws Exception {\r
+                            service.addAdapter(\r
+                                    g.getResource(uri),\r
+                                    interface_,\r
+                                    contextClass,\r
+                                    ((Class<?>)b.loadClass(clazz))\r
+                                    .asSubclass(Adapter.class).newInstance());\r
+                        }\r
+\r
+                    }, fileName);\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    private <T> void handleBaseType(Loader b, final Class<T> interface_, Node node, String fileName) {\r
+        try {\r
+            NamedNodeMap attr = node.getAttributes();\r
+            final String uri = attr.getNamedItem(URI).getNodeValue();\r
+            addInstaller(\r
+                    new AdapterInstaller() {\r
+\r
+                        @Override\r
+                        public void install(ReadGraph g, AdaptionService service) throws Exception {\r
+                            service.declareAdapter(\r
+                                    g.getResource(uri),\r
+                                    interface_);\r
+                        }\r
+\r
+                    }, fileName);\r
+        } catch (Exception e) {\r
+            handleException(e, fileName);\r
+        }\r
+    }\r
+\r
+    public void updateAdaptionService(Session s, final AdaptionService service) throws DatabaseException {\r
+        s.syncRequest(new ReadRequest() {\r
+            @Override\r
+            public void run(ReadGraph g) {\r
+                for(AdapterInstaller t : installers) {\r
+                    try {\r
+                        t.install(g, service);\r
+                    } catch (Exception e) {\r
+                        AdapterRegistry2.this.handleException(e, t);\r
+                    }\r
+                }\r
+            }\r
+        });\r
+    }\r
+\r
+    public void initialize(ClassLoader b, String schemaURL, File[] files) {\r
+\r
+        try {\r
+               \r
+            DocumentBuilderFactory factory =\r
+                DocumentBuilderFactory.newInstance();\r
+            \r
+            if(schemaURL != null) {\r
+            \r
+                   factory.setValidating(true);\r
+                   factory.setAttribute(\r
+                           "http://java.sun.com/xml/jaxp/properties/schemaLanguage",\r
+                   "http://www.w3.org/2001/XMLSchema");\r
+                   factory.setAttribute(\r
+                           "http://java.sun.com/xml/jaxp/properties/schemaSource", schemaURL);\r
+                   \r
+            }\r
+\r
+            // TODO Listen bundles (install/uninstall)\r
+            if (exceptions.isEmpty())\r
+                for (final File f : files) {\r
+//                        String fileName = new Path(b.getLocation()).append(file.getPath()).toString();\r
+                        try {\r
+                            DocumentBuilder builder = factory.newDocumentBuilder();\r
+                            builder.setErrorHandler(new ErrorHandler() {\r
+\r
+                                @Override\r
+                                public void error(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    // TODO Put this error somewhere\r
+                                    System.err.println("Parse error at " + f.getAbsolutePath() + \r
+//                                            + b.getSymbolicName() + "/adapters.xml" +\r
+                                            " line " + exception.getLineNumber() +\r
+                                            " column " + exception.getColumnNumber() + ":");\r
+                                    System.err.println(exception.getMessage());\r
+                                }\r
+\r
+                                @Override\r
+                                public void fatalError(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    error(exception);\r
+                                }\r
+\r
+                                @Override\r
+                                public void warning(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    error(exception);\r
+                                }\r
+\r
+                            });\r
+                            //System.out.println("bundle=" + b.getSymbolicName());\r
+                            Document doc = builder.parse(f);\r
+                            handleAdaptersDocument(loader(b), doc, f.getAbsolutePath());\r
+                        } catch (Exception e) {\r
+                            handleException(e, f.getAbsolutePath());\r
+\r
+                        }\r
+                    }\r
+        } catch (Exception e) {\r
+            handleException(e, "(no file name available)");\r
+        }\r
+       \r
+    }\r
+    \r
+    public void initialize(BundleContext context) {\r
+       \r
+        try {\r
+               \r
+            DocumentBuilderFactory factory =\r
+                DocumentBuilderFactory.newInstance();\r
+            factory.setValidating(true);\r
+            factory.setAttribute(\r
+                    "http://java.sun.com/xml/jaxp/properties/schemaLanguage",\r
+            "http://www.w3.org/2001/XMLSchema");\r
+            factory.setAttribute(\r
+                    "http://java.sun.com/xml/jaxp/properties/schemaSource",\r
+                    context.getBundle().getResource("adapters.xsd").toString());\r
+\r
+            // TODO Listen bundles (install/uninstall)\r
+            if (exceptions.isEmpty())\r
+                for (final Bundle b : context.getBundles()) {\r
+                    URL file = b.getEntry(ADAPTERS_FILE);\r
+                    if (file != null) {\r
+                        String fileName = new Path(b.getLocation()).append(file.getPath()).toString();\r
+                        try {\r
+                            DocumentBuilder builder = factory.newDocumentBuilder();\r
+                            builder.setErrorHandler(new ErrorHandler() {\r
+\r
+                                @Override\r
+                                public void error(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    // TODO Put this error somewhere\r
+                                    System.err.println("Parse error at "\r
+                                            + b.getSymbolicName() + "/adapters.xml" +\r
+                                            " line " + exception.getLineNumber() +\r
+                                            " column " + exception.getColumnNumber() + ":");\r
+                                    System.err.println(exception.getMessage());\r
+                                }\r
+\r
+                                @Override\r
+                                public void fatalError(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    error(exception);\r
+                                }\r
+\r
+                                @Override\r
+                                public void warning(SAXParseException exception)\r
+                                throws SAXException {\r
+                                    error(exception);\r
+                                }\r
+\r
+                            });\r
+                            \r
+                            //System.out.println("bundle=" + b.getSymbolicName());\r
+                            String text = FileUtils.getContents(file);\r
+                            text = OntologyVersions.getInstance().currentVersion(text);\r
+                            StringReader reader = new StringReader( text );\r
+                            InputSource inputSource = new InputSource( reader );\r
+                            Document doc = builder.parse( inputSource );\r
+                            reader.close();                                            \r
+                            handleAdaptersDocument(loader(b), doc, fileName);\r
+                        } catch (Exception e) {\r
+                            handleException(e, fileName);\r
+\r
+                        }\r
+                    }\r
+                }\r
+        } catch (Exception e) {\r
+            handleException(e, "(no file name available)");\r
+        }\r
+    }\r
+\r
+       interface Loader {\r
+               Class<?> loadClass(String name) throws ClassNotFoundException ;\r
+               Bundle getBundle();\r
+       }\r
+\r
+       private Loader loader(final Bundle b) {\r
+               return new Loader() {\r
+\r
+                       @Override\r
+                       public Class<?> loadClass(String name) throws ClassNotFoundException {\r
+                               return b.loadClass(name);\r
+                       }\r
+                       \r
+                       @Override\r
+                       public Bundle getBundle() {\r
+                               return b;\r
+                       }\r
+                       \r
+               };\r
+       }\r
+\r
+       private Loader loader(final ClassLoader b) {\r
+               return new Loader() {\r
+\r
+                       @Override\r
+                       public Class<?> loadClass(String name) throws ClassNotFoundException {\r
+                               return b.loadClass(name);\r
+                       }\r
+\r
+                       @Override\r
+                       public Bundle getBundle() {\r
+                               return null;\r
+                       }\r
+                       \r
+               };\r
+       }\r
+       \r
+}\r