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

org.apache.cayenne.gen.MapClassGenerator Maven / Gradle / Ivy

There is a newer version: 2.0.4
Show newest version
/*****************************************************************
 *   Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 ****************************************************************/


package org.apache.cayenne.gen;

import java.io.Writer;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;

import org.apache.cayenne.CayenneDataObject;
import org.apache.cayenne.CayenneRuntimeException;
import org.apache.cayenne.PersistentObject;
import org.apache.cayenne.map.DataMap;
import org.apache.cayenne.map.ObjEntity;

import foundrylogic.vpp.VPPConfig;

/**
 * Generates Java source code for ObjEntities in the DataMap. This class is abstract and
 * does not deal with filesystem issues directly. Concrete subclasses should provide ways
 * to store generated files by implementing {@link #openWriter(ObjEntity, String, String)}
 * and {@link #closeWriter(Writer)}methods.
 * 
 * @author Andrei Adamchik
 */
public abstract class MapClassGenerator {

    public static final String SINGLE_CLASS_TEMPLATE_1_1 = "dotemplates/singleclass.vm";
    public static final String SUBCLASS_TEMPLATE_1_1 = "dotemplates/subclass.vm";
    public static final String SUPERCLASS_TEMPLATE_1_1 = "dotemplates/superclass.vm";

    public static final String SINGLE_CLASS_TEMPLATE_1_2 = "dotemplates/v1_2/singleclass.vm";
    public static final String SUBCLASS_TEMPLATE_1_2 = "dotemplates/v1_2/subclass.vm";
    public static final String SUPERCLASS_TEMPLATE_1_2 = "dotemplates/v1_2/superclass.vm";

    /**
     * @since 1.2
     */
    public static final String CLIENT_SUBCLASS_TEMPLATE_1_2 = "dotemplates/v1_2/client-subclass.vm";

    /**
     * @since 1.2
     */
    public static final String CLIENT_SUPERCLASS_TEMPLATE_1_2 = "dotemplates/v1_2/client-superclass.vm";

    public static final String SINGLE_CLASS_TEMPLATE = SINGLE_CLASS_TEMPLATE_1_1;
    public static final String SUBCLASS_TEMPLATE = SUBCLASS_TEMPLATE_1_1;
    public static final String SUPERCLASS_TEMPLATE = SUPERCLASS_TEMPLATE_1_1;

    public static final String SUPERCLASS_PREFIX = "_";

    protected static final String VERSION_1_1 = ClassGenerator.VERSION_1_1;
    protected static final String VERSION_1_2 = ClassGenerator.VERSION_1_2;

    protected static final String DEFAULT_VERSION = VERSION_1_1;

    protected static final String MODE_DATAMAP = "datamap";
    protected static final String MODE_ENTITY = "entity";

    protected String versionString = DEFAULT_VERSION;

    protected List objEntities;
    protected String superPkg;
    protected DataMap dataMap;
    protected VPPConfig vppConfig;
    protected String mode = MODE_ENTITY;
    protected boolean client;

    public MapClassGenerator() {
    }

    public MapClassGenerator(DataMap dataMap) {
        this(dataMap, new ArrayList(dataMap.getObjEntities()));
    }

    /**
     * Creates a new MapClassGenerator.
     * 
     * @since 1.2
     */
    public MapClassGenerator(DataMap dataMap, List objEntities) {
        this.dataMap = dataMap;
        this.setObjEntities(objEntities);
    }

    /**
     * @deprecated Since 1.2 use MapClassGenerator(DataMap, List) to provide support for
     *             v1.2 templates.
     */
    public MapClassGenerator(List objEntities) {
        this.objEntities = objEntities;
    }

    protected String defaultSingleClassTemplate() {
        // there is no default single class client template at the moment
        if (client) {
            throw new IllegalStateException(
                    "Default generation for single classes on the client is not supported...");
        }
        else if (VERSION_1_1.equals(versionString)) {
            return SINGLE_CLASS_TEMPLATE_1_1;
        }
        else if (VERSION_1_2.equals(versionString)) {
            return SINGLE_CLASS_TEMPLATE_1_2;
        }
        return SINGLE_CLASS_TEMPLATE;
    }

