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

net.sf.saxon.style.XSLAttributeSet Maven / Gradle / Ivy

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2015 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.Component;
import net.sf.saxon.expr.Expression;
import net.sf.saxon.expr.instruct.AttributeSet;
import net.sf.saxon.expr.instruct.SlotManager;
import net.sf.saxon.expr.instruct.UseAttributeSet;
import net.sf.saxon.expr.parser.Optimizer;
import net.sf.saxon.lib.NamespaceConstant;
import net.sf.saxon.om.*;
import net.sf.saxon.pattern.NodeKindTest;
import net.sf.saxon.trace.LocationKind;
import net.sf.saxon.trans.SymbolicName;
import net.sf.saxon.trans.Visibility;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.tree.AttributeLocation;
import net.sf.saxon.tree.iter.AxisIterator;
import net.sf.saxon.value.Whitespace;

import java.util.ArrayList;
import java.util.List;

/**
 * An xsl:attribute-set element in the stylesheet. 
*/ public class XSLAttributeSet extends StyleElement implements StylesheetComponent { private String nameAtt; // the name of the attribute set as written /*@Nullable*/ private String useAtt; // the value of the use-attribute-sets attribute, as supplied private String visibilityAtt; // the value of the visibility attribute, trimmed private SlotManager stackFrameMap; // needed if variables are used private List attributeSetElements = new ArrayList(); // list of Declarations of XSLAttributeSet objects referenced by this one, within the same package private StructuredQName[] useAttributeSetNames; private List containedInstructions = new ArrayList(); // the compiled form of this attribute set private boolean validated = false; private Visibility visibility; private boolean streamable = false; /** * Get the corresponding Procedure object that results from the compilation of this * StylesheetProcedure */ public AttributeSet getCompiledProcedure() { return (AttributeSet)getPrincipalStylesheetModule().getStylesheetPackage().getComponent( new SymbolicName(StandardNames.XSL_ATTRIBUTE_SET, getObjectName())).getCode(); } public SymbolicName getSymbolicName() { return new SymbolicName(StandardNames.XSL_ATTRIBUTE_SET, getObjectName()); } /** * Check the compatibility of this component with another component that it is overriding * * @param component the overridden component * @throws XPathException if the components are not compatible (differing signatures) */ public void checkCompatibility(Component component) throws XPathException { if (((AttributeSet)component.getCode()).isDeclaredStreamable() && !isDeclaredStreamable()) { compileError("The overridden attribute set is declared streamable, " + "so the overriding attribute set must also be declared streamable"); } } /** * Ask whether this node is a declaration, that is, a permitted child of xsl:stylesheet * (including xsl:include and xsl:import). * * @return true for this element */ @Override public boolean isDeclaration() { return true; } /** * Get the name of this attribute set * * @return the name of the attribute set, as a QName */ public StructuredQName getAttributeSetName() { return getObjectName(); } /** * Ask whether the attribute set is declared streamable * @return true if the attribute streamable="yes" is present */ public boolean isDeclaredStreamable() { return streamable; } public void prepareAttributes() throws XPathException { useAtt = null; String streamableAtt = null; AttributeCollection atts = getAttributeList(); for (int a = 0; a < atts.getLength(); a++) { String f = atts.getQName(a); if (f.equals("name")) { nameAtt = Whitespace.trim(atts.getValue(a)); } else if (f.equals("use-attribute-sets")) { useAtt = atts.getValue(a); } else if (f.equals("streamable")) { streamableAtt = Whitespace.trim(atts.getValue(a)); } else if (f.equals("visibility")) { visibilityAtt = Whitespace.trim(atts.getValue(a)); } else { checkUnknownAttribute(atts.getNodeName(a)); } } if (nameAtt == null) { reportAbsence("name"); setObjectName(new StructuredQName("", "", "attribute-set-error-name")); return; } if (visibilityAtt == null) { visibility = Visibility.PRIVATE; } else { check30attribute("visibility"); visibility = interpretVisibilityValue(visibilityAtt, ""); } if (streamableAtt != null) { streamable = processBooleanAttribute("streamable", streamableAtt); if (streamable && !getConfiguration().isLicensedFeature(Configuration.LicenseFeature.ENTERPRISE_XSLT)) { issueWarning("Request for streaming ignored: this Saxon configuration does not support streaming", this); streamable = false; } } try { setObjectName(makeQName(nameAtt)); } catch (NamespaceException err) { setObjectName(new StructuredQName("", "", "attribute-set-error-name")); compileErrorInAttribute(err.getMessage(), "XTSE0280", "name"); } catch (XPathException err) { setObjectName(new StructuredQName("", "", "attribute-set-error-name")); err.setLocation(new AttributeLocation(this, StructuredQName.fromClarkName("name"))); throw err; } } /** * Get a name identifying the object of the expression, for example a function name, template name, * variable name, key name, element name, etc. This is used only where the name is known statically. * If there is no name, the value will be null. * * @return the name of the object declared in this element, if any */ public StructuredQName getObjectName() { StructuredQName o = super.getObjectName(); if (o == null) { try { prepareAttributes(); o = getObjectName(); } catch (XPathException err) { o = new StructuredQName("saxon", NamespaceConstant.SAXON, "badly-named-attribute-set"); setObjectName(o); } } return o; } public void validate(ComponentDeclaration decl) throws XPathException { if (validated) { return; } checkTopLevel("XTSE0010", true); stackFrameMap = getConfiguration().makeSlotManager(); AxisIterator kids = iterateAxis(AxisInfo.CHILD); NodeInfo child; while ((child = kids.next()) != null) { if (child instanceof XSLAttribute) { if (visibility == Visibility.ABSTRACT) { compileError("An abstract attribute-set must contain no xsl:attribute instructions"); } } else { compileError("Only xsl:attribute is allowed within xsl:attribute-set", "XTSE0010"); } } if (useAtt != null) { if (visibility == Visibility.ABSTRACT) { compileError("An abstract attribute-set must have no @use-attribute-sets attribute"); } // identify any attribute sets that this one refers to useAttributeSetNames = getUsedAttributeSets(useAtt); } validated = true; } public StructuredQName[] getUseAttributeSetNames() { return useAttributeSetNames; } public void index(ComponentDeclaration decl, PrincipalStylesheetModule top) throws XPathException { top.indexAttributeSet(decl); } /** * Check for circularity: specifically, check that this attribute set does not contain * a direct or indirect reference to the one supplied as a parameter * * @param origin the place from which the search started * @throws net.sf.saxon.trans.XPathException * if an error is found */ public void checkCircularity(XSLAttributeSet origin) throws XPathException { if (this == origin) { compileError("The definition of the attribute set is circular", "XTSE0720"); } else { if (!validated) { // if this attribute set isn't validated yet, we don't check it. // The circularity will be detected when the last attribute set in the cycle // gets validated return; } if (attributeSetElements != null) { for (ComponentDeclaration attributeSetElement : attributeSetElements) { XSLAttributeSet element = (XSLAttributeSet) attributeSetElement.getSourceElement(); element.checkCircularity(origin); if (streamable && !element.streamable) { compileError("Attribute-set is declared streamable but references a non-streamable attribute set " + element.getAttributeSetName().getDisplayName(), "XTSE3430"); } } } } } /** * Get the contained xsl:attribute instructions, in compiled form * @return the list of contained instructions. This does not include any xsl:use-attribute-set references. */ public List getContainedInstructions() { return containedInstructions; } /** * Get details of stack frame */ public SlotManager getSlotManager() { return stackFrameMap; } /** * Compile the attribute set * * @param compilation the current compilation episode * @param decl this attribute set declaration * @throws XPathException if a failure is detected */ public void compileDeclaration(Compilation compilation, ComponentDeclaration decl) throws XPathException { if (isActionCompleted(ACTION_COMPILE)) { return; } if (useAtt != null) { // identify any attribute sets that this one refers to List invocations = UseAttributeSet.makeUseAttributeSetInstructions(useAttributeSetNames, this); if (!invocations.isEmpty()) { containedInstructions.add(UseAttributeSet.makeCompositeExpression(invocations)); } // check for circularity, to the extent possible within a single package for (StructuredQName name : useAttributeSetNames) { getPrincipalStylesheetModule().getAttributeSets(name, attributeSetElements); } for (ComponentDeclaration attributeSetElement : attributeSetElements) { ((XSLAttributeSet) attributeSetElement.getSourceElement()).checkCircularity(this); } // check for consistency of streamability attribute if (streamable) { for (ComponentDeclaration attributeSetElement : attributeSetElements) { if (!((XSLAttributeSet) attributeSetElement.getSourceElement()).streamable) { compileError("Attribute set is declared streamable, " + "but references an attribute set that is not declared streamable", "XTSE0730"); } } } } XSLAttribute node; AxisIterator iter = iterateAxis(AxisInfo.CHILD, NodeKindTest.ELEMENT); while ((node = (XSLAttribute) iter.next()) != null) { Expression inst = node.compile(compilation, decl); inst.setRetainedStaticContext(makeRetainedStaticContext()); inst = inst.simplify(); if (getConfiguration().isCompileWithTracing()) { inst = makeTraceInstruction(this, inst); } containedInstructions.add(inst); } setActionCompleted(ACTION_COMPILE); } /** * Optimize the stylesheet construct * * @param declaration this attribute set declaration */ public void optimize(ComponentDeclaration declaration) throws XPathException { // Already done earlier } /** * Get the type of construct. This will be a constant in * class {@link LocationKind}. This method is part of * the {@link net.sf.saxon.trace.InstructionInfo} interface */ public int getConstructType() { return StandardNames.XSL_ATTRIBUTE_SET; } /** * Generate byte code if appropriate * * @param opt the optimizer */ public void generateByteCode(Optimizer opt) {} // private void checkStreamability() throws XPathException { ////#ifdefined STREAM // if (streamable) { // ContextItemStaticInfo info = new ContextItemStaticInfo(AnyItemType.getInstance(), false, true); // procedure.getBody().getStreamability(false, info, null); // if (procedure.getBody().getSweep() != Sweep.MOTIONLESS) { // compileError("The attribute set is declared streamable but it is not motionless", "XTSE3430"); // } // } ////#endif // } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy