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

org.codehaus.modello.plugin.xsd.XsdGenerator Maven / Gradle / Ivy

Go to download

Modello XSD Plugin generates an XML Schema from the model to be able to validate XML content.

The newest version!
package org.codehaus.modello.plugin.xsd;

/*
 * Copyright (c) 2005, Codehaus.org
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy of
 * this software and associated documentation files (the "Software"), to deal in
 * the Software without restriction, including without limitation the rights to
 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
 * of the Software, and to permit persons to whom the Software is furnished to do
 * so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

import javax.inject.Named;

import java.io.File;
import java.io.IOException;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;

import org.codehaus.modello.ModelloException;
import org.codehaus.modello.ModelloParameterConstants;
import org.codehaus.modello.model.Model;
import org.codehaus.modello.model.ModelAssociation;
import org.codehaus.modello.model.ModelClass;
import org.codehaus.modello.model.ModelField;
import org.codehaus.modello.plugin.xsd.metadata.XsdClassMetadata;
import org.codehaus.modello.plugins.xml.AbstractXmlGenerator;
import org.codehaus.modello.plugins.xml.metadata.XmlAssociationMetadata;
import org.codehaus.modello.plugins.xml.metadata.XmlFieldMetadata;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.util.io.CachingWriter;
import org.codehaus.plexus.util.xml.PrettyPrintXMLWriter;
import org.codehaus.plexus.util.xml.XMLWriter;

/**
 * @author Brett Porter
 */
@Named("xsd")
public class XsdGenerator extends AbstractXmlGenerator {
    /**
     * Value standing for any element name (used on xml.tagName)
     */
    private static final String ANY_NAME = "*";

    protected static final String LS = System.lineSeparator();

    @Override
    public void generate(Model model, Map parameters) throws ModelloException {
        initialize(model, parameters);

        try {
            generateXsd(parameters);
        } catch (IOException ex) {
            throw new ModelloException("Exception while generating xsd.", ex);
        }
    }

    private void generateXsd(Map parameters) throws IOException, ModelloException {
        Model objectModel = getModel();

        File directory = getOutputDirectory();

        if (isPackageWithVersion()) {
            directory = new File(directory, getGeneratedVersion().toString());
        }

        if (!directory.exists()) {
            directory.mkdirs();
        }

        // we assume parameters not null
        String xsdFileName = (String) parameters.get(ModelloParameterConstants.OUTPUT_XSD_FILE_NAME);
        boolean enforceMandatoryElements =
                Boolean.parseBoolean((String) parameters.get(ModelloParameterConstants.XSD_ENFORCE_MANDATORY_ELEMENTS));

        File f = new File(directory, objectModel.getId() + "-" + getGeneratedVersion() + ".xsd");

        if (xsdFileName != null) {
            f = new File(directory, xsdFileName);
        }

        try (Writer writer = new CachingWriter(f, StandardCharsets.UTF_8)) {
            XMLWriter w = new PrettyPrintXMLWriter(writer);

            writer.append("").write(LS);

            initHeader(w);

            // TODO: the writer should be knowledgeable of namespaces, but this works
            w.startElement("xs:schema");
            w.addAttribute("xmlns:xs", "http://www.w3.org/2001/XMLSchema");
            w.addAttribute("elementFormDefault", "qualified");

            ModelClass root = objectModel.getClass(objectModel.getRoot(getGeneratedVersion()), getGeneratedVersion());

            String namespace = XsdModelHelper.getNamespace(root.getModel(), getGeneratedVersion());

            w.addAttribute("xmlns", namespace);

            String targetNamespace =
                    XsdModelHelper.getTargetNamespace(root.getModel(), getGeneratedVersion(), namespace);

            // add targetNamespace if attribute is not blank (specifically set to avoid a target namespace)
            if (StringUtils.isNotBlank(targetNamespace)) {
                w.addAttribute("targetNamespace", targetNamespace);
            }

            w.startElement("xs:element");
            String tagName = resolveTagName(root);
            w.addAttribute("name", tagName);
            w.addAttribute("type", root.getName());

            writeClassDocumentation(w, root);

            w.endElement();

            // Element descriptors
            // Traverse from root so "abstract" models aren't included
            int initialCapacity = objectModel.getClasses(getGeneratedVersion()).size();
            writeComplexTypeDescriptor(w, objectModel, root, new HashSet<>(initialCapacity), enforceMandatoryElements);

            w.endElement();
        }
    }

    private static void writeClassDocumentation(XMLWriter w, ModelClass modelClass) {
        writeDocumentation(w, modelClass.getVersionRange().toString(), modelClass.getDescription());
    }

    private static void writeFieldDocumentation(XMLWriter w, ModelField field) {
        writeDocumentation(w, field.getVersionRange().toString(), field.getDescription());
    }

    private static void writeDocumentation(XMLWriter w, String version, String description) {
        if (version != null || description != null) {
            w.startElement("xs:annotation");

            if (version != null) {
                w.startElement("xs:documentation");
                w.addAttribute("source", "version");
                w.writeText(version);
                w.endElement();
            }

            if (description != null) {
                w.startElement("xs:documentation");
                w.addAttribute("source", "description");
                w.writeText(description);
                w.endElement();
            }

            w.endElement();
        }
    }

    private void writeComplexTypeDescriptor(
            XMLWriter w,
            Model objectModel,
            ModelClass modelClass,
            Set written,
            boolean enforceMandatoryElements) {
        written.add(modelClass);

        w.startElement("xs:complexType");
        w.addAttribute("name", modelClass.getName());

        List fields = getFieldsForXml(modelClass, getGeneratedVersion());

        ModelField contentField = getContentField(fields);

        boolean hasContentField = contentField != null;

        List attributeFields = getXmlAttributeFields(fields);

        fields.removeAll(attributeFields);

        if (hasContentField) {
            // yes it's only an extension of xs:string
            w.startElement("xs:simpleContent");

            w.startElement("xs:extension");

            w.addAttribute("base", getXsdType(contentField.getType()));
        }

        writeClassDocumentation(w, modelClass);

        Set toWrite = new HashSet<>();

        if (!fields.isEmpty()) {
            XsdClassMetadata xsdClassMetadata = (XsdClassMetadata) modelClass.getMetadata(XsdClassMetadata.ID);
            boolean compositorAll = XsdClassMetadata.COMPOSITOR_ALL.equals(xsdClassMetadata.getCompositor());

            if (!hasContentField) {
                if (compositorAll) {
                    w.startElement("xs:all");
                } else {
                    w.startElement("xs:sequence");
                }
            }

            for (ModelField field : fields) {
                XmlFieldMetadata xmlFieldMetadata = (XmlFieldMetadata) field.getMetadata(XmlFieldMetadata.ID);

                String fieldTagName = resolveTagName(field, xmlFieldMetadata);

                if (!hasContentField) {
                    if (fieldTagName.equals(ANY_NAME)) {
                        w.startElement("xs:any");
                        w.addAttribute("minOccurs", "0");
                        w.addAttribute("maxOccurs", "unbounded");
                        w.addAttribute("processContents", "skip");
                        w.endElement();
                        continue;
                    }
                    w.startElement("xs:element");

                    if (!enforceMandatoryElements || !field.isRequired()) {
                        // Usually, would only do this if the field is not "required", but due to inheritance, it may be
                        // present, even if not here, so we need to let it slide
                        w.addAttribute("minOccurs", "0");
                    }
                }

                String xsdType = getXsdType(field.getType());

                if ("Date".equals(field.getType()) && "long".equals(xmlFieldMetadata.getFormat())) {
                    xsdType = getXsdType("long");
                }

                if (xmlFieldMetadata.isContent()) {
                    // nothing to add
                } else if ((xsdType != null) || "char".equals(field.getType()) || "Character".equals(field.getType())) {
                    w.addAttribute("name", fieldTagName);

                    if (xsdType != null) {
                        // schema built-in datatype
                        w.addAttribute("type", xsdType);
                    }

                    if (field.getDefaultValue() != null) {
                        // \0 is the implicit default value for char/Character but \0 isn't a valid XML char
                        if (field.getDefaultValue() != "\0") {
                            w.addAttribute("default", field.getDefaultValue());
                        }
                    }

                    writeFieldDocumentation(w, field);

                    if (xsdType == null) {
                        writeCharElement(w);
                    }
                } else {
                    // TODO cleanup/split this part it's no really human readable :-)
                    if (isInnerAssociation(field)) {
                        ModelAssociation association = (ModelAssociation) field;
                        ModelClass fieldModelClass = objectModel.getClass(association.getTo(), getGeneratedVersion());

                        toWrite.add(fieldModelClass);

                        if (association.isManyMultiplicity()) {
                            XmlAssociationMetadata xmlAssociationMetadata = (XmlAssociationMetadata)
                                    association.getAssociationMetadata(XmlAssociationMetadata.ID);

                            if (xmlAssociationMetadata.isWrappedItems()) {
                                w.addAttribute("name", fieldTagName);
                                writeFieldDocumentation(w, field);

                                writeListElement(
                                        w, xmlFieldMetadata, xmlAssociationMetadata, field, fieldModelClass.getName());
                            } else {
                                if (compositorAll) {
                                    // xs:all does not accept maxOccurs="unbounded", xs:sequence MUST be used
                                    // to be able to represent this constraint
                                    throw new IllegalStateException(
                                            field.getName() + " field is declared as xml.listStyle=\"flat\" "
                                                    + "then class " + modelClass.getName()
                                                    + " MUST be declared as xsd.compositor=\"sequence\"");
                                }

                                w.addAttribute("name", resolveTagName(fieldTagName, xmlAssociationMetadata));

                                w.addAttribute("type", fieldModelClass.getName());
                                w.addAttribute("maxOccurs", "unbounded");

                                writeFieldDocumentation(w, field);
                            }
                        } else {
                            // not many multiplicity
                            w.addAttribute("name", fieldTagName);
                            w.addAttribute("type", fieldModelClass.getName());
                            writeFieldDocumentation(w, field);
                        }
                    } else // not inner association
                    {
                        w.addAttribute("name", fieldTagName);

                        writeFieldDocumentation(w, field);

                        if (List.class.getName().equals(field.getType())
                                || Set.class.getName().equals(field.getType())) {
                            ModelAssociation association = (ModelAssociation) field;

                            XmlAssociationMetadata xmlAssociationMetadata = (XmlAssociationMetadata)
                                    association.getAssociationMetadata(XmlAssociationMetadata.ID);

                            writeListElement(w, xmlFieldMetadata, xmlAssociationMetadata, field, getXsdType("String"));
                        } else if (Properties.class.getName().equals(field.getType())
                                || "DOM".equals(field.getType())) {
                            writePropertiesElement(w);
                        } else {
                            throw new IllegalStateException("Non-association field of a non-primitive type '"
                                    + field.getType() + "' for '" + field.getName() + "' in '"
                                    + modelClass.getName() + "' model class");
                        }
                    }
                }
                if (!hasContentField) {
                    w.endElement();
                }
            } // end fields iterator

            if (!hasContentField) {
                w.endElement(); // xs:all or xs:sequence
            }
        }

        for (ModelField field : attributeFields) {
            XmlFieldMetadata xmlFieldMetadata = (XmlFieldMetadata) field.getMetadata(XmlFieldMetadata.ID);

            w.startElement("xs:attribute");

            String xsdType = getXsdType(field.getType());

            String tagName = resolveTagName(field, xmlFieldMetadata);

            w.addAttribute("name", tagName);

            if (xsdType != null) {
                w.addAttribute("type", xsdType);
            }

            if (field.getDefaultValue() != null) {
                w.addAttribute("default", field.getDefaultValue());
            }

            w.addAttribute("use", field.isRequired() ? "required" : "optional");

            writeFieldDocumentation(w, field);

            if ("char".equals(field.getType()) || "Character".equals(field.getType())) {
                writeCharElement(w);
            } else if (xsdType == null) {
                throw new IllegalStateException("Attribute field of a non-primitive type '" + field.getType()
                        + "' for '" + field.getName() + "' in '" + modelClass.getName() + "' model class");
            }

            w.endElement();
        }

        if (hasContentField) {
            w.endElement(); // xs:extension

            w.endElement(); // xs:simpleContent
        }

        w.endElement(); // xs:complexType

        for (ModelClass fieldModelClass : toWrite) {
            if (!written.contains(fieldModelClass)) {
                writeComplexTypeDescriptor(w, objectModel, fieldModelClass, written, enforceMandatoryElements);
            }
        }
    }

    private static void writeCharElement(XMLWriter w) {
        // a char, described as a simpleType base on string with a length restriction to 1
        w.startElement("xs:simpleType");

        w.startElement("xs:restriction");
        w.addAttribute("base", "xs:string");

        w.startElement("xs:length");
        w.addAttribute("value", "1");
        w.addAttribute("fixed", "true");

        w.endElement();

        w.endElement();

        w.endElement();
    }

    private static void writePropertiesElement(XMLWriter w) {
        w.startElement("xs:complexType");

        w.startElement("xs:sequence");

        w.startElement("xs:any");
        w.addAttribute("minOccurs", "0");
        w.addAttribute("maxOccurs", "unbounded");
        w.addAttribute("processContents", "skip");

        w.endElement();

        w.endElement();

        w.endElement();
    }

    private void writeListElement(
            XMLWriter w,
            XmlFieldMetadata xmlFieldMetadata,
            XmlAssociationMetadata xmlAssociationMetadata,
            ModelField field,
            String type) {
        String fieldTagName = resolveTagName(field, xmlFieldMetadata);

        String valuesTagName = resolveTagName(fieldTagName, xmlAssociationMetadata);

        w.startElement("xs:complexType");

        w.startElement("xs:sequence");

        if (valuesTagName.equals(ANY_NAME)) {
            w.startElement("xs:any");
            w.addAttribute("processContents", "skip");
        } else {
            w.startElement("xs:element");
            w.addAttribute("type", type);
            w.addAttribute("name", valuesTagName);
        }
        w.addAttribute("minOccurs", "0");
        w.addAttribute("maxOccurs", "unbounded");

        w.endElement();

        w.endElement();

        w.endElement();
    }

    private static String getXsdType(String type) {
        if ("String".equals(type)) {
            return "xs:string";
        } else if ("boolean".equals(type) || "Boolean".equals(type)) {
            return "xs:boolean";
        } else if ("byte".equals(type) || "Byte".equals(type)) {
            return "xs:byte";
        } else if ("short".equals(type) || "Short".equals(type)) {
            return "xs:short";
        } else if ("int".equals(type) || "Integer".equals(type)) {
            return "xs:int";
        } else if ("long".equals(type) || "Long".equals(type)) {
            return "xs:long";
        } else if ("float".equals(type) || "Float".equals(type)) {
            return "xs:float";
        } else if ("double".equals(type) || "Double".equals(type)) {
            return "xs:double";
        } else if ("Date".equals(type)) {
            return "xs:dateTime";
        } else {
            return null;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy