blob: abee52b02c7492284e1074e4bba2cd64ca967eca [file] [log] [blame]
/*
* reserved comment block
* DO NOT REMOVE OR ALTER!
*/
/*
* 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 com.sun.org.apache.xerces.internal.impl.xs.identity;
import com.sun.org.apache.xerces.internal.impl.xpath.XPathException;
import com.sun.org.apache.xerces.internal.impl.xs.util.ShortListImpl;
import com.sun.org.apache.xerces.internal.util.SymbolTable;
import com.sun.org.apache.xerces.internal.util.XMLChar;
import com.sun.org.apache.xerces.internal.xni.NamespaceContext;
import com.sun.org.apache.xerces.internal.xs.ShortList;
import com.sun.org.apache.xerces.internal.xs.XSComplexTypeDefinition;
import com.sun.org.apache.xerces.internal.xs.XSConstants;
import com.sun.org.apache.xerces.internal.xs.XSTypeDefinition;
/**
* Schema identity constraint field.
*
* @xerces.internal
*
* @author Andy Clark, IBM
*/
public class Field {
//
// Data
//
/** Field XPath. */
protected final Field.XPath fXPath;
/** Identity constraint. */
protected final IdentityConstraint fIdentityConstraint;
//
// Constructors
//
/** Constructs a field. */
public Field(Field.XPath xpath,
IdentityConstraint identityConstraint) {
fXPath = xpath;
fIdentityConstraint = identityConstraint;
} // <init>(Field.XPath,IdentityConstraint)
//
// Public methods
//
/** Returns the field XPath. */
public com.sun.org.apache.xerces.internal.impl.xpath.XPath getXPath() {
return fXPath;
} // getXPath():org.apache.xerces.impl.v1.schema.identity.XPath
/** Returns the identity constraint. */
public IdentityConstraint getIdentityConstraint() {
return fIdentityConstraint;
} // getIdentityConstraint():IdentityConstraint
// factory method
/** Creates a field matcher. */
public XPathMatcher createMatcher(ValueStore store) {
return new Field.Matcher(fXPath, store);
} // createMatcher(ValueStore):XPathMatcher
//
// Object methods
//
/** Returns a string representation of this object. */
public String toString() {
return fXPath.toString();
} // toString():String
//
// Classes
//
/**
* Field XPath.
*
* @author Andy Clark, IBM
*/
public static class XPath
extends com.sun.org.apache.xerces.internal.impl.xpath.XPath {
//
// Constructors
//
/** Constructs a field XPath expression. */
public XPath(String xpath,
SymbolTable symbolTable,
NamespaceContext context) throws XPathException {
super(fixupXPath(xpath), symbolTable, context);
// verify that only one attribute is selected per branch
for (int i=0;i<fLocationPaths.length;i++) {
for(int j=0; j<fLocationPaths[i].steps.length; j++) {
com.sun.org.apache.xerces.internal.impl.xpath.XPath.Axis axis =
fLocationPaths[i].steps[j].axis;
if (axis.type == XPath.Axis.ATTRIBUTE &&
(j < fLocationPaths[i].steps.length-1)) {
throw new XPathException("c-fields-xpaths");
}
}
}
} // <init>(String,SymbolTable,NamespacesContext)
/** Fixup XPath expression. Avoid creating a new String if possible. */
private static String fixupXPath(String xpath) {
final int end = xpath.length();
int offset = 0;
boolean whitespace = true;
char c;
// NOTE: We have to prefix the field XPath with "./" in
// order to handle selectors such as "@attr" that
// select the attribute because the fields could be
// relative to the selector element. -Ac
// Unless xpath starts with a descendant node -Achille Fokoue
// ... or a / or a . - NG
for (; offset < end; ++offset) {
c = xpath.charAt(offset);
if (whitespace) {
if (!XMLChar.isSpace(c)) {
if (c == '.' || c == '/') {
whitespace = false;
}
else if (c != '|') {
return fixupXPath2(xpath, offset, end);
}
}
}
else if (c == '|') {
whitespace = true;
}
}
return xpath;
} // fixupXPath(String):String
private static String fixupXPath2(String xpath, int offset, final int end) {
StringBuffer buffer = new StringBuffer(end + 2);
for (int i = 0; i < offset; ++i) {
buffer.append(xpath.charAt(i));
}
buffer.append("./");
boolean whitespace = false;
char c;
for (; offset < end; ++offset) {
c = xpath.charAt(offset);
if (whitespace) {
if (!XMLChar.isSpace(c)) {
if (c == '.' || c == '/') {
whitespace = false;
}
else if (c != '|') {
buffer.append("./");
whitespace = false;
}
}
}
else if (c == '|') {
whitespace = true;
}
buffer.append(c);
}
return buffer.toString();
} // fixupXPath2(String, int, int):String
} // class XPath
/**
* Field matcher.
*
* @author Andy Clark, IBM
*/
protected class Matcher
extends XPathMatcher {
//
// Data
//
/** Value store for data values. */
protected final ValueStore fStore;
/** A flag indicating whether the field is allowed to match a value. */
protected boolean fMayMatch = true;
//
// Constructors
//
/** Constructs a field matcher. */
public Matcher(Field.XPath xpath, ValueStore store) {
super(xpath);
fStore = store;
} // <init>(Field.XPath,ValueStore)
//
// XPathHandler methods
//
/**
* This method is called when the XPath handler matches the
* XPath expression.
*/
protected void matched(Object actualValue, short valueType, ShortList itemValueType, boolean isNil) {
super.matched(actualValue, valueType, itemValueType, isNil);
if(isNil && (fIdentityConstraint.getCategory() == IdentityConstraint.IC_KEY)) {
String code = "KeyMatchesNillable";
fStore.reportError(code,
new Object[]{fIdentityConstraint.getElementName(), fIdentityConstraint.getIdentityConstraintName()});
}
fStore.addValue(Field.this, fMayMatch, actualValue, convertToPrimitiveKind(valueType), convertToPrimitiveKind(itemValueType));
// once we've stored the value for this field, we set the mayMatch
// member to false so that in the same scope, we don't match any more
// values (and throw an error instead).
fMayMatch = false;
} // matched(String)
private short convertToPrimitiveKind(short valueType) {
/** Primitive datatypes. */
if (valueType <= XSConstants.NOTATION_DT) {
return valueType;
}
/** Types derived from string. */
if (valueType <= XSConstants.ENTITY_DT) {
return XSConstants.STRING_DT;
}
/** Types derived from decimal. */
if (valueType <= XSConstants.POSITIVEINTEGER_DT) {
return XSConstants.DECIMAL_DT;
}
/** Other types. */
return valueType;
}
private ShortList convertToPrimitiveKind(ShortList itemValueType) {
if (itemValueType != null) {
int i;
final int length = itemValueType.getLength();
for (i = 0; i < length; ++i) {
short type = itemValueType.item(i);
if (type != convertToPrimitiveKind(type)) {
break;
}
}
if (i != length) {
final short [] arr = new short[length];
for (int j = 0; j < i; ++j) {
arr[j] = itemValueType.item(j);
}
for(; i < length; ++i) {
arr[i] = convertToPrimitiveKind(itemValueType.item(i));
}
return new ShortListImpl(arr, arr.length);
}
}
return itemValueType;
}
protected void handleContent(XSTypeDefinition type, boolean nillable, Object actualValue, short valueType, ShortList itemValueType) {
if (type == null ||
type.getTypeCategory() == XSTypeDefinition.COMPLEX_TYPE &&
((XSComplexTypeDefinition) type).getContentType()
!= XSComplexTypeDefinition.CONTENTTYPE_SIMPLE) {
// the content must be simpleType content
fStore.reportError( "cvc-id.3", new Object[] {
fIdentityConstraint.getName(),
fIdentityConstraint.getElementName()});
}
fMatchedString = actualValue;
matched(fMatchedString, valueType, itemValueType, nillable);
} // handleContent(XSElementDecl, String)
} // class Matcher
} // class Field