net.sf.saxon.style.XSLMap Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of Saxon-HE Show documentation
Show all versions of Saxon-HE Show documentation
The XSLT and XQuery Processor
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2018-2023 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.style;
import net.sf.saxon.Configuration;
import net.sf.saxon.expr.Expression;
import net.sf.saxon.expr.Literal;
import net.sf.saxon.expr.StaticProperty;
import net.sf.saxon.expr.instruct.SequenceInstr;
import net.sf.saxon.expr.parser.RoleDiagnostic;
import net.sf.saxon.expr.parser.TypeChecker;
import net.sf.saxon.lib.Feature;
import net.sf.saxon.ma.map.HashTrieMap;
import net.sf.saxon.ma.map.MapFunctionSet;
import net.sf.saxon.ma.map.MapType;
import net.sf.saxon.om.AttributeInfo;
import net.sf.saxon.om.NamespaceUri;
import net.sf.saxon.om.NodeName;
import net.sf.saxon.trans.SaxonErrorCode;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.type.ItemType;
import net.sf.saxon.value.QNameValue;
import net.sf.saxon.value.SequenceType;
import net.sf.saxon.value.StringValue;
import java.util.function.Supplier;
/**
* Handler for xsl:map instructions in an XSLT 3.0 stylesheet.
*/
public class XSLMap extends StyleElement {
private Expression select = null;
private Expression onDuplicates = null;
/**
* Determine whether this node is an instruction.
*
* @return true - it is an instruction
*/
@Override
public boolean isInstruction() {
return true;
}
/**
* Determine the type of item returned by this instruction (only relevant if
* it is an instruction).
*
* @return the item type returned
*/
protected ItemType getReturnedItemType() {
return MapType.ANY_MAP_TYPE;
}
/**
* Determine whether this type of element is allowed to contain a sequence constructor
*
* @return true: yes, it may contain a sequence constructor
*/
@Override
protected boolean mayContainSequenceConstructor() {
return true;
}
@Override
protected void prepareAttributes() {
for (AttributeInfo att : attributes()) {
NodeName attName = att.getNodeName();
String f = attName.getDisplayName();
String value = att.getValue();
if (attName.getLocalPart().equals("on-duplicates")) {
if (attName.getNamespaceUri().isEmpty()) {
requireXslt40("on-duplicates");
onDuplicates = makeExpression(value, att);
} else if (attName.hasURI(NamespaceUri.SAXON)) {
if (getConfiguration().isLicensedFeature(Configuration.LicenseFeature.PROFESSIONAL_EDITION)) {
onDuplicates = makeExpression(value, att);
} else {
issueWarning("saxon:on-duplicates ignored - requires Saxon-PE license",
SaxonErrorCode.SXWN9013);
}
}
} else {
checkUnknownAttribute(attName);
}
}
}
@Override
public Expression compile(Compilation exec, ComponentDeclaration decl) throws XPathException {
select = compileSequenceConstructor(exec, decl, false);
select = select.simplify();
// Custom type-checking; the checking performed by map:merge() gives poor diagnostics
TypeChecker tc = getConfiguration().getTypeChecker(false);
Supplier role =
() -> new RoleDiagnostic(RoleDiagnostic.MISC, "xsl:map sequence constructor", 0, "XTTE3375");
select = tc.staticTypeCheck(
select,
SequenceType.makeSequenceType(MapType.ANY_MAP_TYPE, StaticProperty.ALLOWS_ZERO_OR_MORE),
role, makeExpressionVisitor());
Expression optionsExp;
if (onDuplicates != null) {
optionsExp = MapFunctionSet.getInstance(31).makeFunction("entry", 2).makeFunctionCall(
Literal.makeLiteral(new QNameValue("", NamespaceUri.SAXON, "on-duplicates")),
onDuplicates);
} else {
HashTrieMap options = new HashTrieMap();
options.initialPut(StringValue.bmp("duplicates"), StringValue.bmp("reject"));
options.initialPut(new QNameValue("", NamespaceUri.SAXON, "duplicates-error-code"),
StringValue.bmp("XTDE3365"));
optionsExp = Literal.makeLiteral(options, select);
}
Expression exp = MapFunctionSet.getInstance(31).makeFunction("merge", 2)
.makeFunctionCall(select, optionsExp);
if (getConfiguration().getBooleanProperty(Feature.STRICT_STREAMABILITY)) {
exp = new SequenceInstr(exp);
}
return exp;
}
}