    protected String defaultSubclassTemplate() {
        // client templates are always 1.2
        if (client) {
            return CLIENT_SUBCLASS_TEMPLATE_1_2;
        }
        else if (VERSION_1_1.equals(versionString)) {
            return SUBCLASS_TEMPLATE_1_1;
        }
        else if (VERSION_1_2.equals(versionString)) {
            return SUBCLASS_TEMPLATE_1_2;
        }
        return SUBCLASS_TEMPLATE;
    }

    protected String defaultSuperclassTemplate() {
        // client templates are always 1.2
        if (client) {
            return CLIENT_SUPERCLASS_TEMPLATE_1_2;
        }
        else if (VERSION_1_1.equals(versionString)) {
            return SUPERCLASS_TEMPLATE_1_1;
        }
        else if (VERSION_1_2.equals(versionString)) {
            return SUPERCLASS_TEMPLATE_1_2;
        }
        return SUPERCLASS_TEMPLATE;
    }

    /**
     * Creates a Writer to output source code for a given ObjEntity and Java class.
     * 
     * @return Writer to store generated class source code or null if this class
     *         generation should be skipped.
     */
    public abstract Writer openWriter(ObjEntity entity, String pkgName, String className)
            throws Exception;

    /**
     * Closes writer after class code has been successfully written by
     * ClassGenerationInfo.
     */
    public abstract void closeWriter(Writer out) throws Exception;

    /**
     * Runs class generation. Produces a pair of Java classes for each ObjEntity in the
     * map. Uses default Cayenne templates for classes.
     */
    public void generateClassPairs() throws Exception {
        generateClassPairs(
                defaultSubclassTemplate(),
                defaultSuperclassTemplate(),
                SUPERCLASS_PREFIX);
    }

    /**
     * Runs class generation. Produces a pair of Java classes for each ObjEntity in the
     * map. This allows developers to use generated subclass  for their custom
     * code, while generated superclass  will contain Cayenne code. Superclass will
     * be generated in the same package, its class name will be derived from the class
     * name by adding a superPrefix.
     */
    public void generateClassPairs(
            String classTemplate,
            String superTemplate,
            String superPrefix) throws Exception {

        // note - client templates ignore version and unconditionally use 1.2
        if (client) {
            generateClientClassPairs_1_2(classTemplate, superTemplate, superPrefix);
        }
        else if (VERSION_1_1.equals(versionString)) {
            generateClassPairs_1_1(classTemplate, superTemplate, superPrefix);
        }
        else if (VERSION_1_2.equals(versionString)) {
            generateClassPairs_1_2(classTemplate, superTemplate, superPrefix);
        }
        else {
            throw new IllegalStateException("Illegal Version in generateClassPairs: "
                    + versionString);
        }
    }

    /**
     * Runs class generation. Produces a pair of Java classes for each ObjEntity in the
     * map. This allows developers to use generated subclass  for their custom
     * code, while generated superclass  will contain Cayenne code. Superclass will
     * be generated in the same package, its class name will be derived from the class
     * name by adding a superPrefix.
     */
    private void generateClassPairs_1_1(
            String classTemplate,
            String superTemplate,
            String superPrefix) throws Exception {

        ClassGenerator mainGenSetup = new ClassGenerator(classTemplate, versionString);
        ClassGenerator superGenSetup = new ClassGenerator(superTemplate, versionString);

        ClassGenerationInfo mainGen = mainGenSetup.getClassGenerationInfo();
        ClassGenerationInfo superGen = superGenSetup.getClassGenerationInfo();

        // prefix is needed for both generators
        mainGen.setSuperPrefix(superPrefix);
        superGen.setSuperPrefix(superPrefix);

        // Iterate only once if this is datamap mode
        Iterator it = objEntities.iterator();
        if (MODE_ENTITY.equals(mode)) {
            it = objEntities.iterator();
        }
        else {
            if (objEntities.isEmpty()) {
                it = objEntities.iterator();
            }
            else {
                it = Collections.singleton(objEntities.get(0)).iterator();
            }
        }

        while (it.hasNext()) {
            ObjEntity ent = (ObjEntity) it.next();

            // 1. do the superclass
            initClassGenerator_1_1(superGen, ent, true);

            Writer superOut = openWriter(ent, superGen.getPackageName(), superPrefix
                    + superGen.getClassName());

            if (superOut != null) {
                superGenSetup.generateClass(superOut, ent);
                closeWriter(superOut);
            }

            // 2. do the main class
            initClassGenerator_1_1(mainGen, ent, false);
            Writer mainOut = openWriter(ent, mainGen.getPackageName(), mainGen
                    .getClassName());
            if (mainOut != null) {
                mainGenSetup.generateClass(mainOut, ent);
                closeWriter(mainOut);
            }
        }
    }

    private void generateClientClassPairs_1_2(
            String classTemplate,
            String superTemplate,
            String superPrefix) throws Exception {

        ClassGenerator mainGenSetup = new ClassGenerator(
                classTemplate,
                ClassGenerator.VERSION_1_2,
                vppConfig);
        ClassGenerator superGenSetup = new ClassGenerator(
                superTemplate,
                ClassGenerator.VERSION_1_2,
                vppConfig);

        // Iterate only once if this is datamap mode
        Iterator it = objEntities.iterator();
        if (MODE_ENTITY.equals(mode)) {
            it = objEntities.iterator();
        }
        else {
            if (objEntities.isEmpty()) {
                it = objEntities.iterator();
            }
            else {
                it = Collections.singleton(objEntities.get(0)).iterator();
            }
        }

        while (it.hasNext()) {
            ObjEntity entity = (ObjEntity) it.next();

            // use client name, and if not specified use regular class name
            String fqnSubClass = entity.getClientClassName();
            if (fqnSubClass == null) {
                fqnSubClass = entity.getClassName();
            }

            // use PersistentObject instead of CayenneDataObject as base ...
            String fqnBaseClass = (entity.getClientSuperClassName() != null) ? entity
                    .getClientSuperClassName() : PersistentObject.class.getName();

            StringUtils stringUtils = StringUtils.getInstance();

            String subClassName = stringUtils.stripPackageName(fqnSubClass);
            String subPackageName = stringUtils.stripClass(fqnSubClass);

            String superClassName = superPrefix
                    + stringUtils.stripPackageName(fqnSubClass);

            String superPackageName = this.superPkg;
            String fqnSuperClass = superPackageName + "." + superClassName;

            Writer superOut = openWriter(entity, superPackageName, superClassName);

            if (superOut != null) {
                superGenSetup.generateClass(
                        superOut,
                        dataMap,
                        entity,
                        fqnBaseClass,
                        fqnSuperClass,
                        fqnSubClass);
                closeWriter(superOut);
            }

            Writer mainOut = openWriter(entity, subPackageName, subClassName);
            if (mainOut != null) {
                mainGenSetup.generateClass(
                        mainOut,
                        dataMap,
                        entity,
                        fqnBaseClass,
                        fqnSuperClass,
                        fqnSubClass);
                closeWriter(mainOut);
            }
        }
    }

    /**
     * Runs class generation. Produces a pair of Java classes for each ObjEntity in the
     * map. This allows developers to use generated subclass  for their custom
     * code, while generated superclass  will contain Cayenne code. Superclass will
     * be generated in the same package, its class name will be derived from the class
     * name by adding a superPrefix.
     */
    private void generateClassPairs_1_2(
            String classTemplate,
            String superTemplate,
            String superPrefix) throws Exception {

        ClassGenerator mainGenSetup = new ClassGenerator(
                classTemplate,
                versionString,
                vppConfig);
        ClassGenerator superGenSetup = new ClassGenerator(
                superTemplate,
                versionString,
                vppConfig);

        // Iterate only once if this is datamap mode
        Iterator it = objEntities.iterator();
        if (MODE_ENTITY.equals(mode)) {
            it = objEntities.iterator();
        }
        else {
            if (objEntities.isEmpty()) {
                it = objEntities.iterator();
            }
            else {
                it = Collections.singleton(objEntities.get(0)).iterator();
            }
        }

        while (it.hasNext()) {
            ObjEntity ent = (ObjEntity) it.next();

            String fqnSubClass = ent.getClassName();
            String fqnBaseClass = (null != ent.getSuperClassName()) ? ent
                    .getSuperClassName() : CayenneDataObject.class.getName();

            StringUtils stringUtils = StringUtils.getInstance();

            String subClassName = stringUtils.stripPackageName(fqnSubClass);
            String subPackageName = stringUtils.stripClass(fqnSubClass);

            String superClassName = superPrefix
                    + stringUtils.stripPackageName(fqnSubClass);

            String superPackageName = this.superPkg;
            String fqnSuperClass = superPackageName + "." + superClassName;

            Writer superOut = openWriter(ent, superPackageName, superClassName);

            if (superOut != null) {
                superGenSetup.generateClass(
                        superOut,
                        dataMap,
                        ent,
                        fqnBaseClass,
                        fqnSuperClass,
                        fqnSubClass);
                closeWriter(superOut);
            }

            Writer mainOut = openWriter(ent, subPackageName, subClassName);
            if (mainOut != null) {
                mainGenSetup.generateClass(
                        mainOut,
                        dataMap,
                        ent,
                        fqnBaseClass,
                        fqnSuperClass,
                        fqnSubClass);
                closeWriter(mainOut);
            }
        }
    }

    /**
     * Runs class generation. Produces a single Java class for each ObjEntity in the map.
     * Uses default Cayenne templates for classes.
     */
    public void generateSingleClasses() throws Exception {
        generateSingleClasses(defaultSingleClassTemplate(), SUPERCLASS_PREFIX);
    }

    /**
     * Runs class generation. Produces a single Java class for each ObjEntity in the map.
     */
    private void generateSingleClasses_1_1(String classTemplate) throws Exception {
        ClassGenerator gen = new ClassGenerator(classTemplate, versionString);

        // Iterate only once if this is datamap mode
        Iterator it = objEntities.iterator();
        if (MODE_ENTITY.equals(mode)) {
            it = objEntities.iterator();
        }
        else {
            if (objEntities.isEmpty()) {
                it = objEntities.iterator();
            }
            else {
                it = Collections.singleton(objEntities.get(0)).iterator();
            }
        }

        while (it.hasNext()) {
            ObjEntity ent = (ObjEntity) it.next();

            initClassGenerator_1_1(gen.getClassGenerationInfo(), ent, false);
            Writer out = openWriter(
                    ent,
                    gen.getClassGenerationInfo().getPackageName(),
                    gen.getClassGenerationInfo().getClassName());
            if (out == null) {
                continue;
            }

            gen.generateClass(out, ent);
            closeWriter(out);
        }
    }

    /**
     * Runs class generation. Produces a single Java class for each ObjEntity in the map.
     */
    private void generateSingleClasses_1_2(String classTemplate, String superPrefix)
            throws Exception {
        ClassGenerator gen = new ClassGenerator(classTemplate, versionString, vppConfig);

        // Iterate only once if this is datamap mode
        Iterator it = objEntities.iterator();
        if (MODE_ENTITY.equals(mode)) {
            it = objEntities.iterator();
        }
        else {
            if (objEntities.isEmpty()) {
                it = objEntities.iterator();
            }
            else {
                it = Collections.singleton(objEntities.get(0)).iterator();
            }
        }

        while (it.hasNext()) {
            ObjEntity ent = (ObjEntity) it.next();

            String fqnSubClass = ent.getClassName();
            String fqnBaseClass = (null != ent.getSuperClassName()) ? ent
                    .getSuperClassName() : CayenneDataObject.class.getName();

            StringUtils stringUtils = StringUtils.getInstance();

            String subClassName = stringUtils.stripPackageName(fqnSubClass);
            String subPackageName = stringUtils.stripClass(fqnSubClass);

            String superClassName = superPrefix
                    + stringUtils.stripPackageName(fqnSubClass);

            String superPackageName = this.superPkg;
            String fqnSuperClass = superPackageName + "." + superClassName;

            Writer out = openWriter(ent, subPackageName, subClassName);
            if (out == null) {
                continue;
            }

            gen
                    .generateClass(
                            out,
                            dataMap,
                            ent,
                            fqnBaseClass,
                            fqnSuperClass,
                            fqnSubClass);
            closeWriter(out);
        }
    }

    /**
     * Runs class generation. Produces a single Java class for each ObjEntity in the map.
     * 
     * @deprecated Use generateSingleClasses(String classTemplate, String superPrefix)
     *             instead.
     */
    public void generateSingleClasses(String classTemplate) throws Exception {
        generateSingleClasses(classTemplate, SUPERCLASS_PREFIX);
    }

    /**
     * Runs class generation. Produces a single Java class for each ObjEntity in the map.
     */
    public void generateSingleClasses(String classTemplate, String superPrefix)
            throws Exception {
        // note - client templates ignore version and automatically switch to 1.2
        if (client) {
            generateSingleClasses_1_2(classTemplate, superPrefix);
        }
        else if (VERSION_1_1.equals(versionString)) {
            generateSingleClasses_1_1(classTemplate);
        }
        else if (VERSION_1_2.equals(versionString)) {
            generateSingleClasses_1_2(classTemplate, superPrefix);
        }
        else {
            throw new IllegalStateException("Illegal Version in generateClassPairs: "
                    + versionString);
        }
    }

    /** Initializes ClassGenerationInfo with class name and package of a generated class. */
    protected void initClassGenerator_1_1(
            ClassGenerationInfo gen,
            ObjEntity entity,
            boolean superclass) {

        // figure out generator properties
        String fullClassName = entity.getClassName();
        int i = fullClassName.lastIndexOf(".");

        String pkg = null;
        String spkg = null;
        String cname = null;

        // dot in first or last position is invalid
        if (i == 0 || i + 1 == fullClassName.length()) {
            throw new CayenneRuntimeException("Invalid class mapping: " + fullClassName);
        }
        else if (i < 0) {
            pkg = (superclass) ? superPkg : null;
            spkg = (superclass) ? null : superPkg;
            cname = fullClassName;
        }
        else {
            cname = fullClassName.substring(i + 1);
            pkg = (superclass && superPkg != null) ? superPkg : fullClassName.substring(
                    0,
                    i);

            spkg = (!superclass && superPkg != null && !pkg.equals(superPkg))
                    ? superPkg
                    : null;
        }

        // init generator
        gen.setPackageName(pkg);
        gen.setClassName(cname);
        if (entity.getSuperClassName() != null) {
            gen.setSuperClassName(entity.getSuperClassName());
        }
        else {
            gen.setSuperClassName(CayenneDataObject.class.getName());
        }
        gen.setSuperPackageName(spkg);
    }

    /**
     * Returns "superPkg" property value - a name of a superclass package that should be
     * used for all generated superclasses.
     */
    public String getSuperPkg() {
        return superPkg;
    }

    /**
     * Sets "superPkg" property value.
     */
    public void setSuperPkg(String superPkg) {
        this.superPkg = superPkg;
    }

    /**
     * Returns whether a default client object template will be used.
     * 
     * @since 1.2
     */
    public boolean isClient() {
        return client;
    }

    /**
     * Sets whether a default client object template should be used.
     * 
     * @since 1.2
     */
    public void setClient(boolean client) {
        this.client = client;
    }

    /**
     * @return Returns the dataMap.
     */
    public DataMap getDataMap() {
        return dataMap;
    }

    /**
     * @param dataMap The dataMap to set.
     */
    public void setDataMap(DataMap dataMap) {
        this.dataMap = dataMap;
    }

    public List getObjEntities() {
        return objEntities;
    }

    /**
     * Initializes internal ObjEntities list. This method creates a copy of the provided
     * list to allow its indepdendent modification and also filters out entities that do
     * not require class generation.
     */
    public void setObjEntities(List objEntities) {
        this.objEntities = objEntities != null
                ? new ArrayList(objEntities)
                : new ArrayList();

        // remove generic entities...
        Iterator it = objEntities.iterator();
        while (it.hasNext()) {
            ObjEntity e = (ObjEntity) it.next();
            if (e.isGeneric()) {
                it.remove();
            }
        }
    }

    /**
     * @return Returns the versionString.
     */
    public String getVersionString() {
        return versionString;
    }

    /**
     * @param versionString The versionString to set.
     */
    public void setVersionString(String versionString) {
        if ((false == VERSION_1_1.equals(versionString))
                && (false == VERSION_1_2.equals(versionString))) {
            throw new IllegalStateException("'version' must be '"
                    + VERSION_1_1
                    + "' or '"
                    + VERSION_1_2
                    + "', but was '"
                    + versionString
                    + "'");
        }
        this.versionString = versionString;
    }

    /**
     * @return Returns the vppConfig.
     */
    public VPPConfig getVppConfig() {
        return vppConfig;
    }

    /**
     * @param vppConfig The vppConfig to set.
     */
    public void setVppConfig(VPPConfig vppConfig) {
        this.vppConfig = vppConfig;
    }

    /**
     * @param mode use "entity" for per-entity generation and "datamap" for per-datamap
     *            generation.
     */
    public void setMode(String mode) {
        if ((false == MODE_ENTITY.equals(mode)) && (false == MODE_DATAMAP.equals(mode))) {
            throw new IllegalStateException("'mode' must be '"
                    + MODE_ENTITY
                    + "' or '"
                    + MODE_DATAMAP
                    + "', but was '"
                    + mode
                    + "'");
        }
        this.mode = mode;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy