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

com.sun.msv.verifier.identity.PathMatcher Maven / Gradle / Ivy

There is a newer version: 2.3.0
Show newest version
/*
 * Copyright (c) 2001-2013 Oracle and/or its affiliates. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 *   - Redistributions of source code must retain the above copyright
 *     notice, this list of conditions and the following disclaimer.
 *
 *   - Redistributions in binary form must reproduce the above copyright
 *     notice, this list of conditions and the following disclaimer in the
 *     documentation and/or other materials provided with the distribution.
 *
 *   - Neither the name of Oracle nor the names of its
 *     contributors may be used to endorse or promote products derived
 *     from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package com.sun.msv.verifier.identity;

import org.relaxng.datatype.Datatype;
import org.xml.sax.SAXException;

import com.sun.msv.grammar.xmlschema.XPath;

/**
 * Base implementation of XPath matching engine.
 * 
 * It only supports the subset defined in XML Schema Part 1. Extra care
 * must be taken to call the testInitialMatch method after the creation of an object.
 * 
 * Match to an attribute is not supported. It is implemented in FieldPathMatcher
 * class.
 * 
 * The onMatched method is called when the specified XPath matches the current element.
 * Derived classes should implement this method to do something useful.
 * 
 * @author Kohsuke KAWAGUCHI
 */
public abstract class PathMatcher extends MatcherBundle {
    
    protected PathMatcher( IDConstraintChecker owner, XPath[] paths ) {
        super(owner);
        children = new Matcher[paths.length];
        for( int i=0; i
     * This class implements the matching engine for single
     * 
     * "Path".
     * 
     * 

* The outer PathMatcher uses multiple instances of this class * and thereby implements the matching engine for the whole "Selector". * *

* This class only supports the subset defined in XML Schema Part 1. Extra care * must be taken to call the testInitialMatch method * after the creation of an object. * *

* When a match is found, this class notifies the parent object by using a flag. * * */ private class SinglePathMatcher extends Matcher { /** * stores matched steps. * first dimension is expanded as the depth goes deep. * second dimension is always equal to the size of steps. */ private boolean[][] activeSteps; // private short currentDepth=0; protected final XPath path; /** * this flag is set to true when the path contains an attribute step * and the current element matches the element part of the path. * * When this flag is true, we need to honor the onAttribute event * and check if the path really matches to the attribute. */ private boolean elementMatched = false; protected SinglePathMatcher( XPath path ) { super(PathMatcher.this.owner); this.path = path; activeSteps = new boolean[4][]; /* activeSteps[i][0] is used to represent an implicit "root". For example, when XPath is "//A/B", [0]:root [1]:A [2]:B (initial state) 1 0 0 (1 indicates "active") [0] is initialized to 1. (startElement(X)) (step:1 shift to right) 1(*1) 1 0 *1 [0] will be populated by isAnyDescendant field. In this case, since isAnyDescendant ("//") is true, [0] is set to true after shift. This indicates that new element X can possibly be used as the implicit root. (step:2 perform name test) 1 0 0 root is excluded from the test. Since A doesn't match X, the corresponding field is set to false. (startElement(A)) (step:1 shift to right) 1 1 0 (step:2 perform name test) 1 1 0 (startElement(B)) (step:1 shift to right) 1 1 1 (step:2 perform name test) 1 0 1 (*2) *2 Now that the right most slot is true, this element B matches XPath. */ activeSteps[0] = new boolean[path.steps.length+1]; activeSteps[0][0] = true; // initialization // we only need an empty buffer for activeStep[0]. // other slots are filled on demand. if(path.steps.length==0) { // if the step is length 0, (that is, ".") // it is an immediate match. if( path.attributeStep==null ) // report to the parent PathMatcher that a match was found matchFound = true; else elementMatched = true; } } protected void startElement( String namespaceURI, String localName ) throws SAXException { elementMatched = false; // reset the flag. final int depth = getDepth(); if(depth==activeSteps.length-1) { // if the buffer is used up, expand buffer boolean[][] newBuf = new boolean[depth*2][]; System.arraycopy( activeSteps, 0, newBuf, 0, activeSteps.length ); activeSteps = newBuf; } // currentDepth++; int len = path.steps.length; boolean[] prvBuf = activeSteps[depth-1]; boolean[] curBuf = activeSteps[depth]; if(curBuf==null) activeSteps[depth]=curBuf=new boolean[len+1/*implicit root*/]; // shift to right if(len!=0) { System.arraycopy( prvBuf, 0, curBuf, 1, len ); curBuf[0] = path.isAnyDescendant; } // perform name test and deactivate unmatched steps for( int i=1; i<=len; i++ ) // exclude root from test. if( curBuf[i] && !path.steps[i-1].accepts(namespaceURI,localName) ) curBuf[i] = false; if( curBuf[len] ) { // this element matched this path if( path.attributeStep==null ) // report to the parent PathMatcher that a match was found matchFound = true; else elementMatched = true; } } protected void onAttribute( String namespaceURI, String localName, String value, Datatype type ) throws SAXException { // attribute step is not tested when the parent element doesn't match // the parent XPath expression. if( !elementMatched ) return; if( path.attributeStep.accepts(namespaceURI,localName) ) // report to the parent PathMatcher that a match was found matchFound = true; // keep the elementMatched flag as-is. } protected void endElement( Datatype dt ) { elementMatched = false; // reset the flag. // currentDepth--; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy