net.sf.saxon.expr.instruct.CopyOf Maven / Gradle / Ivy
Show all versions of saxon-he Show documentation
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2013 Saxonica Limited.
// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// This Source Code Form is "Incompatible With Secondary Licenses", as defined by the Mozilla Public License, v. 2.0.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
package net.sf.saxon.expr.instruct;
import net.sf.saxon.Configuration;
import net.sf.saxon.Controller;
import net.sf.saxon.event.*;
import net.sf.saxon.expr.*;
import net.sf.saxon.expr.parser.*;
import net.sf.saxon.expr.sort.DocumentSorter;
import net.sf.saxon.lib.NamespaceConstant;
import net.sf.saxon.lib.ParseOptions;
import net.sf.saxon.lib.Validation;
import net.sf.saxon.om.*;
import net.sf.saxon.pattern.AnyNodeTest;
import net.sf.saxon.pattern.ContentTypeTest;
import net.sf.saxon.pattern.NodeKindTest;
import net.sf.saxon.pattern.NodeTest;
import net.sf.saxon.trace.ExpressionPresenter;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.tree.wrapper.VirtualCopy;
import net.sf.saxon.tree.wrapper.VirtualUntypedCopy;
import net.sf.saxon.type.*;
import net.sf.saxon.value.SequenceType;
import net.sf.saxon.value.Whitespace;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Iterator;
import java.util.List;
/**
* An xsl:copy-of element in the stylesheet.
*/
public class CopyOf extends Instruction implements ValidatingInstruction {
private Expression select;
private boolean copyNamespaces;
private int validation;
private SchemaType schemaType;
private boolean requireDocumentOrElement = false;
private boolean rejectDuplicateAttributes;
private boolean readOnce = false;
private boolean validating;
private boolean copyLineNumbers = true;
private boolean copyForUpdate = false;
private String staticBaseUri;
/**
* Create an xsl:copy-of instruction (also used in XQuery for implicit copying)
* @param select expression that selects the nodes to be copied
* @param copyNamespaces true if namespaces are to be copied
* @param validation validation mode for the result tree
* @param schemaType schema type for validating the result tree
* @param rejectDuplicateAttributes true if duplicate attributes are to be rejected (XQuery). False
* if duplicates are handled by discarding all but the first (XSLT).
*/
public CopyOf(Expression select,
boolean copyNamespaces,
int validation,
SchemaType schemaType,
boolean rejectDuplicateAttributes) {
this.select = select;
this.copyNamespaces = copyNamespaces;
this.validation = validation;
this.schemaType = schemaType;
validating = schemaType != null || validation != Validation.PRESERVE;
this.rejectDuplicateAttributes = rejectDuplicateAttributes;
adoptChildExpression(select);
}
/**
* Set the select expression
* @param select the new select expression
*/
public void setSelectExpression(Expression select) {
this.select = select;
}
/**
* Get the expression that selects the nodes to be copied
* @return the select expression
*/
public Expression getSelectExpression() {
return select;
}
/**
* Get the validation mode
* @return the validation mode
*/
public int getValidationAction() {
return validation;
}
/**
* Test if the instruction is doing validation
* @return true if it is
*/
public boolean isValidating() {
return validating;
}
/**
* Get the schema type to be used for validation
* @return the schema type, or null if not validating against a type
*/
public SchemaType getSchemaType() {
return schemaType;
}
/**
* Set the static base URI of the xsl:copy-of instruction
* @param base the static base URI
*/
public void setStaticBaseUri(String base) {
staticBaseUri = base;
}
/**
* Set the "saxon:read-once" optimization mode
* @param b true to enable the optimization
*/
public void setReadOnce(boolean b) {
readOnce = b;
}
/**
* Set whether line numbers are to be copied from the source to the result.
* Default is false.
* @param copy true if line numbers are to be copied
*/
public void setCopyLineNumbers(boolean copy) {
copyLineNumbers = copy;
}
/**
* Determine whether this instruction creates new nodes.
* The result depends on the type of the select expression.
*/
public final boolean createsNewNodes() {
Executable exec = getExecutable();
if (exec == null) {
return true; // This shouldn't happen, but we err on the safe side
}
final TypeHierarchy th = exec.getConfiguration().getTypeHierarchy();
return !select.getItemType(th).isPlainType();
}
/**
* Get the name of this instruction, for diagnostics and tracing
*/
public int getInstructionNameCode() {
return StandardNames.XSL_COPY_OF;
}
/**
* For XQuery, the operand (select) must be a single element or document node.
* @param requireDocumentOrElement true if the argument must be a single element or document node
*/
public void setRequireDocumentOrElement(boolean requireDocumentOrElement) {
this.requireDocumentOrElement = requireDocumentOrElement;
}
/**
* Test whether this expression requires a document or element node
* @return true if this expression requires the value of the argument to be a document or element node,
* false if there is no such requirement
*/
public boolean isDocumentOrElementRequired() {
return requireDocumentOrElement;
}
/**
* Set whether this instruction is creating a copy for the purpose of updating (XQuery transform expression)
* @param forUpdate true if this copy is being created to support an update
*/
public void setCopyForUpdate(boolean forUpdate) {
copyForUpdate = forUpdate;
}
/**
* Ask whether this instruction is creating a copy for the purpose of updating (XQuery transform expression)
* @return true if this copy is being created to support an update
*/
public boolean isCopyForUpdate() {
return copyForUpdate;
}
/**
* An implementation of Expression must provide at least one of the methods evaluateItem(), iterate(), or process().
* This method indicates which of these methods is provided. This implementation provides both iterate() and
* process() methods natively.
*/
public int getImplementationMethod() {
return ITERATE_METHOD | PROCESS_METHOD | WATCH_METHOD;
}
/**
* Determine whether namespaces are to be copied or not
* @return true if namespaces are to be copied (the default)
*/
public boolean isCopyNamespaces() {
return copyNamespaces;
}
/**
* Copy an expression. This makes a deep copy.
*
* @return the copy of the original expression
*/
/*@NotNull*/
public Expression copy() {
CopyOf c = new CopyOf(select.copy(), copyNamespaces, validation, schemaType, rejectDuplicateAttributes);
c.setContainer(getContainer());
// we don't normally setContainer() in the copy() method, but it's needed here because of the
// call on getContainer() in computeSpecialProperties()
c.setCopyForUpdate(copyForUpdate);
c.setCopyLineNumbers(copyLineNumbers);
c.setReadOnce(readOnce);
c.setStaticBaseUri(staticBaseUri);
return c;
}
/*@NotNull*/
public Expression simplify(ExpressionVisitor visitor) throws XPathException {
select = visitor.simplify(select);
return this;
}
/*@NotNull*/
public ItemType getItemType(TypeHierarchy th) {
Executable exec = getExecutable();
assert exec != null;
ItemType in = select.getItemType(th);
if (!exec.isSchemaAware()) {
return in;
}
Configuration config = exec.getConfiguration();
if (schemaType != null) {
int e = th.relationship(in, NodeKindTest.ELEMENT);
if (e == TypeHierarchy.SAME_TYPE || e == TypeHierarchy.SUBSUMED_BY) {
return new ContentTypeTest(Type.ELEMENT, schemaType, config, false);
}
int a = th.relationship(in, NodeKindTest.ATTRIBUTE);
if (a == TypeHierarchy.SAME_TYPE || a == TypeHierarchy.SUBSUMED_BY) {
return new ContentTypeTest(Type.ATTRIBUTE, schemaType, config, false);
}
} else switch (validation) {
case Validation.PRESERVE:
return in;
case Validation.STRIP: {
int e = th.relationship(in, NodeKindTest.ELEMENT);
if (e == TypeHierarchy.SAME_TYPE || e == TypeHierarchy.SUBSUMED_BY) {
return new ContentTypeTest(Type.ELEMENT, Untyped.getInstance(), config, false);
}
int a = th.relationship(in, NodeKindTest.ATTRIBUTE);
if (a == TypeHierarchy.SAME_TYPE || a == TypeHierarchy.SUBSUMED_BY) {
return new ContentTypeTest(Type.ATTRIBUTE, BuiltInAtomicType.UNTYPED_ATOMIC, config, false);
}
if (e != TypeHierarchy.DISJOINT || a != TypeHierarchy.DISJOINT) {
// it might be an element or attribute
return AnyNodeTest.getInstance();
} else {
// it can't be an element or attribute, so stripping type annotations can't affect it
return in;
}
}
case Validation.STRICT:
case Validation.LAX:
if (in instanceof NodeTest) {
int fp = ((NodeTest)in).getFingerprint();
if (fp != -1) {
int e = th.relationship(in, NodeKindTest.ELEMENT);
if (e == TypeHierarchy.SAME_TYPE || e == TypeHierarchy.SUBSUMED_BY) {
SchemaDeclaration elem = config.getElementDeclaration(fp);
if (elem != null) {
return new ContentTypeTest(Type.ELEMENT, elem.getType(), config, false);
} else {
// Although there is no element declaration now, there might be one at run-time
return new ContentTypeTest(Type.ELEMENT, AnyType.getInstance(), config, false);
}
}
int a = th.relationship(in, NodeKindTest.ATTRIBUTE);
if (a == TypeHierarchy.SAME_TYPE || a == TypeHierarchy.SUBSUMED_BY) {
SchemaDeclaration attr = config.getElementDeclaration(fp);
if (attr != null) {
return new ContentTypeTest(Type.ATTRIBUTE, attr.getType(), config, false);
} else {
// Although there is no attribute declaration now, there might be one at run-time
return new ContentTypeTest(Type.ATTRIBUTE, AnySimpleType.getInstance(), config, false);
}
}
} else {
int e = th.relationship(in, NodeKindTest.ELEMENT);
if (e == TypeHierarchy.SAME_TYPE || e == TypeHierarchy.SUBSUMED_BY) {
return NodeKindTest.ELEMENT;
}
int a = th.relationship(in, NodeKindTest.ATTRIBUTE);
if (a == TypeHierarchy.SAME_TYPE || a == TypeHierarchy.SUBSUMED_BY) {
return NodeKindTest.ATTRIBUTE;
}
}
return AnyNodeTest.getInstance();
} else if (in instanceof AtomicType) {
return in;
} else {
return AnyItemType.getInstance();
}
}
return select.getItemType(th);
}
public int getCardinality() {
return select.getCardinality();
}
public int getDependencies() {
return select.getDependencies();
}
protected void promoteInst(PromotionOffer offer) throws XPathException {
select = doPromotion(select, offer);
}
/*@NotNull*/
public Expression typeCheck(ExpressionVisitor visitor, ExpressionVisitor.ContextItemType contextItemType) throws XPathException {
select = visitor.typeCheck(select, contextItemType);
if (isDocumentOrElementRequired()) {
// this implies the expression is actually an XQuery validate{} expression, hence the error messages
RoleLocator role = new RoleLocator(RoleLocator.TYPE_OP, "validate", 0);
role.setErrorCode("XQTY0030");
select = TypeChecker.staticTypeCheck(select, SequenceType.SINGLE_NODE, false, role, visitor);
TypeHierarchy th = visitor.getConfiguration().getTypeHierarchy();
ItemType t = select.getItemType(th);
if (th.isSubType(t, NodeKindTest.ATTRIBUTE)) {
throw new XPathException("validate{} expression cannot be applied to an attribute", "XQTY0030");
}
if (th.isSubType(t, NodeKindTest.TEXT)) {
throw new XPathException("validate{} expression cannot be applied to a text node", "XQTY0030");
}
if (th.isSubType(t, NodeKindTest.COMMENT)) {
throw new XPathException("validate{} expression cannot be applied to a comment node", "XQTY0030");
}
if (th.isSubType(t, NodeKindTest.PROCESSING_INSTRUCTION)) {
throw new XPathException("validate{} expression cannot be applied to a processing instruction node", "XQTY0030");
}
if (th.isSubType(t, NodeKindTest.NAMESPACE)) {
throw new XPathException("validate{} expression cannot be applied to a namespace node", "XQTY0030");
}
}
adoptChildExpression(select);
return this;
}
/*@NotNull*/
public Expression optimize(ExpressionVisitor visitor, ExpressionVisitor.ContextItemType contextItemType) throws XPathException {
if (readOnce) {
Expression optcopy;
if (!(select instanceof DocumentSorter)) {
optcopy = visitor.getConfiguration().obtainOptimizer().optimizeCopy(select);
if (optcopy != null) {
return optcopy;
}
}
select = visitor.optimize(select, contextItemType);
optcopy = visitor.getConfiguration().obtainOptimizer().optimizeCopy(select);
if (optcopy != null) {
return optcopy;
}
} else {
select = visitor.optimize(select, contextItemType);
}
if (Literal.isEmptySequence(select)) {
return select;
}
adoptChildExpression(select);
final TypeHierarchy th = visitor.getConfiguration().getTypeHierarchy();
if (select.getItemType(th).isPlainType()) {
return select;
}
return this;
}
/**
* Diagnostic print of expression structure. The abstract expression tree
* is written to the supplied output destination.
*/
public void explain(ExpressionPresenter out) {
out.startElement("copyOf");
out.emitAttribute("validation", Validation.toString(validation));
select.explain(out);
out.endElement();
}
/*@NotNull*/
public Iterator iterateSubExpressions() {
return new MonoIterator(select);
}
/**
* Get the immediate sub-expressions of this expression, with information about the relationship
* of each expression to its parent expression. Default implementation
* works off the results of iterateSubExpressions()
*
* @return an iterator containing the sub-expressions of this expression
*/
/*@NotNull*/
public Iterator iterateSubExpressionInfo() {
SubExpressionInfo info = new SubExpressionInfo(select, true, false, NODE_VALUE_CONTEXT);
return new MonoIterator(info);
}
/**
* Replace one subexpression by a replacement subexpression
* @param original the original subexpression
* @param replacement the replacement subexpression
* @return true if the original subexpression is found
*/
public boolean replaceSubExpression(Expression original, Expression replacement) {
boolean found = false;
if (select == original) {
select = replacement;
found = true;
}
return found;
}
/**
* Add a representation of this expression to a PathMap. The PathMap captures a map of the nodes visited
* by an expression in a source tree.
*
* The default implementation of this method assumes that an expression does no navigation other than
* the navigation done by evaluating its subexpressions, and that the subexpressions are evaluated in the
* same context as the containing expression. The method must be overridden for any expression
* where these assumptions do not hold. For example, implementations exist for AxisExpression, ParentExpression,
* and RootExpression (because they perform navigation), and for the doc(), document(), and collection()
* functions because they create a new navigation root. Implementations also exist for PathExpression and
* FilterExpression because they have subexpressions that are evaluated in a different context from the
* calling expression.
*
* @param pathMap the PathMap to which the expression should be added
* @param pathMapNodeSet the PathMapNodeSet to which the paths embodied in this expression should be added
* @return the pathMapNodeSet representing the points in the source document that are both reachable by this
* expression, and that represent possible results of this expression. For an expression that does
* navigation, it represents the end of the arc in the path map that describes the navigation route. For other
* expressions, it is the same as the input pathMapNode.
*/
public PathMap.PathMapNodeSet addToPathMap(PathMap pathMap, PathMap.PathMapNodeSet pathMapNodeSet) {
PathMap.PathMapNodeSet result = super.addToPathMap(pathMap, pathMapNodeSet);
result.setReturnable(false);
TypeHierarchy th = getExecutable().getConfiguration().getTypeHierarchy();
ItemType type = getItemType(th);
if (th.relationship(type, NodeKindTest.ELEMENT) != TypeHierarchy.DISJOINT ||
th.relationship(type, NodeKindTest.DOCUMENT) != TypeHierarchy.DISJOINT) {
result.addDescendants();
}
return new PathMap.PathMapNodeSet(pathMap.makeNewRoot(this));
}
/**
* Process this xsl:copy-of instruction
*
* @param context the dynamic context for the transformation
* @return null - this implementation of the method never returns a TailCall
*/
public TailCall processLeavingTail(XPathContext context) throws XPathException {
Controller controller = context.getController();
SequenceReceiver out = context.getReceiver();
boolean copyBaseURI = (out.getSystemId() == null);
// if the copy is being attached to an existing parent, it inherits the base URI of the parent
int copyOptions = CopyOptions.TYPE_ANNOTATIONS;
if (copyNamespaces) {
copyOptions |= CopyOptions.ALL_NAMESPACES;
}
if (copyForUpdate) {
copyOptions |= CopyOptions.FOR_UPDATE;
}
//int whichNamespaces = (copyNamespaces ? NodeInfo.ALL_NAMESPACES : NodeInfo.NO_NAMESPACES);
SequenceIterator iter = select.iterate(context);
while (true) {
Item item = iter.next();
if (item == null) {
break;
}
if (item instanceof NodeInfo) {
NodeInfo source = (NodeInfo) item;
int kind = source.getNodeKind();
if (requireDocumentOrElement &&
!(kind == Type.ELEMENT || kind == Type.DOCUMENT)) {
XPathException e = new XPathException("Operand of validate expression must be a document or element node");
e.setXPathContext(context);
e.setErrorCode("XQTY0030");
throw e;
}
final Configuration config = controller.getConfiguration();
switch (kind) {
case Type.ELEMENT: {
Receiver eval = out;
if (validating) {
ParseOptions options = new ParseOptions();
options.setSchemaValidationMode(validation);
options.setTopLevelType(schemaType);
options.setTopLevelElement(new NameOfNode(source));
eval = config.getElementValidator(out, options, locationId);
}
if (copyBaseURI) {
eval.setSystemId(computeNewBaseUri(source));
}
Receiver savedReceiver = null;
PipelineConfiguration savedPipe = null;
if (copyLineNumbers) {
savedReceiver = eval;
PipelineConfiguration pipe = eval.getPipelineConfiguration();
savedPipe = new PipelineConfiguration(pipe);
LocationCopier copier = new LocationCopier(false);
pipe.setLocationProvider(copier);
pipe.setComponent(CopyInformee.class.getName(), copier);
}
source.copy(eval, copyOptions, locationId);
if (copyLineNumbers) {
eval = savedReceiver;
eval.setPipelineConfiguration(savedPipe);
}
break;
}
case Type.ATTRIBUTE:
if (schemaType != null && schemaType.isComplexType()) {
XPathException e = new XPathException("When copying an attribute with schema validation, the requested type must not be a complex type");
e.setLocator(this);
e.setXPathContext(context);
e.setErrorCode("XTTE1535");
throw dynamicError(this, e, context);
}
try {
copyAttribute(source, (SimpleType)schemaType, validation, this, context, rejectDuplicateAttributes);
} catch (NoOpenStartTagException err) {
XPathException e = new XPathException(err.getMessage());
e.setLocator(this);
e.setXPathContext(context);
e.setErrorCodeQName(err.getErrorCodeQName());
throw dynamicError(this, e, context);
}
break;
case Type.TEXT:
out.characters(source.getStringValueCS(), locationId, 0);
break;
case Type.PROCESSING_INSTRUCTION:
if (copyBaseURI) {
out.setSystemId(source.getBaseURI());
}
out.processingInstruction(source.getDisplayName(), source.getStringValueCS(), locationId, 0);
break;
case Type.COMMENT:
out.comment(source.getStringValueCS(), locationId, 0);
break;
case Type.NAMESPACE:
try {
source.copy(out, 0, locationId);
} catch (NoOpenStartTagException err) {
XPathException e = new XPathException(err.getMessage());
e.setXPathContext(context);
e.setErrorCodeQName(err.getErrorCodeQName());
throw dynamicError(this, e, context);
}
break;
case Type.DOCUMENT: {
ParseOptions options = new ParseOptions();
options.setSchemaValidationMode(validation);
options.setStripSpace(Whitespace.NONE);
options.setTopLevelType(schemaType);
Receiver val = config.getDocumentValidator(out, source.getBaseURI(), options);
//val.setPipelineConfiguration(out.getPipelineConfiguration());
if (copyBaseURI) {
val.setSystemId(source.getBaseURI());
}
Receiver savedReceiver = null;
PipelineConfiguration savedPipe = null;
if (copyLineNumbers) {
savedReceiver = val;
savedPipe = new PipelineConfiguration(val.getPipelineConfiguration());
LocationCopier copier = new LocationCopier(true);
val.getPipelineConfiguration().setLocationProvider(copier);
val.getPipelineConfiguration().setComponent(CopyInformee.class.getName(), copier);
}
source.copy(val, copyOptions, locationId);
if (copyLineNumbers) {
val = savedReceiver;
val.setPipelineConfiguration(savedPipe);
}
break;
}
default:
throw new IllegalArgumentException("Unknown node kind " + source.getNodeKind());
}
} else {
out.append(item, locationId, NodeInfo.ALL_NAMESPACES);
}
}
return null;
}
private String computeNewBaseUri(NodeInfo source) {
// These rules are the rules for xsl:copy-of instruction in XSLT. The same code is used to support the
// validate{} expression in XQuery. XQuery says nothing about the base URI of a node that results
// from a validate{} expression, so until it does, we might as well use the same logic.
String newBaseUri;
String xmlBase = source.getAttributeValue(NamespaceConstant.XML, "base");
if (xmlBase != null) {
try {
URI xmlBaseUri = new URI(xmlBase);
if (xmlBaseUri.isAbsolute()) {
newBaseUri = xmlBase;
} else if (staticBaseUri != null) {
URI sbu = new URI(staticBaseUri);
URI abs = sbu.resolve(xmlBaseUri);
newBaseUri = abs.toString();
} else {
newBaseUri = source.getBaseURI();
}
} catch (URISyntaxException err) {
newBaseUri = source.getBaseURI();
}
} else {
newBaseUri = source.getBaseURI();
}
return newBaseUri;
}
/**
* Method shared by xsl:copy and xsl:copy-of to copy an attribute node
* @param source the node to be copied
* @param schemaType the simple type against which the value is to be validated, if any
* @param validation one of preserve, strip, strict, lax
* @param instruction the calling instruction, used for diagnostics
* @param context the dynamic context
* @param rejectDuplicates true if duplicate attributes with the same name are disallowed (XQuery)
* @throws XPathException if a failure occurs
*/
static void copyAttribute(NodeInfo source,
SimpleType schemaType,
int validation,
Instruction instruction,
XPathContext context,
boolean rejectDuplicates)
throws XPathException {
SimpleType annotation = BuiltInAtomicType.UNTYPED_ATOMIC;
int opt = 0;
if (rejectDuplicates) {
opt |= ReceiverOptions.REJECT_DUPLICATES;
}
CharSequence value = source.getStringValueCS();
if (schemaType != null) {
if (schemaType.isSimpleType()) {
if (schemaType.isNamespaceSensitive()) {
XPathException err = new XPathException("Cannot create a parentless attribute whose " +
"type is namespace-sensitive (such as xs:QName)");
err.setErrorCode("XTTE1545");
err.setXPathContext(context);
err.setLocator(instruction);
throw err;
}
try {
ValidationFailure err = schemaType.validateContent(
value, DummyNamespaceResolver.getInstance(), context.getConfiguration().getConversionRules());
if (err != null) {
throw new ValidationException("Attribute being copied does not match the required type. " +
err.getMessage());
}
annotation = schemaType;
} catch (UnresolvedReferenceException ure) {
throw new ValidationException(ure);
}
} else {
XPathException e = new XPathException("Cannot validate an attribute against a complex type");
e.setXPathContext(context);
e.setErrorCode("XTTE1535"); // See spec bug 13001
e.setIsTypeError(true);
throw e;
}
} else if (validation == Validation.STRICT || validation == Validation.LAX) {
try {
annotation = context.getConfiguration().validateAttribute(source.getNameCode(), value, validation);
} catch (ValidationException e) {
XPathException err = XPathException.makeXPathException(e);
err.setErrorCodeQName(e.getErrorCodeQName());
err.setXPathContext(context);
err.setLocator(instruction);
err.setIsTypeError(true);
throw err;
}
} else if (validation == Validation.PRESERVE) {
annotation = (SimpleType)source.getSchemaType();
if (!annotation.equals(BuiltInAtomicType.UNTYPED_ATOMIC) && annotation.isNamespaceSensitive()) {
XPathException err = new XPathException("Cannot preserve type annotation when copying an attribute with namespace-sensitive content");
err.setErrorCode((instruction.getHostLanguage() == Configuration.XSLT ? "XTTE0950" : "XQTY0086"));
err.setIsTypeError(true);
err.setXPathContext(context);
err.setLocator(instruction);
throw err;
}
}
context.getReceiver().attribute(new NameOfNode(source), annotation, value, instruction.getLocationId(), opt);
}
/*@NotNull*/
public SequenceIterator extends Item> iterate(XPathContext context) throws XPathException {
final Controller controller = context.getController();
assert controller != null;
if (schemaType == null && copyNamespaces && !copyForUpdate) {
if (validation == Validation.PRESERVE) {
// create a virtual copy of the underlying nodes
ItemMappingFunction- copier = new ItemMappingFunction
- () {
public Item mapItem(Item item) {
if (item instanceof NodeInfo) {
VirtualCopy vc = VirtualCopy.makeVirtualCopy((NodeInfo)item, (NodeInfo) item);
long documentNumber =
controller.getConfiguration().getDocumentNumberAllocator().allocateDocumentNumber();
vc.setDocumentNumber(documentNumber);
if (((NodeInfo)item).getNodeKind() == Type.ELEMENT) {
vc.setSystemId(computeNewBaseUri((NodeInfo)item));
}
return vc;
} else {
return item;
}
}
};
return new ItemMappingIterator
- ((SequenceIterator
- )select.iterate(context), copier, true);
} else if (validation == Validation.STRIP) {
// create a virtual copy of the underlying nodes
ItemMappingFunction
- copier = new ItemMappingFunction
- () {
public Item mapItem(Item item) {
if (!(item instanceof NodeInfo)) {
return item;
}
VirtualCopy vc = VirtualUntypedCopy.makeVirtualUntypedCopy((NodeInfo) item, (NodeInfo) item);
long documentNumber =
controller.getConfiguration().getDocumentNumberAllocator().allocateDocumentNumber();
vc.setDocumentNumber(documentNumber);
if (((NodeInfo)item).getNodeKind() == Type.ELEMENT) {
vc.setSystemId(computeNewBaseUri((NodeInfo)item));
}
return vc;
}
};
return new ItemMappingIterator
- ((SequenceIterator
- )select.iterate(context), copier, true);
}
}
SequenceReceiver saved = context.getReceiver();
PipelineConfiguration pipe = controller.makePipelineConfiguration();
SequenceOutputter out = new SequenceOutputter(pipe);
pipe.setHostLanguage(getHostLanguage());
context.setReceiver(out);
try {
process(context);
} catch (XPathException err) {
if (err instanceof ValidationException) {
((ValidationException) err).setSourceLocator(this);
((ValidationException) err).setSystemId(getSystemId());
}
err.maybeSetLocation(this);
throw err;
}
context.setReceiver(saved);
return out.getSequence().iterate();
}
}