All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jvnet.hk2.config.DomDocument Maven / Gradle / Ivy

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 2007-2011 Oracle and/or its affiliates. All rights reserved.
 *
 * The contents of this file are subject to the terms of either the GNU
 * General Public License Version 2 only ("GPL") or the Common Development
 * and Distribution License("CDDL") (collectively, the "License").  You
 * may not use this file except in compliance with the License.  You can
 * obtain a copy of the License at
 * https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
 * or packager/legal/LICENSE.txt.  See the License for the specific
 * language governing permissions and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file at packager/legal/LICENSE.txt.
 *
 * GPL Classpath Exception:
 * Oracle designates this particular file as subject to the "Classpath"
 * exception as provided by Oracle in the GPL Version 2 section of the License
 * file that accompanied this code.
 *
 * Modifications:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 *
 * Contributor(s):
 * If you wish your version of this file to be governed by only the CDDL or
 * only the GPL Version 2, indicate your decision by adding "[Contributor]
 * elects to include this software in this distribution under the [CDDL or GPL
 * Version 2] license."  If you don't indicate a single choice of license, a
 * recipient has the option to distribute your version of this file under
 * either the CDDL, the GPL Version 2 or to extend the choice of license to
 * its licensees as provided above.  However, if you add GPL Version 2 code
 * and therefore, elected the GPL Version 2 license, then the option applies
 * only if the new code is made subject to such option by the copyright
 * holder.
 */
package org.jvnet.hk2.config;

import org.jvnet.hk2.component.Habitat;
import org.jvnet.hk2.component.Inhabitant;
import org.jvnet.hk2.component.ComponentException;
import org.jvnet.hk2.component.MultiMap;

import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.XMLStreamWriter;
import javax.xml.stream.XMLStreamException;
import java.util.*;

/**
 * Represents a whole DOM tree.
 *
 * @author Kohsuke Kawaguchi
 */
public class DomDocument {
    /**
     * A hook to perform variable replacements on the attribute/element values
     * that are found in the configuration file.
     * The translation happens lazily when objects are actually created, not when
     * configuration is parsed, so this allows circular references —
     * {@link Translator} may refer to objects in the configuration file being read.
     */
    private volatile Translator translator = Translator.NOOP;

    protected final Map,ConfigModel> models = new HashMap, ConfigModel>();
    private final MultiMap> implementorsOf = new MultiMap>();

    /*package*/ final Habitat habitat;

    /*package*/ T root;

    private final Map validators = new HashMap();
    
    /*package*/ static final List PRIMS = Collections.unmodifiableList(Arrays.asList(
    "boolean", "char", "int", "java.lang.Boolean", "java.lang.Character", "java.lang.Integer"));
    
    public DomDocument(Habitat habitat) {
        this.habitat = habitat;
        for (String prim : PRIMS) {
            validators.put(prim, new PrimitiveDataType(prim) );
        }
    }

    public Dom getRoot() {
        return root;
    }

    public Translator getTranslator() {
        return translator;
    }

    public void setTranslator(Translator translator) {
        this.translator = translator;
    }

    /**
     * Creates {@link ConfigModel} for the given {@link ConfigInjector} if we haven't done so.
     */
    /*package*/ ConfigModel buildModel(Inhabitant i) {
        ConfigModel m = models.get(i);
        if(m==null)
            m = new ConfigModel(this,i,i.metadata());
        return m;
    }

    /**
     * Obtains a {@link ConfigModel} for the given class (Which should have {@link Configured} annotation on it.)
     */
    public ConfigModel buildModel(Class clazz) {
        return buildModel(clazz.getName());
    }

    /**
     * Obtains a {@link ConfigModel} for the given class (Which should have {@link Configured} annotation on it.)
     */
    public ConfigModel buildModel(String fullyQualifiedClassName) {
        Inhabitant i = habitat.getInhabitantByAnnotation(InjectionTarget.class, fullyQualifiedClassName);
        if(i==null)
            throw new ComponentException("ConfigInjector for %s is not found, is it annotated with @Configured",fullyQualifiedClassName);
        return buildModel(i);
    }

    /**
     * Obtains the {@link ConfigModel} from the "global" element name.
     *
     * 

* This method uses {@link #buildModel} to lazily build models if necessary. * * @return * Null if no configurable component is registered under the given global element name. */ public ConfigModel getModelByElementName(String elementName) { Inhabitant i = habitat.getInhabitant(ConfigInjector.class, elementName); if(i==null) return null; return buildModel(i); } /** * Calculates all @Configured interfaces subclassing the passed interface type. * * @param intf a @Configured interface * @return List of all @Configured subclasses * @throws ClassNotFoundException */ public synchronized List getAllModelsImplementing(Class intf) throws ClassNotFoundException { if (implementorsOf.size()==0) { initXRef(); } return implementorsOf.getOne(intf); } /** * probably a bit slow, calculates all the @Configured interfaces subclassing, useful * to find all possible subclasses of a type. * * @throws ClassNotFoundException */ private void initXRef() throws ClassNotFoundException { // force initialization of all the config models. for (Inhabitant i : habitat.getInhabitants(ConfigInjector.class)) { buildModel(i); } for (ConfigModel cm : models.values()) { Class targetType = cm.classLoaderHolder.get().loadClass(cm.targetTypeName); do { Class[] intfs = targetType.getInterfaces(); for (Class intf : intfs) { if (intf.isAnnotationPresent(Configured.class)) { List models = implementorsOf.getOne(intf); if (models==null) { models = new ArrayList(); implementorsOf.add(intf, models); } models.add(cm); } } targetType = targetType.getSuperclass(); } while (targetType!=null); } } // TODO: to be removed once we make sure that no one is using it anymore @Deprecated public ConfigModel getModel(Class c) { return buildModel(c); } public Dom make(Habitat habitat, XMLStreamReader in, T parent, ConfigModel model) { return new Dom(habitat,this,parent,model,in); } /** * Writes back the whole DOM tree as an XML document. * *

* To support writing a subtree, this method doesn't invoke the start/endDocument * events. Those are the responsibility of the caller. * * @param w * Receives XML infoset stream. */ public void writeTo(XMLStreamWriter w) throws XMLStreamException { root.writeTo(null,w); } /*package*/ DataType getValidator(String dataType) { synchronized(validators) { DataType validator = validators.get(dataType); if (validator != null) return (validator); } Collection dtfh = habitat.getAllByContract(DataType.class); synchronized(validators) { for (DataType dt : dtfh) { validators.put(dt.getClass().getName(), dt); } return (validators.get(dataType)); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy