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

org.apache.xerces.impl.xs.models.XS11AllCM Maven / Gradle / Ivy

/*
 * 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.xerces.impl.xs.models;

import java.util.ArrayList;
import java.util.List;
import java.util.Vector;

import org.apache.xerces.impl.Constants;
import org.apache.xerces.impl.xs.SchemaSymbols;
import org.apache.xerces.impl.xs.SubstitutionGroupHandler;
import org.apache.xerces.impl.xs.XMLSchemaException;
import org.apache.xerces.impl.xs.XSConstraints;
import org.apache.xerces.impl.xs.XSElementDecl;
import org.apache.xerces.impl.xs.XSElementDeclHelper;
import org.apache.xerces.impl.xs.XSOpenContentDecl;
import org.apache.xerces.impl.xs.XSParticleDecl;
import org.apache.xerces.impl.xs.XSWildcardDecl;
import org.apache.xerces.xni.QName;

/**
 * XSAll11CM implements XSCMValidator and handles <all>.
 *
 * @xerces.internal
 *
 * @author Khaled Noaman, IBM
 * @version $Id: XS11AllCM.java 1155073 2011-08-08 20:02:50Z knoaman $
 */
public class XS11AllCM implements XSCMValidator, XS11CMRestriction.XS11CM {

    //
    // Constants
    //

    // start the content model: did not see any children
    private static final short STATE_START = 0;
    private static final short STATE_CHILD = 1;
    private static final short STATE_SUFFIX = 2;

    //
    // Data
    //
    private final boolean fHasOptionalContent;

    private final XSElementDecl fElements[];
    private final XSWildcardDecl fWildcards[];
    private final int fMinOccurs[], fMaxOccurs[];
    private final int fNumElements, fNumTotal;

    private final XSOpenContentDecl fOpenContent;

    //
    // Constructors
    //

    public XS11AllCM (boolean hasOptionalContent, int size, XSParticleDecl[] particles,
            XSOpenContentDecl openContent) {
        fHasOptionalContent = hasOptionalContent;

        // Index 0 is not used.
        int numE = 1;
        for (int i = 0; i < size; i++) {
            if (particles[i].fType == XSParticleDecl.PARTICLE_ELEMENT) {
                numE++;
            }
        }
        fNumElements = numE;
        fNumTotal = size+1;
        
        if (numE > 1) {
            fElements = new XSElementDecl[numE];
        }
        else {
            fElements = null;
        }
        if (fNumTotal > numE) {
            fWildcards = new XSWildcardDecl[fNumTotal];
        }
        else {
            fWildcards = null;
        }
        if (fNumTotal > 1) {
            fMinOccurs = new int[fNumTotal];
            fMaxOccurs = new int[fNumTotal];
        }
        else {
            fMinOccurs = null;
            fMaxOccurs = null;
        }

        int numW = numE;
        numE = 1;
        for (int i = 0; i < size; i++) {
            XSParticleDecl particle = particles[i];
            if (particle.fType == XSParticleDecl.PARTICLE_ELEMENT) {
                fElements[numE] = (XSElementDecl)particle.fValue;
                fMinOccurs[numE] = particle.fMinOccurs;
                fMaxOccurs[numE] = particle.fMaxOccurs;
                numE++;
            }
            else {
                fWildcards[numW] = (XSWildcardDecl)particle.fValue;
                fMinOccurs[numW] = particle.fMinOccurs;
                fMaxOccurs[numW] = particle.fMaxOccurs;
                numW++;
            }
        }

        fOpenContent = openContent;
    }

    //
    // XSCMValidator methods
    //

    /**
     * This methods to be called on entering a first element whose type
     * has this content model. It will return the initial state of the
     * content model
     *
     * @return Start state of the content model
     */
    public int[] startContentModel() {
        int[] state = new int[fNumTotal];
        for (int i = 0; i < fNumTotal; i++) {
            state[i] = STATE_START;
        }
        return state;
    }

    // convenient method: when error occurs, to find a matching decl
    // from the candidate elements.
    Object findMatchingDecl(QName elementName, SubstitutionGroupHandler subGroupHandler) {
        Object matchingDecl = findMatchingElemDecl(elementName, subGroupHandler);

        if (matchingDecl != null) {
            return matchingDecl;
        }

        for (int i = fNumElements; i < fNumTotal; i++) {
            if (fWildcards[i].allowQName(elementName)) {
                return fWildcards[i];
            }
        }

        return null;
    }

    // convenient method: to find a matching element decl 
    public XSElementDecl findMatchingElemDecl(QName elementName, SubstitutionGroupHandler subGroupHandler) {
        for (int i = 1; i < fNumElements; i++) {
            final XSElementDecl matchingDecl = subGroupHandler.getMatchingElemDecl(elementName, fElements[i], Constants.SCHEMA_VERSION_1_1);
            if (matchingDecl != null) {
                return matchingDecl;
            }
        }

        return null;
    }

    public boolean allowExpandedName(XSWildcardDecl wildcard,
                                     QName curElem,
                                     SubstitutionGroupHandler subGroupHandler,
                                     XSElementDeclHelper eDeclHelper) {
        if (wildcard.allowQName(curElem)) {
            if (wildcard.fDisallowedSibling && findMatchingElemDecl(curElem, subGroupHandler) != null) {
                return false;
            }
            if (wildcard.fDisallowedDefined && eDeclHelper.getGlobalElementDecl(curElem) != null) {
                return false;
            }
            return true;
        }
        return false;
    }

    /**
     * The method corresponds to one transition in the content model.
     *
     * @param elementName
     * @param currentState  Current state
     * @return an element decl object
     */
    public Object oneTransition (QName elementName, int[] currentState,
            SubstitutionGroupHandler subGroupHandler, XSElementDeclHelper eDeclHelper) {
        // error state
        if (currentState[0] < 0) {
            currentState[0] = XSCMValidator.SUBSEQUENT_ERROR;
            return findMatchingDecl(elementName, subGroupHandler);
        }

        // open content - suffix mode
        if (currentState[0] == STATE_SUFFIX) {
            if (allowExpandedName(fOpenContent.fWildcard, elementName, subGroupHandler, eDeclHelper)) {
                return fOpenContent;
            }
            else { // error
                currentState[0] = XSCMValidator.FIRST_ERROR;
                return findMatchingDecl(elementName, subGroupHandler);
            }
        }

        // seen child
        currentState[0] = STATE_CHILD;

        for (int i = 1; i < fNumElements; i++) {
            if (currentState[i] == fMaxOccurs[i]) {
                continue;
            }
            Object matchingDecl = subGroupHandler.getMatchingElemDecl(elementName, fElements[i], Constants.SCHEMA_VERSION_1_1);
            if (matchingDecl != null) {
                // found the decl, mark this element as "seen".
                ++currentState[i];
                return matchingDecl;
            }
        }

        for (int i = fNumElements; i < fNumTotal; i++) {
            if (currentState[i] == fMaxOccurs[i]) {
                continue;
            }
            if (allowExpandedName(fWildcards[i], elementName, subGroupHandler, eDeclHelper)) {
                // found the decl, mark this element as "seen".
                ++currentState[i];
                return fWildcards[i];
            }
        }

        // apply open content, if present
        if (fOpenContent != null) {
            if (fOpenContent.fMode == XSOpenContentDecl.MODE_SUFFIX) {
                if (isFinal(currentState)) {
                    currentState[0] = STATE_SUFFIX;
                }
                else {
                    currentState[0] = XSCMValidator.FIRST_ERROR;
                    return findMatchingDecl(elementName, subGroupHandler);
                }
            }
            if (allowExpandedName(fOpenContent.fWildcard, elementName, subGroupHandler, eDeclHelper)) {
            //if (fOpenContent.fWildcard.allowQName(elementName)) {
                return fOpenContent;
            }
        }

        // couldn't find the decl, change to error state.
        currentState[0] = XSCMValidator.FIRST_ERROR;
        return findMatchingDecl(elementName, subGroupHandler);
    }

    /**
     * The method indicates the end of list of children
     *
     * @param currentState  Current state of the content model
     * @return true if the last state was a valid final state
     */
    public boolean endContentModel (int[] currentState) {
        int state = currentState[0];

        if (state == XSCMValidator.FIRST_ERROR || state == XSCMValidator.SUBSEQUENT_ERROR) {
            return false;
        }

        return isFinal(currentState);
    }

    /**
     * check whether this content violates UPA constraint.
     *
     * @param subGroupHandler the substitution group handler
     * @param xsConstraints the XML Schema Constraint checker
     * @return true if this content model contains other or list wildcard
     */
    public boolean checkUniqueParticleAttribution(SubstitutionGroupHandler subGroupHandler, XSConstraints xsConstraints) throws XMLSchemaException {
        // check whether there is conflict between any two leaves
        for (int i = 1; i < fNumElements; i++) {
            for (int j = i+1; j < fNumElements; j++) {
                if (xsConstraints.overlapUPA(fElements[i], fElements[j], subGroupHandler)) {
                    // REVISIT: do we want to report all errors? or just one?
                    throw new XMLSchemaException("cos-nonambig", new Object[]{fElements[i].toString(),
                                                                              fElements[j].toString()});
                }
            }
        }
        for (int i = fNumElements; i < fNumTotal; i++) {
            for (int j = i+1; j < fNumTotal; j++) {
                if (xsConstraints.overlapUPA(fWildcards[i], fWildcards[j])) {
                    // REVISIT: do we want to report all errors? or just one?
                    throw new XMLSchemaException("cos-nonambig", new Object[]{fWildcards[i].toString(),
                                                                              fWildcards[j].toString()});
                }
            }
        }
        return false;
    }

    /**
     * Check which elements are valid to appear at this point. This method also
     * works if the state is in error, in which case it returns what should
     * have been seen.
     *
     * @param state  the current state
     * @return       a Vector whose entries are instances of
     *               either XSWildcardDecl or XSElementDecl.
     */
    public Vector whatCanGoHere(int[] state) {
        Vector ret = new Vector();

        // handle element declarations
        for (int i = 1; i < fNumElements; i++) {
            // we only try to look for a matching decl if we have not seen
            // this element yet or we have seen it less times than its maxOccurs.
            if (state[i] == STATE_START || state[i] < fMaxOccurs[i]) {
                ret.addElement(fElements[i]);
            }
        }

        // only add wildcards if no element can be matched
        if (ret.size() == 0) {
            for (int i = fNumElements; i < fNumTotal; i++) {
                if (state[i] == STATE_START || state[i] < fMaxOccurs[i]) {
                    ret.addElement(fWildcards[i]);
                }
            }
        }

        // if 'ret' is empty and we have an open content, add the open content to 'ret'
        if (ret.size() == 0 && fOpenContent != null) {
            ret.add(fOpenContent);
        }
        return ret;
    }
    
    public int [] occurenceInfo(int[] state) {
        // REVISIT: maxOccurs > 1 is allowed by  in XML Schema 1.1
        return null;
    }
    
    public String getTermName(int termId) {
        // REVISIT: maxOccurs > 1 is allowed by  in XML Schema 1.1
        return null;
    }

    public boolean isCompactedForUPA() {
        return false;
    }

    private boolean isFinal(int[] currentState) {
        // If  has minOccurs of zero and there are
        // no children to validate, it is trivially valid
        //
        // XML Schema 1.1
        // reached final state and doing suffix validation against open content
        if ((fHasOptionalContent && currentState[0] == STATE_START) || currentState[0] == STATE_SUFFIX) {
            return true;
        }

        for (int i = 1; i < fNumTotal; i++) {
            // if one element is required, but not present, then error
            if (currentState[i] < fMinOccurs[i]) {
                return false;
            }
        }

        return true;
    }

    public XSElementDecl nextElementTransition(int[] s, int[] sn, int[] index) {
        for (int idx = index[0] == -1 ? 1 : index[0] + 1; idx < fNumElements; idx++) {
            if (isAllowedTransition(s, sn, idx)) {
                index[0] = idx;
                return fElements[idx];
            }
        }
        index[0] = -1;
        return null;
    }
    public XSWildcardDecl nextWildcardTransition(int[] s, int[] sn, int[] index) {
        int idx = index[0] == -1 ? fNumElements : index[0] + 1;
        for (; idx < fNumTotal; idx++) {
            if (isAllowedTransition(s, sn, idx)) {
                index[0] = idx;
                return fWildcards[idx];
            }
        }
        if (idx == fNumTotal && isOpenContentAllowed(s, sn)) {
            index[0] = fNumTotal;
            return fOpenContent.fWildcard;
        }
        index[0] = -1;
        return null;
    }
    private boolean isAllowedTransition(int[] s, int[] sn, int index) {
        // Already used suffix open content. Can't make any other transition.
        if (s[0] == STATE_SUFFIX) {
            return false;
        }
        // Already seen all allowed occurrences.
        if (s[index] == fMaxOccurs[index]) {
            return false;
        }
        // Allowed transition. Update next state.
        if (sn != null) {
            System.arraycopy(s, 0, sn, 0, s.length);
            sn[0] = STATE_CHILD;
            // Only increase count if meaningful. i.e. don't exceed minOccurs
            // if max is unbounded.
            if (sn[index] == 0 || sn[index] < fMinOccurs[index] || fMaxOccurs[index] != SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                sn[index]++;
            }
        }
        return true;
    }
    private boolean isOpenContentAllowed(int[] s, int[] sn) {
        if (fOpenContent == null) {
            return false;
        }
        if (fOpenContent.fMode == XSOpenContentDecl.MODE_SUFFIX) {
            // Suffix open content can only kick in on final states
            if (isFinal(s)) {
                sn[0] = STATE_SUFFIX;
                return true;
            }
            return false;
        }
        
        // Interleave: keep old state
        System.arraycopy(s, 0, sn, 0, s.length);
        return true;
    }
    public boolean isOpenContent(XSWildcardDecl w) {
        return fOpenContent != null && fOpenContent.fWildcard == w;
    }
    public List getDefinedNames(SubstitutionGroupHandler subGroupHandler) {
        List ret = new ArrayList();
        // Index starts at 1. Get names of all elements and their sub-groups
        for (int i = 1; i < fNumElements; i++) {
            XSElementDecl e = fElements[i];
            ret.add(e.fTargetNamespace);
            ret.add(e.fName);
            if (e.fScope == XSElementDecl.SCOPE_GLOBAL) {
                XSElementDecl[] es = subGroupHandler.getSubstitutionGroup(e, Constants.SCHEMA_VERSION_1_1);
                for (int j = 0; j < es.length; j++) {
                    ret.add(es[j].fTargetNamespace);
                    ret.add(es[j].fName);
                }
            }
        }
        return ret;
    }
    public void optimizeStates(XS11CMRestriction.XS11CM base, int[] b, int[] d, int indexb) {
    }

    // Constructor only used by copy()
    private XS11AllCM(boolean hasOptionalContent, XSElementDecl[] elements,
            XSWildcardDecl[] wildcards, int[] minOccurs, int[] maxOccurs,
            int numElements, int numTotal, XSOpenContentDecl openContent) {
        super();
        fHasOptionalContent = hasOptionalContent;
        fElements = elements;
        fWildcards = wildcards;
        fMinOccurs = minOccurs;
        fMaxOccurs = maxOccurs;
        fNumElements = numElements;
        fNumTotal = numTotal;
        fOpenContent = openContent;
    }

    // Make a copy to be modified
    XS11AllCM copy() {
        int[] minOccurs, maxOccurs;
        if (fNumTotal > 1) {
            minOccurs = new int[fNumTotal];
            maxOccurs = new int[fNumTotal];
            System.arraycopy(fMinOccurs, 0, minOccurs, 0, fNumTotal);
            System.arraycopy(fMaxOccurs, 0, maxOccurs, 0, fNumTotal);
        }
        else {
            minOccurs = null;
            maxOccurs = null;
        }

        return new XS11AllCM(fHasOptionalContent, fElements,
                fWildcards, minOccurs, maxOccurs,
                fNumElements, fNumTotal, fOpenContent);
    }
    // Collect occurrence information as the derived content model.
    // Add min/maxOccurs of the specified derived element to the entry
    // for its corresponding base element
    void collectOccurs(int[] min, int[] max, int b, int d) {
        min[b] += fMinOccurs[d];
        if (max[b] != SchemaSymbols.OCCURRENCE_UNBOUNDED) {
            if (fMaxOccurs[d] == SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                max[b] = SchemaSymbols.OCCURRENCE_UNBOUNDED;
            }
            else {
                max[b] += fMaxOccurs[d];
            }
        }
    }
    // Attempt to reduce min/maxOccurs from the base content model.
    boolean removeAsBase(int[] min, int[] max, int[] used) {
        for (int i = 1; i < fNumElements; i++) {
            // Base has more minOccurs, can't be satisfied by derived. Error.
            if (fMinOccurs[i] > min[i]) {
                return false;
            }
            // Min in base is no long significant.
            fMinOccurs[i] = 0;
            // 2 cases. a) If base matches a single derived, then it's OK
            // for derived to have bigger maxOccurs. Reduce both to make max
            // in base 0. b) If base matches more than 1 derived, then it has
            // to have big enough maxOccurs to cover all derived elements,
            // otherwise we can't decide which derived to consume first.
            // max of the base will be whatever is left, because it's possible
            // for it to match derived wildcard.
            if (fMaxOccurs[i] != SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                if (max[i] == SchemaSymbols.OCCURRENCE_UNBOUNDED ||
                        fMaxOccurs[i] < max[i]) {
                    // Base doesn't have sufficient maxOccurs
                    if (used[i] > 1) {
                        // Not OK if there are more than one derived.
                        // Mark base as not usable.
                        used[i] = -1;
                        continue;
                    }
                    // OK if there is only one derived. #a case above.
                    // Base becomes 0; derived is subtracted.
                    if (max[i] != SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                        max[i] -= fMaxOccurs[i];
                    }
                    fMaxOccurs[i] = 0;
                }
                else {
                    // Base has sufficient maxOccurs. Subtract from base and
                    // make derived 0.
                    fMaxOccurs[i] -= max[i];
                    max[i] = 0;
                }
            }
            else {
                // Base has sufficient maxOccurs = unbounded.
                if (max[i] == SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                    // If derived is also unbounded, make both 0
                    fMaxOccurs[i] = 0;
                }
                max[i] = 0;
            }
        }

        return true;
    }
    // Attempt to reduce min/maxOccurs from the derived content model.
    void removeAsDerived(int[] max, int[] used, int[] match) {
        for (int i = 1; i < fNumElements; i++) {
            int b = match[i];
            // Only if the derived element has a base match that's usable
            if (b < 0 || used[b] < 0) {
                continue;
            }
            // min must have been satisfied. Max is 0 for 1-many matches.
            // max can only be >0 for 1-1 match.
            fMinOccurs[i] = 0;
            fMaxOccurs[i] = max[b];
        }
    }
    int minOccurs(int index) {
        return fMinOccurs[index];
    }
    int maxOccurs(int index) {
        return fMaxOccurs[index];
    }
    boolean isUnbounded(int index) {
        return index < fNumTotal ? fMaxOccurs[index] == SchemaSymbols.OCCURRENCE_UNBOUNDED : true;
    }
    boolean hasOptionalContent() {
        return fHasOptionalContent;
    }
    int totalMin() {
        int ret = 0;
        for (int i = 1; i < fNumTotal; i++) {
            ret += fMinOccurs[i];
        }
        return ret;
    }
    int min(int i) {
        return fMinOccurs[i];
    }
    XSOpenContentDecl getOpenContent() {
        return fOpenContent;
    }
    // Called to know the size of the state machine if this is viewed as DFA
    int calOccurs() {
        long ret = 1;
        for (int i = 1; i < fNumTotal; i++) {
            // Multiply all maxOccurs. If max is unbounded, then use min.
            int occ = fMaxOccurs[i];
            if (occ == 0) {
                continue;
            }
            if (occ == SchemaSymbols.OCCURRENCE_UNBOUNDED) {
                occ = fMinOccurs[i] == 0 ? 1 : fMinOccurs[i];
            }
            ret *= occ + 1;
            if (ret > Integer.MAX_VALUE) {
                return -1;
            }
        }
        return (int)ret;
    }
} // class XSAll11CM




© 2015 - 2024 Weber Informatics LLC | Privacy Policy