Search in sources :

Example 1 with Choice

use of org.eclipse.persistence.internal.oxm.schema.model.Choice in project eclipselink by eclipse-ee4j.

the class SchemaGenerator method buildSchemaComponentsForXPath.

/**
 * This method will build element/complexType/typedefparticle components for a given xml-path,
 * and return an XmlPathResult instance containg the sequence that the target should be added
 * to, as well as the current schema - which could be different than the working schema used
 * before calling this method in the case of a prefixed path element from a different namespace.
 * Regarding the path 'target', if the xml-path was "contact-info/address/street", "street"
 * would be the target.  In this case the sequence containing the "address" element would be
 * set in the XmlPathResult to be returned.
 *
 * The exception case is an 'any', where we want to process the last path element before
 * returning - this is necessary due to the fact that an Any will be added to the sequence
 * in place of the last path element by the calling method.
 */
private AddToSchemaResult buildSchemaComponentsForXPath(XPathFragment frag, AddToSchemaResult xpr, boolean isChoice, Property next) {
    boolean isAny = next.isAny() || next.isAnyAttribute();
    TypeDefParticle currentParticle = xpr.particle;
    Schema workingSchema = xpr.schema;
    // each nested choice on a collection will be unbounded
    boolean isUnbounded = false;
    if (currentParticle != null) {
        isUnbounded = (currentParticle.getMaxOccurs() != null && currentParticle.getMaxOccurs() == Occurs.UNBOUNDED);
    }
    // don't process the last frag; that will be handled by the calling method if necessary
    // note that we may need to process the last frag if it has a namespace or is an 'any'
    boolean lastFrag = (frag.getNextFragment() == null || frag.getNextFragment().nameIsText());
    // if the element is already in the sequence we don't want the calling method to add a second one
    if (lastFrag && (elementExistsInParticle(frag.getLocalName(), frag.getShortName(), currentParticle) != null)) {
        xpr.particle = null;
        return xpr;
    }
    // if the current element exists, use it; otherwise create a new one
    Element currentElement = elementExistsInParticle(frag.getLocalName(), frag.getShortName(), currentParticle);
    boolean currentElementExists = (currentElement != null);
    if (!currentElementExists) {
        currentElement = new Element();
        // don't set the element name yet, as it may end up being a ref
        ComplexType cType = new ComplexType();
        TypeDefParticle particle = null;
        if (isChoice) {
            particle = new Choice();
            if (isUnbounded) {
                particle.setMaxOccurs(Occurs.UNBOUNDED);
            }
        } else {
            XPathFragment nextFragment = frag.getNextFragment();
            if (frag.containsIndex() || frag.getPredicate() != null || (!next.isXmlList() && null != nextFragment && nextFragment.isAttribute() && helper.isCollectionType(next.getType()))) {
                currentElement.setMaxOccurs(Occurs.UNBOUNDED);
            }
            particle = new Sequence();
        }
        cType.setTypeDefParticle(particle);
        currentElement.setComplexType(cType);
    } else {
        // if the current element already exists, we may need to change it's type
        XPathFragment nextFrag = frag.getNextFragment();
        if (nextFrag != null && nextFrag.isAttribute()) {
            if (currentElement.getType() != null && currentElement.getComplexType() == null) {
                // there's already a text mapping to this element, so
                // attributes can be added by making it complex with
                // simple content.
                SimpleType type = currentElement.getSimpleType();
                if (type != null) {
                    ComplexType cType = new ComplexType();
                    cType.setSimpleContent(new SimpleContent());
                    Extension extension = new Extension();
                    extension.setBaseType(type.getRestriction().getBaseType());
                    cType.getSimpleContent().setExtension(extension);
                    currentElement.setSimpleType(null);
                    currentElement.setComplexType(cType);
                } else {
                    String eType = currentElement.getType();
                    ComplexType cType = new ComplexType();
                    SimpleContent sContent = new SimpleContent();
                    Extension extension = new Extension();
                    extension.setBaseType(eType);
                    sContent.setExtension(extension);
                    cType.setSimpleContent(sContent);
                    currentElement.setType(null);
                    currentElement.setComplexType(cType);
                }
            }
        }
    }
    // may need to create a ref, depending on the namespace
    Element globalElement = null;
    String fragUri = frag.getNamespaceURI();
    if (fragUri != null) {
        Schema fragSchema = getSchemaForNamespace(fragUri);
        String targetNS = workingSchema.getTargetNamespace();
        // handle Attribute case
        if (frag.isAttribute()) {
            if (fragSchema == null || (fragSchema.isAttributeFormDefault() && !fragUri.equals(targetNS)) || (!fragSchema.isAttributeFormDefault() && fragUri.length() > 0)) {
                // if fragSchema is null, just generate the ref
                if (fragSchema != null) {
                    Attribute globalAttribute = null;
                    globalAttribute = fragSchema.getTopLevelAttributes().get(frag.getLocalName());
                    if (globalAttribute == null) {
                        globalAttribute = createGlobalAttribute(frag, workingSchema, fragSchema, next);
                    }
                } else {
                    // may need to add an import
                    addImportIfRequired(workingSchema, null, fragUri);
                }
                // add the attribute ref to the current element
                String attributeRefName;
                if (fragUri.equals(targetNS)) {
                    String prefix = fragSchema.getNamespaceResolver().resolveNamespaceURI(fragUri);
                    attributeRefName = prefix + COLON + frag.getLocalName();
                } else {
                    attributeRefName = frag.getShortName();
                }
                TypeDefParticleOwner type;
                if (currentParticle != null) {
                    type = currentParticle.getOwner();
                } else {
                    type = xpr.simpleContentType;
                }
                if (type instanceof ComplexType) {
                    createRefAttribute(attributeRefName, (ComplexType) type);
                }
                // set the frag's schema as it may be different than the current schema
                xpr.schema = fragSchema;
                // ref case - indicate to the calling method that there's nothing to do
                xpr.particle = null;
            }
            // since we are dealing with an attribute, we are on the last fragment; return
            return xpr;
        }
        // here we are dealing with an Element
        if ((fragSchema.isElementFormDefault() && !fragUri.equals(targetNS)) || (!fragSchema.isElementFormDefault() && fragUri.length() > 0)) {
            // must generate a global element and create a reference to it
            // if the global element exists, use it; otherwise create a new one
            globalElement = fragSchema.getTopLevelElements().get(frag.getLocalName());
            if (globalElement == null) {
                globalElement = createGlobalElement(frag, workingSchema, fragSchema, isChoice, isUnbounded, next, (lastFrag && !isAny));
            }
            // if the current element doesn't exist set a ref and add it to the sequence
            if (!currentElementExists) {
                // use prefix from the working schema's resolver - add prefix/uri pair if necessary
                String fragPrefix = workingSchema.getNamespaceResolver().resolveNamespaceURI(fragUri);
                if (fragPrefix == null) {
                    fragPrefix = workingSchema.getNamespaceResolver().generatePrefix(frag.getPrefix());
                    workingSchema.getNamespaceResolver().put(fragPrefix, fragUri);
                }
                currentElement = createRefElement(fragPrefix + COLON + frag.getLocalName(), currentParticle);
                if (frag.containsIndex() || frag.getPredicate() != null || helper.isCollectionType(next.getType())) {
                    currentElement.setMaxOccurs(Occurs.UNBOUNDED);
                }
                currentElementExists = true;
            }
            // set the frag's schema as it may be different than the current schema
            xpr.schema = fragSchema;
            // at this point, if we are dealing with the last fragment we will need to return
            if (lastFrag) {
                // add a second one...unless we're dealing with an 'any'
                if (isAny) {
                    // set the particle that the 'any' will be added to by the calling method
                    xpr.particle = globalElement.getComplexType().getTypeDefParticle();
                    return xpr;
                }
                // ref case - indicate to the calling method that there's nothing to do
                xpr.particle = null;
                return xpr;
            }
            // make the global element current
            currentElement = globalElement;
        }
    }
    if (!lastFrag || (lastFrag && isAny)) {
        // if we didn't process a global element, and the current element isn't already in the sequence, add it
        if (!currentElementExists && globalElement == null) {
            currentElement.setName(frag.getLocalName());
            Integer minOccurs = next.getMinOccurs();
            if (minOccurs != null)
                currentElement.setMinOccurs(String.valueOf(minOccurs));
            else
                currentElement.setMinOccurs(Occurs.ZERO);
            currentParticle.addElement(currentElement);
        }
        // set the correct particle to use/return
        if (currentElement.getComplexType() != null) {
            if (currentElement.getComplexType().getTypeDefParticle() == null) {
                // complexType with simple-content
                xpr.simpleContentType = currentElement.getComplexType();
                xpr.particle = null;
            } else {
                xpr.particle = currentElement.getComplexType().getTypeDefParticle();
            }
        } else {
            // If there's no complex type, we're building the path through an element with
            // a simple type. In order to build the path through this
            // element, switch to a complex type with simple content.
            SimpleType type = currentElement.getSimpleType();
            if (type != null) {
                ComplexType cType = new ComplexType();
                cType.setSimpleContent(new SimpleContent());
                Extension extension = new Extension();
                extension.setBaseType(type.getRestriction().getBaseType());
                cType.getSimpleContent().setExtension(extension);
                currentElement.setSimpleType(null);
                currentElement.setComplexType(cType);
                xpr.particle = null;
                xpr.simpleContentType = cType;
            } else {
                String eType = currentElement.getType();
                ComplexType cType = new ComplexType();
                SimpleContent sContent = new SimpleContent();
                Extension extension = new Extension();
                extension.setBaseType(eType);
                sContent.setExtension(extension);
                cType.setSimpleContent(sContent);
                currentElement.setType(null);
                currentElement.setComplexType(cType);
                xpr.particle = null;
                xpr.simpleContentType = cType;
            }
        }
    }
    // if we're on the last fragment, we're done
    if (lastFrag) {
        return xpr;
    }
    // call back into this method to process the next path element
    return buildSchemaComponentsForXPath(frag.getNextFragment(), xpr, isChoice, next);
}
Also used : TypeDefParticle(org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticle) Choice(org.eclipse.persistence.internal.oxm.schema.model.Choice) AnyAttribute(org.eclipse.persistence.internal.oxm.schema.model.AnyAttribute) Attribute(org.eclipse.persistence.internal.oxm.schema.model.Attribute) Schema(org.eclipse.persistence.internal.oxm.schema.model.Schema) XmlVirtualAccessMethodsSchema(org.eclipse.persistence.jaxb.xmlmodel.XmlVirtualAccessMethodsSchema) Element(org.eclipse.persistence.internal.oxm.schema.model.Element) XPathFragment(org.eclipse.persistence.internal.oxm.XPathFragment) Sequence(org.eclipse.persistence.internal.oxm.schema.model.Sequence) Extension(org.eclipse.persistence.internal.oxm.schema.model.Extension) SimpleType(org.eclipse.persistence.internal.oxm.schema.model.SimpleType) TypeDefParticleOwner(org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticleOwner) SimpleContent(org.eclipse.persistence.internal.oxm.schema.model.SimpleContent) ComplexType(org.eclipse.persistence.internal.oxm.schema.model.ComplexType)

Example 2 with Choice

use of org.eclipse.persistence.internal.oxm.schema.model.Choice in project eclipselink by eclipse-ee4j.

the class SchemaGenerator method addToSchemaType.

public void addToSchemaType(TypeInfo ownerTypeInfo, java.util.List<Property> properties, TypeDefParticle compositor, ComplexType type, Schema workingSchema) {
    // If there are no properties we don't want a sequence/choice or all tag written out
    if (properties.size() == 0) {
        type.setAll(null);
        type.setSequence(null);
        type.setChoice(null);
        ownerTypeInfo.setCompositor(null);
        return;
    }
    boolean extAnyAdded = false;
    // generate schema components for each property
    for (Property next : properties) {
        if (next == null) {
            continue;
        }
        Schema currentSchema = workingSchema;
        TypeDefParticle parentCompositor = compositor;
        boolean isChoice = (parentCompositor instanceof Choice);
        ComplexType parentType = type;
        // ignore transient and inverse reference/non-writeable properties
        if (!next.isTransient() && !(next.isInverseReference() && !next.isWriteableInverseReference())) {
            // handle xml extensions
            if (next.isVirtual()) {
                boolean extSchemaAny = false;
                if (ownerTypeInfo.getXmlVirtualAccessMethods().getSchema() != null) {
                    extSchemaAny = ownerTypeInfo.getXmlVirtualAccessMethods().getSchema().equals(XmlVirtualAccessMethodsSchema.ANY);
                }
                if (extSchemaAny && !next.isAttribute()) {
                    if (!extAnyAdded) {
                        addAnyToSchema(next, compositor, true, Constants.ANY_NAMESPACE_ANY);
                        extAnyAdded = true;
                        continue;
                    } else {
                        // any already added
                        continue;
                    }
                } else {
                // proceed, adding the schema element as usual
                }
            }
            // handle transformers
            if (next.isSetXmlTransformation() && next.getXmlTransformation().isSetXmlWriteTransformers()) {
                addTransformerToSchema(next, ownerTypeInfo, compositor, type, workingSchema);
                continue;
            }
            // handle XmlJoinNodes
            if (next.isSetXmlJoinNodes()) {
                addXmlJoinNodesToSchema(next, parentCompositor, currentSchema, parentType);
                continue;
            }
            // deal with xml-path case
            if (next.getXmlPath() != null) {
                // create schema components based on the XmlPath
                AddToSchemaResult xpr = addXPathToSchema(next, parentCompositor, currentSchema, isChoice, type);
                // if the returned object or schema component is null there is nothing to do
                if (xpr == null || ((parentCompositor = xpr.particle) == null && xpr.simpleContentType == null)) {
                    continue;
                }
                // now process the property as per usual, adding to the created schema component
                if (xpr.schema == null) {
                    // no need to generate the attribute in the target schema
                    continue;
                }
                currentSchema = xpr.schema;
                if (parentCompositor == null) {
                    parentType = xpr.simpleContentType;
                } else if (parentCompositor.getOwner() instanceof ComplexType) {
                    parentType = ((ComplexType) parentCompositor.getOwner());
                }
            // deal with the XmlElementWrapper case
            } else if (!isChoice && !next.isMap() && next.isSetXmlElementWrapper()) {
                AddToSchemaResult asr = addXmlElementWrapperToSchema(next, currentSchema, compositor);
                // if returned object is null there is nothing to do
                if (asr == null) {
                    continue;
                }
                // the returned object contains ComplexType and TypeDefParticles to use
                parentType = asr.type;
                parentCompositor = asr.particle;
            }
            // handle mixed content
            if (next.isMixedContent()) {
                parentType.setMixed(true);
            }
            // handle attribute
            if (next.isAttribute() && !next.isAnyAttribute()) {
                addAttributeToSchema(buildAttribute(next, currentSchema), next.getSchemaName(), currentSchema, parentType);
            // handle any attribute
            } else if (next.isAnyAttribute()) {
                addAnyAttributeToSchema(parentType);
            // handle choice
            } else if (next.isChoice()) {
                addChoiceToSchema(next, ownerTypeInfo, parentType, parentCompositor, currentSchema);
            // handle reference
            } else if (next.isReference()) {
                addReferenceToSchema(next, currentSchema, parentCompositor);
            // handle any
            } else if (next.isAny() || next.getVariableAttributeName() != null) {
                addAnyToSchema(next, parentCompositor);
            // add an element
            } else if (!(ownerTypeInfo.getXmlValueProperty() != null && ownerTypeInfo.getXmlValueProperty() == next)) {
                Element element = buildElement(next, parentCompositor instanceof All, currentSchema, ownerTypeInfo);
                addElementToSchema(element, next.getSchemaName().getNamespaceURI(), next.isPositional(), parentCompositor, currentSchema, ownerTypeInfo.getJavaClass().getPackageName());
            }
        }
    }
}
Also used : All(org.eclipse.persistence.internal.oxm.schema.model.All) TypeDefParticle(org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticle) Choice(org.eclipse.persistence.internal.oxm.schema.model.Choice) Schema(org.eclipse.persistence.internal.oxm.schema.model.Schema) XmlVirtualAccessMethodsSchema(org.eclipse.persistence.jaxb.xmlmodel.XmlVirtualAccessMethodsSchema) Element(org.eclipse.persistence.internal.oxm.schema.model.Element) ComplexType(org.eclipse.persistence.internal.oxm.schema.model.ComplexType)

Example 3 with Choice

use of org.eclipse.persistence.internal.oxm.schema.model.Choice in project eclipselink by eclipse-ee4j.

the class SchemaGenerator method addAnyToSchema.

/**
 * Convenience method for processing an any property. Required
 * schema components will be generated and set accordingly.
 *
 * @param property the choice property to be processed
 * @param compositor the sequence/choice/all to modify
 * @param isCollection if true will be unbounded
 * @param anyNamespace value for the Any's namespace attribute
 */
private void addAnyToSchema(Property property, TypeDefParticle compositor, boolean isCollection, String anyNamespace) {
    Any any = new Any();
    any.setNamespace(anyNamespace);
    if (property.isLax()) {
        any.setProcessContents(Any.LAX);
    } else {
        any.setProcessContents(SKIP);
    }
    if (isCollection) {
        any.setMinOccurs(Occurs.ZERO);
        any.setMaxOccurs(Occurs.UNBOUNDED);
    }
    if (compositor instanceof Sequence) {
        ((Sequence) compositor).addAny(any);
    } else if (compositor instanceof Choice) {
        ((Choice) compositor).addAny(any);
    }
}
Also used : Choice(org.eclipse.persistence.internal.oxm.schema.model.Choice) Sequence(org.eclipse.persistence.internal.oxm.schema.model.Sequence) Any(org.eclipse.persistence.internal.oxm.schema.model.Any)

Example 4 with Choice

use of org.eclipse.persistence.internal.oxm.schema.model.Choice in project eclipselink by eclipse-ee4j.

the class SchemaModelGenerator method processChoiceMapping.

/**
 * Process a given XMLChoiceMapping.
 */
protected void processChoiceMapping(Map<Field, Class<?>> fieldToClassMap, List<XMLChoiceFieldToClassAssociation> choiceFieldToClassList, Sequence seq, ComplexType ct, HashMap<String, Schema> schemaForNamespace, Schema workingSchema, SchemaModelGeneratorProperties properties, List<Descriptor> descriptors, boolean isCollection) {
    Choice theChoice = new Choice();
    if (isCollection) {
        theChoice.setMaxOccurs(Occurs.UNBOUNDED);
    }
    for (XMLChoiceFieldToClassAssociation next : choiceFieldToClassList) {
        Field field = next.getXmlField();
        Element element = buildElement(field.getXPathFragment().getShortName(), Occurs.ZERO, null);
        QName schemaTypeQName = field.getSchemaType();
        if (schemaTypeQName != null) {
            element.setType(getSchemaTypeString(schemaTypeQName, workingSchema));
        } else {
            element = processReferenceDescriptor(element, getDescriptorByClass(fieldToClassMap.get(field), descriptors), schemaForNamespace, workingSchema, properties, descriptors, field, false);
        }
        theChoice.addElement(element);
    }
    seq.addChoice(theChoice);
}
Also used : Field(org.eclipse.persistence.internal.oxm.mappings.Field) XMLChoiceFieldToClassAssociation(org.eclipse.persistence.internal.oxm.XMLChoiceFieldToClassAssociation) Choice(org.eclipse.persistence.internal.oxm.schema.model.Choice) QName(javax.xml.namespace.QName) Element(org.eclipse.persistence.internal.oxm.schema.model.Element)

Example 5 with Choice

use of org.eclipse.persistence.internal.oxm.schema.model.Choice in project eclipselink by eclipse-ee4j.

the class SchemaGenerator method createGlobalElement.

/**
 * Create a global element.  An import is added if necessary.  This method
 * will typically be called when processing an XPath and a prefixed path
 * element is encountered the requires an element ref.
 *
 * @param frag XPathFragment which wil lbe used to create the global element
 * @param workingSchema current schema
 * @param fragSchema frag's schema
 * @param isChoice indicates if we need to construct a choice
 * @param isUnbounded maxOccurs setting for choice
 * @param prop property which owns the xml-path
 * @param shouldSetType if this is the last fragment in the xml-path and not an 'any', we should set the type
 */
public Element createGlobalElement(XPathFragment frag, Schema workingSchema, Schema fragSchema, boolean isChoice, boolean isUnbounded, Property prop, boolean shouldSetType) {
    Element gElement = new Element();
    gElement.setName(frag.getLocalName());
    if (shouldSetType) {
        gElement.setType(getQualifiedTypeName(prop, fragSchema));
    } else {
        ComplexType gCType = new ComplexType();
        TypeDefParticle particle;
        if (isChoice) {
            particle = new Choice();
            if (isUnbounded) {
                particle.setMaxOccurs(Occurs.UNBOUNDED);
            }
        } else {
            particle = new Sequence();
        }
        gCType.setTypeDefParticle(particle);
        gElement.setComplexType(gCType);
    }
    fragSchema.addTopLevelElement(gElement);
    addImportIfRequired(workingSchema, fragSchema, frag.getNamespaceURI());
    return gElement;
}
Also used : TypeDefParticle(org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticle) Choice(org.eclipse.persistence.internal.oxm.schema.model.Choice) Element(org.eclipse.persistence.internal.oxm.schema.model.Element) Sequence(org.eclipse.persistence.internal.oxm.schema.model.Sequence) ComplexType(org.eclipse.persistence.internal.oxm.schema.model.ComplexType)

Aggregations

Choice (org.eclipse.persistence.internal.oxm.schema.model.Choice)7 Element (org.eclipse.persistence.internal.oxm.schema.model.Element)5 Sequence (org.eclipse.persistence.internal.oxm.schema.model.Sequence)5 ComplexType (org.eclipse.persistence.internal.oxm.schema.model.ComplexType)3 Schema (org.eclipse.persistence.internal.oxm.schema.model.Schema)3 TypeDefParticle (org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticle)3 XmlVirtualAccessMethodsSchema (org.eclipse.persistence.jaxb.xmlmodel.XmlVirtualAccessMethodsSchema)3 ArrayList (java.util.ArrayList)1 QName (javax.xml.namespace.QName)1 XMLChoiceFieldToClassAssociation (org.eclipse.persistence.internal.oxm.XMLChoiceFieldToClassAssociation)1 XPathFragment (org.eclipse.persistence.internal.oxm.XPathFragment)1 Field (org.eclipse.persistence.internal.oxm.mappings.Field)1 All (org.eclipse.persistence.internal.oxm.schema.model.All)1 Any (org.eclipse.persistence.internal.oxm.schema.model.Any)1 AnyAttribute (org.eclipse.persistence.internal.oxm.schema.model.AnyAttribute)1 Attribute (org.eclipse.persistence.internal.oxm.schema.model.Attribute)1 Extension (org.eclipse.persistence.internal.oxm.schema.model.Extension)1 SimpleContent (org.eclipse.persistence.internal.oxm.schema.model.SimpleContent)1 SimpleType (org.eclipse.persistence.internal.oxm.schema.model.SimpleType)1 TypeDefParticleOwner (org.eclipse.persistence.internal.oxm.schema.model.TypeDefParticleOwner)1