org.apache.xerces.impl.xs.models.CMNodeFactory Maven / Gradle / Ivy
Show all versions of xercesImpl Show documentation
/*
* 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 org.apache.xerces.impl.Constants;
import org.apache.xerces.impl.XMLErrorReporter;
import org.apache.xerces.impl.dtd.models.CMNode;
import org.apache.xerces.impl.xs.XSMessageFormatter;
import org.apache.xerces.util.SecurityManager;
import org.apache.xerces.xni.parser.XMLComponentManager;
import org.apache.xerces.xni.parser.XMLConfigurationException;
/**
*
* @xerces.internal
*
* @author Neeraj Bajaj
*
* @version $Id: CMNodeFactory.java 911823 2010-02-19 14:38:42Z mrglavas $
*/
public class CMNodeFactory {
/** Property identifier: error reporter. */
private static final String ERROR_REPORTER =
Constants.XERCES_PROPERTY_PREFIX + Constants.ERROR_REPORTER_PROPERTY;
/** property identifier: security manager. */
private static final String SECURITY_MANAGER =
Constants.XERCES_PROPERTY_PREFIX + Constants.SECURITY_MANAGER_PROPERTY;
private static final boolean DEBUG = false ;
//
private static final int MULTIPLICITY = 1 ;
//count of number of nodes created
private int nodeCount = 0;
//No. of nodes allowed.
private int maxNodeLimit ;
/**
* Error reporter. This property identifier is:
* http://apache.org/xml/properties/internal/error-reporter
*/
private XMLErrorReporter fErrorReporter;
// stores defaults for different security holes (maxOccurLimit in current context) if it has
// been set on the configuration.
private SecurityManager fSecurityManager = null;
/** default constructor */
public CMNodeFactory() {
}
public void reset(XMLComponentManager componentManager) {
fErrorReporter = (XMLErrorReporter)componentManager.getProperty(ERROR_REPORTER);
try {
fSecurityManager = (SecurityManager)componentManager.getProperty(SECURITY_MANAGER);
reset();
}
catch (XMLConfigurationException e) {
fSecurityManager = null;
}
}//reset()
public void reset() {
// we are setting the limit of number of nodes to 3 times the maxOccurs value.
if (fSecurityManager != null) {
maxNodeLimit = fSecurityManager.getMaxOccurNodeLimit() * MULTIPLICITY;
}
}
public CMNode getCMLeafNode(int type, Object leaf, int id, int position) {
nodeCountCheck();
return new XSCMLeaf(type, leaf, id, position) ;
}
public CMNode getCMRepeatingLeafNode(int type, Object leaf,
int minOccurs, int maxOccurs, int id, int position) {
nodeCountCheck();
return new XSCMRepeatingLeaf(type, leaf, minOccurs, maxOccurs, id, position);
}
public CMNode getCMUniOpNode(int type, CMNode childNode) {
nodeCountCheck();
return new XSCMUniOp(type, childNode) ;
}
public CMNode getCMBinOpNode(int type, CMNode leftNode, CMNode rightNode) {
nodeCountCheck() ;
return new XSCMBinOp(type, leftNode, rightNode) ;
}
public void nodeCountCheck(){
if( fSecurityManager != null && nodeCount++ > maxNodeLimit){
if(DEBUG){
System.out.println("nodeCount = " + nodeCount ) ;
System.out.println("nodeLimit = " + maxNodeLimit ) ;
}
fErrorReporter.reportError(XSMessageFormatter.SCHEMA_DOMAIN, "maxOccurLimit", new Object[]{ new Integer(maxNodeLimit) }, XMLErrorReporter.SEVERITY_FATAL_ERROR);
// similarly to entity manager behaviour, take into accont
// behaviour if continue-after-fatal-error is set.
nodeCount = 0;
}
}//nodeCountCheck()
//reset the node count
public void resetNodeCount(){
nodeCount = 0 ;
}
/**
* Sets the value of a property. This method is called by the component
* manager any time after reset when a property changes value.
*
* Note: Components should silently ignore properties
* that do not affect the operation of the component.
*
* @param propertyId The property identifier.
* @param value The value of the property.
*
* @throws SAXNotRecognizedException The component should not throw
* this exception.
* @throws SAXNotSupportedException The component should not throw
* this exception.
*/
public void setProperty(String propertyId, Object value)
throws XMLConfigurationException {
// Xerces properties
if (propertyId.startsWith(Constants.XERCES_PROPERTY_PREFIX)) {
final int suffixLength = propertyId.length() - Constants.XERCES_PROPERTY_PREFIX.length();
if (suffixLength == Constants.SECURITY_MANAGER_PROPERTY.length() &&
propertyId.endsWith(Constants.SECURITY_MANAGER_PROPERTY)) {
fSecurityManager = (SecurityManager)value;
maxNodeLimit = (fSecurityManager != null) ? fSecurityManager.getMaxOccurNodeLimit() * MULTIPLICITY : 0 ;
return;
}
if (suffixLength == Constants.ERROR_REPORTER_PROPERTY.length() &&
propertyId.endsWith(Constants.ERROR_REPORTER_PROPERTY)) {
fErrorReporter = (XMLErrorReporter)value;
return;
}
}
} // setProperty(String,Object)
}//CMNodeFactory()