e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
1032 lines
44 KiB
C#
1032 lines
44 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="XmlQueryRuntime.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
// <owner current="true" primary="true">[....]</owner>
|
|
//------------------------------------------------------------------------------
|
|
using System;
|
|
using System.IO;
|
|
using System.Xml;
|
|
using System.Xml.XPath;
|
|
using System.Xml.Schema;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Text;
|
|
using System.Globalization;
|
|
using System.Reflection;
|
|
using System.Reflection.Emit;
|
|
using System.Xml.Xsl.Qil;
|
|
using System.Xml.Xsl.IlGen;
|
|
using System.ComponentModel;
|
|
using MS.Internal.Xml.XPath;
|
|
using System.Runtime.Versioning;
|
|
|
|
namespace System.Xml.Xsl.Runtime {
|
|
using Res = System.Xml.Utils.Res;
|
|
|
|
/// <summary>
|
|
/// XmlQueryRuntime is passed as the first parameter to all generated query methods.
|
|
///
|
|
/// XmlQueryRuntime contains runtime support for generated ILGen queries:
|
|
/// 1. Stack of output writers (stack handles nested document construction)
|
|
/// 2. Manages list of all xml types that are used within the query
|
|
/// 3. Manages list of all atomized names that are used within the query
|
|
/// </summary>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public sealed class XmlQueryRuntime {
|
|
// Early-Bound Library Objects
|
|
private XmlQueryContext ctxt;
|
|
private XsltLibrary xsltLib;
|
|
private EarlyBoundInfo[] earlyInfo;
|
|
private object[] earlyObjects;
|
|
|
|
// Global variables and parameters
|
|
private string[] globalNames;
|
|
private object[] globalValues;
|
|
|
|
// Names, prefix mappings, and name filters
|
|
private XmlNameTable nameTableQuery;
|
|
private string[] atomizedNames; // Names after atomization
|
|
private XmlNavigatorFilter[] filters; // Name filters (contain atomized names)
|
|
private StringPair[][] prefixMappingsList; // Lists of prefix mappings (used to resolve computed names)
|
|
|
|
// Xml types
|
|
private XmlQueryType[] types;
|
|
|
|
// Collations
|
|
private XmlCollation[] collations;
|
|
|
|
// Document ordering
|
|
private DocumentOrderComparer docOrderCmp;
|
|
|
|
// Indexes
|
|
private ArrayList[] indexes;
|
|
|
|
// Output construction
|
|
private XmlQueryOutput output;
|
|
private Stack<XmlQueryOutput> stkOutput;
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Constructors
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// This constructor is internal so that external users cannot construct it (and therefore we do not have to test it separately).
|
|
/// </summary>
|
|
[ResourceConsumption(ResourceScope.Machine)]
|
|
[ResourceExposure(ResourceScope.Machine)]
|
|
internal XmlQueryRuntime(XmlQueryStaticData data, object defaultDataSource, XmlResolver dataSources, XsltArgumentList argList, XmlSequenceWriter seqWrt) {
|
|
Debug.Assert(data != null);
|
|
string[] names = data.Names;
|
|
Int32Pair[] filters = data.Filters;
|
|
WhitespaceRuleLookup wsRules;
|
|
int i;
|
|
|
|
// Early-Bound Library Objects
|
|
wsRules = (data.WhitespaceRules != null && data.WhitespaceRules.Count != 0) ? new WhitespaceRuleLookup(data.WhitespaceRules) : null;
|
|
this.ctxt = new XmlQueryContext(this, defaultDataSource, dataSources, argList, wsRules);
|
|
this.xsltLib = null;
|
|
this.earlyInfo = data.EarlyBound;
|
|
this.earlyObjects = (this.earlyInfo != null) ? new object[earlyInfo.Length] : null;
|
|
|
|
// Global variables and parameters
|
|
this.globalNames = data.GlobalNames;
|
|
this.globalValues = (this.globalNames != null) ? new object[this.globalNames.Length] : null;
|
|
|
|
// Names
|
|
this.nameTableQuery = this.ctxt.QueryNameTable;
|
|
this.atomizedNames = null;
|
|
|
|
if (names != null) {
|
|
// Atomize all names in "nameTableQuery". Use names from the default data source's
|
|
// name table when possible.
|
|
XmlNameTable nameTableDefault = ctxt.DefaultNameTable;
|
|
this.atomizedNames = new string[names.Length];
|
|
|
|
if (nameTableDefault != this.nameTableQuery && nameTableDefault != null) {
|
|
// Ensure that atomized names from the default data source are added to the
|
|
// name table used in this query
|
|
for (i = 0; i < names.Length; i++) {
|
|
string name = nameTableDefault.Get(names[i]);
|
|
this.atomizedNames[i] = this.nameTableQuery.Add(name ?? names[i]);
|
|
}
|
|
}
|
|
else {
|
|
// Enter names into nametable used in this query
|
|
for (i = 0; i < names.Length; i++)
|
|
this.atomizedNames[i] = this.nameTableQuery.Add(names[i]);
|
|
}
|
|
}
|
|
|
|
// Name filters
|
|
this.filters = null;
|
|
if (filters != null) {
|
|
// Construct name filters. Each pair of integers in the filters[] array specifies the
|
|
// (localName, namespaceUri) of the NameFilter to be created.
|
|
this.filters = new XmlNavigatorFilter[filters.Length];
|
|
|
|
for (i = 0; i < filters.Length; i++)
|
|
this.filters[i] = XmlNavNameFilter.Create(this.atomizedNames[filters[i].Left], this.atomizedNames[filters[i].Right]);
|
|
}
|
|
|
|
// Prefix maping lists
|
|
this.prefixMappingsList = data.PrefixMappingsList;
|
|
|
|
// Xml types
|
|
this.types = data.Types;
|
|
|
|
// Xml collations
|
|
this.collations = data.Collations;
|
|
|
|
// Document ordering
|
|
this.docOrderCmp = new DocumentOrderComparer();
|
|
|
|
// Indexes
|
|
this.indexes = null;
|
|
|
|
// Output construction
|
|
this.stkOutput = new Stack<XmlQueryOutput>(16);
|
|
this.output = new XmlQueryOutput(this, seqWrt);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Debugger Utility Methods
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Return array containing the names of all the global variables and parameters used in this query, in this format:
|
|
/// {namespace}prefix:local-name
|
|
/// </summary>
|
|
public string[] DebugGetGlobalNames() {
|
|
return this.globalNames;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the value of a global value having the specified name. Always return the global value as a list of XPathItem.
|
|
/// Return null if there is no global value having the specified name.
|
|
/// </summary>
|
|
public IList DebugGetGlobalValue(string name) {
|
|
for (int idx = 0; idx < this.globalNames.Length; idx++) {
|
|
if (this.globalNames[idx] == name) {
|
|
Debug.Assert(IsGlobalComputed(idx), "Cannot get the value of a global value until it has been computed.");
|
|
Debug.Assert(this.globalValues[idx] is IList<XPathItem>, "Only debugger should call this method, and all global values should have type item* in debugging scenarios.");
|
|
return (IList) this.globalValues[idx];
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Set the value of a global value having the specified name. If there is no such value, this method is a no-op.
|
|
/// </summary>
|
|
public void DebugSetGlobalValue(string name, object value) {
|
|
for (int idx = 0; idx < this.globalNames.Length; idx++) {
|
|
if (this.globalNames[idx] == name) {
|
|
Debug.Assert(IsGlobalComputed(idx), "Cannot get the value of a global value until it has been computed.");
|
|
Debug.Assert(this.globalValues[idx] is IList<XPathItem>, "Only debugger should call this method, and all global values should have type item* in debugging scenarios.");
|
|
|
|
// Always convert "value" to a list of XPathItem using the item* converter
|
|
this.globalValues[idx] = (IList<XPathItem>) XmlAnyListConverter.ItemList.ChangeType(value, typeof(XPathItem[]), null);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert sequence to it's appropriate XSLT type and return to caller.
|
|
/// </summary>
|
|
public object DebugGetXsltValue(IList seq) {
|
|
if (seq != null && seq.Count == 1) {
|
|
XPathItem item = seq[0] as XPathItem;
|
|
if (item != null && !item.IsNode) {
|
|
return item.TypedValue;
|
|
}
|
|
else if (item is RtfNavigator) {
|
|
return ((RtfNavigator) item).ToNavigator();
|
|
}
|
|
}
|
|
|
|
return seq;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Early-Bound Library Objects
|
|
//-----------------------------------------------
|
|
|
|
internal const BindingFlags EarlyBoundFlags = BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static;
|
|
internal const BindingFlags LateBoundFlags = BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static;
|
|
|
|
/// <summary>
|
|
/// Return the object that manages external user context information such as data sources, parameters, extension objects, etc.
|
|
/// </summary>
|
|
public XmlQueryContext ExternalContext {
|
|
get { return this.ctxt; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return the object that manages the state needed to implement various Xslt functions.
|
|
/// </summary>
|
|
public XsltLibrary XsltFunctions {
|
|
get {
|
|
if (this.xsltLib == null) {
|
|
this.xsltLib = new XsltLibrary(this);
|
|
}
|
|
|
|
return this.xsltLib;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the early-bound extension object identified by "index". If it does not yet exist, create an instance using the
|
|
/// corresponding ConstructorInfo.
|
|
/// </summary>
|
|
public object GetEarlyBoundObject(int index) {
|
|
object obj;
|
|
Debug.Assert(this.earlyObjects != null && index < this.earlyObjects.Length, "Early bound object does not exist");
|
|
|
|
obj = this.earlyObjects[index];
|
|
if (obj == null) {
|
|
// Early-bound object does not yet exist, so create it now
|
|
obj = this.earlyInfo[index].CreateObject();
|
|
this.earlyObjects[index] = obj;
|
|
}
|
|
|
|
return obj;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the early bound object identified by "namespaceUri" contains a method that matches "name".
|
|
/// </summary>
|
|
public bool EarlyBoundFunctionExists(string name, string namespaceUri) {
|
|
if (this.earlyInfo == null)
|
|
return false;
|
|
|
|
for (int idx = 0; idx < this.earlyInfo.Length; idx++) {
|
|
if (namespaceUri == this.earlyInfo[idx].NamespaceUri)
|
|
return new XmlExtensionFunction(name, namespaceUri, -1, this.earlyInfo[idx].EarlyBoundType, EarlyBoundFlags).CanBind();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Global variables and parameters
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Return true if the global value specified by idxValue was previously computed.
|
|
/// </summary>
|
|
public bool IsGlobalComputed(int index) {
|
|
return this.globalValues[index] != null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return the value that is bound to the global variable or parameter specified by idxValue.
|
|
/// If the value has not yet been computed, then compute it now and store it in this.globalValues.
|
|
/// </summary>
|
|
public object GetGlobalValue(int index) {
|
|
Debug.Assert(IsGlobalComputed(index), "Cannot get the value of a global value until it has been computed.");
|
|
return this.globalValues[index];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return the value that is bound to the global variable or parameter specified by idxValue.
|
|
/// If the value has not yet been computed, then compute it now and store it in this.globalValues.
|
|
/// </summary>
|
|
public void SetGlobalValue(int index, object value) {
|
|
Debug.Assert(!IsGlobalComputed(index), "Global value should only be set once.");
|
|
this.globalValues[index] = value;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Names, prefix mappings, and name filters
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Return the name table used to atomize all names used by the query.
|
|
/// </summary>
|
|
public XmlNameTable NameTable {
|
|
get { return this.nameTableQuery; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the atomized name at the specified index in the array of names.
|
|
/// </summary>
|
|
public string GetAtomizedName(int index) {
|
|
Debug.Assert(this.atomizedNames != null);
|
|
return this.atomizedNames[index];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the name filter at the specified index in the array of filters.
|
|
/// </summary>
|
|
public XmlNavigatorFilter GetNameFilter(int index) {
|
|
Debug.Assert(this.filters != null);
|
|
return this.filters[index];
|
|
}
|
|
|
|
/// <summary>
|
|
/// XPathNodeType.All: Filters all nodes
|
|
/// XPathNodeType.Attribute: Filters attributes
|
|
/// XPathNodeType.Namespace: Not allowed
|
|
/// XPathNodeType.XXX: Filters all nodes *except* those having XPathNodeType.XXX
|
|
/// </summary>
|
|
public XmlNavigatorFilter GetTypeFilter(XPathNodeType nodeType) {
|
|
if (nodeType == XPathNodeType.All)
|
|
return XmlNavNeverFilter.Create();
|
|
|
|
if (nodeType == XPathNodeType.Attribute)
|
|
return XmlNavAttrFilter.Create();
|
|
|
|
return XmlNavTypeFilter.Create(nodeType);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse the specified tag name (foo:bar) and resolve the resulting prefix. If the prefix cannot be resolved,
|
|
/// then throw an error. Return an XmlQualifiedName.
|
|
/// </summary>
|
|
public XmlQualifiedName ParseTagName(string tagName, int indexPrefixMappings) {
|
|
string prefix, localName, ns;
|
|
|
|
// Parse the tagName as a prefix, localName pair and resolve the prefix
|
|
ParseTagName(tagName, indexPrefixMappings, out prefix, out localName, out ns);
|
|
return new XmlQualifiedName(localName, ns);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse the specified tag name (foo:bar). Return an XmlQualifiedName consisting of the parsed local name
|
|
/// and the specified namespace.
|
|
/// </summary>
|
|
public XmlQualifiedName ParseTagName(string tagName, string ns) {
|
|
string prefix, localName;
|
|
|
|
// Parse the tagName as a prefix, localName pair
|
|
ValidateNames.ParseQNameThrow(tagName, out prefix, out localName);
|
|
return new XmlQualifiedName(localName, ns);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse the specified tag name (foo:bar) and resolve the resulting prefix. If the prefix cannot be resolved,
|
|
/// then throw an error. Return the prefix, localName, and namespace URI.
|
|
/// </summary>
|
|
internal void ParseTagName(string tagName, int idxPrefixMappings, out string prefix, out string localName, out string ns) {
|
|
Debug.Assert(this.prefixMappingsList != null);
|
|
|
|
// Parse the tagName as a prefix, localName pair
|
|
ValidateNames.ParseQNameThrow(tagName, out prefix, out localName);
|
|
|
|
// Map the prefix to a namespace URI
|
|
ns = null;
|
|
foreach (StringPair pair in this.prefixMappingsList[idxPrefixMappings]) {
|
|
if (prefix == pair.Left) {
|
|
ns = pair.Right;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Throw exception if prefix could not be resolved
|
|
if (ns == null) {
|
|
// Check for mappings that are always in-scope
|
|
if (prefix.Length == 0)
|
|
ns = "";
|
|
else if (prefix.Equals("xml"))
|
|
ns = XmlReservedNs.NsXml;
|
|
// It is not correct to resolve xmlns prefix in XPath but removing it would be a breaking change.
|
|
else if (prefix.Equals("xmlns"))
|
|
ns = XmlReservedNs.NsXmlNs;
|
|
else
|
|
throw new XslTransformException(Res.Xslt_InvalidPrefix, prefix);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the nav1's LocalName and NamespaceURI properties equal nav2's corresponding properties.
|
|
/// </summary>
|
|
public bool IsQNameEqual(XPathNavigator n1, XPathNavigator n2) {
|
|
if ((object) n1.NameTable == (object) n2.NameTable) {
|
|
// Use atomized comparison
|
|
return (object) n1.LocalName == (object) n2.LocalName && (object) n1.NamespaceURI == (object) n2.NamespaceURI;
|
|
}
|
|
|
|
return (n1.LocalName == n2.LocalName) && (n1.NamespaceURI == n2.NamespaceURI);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the specified navigator's LocalName and NamespaceURI properties equal the argument names.
|
|
/// </summary>
|
|
public bool IsQNameEqual(XPathNavigator navigator, int indexLocalName, int indexNamespaceUri) {
|
|
if ((object) navigator.NameTable == (object) this.nameTableQuery) {
|
|
// Use atomized comparison
|
|
return ((object) GetAtomizedName(indexLocalName) == (object) navigator.LocalName &&
|
|
(object) GetAtomizedName(indexNamespaceUri) == (object) navigator.NamespaceURI);
|
|
}
|
|
|
|
// Use string comparison
|
|
return (GetAtomizedName(indexLocalName) == navigator.LocalName) && (GetAtomizedName(indexNamespaceUri) == navigator.NamespaceURI);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Xml types
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Get the array of xml types that are used within this query.
|
|
/// </summary>
|
|
internal XmlQueryType[] XmlTypes {
|
|
get { return this.types; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the Xml query type at the specified index in the array of types.
|
|
/// </summary>
|
|
internal XmlQueryType GetXmlType(int idxType) {
|
|
Debug.Assert(this.types != null);
|
|
return this.types[idxType];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Forward call to ChangeTypeXsltArgument(XmlQueryType, object, Type).
|
|
/// </summary>
|
|
public object ChangeTypeXsltArgument(int indexType, object value, Type destinationType) {
|
|
return ChangeTypeXsltArgument(GetXmlType(indexType), value, destinationType);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert from the Clr type of "value" to Clr type "destinationType" using V1 Xslt rules.
|
|
/// These rules include converting any Rtf values to Nodes.
|
|
/// </summary>
|
|
internal object ChangeTypeXsltArgument(XmlQueryType xmlType, object value, Type destinationType) {
|
|
Debug.Assert(XmlILTypeHelper.GetStorageType(xmlType).IsAssignableFrom(value.GetType()),
|
|
"Values passed to ChangeTypeXsltArgument should be in ILGen's default Clr representation.");
|
|
Debug.Assert(destinationType == XsltConvert.ObjectType || !destinationType.IsAssignableFrom(value.GetType()),
|
|
"No need to call ChangeTypeXsltArgument since value is already assignable to destinationType " + destinationType);
|
|
|
|
switch (xmlType.TypeCode) {
|
|
case XmlTypeCode.String:
|
|
if (destinationType == XsltConvert.DateTimeType)
|
|
value = XsltConvert.ToDateTime((string) value);
|
|
break;
|
|
|
|
case XmlTypeCode.Double:
|
|
if (destinationType != XsltConvert.DoubleType)
|
|
value = Convert.ChangeType(value, destinationType, CultureInfo.InvariantCulture);
|
|
break;
|
|
|
|
case XmlTypeCode.Node:
|
|
Debug.Assert(xmlType != XmlQueryTypeFactory.Node && xmlType != XmlQueryTypeFactory.NodeS,
|
|
"Rtf values should have been eliminated by caller.");
|
|
|
|
if (destinationType == XsltConvert.XPathNodeIteratorType) {
|
|
value = new XPathArrayIterator((IList) value);
|
|
}
|
|
else if (destinationType == XsltConvert.XPathNavigatorArrayType) {
|
|
// Copy sequence to XPathNavigator[]
|
|
IList<XPathNavigator> seq = (IList<XPathNavigator>) value;
|
|
XPathNavigator[] navArray = new XPathNavigator[seq.Count];
|
|
|
|
for (int i = 0; i < seq.Count; i++)
|
|
navArray[i] = seq[i];
|
|
|
|
value = navArray;
|
|
}
|
|
break;
|
|
|
|
case XmlTypeCode.Item: {
|
|
// Only typeof(object) is supported as a destination type
|
|
if (destinationType != XsltConvert.ObjectType)
|
|
throw new XslTransformException(Res.Xslt_UnsupportedClrType, destinationType.Name);
|
|
|
|
// Convert to default, backwards-compatible representation
|
|
// 1. NodeSet: System.Xml.XPath.XPathNodeIterator
|
|
// 2. Rtf: System.Xml.XPath.XPathNavigator
|
|
// 3. Other: Default V1 representation
|
|
IList<XPathItem> seq = (IList<XPathItem>) value;
|
|
if (seq.Count == 1) {
|
|
XPathItem item = seq[0];
|
|
|
|
if (item.IsNode) {
|
|
// Node or Rtf
|
|
RtfNavigator rtf = item as RtfNavigator;
|
|
if (rtf != null)
|
|
value = rtf.ToNavigator();
|
|
else
|
|
value = new XPathArrayIterator((IList) value);
|
|
}
|
|
else {
|
|
// Atomic value
|
|
value = item.TypedValue;
|
|
}
|
|
}
|
|
else {
|
|
// Nodeset
|
|
value = new XPathArrayIterator((IList) value);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
Debug.Assert(destinationType.IsAssignableFrom(value.GetType()), "ChangeType from type " + value.GetType().Name + " to type " + destinationType.Name + " failed");
|
|
return value;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Forward call to ChangeTypeXsltResult(XmlQueryType, object)
|
|
/// </summary>
|
|
public object ChangeTypeXsltResult(int indexType, object value) {
|
|
return ChangeTypeXsltResult(GetXmlType(indexType), value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert from the Clr type of "value" to the default Clr type that ILGen uses to represent the xml type, using
|
|
/// the conversion rules of the xml type.
|
|
/// </summary>
|
|
internal object ChangeTypeXsltResult(XmlQueryType xmlType, object value) {
|
|
if (value == null)
|
|
throw new XslTransformException(Res.Xslt_ItemNull, string.Empty);
|
|
|
|
switch (xmlType.TypeCode) {
|
|
case XmlTypeCode.String:
|
|
if (value.GetType() == XsltConvert.DateTimeType)
|
|
value = XsltConvert.ToString((DateTime) value);
|
|
break;
|
|
|
|
case XmlTypeCode.Double:
|
|
if (value.GetType() != XsltConvert.DoubleType)
|
|
value = ((IConvertible) value).ToDouble(null);
|
|
|
|
break;
|
|
|
|
case XmlTypeCode.Node:
|
|
if (!xmlType.IsSingleton) {
|
|
XPathArrayIterator iter = value as XPathArrayIterator;
|
|
|
|
// Special-case XPathArrayIterator in order to avoid copies
|
|
if (iter != null && iter.AsList is XmlQueryNodeSequence) {
|
|
value = iter.AsList as XmlQueryNodeSequence;
|
|
}
|
|
else {
|
|
// Iterate over list and ensure it only contains nodes
|
|
XmlQueryNodeSequence seq = new XmlQueryNodeSequence();
|
|
IList list = value as IList;
|
|
|
|
if (list != null) {
|
|
for (int i = 0; i < list.Count; i++)
|
|
seq.Add(EnsureNavigator(list[i]));
|
|
}
|
|
else {
|
|
foreach (object o in (IEnumerable) value)
|
|
seq.Add(EnsureNavigator(o));
|
|
}
|
|
|
|
value = seq;
|
|
}
|
|
|
|
// Always sort node-set by document order
|
|
value = ((XmlQueryNodeSequence) value).DocOrderDistinct(this.docOrderCmp);
|
|
}
|
|
break;
|
|
|
|
case XmlTypeCode.Item: {
|
|
Type sourceType = value.GetType();
|
|
IXPathNavigable navigable;
|
|
|
|
// If static type is item, then infer type based on dynamic value
|
|
switch (XsltConvert.InferXsltType(sourceType).TypeCode) {
|
|
case XmlTypeCode.Boolean:
|
|
value = new XmlQueryItemSequence(new XmlAtomicValue(XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.Boolean), value));
|
|
break;
|
|
|
|
case XmlTypeCode.Double:
|
|
value = new XmlQueryItemSequence(new XmlAtomicValue(XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.Double), ((IConvertible) value).ToDouble(null)));
|
|
break;
|
|
|
|
case XmlTypeCode.String:
|
|
if (sourceType == XsltConvert.DateTimeType)
|
|
value = new XmlQueryItemSequence(new XmlAtomicValue(XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.String), XsltConvert.ToString((DateTime) value)));
|
|
else
|
|
value = new XmlQueryItemSequence(new XmlAtomicValue(XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.String), value));
|
|
break;
|
|
|
|
case XmlTypeCode.Node:
|
|
// Support XPathNavigator[]
|
|
value = ChangeTypeXsltResult(XmlQueryTypeFactory.NodeS, value);
|
|
break;
|
|
|
|
case XmlTypeCode.Item:
|
|
// Support XPathNodeIterator
|
|
if (value is XPathNodeIterator) {
|
|
value = ChangeTypeXsltResult(XmlQueryTypeFactory.NodeS, value);
|
|
break;
|
|
}
|
|
|
|
// Support IXPathNavigable and XPathNavigator
|
|
navigable = value as IXPathNavigable;
|
|
if (navigable != null) {
|
|
if (value is XPathNavigator)
|
|
value = new XmlQueryNodeSequence((XPathNavigator) value);
|
|
else
|
|
value = new XmlQueryNodeSequence(navigable.CreateNavigator());
|
|
break;
|
|
}
|
|
|
|
throw new XslTransformException(Res.Xslt_UnsupportedClrType, sourceType.Name);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
Debug.Assert(XmlILTypeHelper.GetStorageType(xmlType).IsAssignableFrom(value.GetType()), "Xml type " + xmlType + " is not represented in ILGen as " + value.GetType().Name);
|
|
return value;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Ensure that "value" is a navigator and not null.
|
|
/// </summary>
|
|
private static XPathNavigator EnsureNavigator(object value) {
|
|
XPathNavigator nav = value as XPathNavigator;
|
|
|
|
if (nav == null)
|
|
throw new XslTransformException(Res.Xslt_ItemNull, string.Empty);
|
|
|
|
return nav;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the type of every item in "seq" matches the xml type identified by "idxType".
|
|
/// </summary>
|
|
public bool MatchesXmlType(IList<XPathItem> seq, int indexType) {
|
|
XmlQueryType typBase = GetXmlType(indexType);
|
|
XmlQueryCardinality card;
|
|
|
|
switch (seq.Count) {
|
|
case 0: card = XmlQueryCardinality.Zero; break;
|
|
case 1: card = XmlQueryCardinality.One; break;
|
|
default: card = XmlQueryCardinality.More; break;
|
|
}
|
|
|
|
if (!(card <= typBase.Cardinality))
|
|
return false;
|
|
|
|
typBase = typBase.Prime;
|
|
for (int i = 0; i < seq.Count; i++) {
|
|
if (!CreateXmlType(seq[0]).IsSubtypeOf(typBase))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the type of "item" matches the xml type identified by "idxType".
|
|
/// </summary>
|
|
public bool MatchesXmlType(XPathItem item, int indexType) {
|
|
return CreateXmlType(item).IsSubtypeOf(GetXmlType(indexType));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the type of "seq" is a subtype of a singleton type identified by "code".
|
|
/// </summary>
|
|
public bool MatchesXmlType(IList<XPathItem> seq, XmlTypeCode code) {
|
|
if (seq.Count != 1)
|
|
return false;
|
|
|
|
return MatchesXmlType(seq[0], code);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return true if the type of "item" is a subtype of the type identified by "code".
|
|
/// </summary>
|
|
public bool MatchesXmlType(XPathItem item, XmlTypeCode code) {
|
|
// All atomic type codes appear after AnyAtomicType
|
|
if (code > XmlTypeCode.AnyAtomicType)
|
|
return !item.IsNode && item.XmlType.TypeCode == code;
|
|
|
|
// Handle node code and AnyAtomicType
|
|
switch (code) {
|
|
case XmlTypeCode.AnyAtomicType: return !item.IsNode;
|
|
case XmlTypeCode.Node: return item.IsNode;
|
|
case XmlTypeCode.Item: return true;
|
|
default:
|
|
if (!item.IsNode)
|
|
return false;
|
|
|
|
switch (((XPathNavigator) item).NodeType) {
|
|
case XPathNodeType.Root: return code == XmlTypeCode.Document;
|
|
case XPathNodeType.Element: return code == XmlTypeCode.Element;
|
|
case XPathNodeType.Attribute: return code == XmlTypeCode.Attribute;
|
|
case XPathNodeType.Namespace: return code == XmlTypeCode.Namespace;
|
|
case XPathNodeType.Text: return code == XmlTypeCode.Text;
|
|
case XPathNodeType.SignificantWhitespace: return code == XmlTypeCode.Text;
|
|
case XPathNodeType.Whitespace: return code == XmlTypeCode.Text;
|
|
case XPathNodeType.ProcessingInstruction: return code == XmlTypeCode.ProcessingInstruction;
|
|
case XPathNodeType.Comment: return code == XmlTypeCode.Comment;
|
|
}
|
|
break;
|
|
}
|
|
|
|
Debug.Fail("XmlTypeCode " + code + " was not fully handled.");
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create an XmlQueryType that represents the type of "item".
|
|
/// </summary>
|
|
private XmlQueryType CreateXmlType(XPathItem item) {
|
|
if (item.IsNode) {
|
|
// Rtf
|
|
RtfNavigator rtf = item as RtfNavigator;
|
|
if (rtf != null)
|
|
return XmlQueryTypeFactory.Node;
|
|
|
|
// Node
|
|
XPathNavigator nav = (XPathNavigator) item;
|
|
switch (nav.NodeType) {
|
|
case XPathNodeType.Root:
|
|
case XPathNodeType.Element:
|
|
if (nav.XmlType == null)
|
|
return XmlQueryTypeFactory.Type(nav.NodeType, XmlQualifiedNameTest.New(nav.LocalName, nav.NamespaceURI), XmlSchemaComplexType.UntypedAnyType, false);
|
|
|
|
return XmlQueryTypeFactory.Type(nav.NodeType, XmlQualifiedNameTest.New(nav.LocalName, nav.NamespaceURI), nav.XmlType, nav.SchemaInfo.SchemaElement.IsNillable);
|
|
|
|
case XPathNodeType.Attribute:
|
|
if (nav.XmlType == null)
|
|
return XmlQueryTypeFactory.Type(nav.NodeType, XmlQualifiedNameTest.New(nav.LocalName, nav.NamespaceURI), DatatypeImplementation.UntypedAtomicType, false);
|
|
|
|
return XmlQueryTypeFactory.Type(nav.NodeType, XmlQualifiedNameTest.New(nav.LocalName, nav.NamespaceURI), nav.XmlType, false);
|
|
}
|
|
|
|
return XmlQueryTypeFactory.Type(nav.NodeType, XmlQualifiedNameTest.Wildcard, XmlSchemaComplexType.AnyType, false);
|
|
}
|
|
|
|
// Atomic value
|
|
return XmlQueryTypeFactory.Type((XmlSchemaSimpleType)item.XmlType, true);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Xml collations
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Get a collation that was statically created.
|
|
/// </summary>
|
|
public XmlCollation GetCollation(int index) {
|
|
Debug.Assert(this.collations != null);
|
|
return this.collations[index];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create a collation from a string.
|
|
/// </summary>
|
|
public XmlCollation CreateCollation(string collation) {
|
|
return XmlCollation.Create(collation);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Document Ordering and Identity
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Compare the relative positions of two navigators. Return -1 if navThis is before navThat, 1 if after, and
|
|
/// 0 if they are positioned to the same node.
|
|
/// </summary>
|
|
public int ComparePosition(XPathNavigator navigatorThis, XPathNavigator navigatorThat) {
|
|
return this.docOrderCmp.Compare(navigatorThis, navigatorThat);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get a comparer which guarantees a stable ordering among nodes, even those from different documents.
|
|
/// </summary>
|
|
public IList<XPathNavigator> DocOrderDistinct(IList<XPathNavigator> seq) {
|
|
if (seq.Count <= 1)
|
|
return seq;
|
|
|
|
XmlQueryNodeSequence nodeSeq = (XmlQueryNodeSequence) seq;
|
|
if (nodeSeq == null)
|
|
nodeSeq = new XmlQueryNodeSequence(seq);
|
|
|
|
return nodeSeq.DocOrderDistinct(this.docOrderCmp);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generate a unique string identifier for the specified node. Do this by asking the navigator for an identifier
|
|
/// that is unique within the document, and then prepend a document index.
|
|
/// </summary>
|
|
public string GenerateId(XPathNavigator navigator) {
|
|
return string.Concat("ID", this.docOrderCmp.GetDocumentIndex(navigator).ToString(CultureInfo.InvariantCulture), navigator.UniqueId);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Indexes
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// If an index having the specified Id has already been created over the "context" document, then return it
|
|
/// in "index" and return true. Otherwise, create a new, empty index and return false.
|
|
/// </summary>
|
|
public bool FindIndex(XPathNavigator context, int indexId, out XmlILIndex index) {
|
|
XPathNavigator navRoot;
|
|
ArrayList docIndexes;
|
|
Debug.Assert(context != null);
|
|
|
|
// Get root of document
|
|
navRoot = context.Clone();
|
|
navRoot.MoveToRoot();
|
|
|
|
// Search pre-existing indexes in order to determine whether the specified index has already been created
|
|
if (this.indexes != null && indexId < this.indexes.Length) {
|
|
docIndexes = (ArrayList) this.indexes[indexId];
|
|
if (docIndexes != null) {
|
|
// Search for an index defined over the specified document
|
|
for (int i = 0; i < docIndexes.Count; i += 2) {
|
|
// If we find a matching document, then return the index saved in the next slot
|
|
if (((XPathNavigator) docIndexes[i]).IsSamePosition(navRoot)) {
|
|
index = (XmlILIndex) docIndexes[i + 1];
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return a new, empty index
|
|
index = new XmlILIndex();
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Add a newly built index over the specified "context" document to the existing collection of indexes.
|
|
/// </summary>
|
|
public void AddNewIndex(XPathNavigator context, int indexId, XmlILIndex index) {
|
|
XPathNavigator navRoot;
|
|
ArrayList docIndexes;
|
|
Debug.Assert(context != null);
|
|
|
|
// Get root of document
|
|
navRoot = context.Clone();
|
|
navRoot.MoveToRoot();
|
|
|
|
// Ensure that a slot exists for the new index
|
|
if (this.indexes == null) {
|
|
this.indexes = new ArrayList[indexId + 4];
|
|
}
|
|
else if (indexId >= this.indexes.Length) {
|
|
// Resize array
|
|
ArrayList[] indexesNew = new ArrayList[indexId + 4];
|
|
Array.Copy(this.indexes, 0, indexesNew, 0, this.indexes.Length);
|
|
this.indexes = indexesNew;
|
|
}
|
|
|
|
docIndexes = (ArrayList) this.indexes[indexId];
|
|
if (docIndexes == null) {
|
|
docIndexes = new ArrayList();
|
|
this.indexes[indexId] = docIndexes;
|
|
}
|
|
|
|
docIndexes.Add(navRoot);
|
|
docIndexes.Add(index);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Output construction
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Get output writer object.
|
|
/// </summary>
|
|
public XmlQueryOutput Output {
|
|
get { return this.output; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Start construction of a nested sequence of items. Return a new XmlQueryOutput that will be
|
|
/// used to construct this new sequence.
|
|
/// </summary>
|
|
public void StartSequenceConstruction(out XmlQueryOutput output) {
|
|
// Push current writer
|
|
this.stkOutput.Push(this.output);
|
|
|
|
// Create new writers
|
|
output = this.output = new XmlQueryOutput(this, new XmlCachedSequenceWriter());
|
|
}
|
|
|
|
/// <summary>
|
|
/// End construction of a nested sequence of items and return the items as an IList<XPathItem>
|
|
/// internal class. Return previous XmlQueryOutput.
|
|
/// </summary>
|
|
public IList<XPathItem> EndSequenceConstruction(out XmlQueryOutput output) {
|
|
IList<XPathItem> seq = ((XmlCachedSequenceWriter) this.output.SequenceWriter).ResultSequence;
|
|
|
|
// Restore previous XmlQueryOutput
|
|
output = this.output = this.stkOutput.Pop();
|
|
|
|
return seq;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Start construction of an Rtf. Return a new XmlQueryOutput object that will be used to construct this Rtf.
|
|
/// </summary>
|
|
public void StartRtfConstruction(string baseUri, out XmlQueryOutput output) {
|
|
// Push current writer
|
|
this.stkOutput.Push(this.output);
|
|
|
|
// Create new XmlQueryOutput over an Rtf writer
|
|
output = this.output = new XmlQueryOutput(this, new XmlEventCache(baseUri, true));
|
|
}
|
|
|
|
/// <summary>
|
|
/// End construction of an Rtf and return it as an RtfNavigator. Return previous XmlQueryOutput object.
|
|
/// </summary>
|
|
public XPathNavigator EndRtfConstruction(out XmlQueryOutput output) {
|
|
XmlEventCache events;
|
|
|
|
events = (XmlEventCache) this.output.Writer;
|
|
|
|
// Restore previous XmlQueryOutput
|
|
output = this.output = this.stkOutput.Pop();
|
|
|
|
// Return Rtf as an RtfNavigator
|
|
events.EndEvents();
|
|
return new RtfTreeNavigator(events, this.nameTableQuery);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Construct a new RtfTextNavigator from the specified "text". This is much more efficient than calling
|
|
/// StartNodeConstruction(), StartRtf(), WriteString(), EndRtf(), and EndNodeConstruction().
|
|
/// </summary>
|
|
public XPathNavigator TextRtfConstruction(string text, string baseUri) {
|
|
return new RtfTextNavigator(text, baseUri);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------
|
|
// Miscellaneous
|
|
//-----------------------------------------------
|
|
|
|
/// <summary>
|
|
/// Report query execution information to event handler.
|
|
/// </summary>
|
|
public void SendMessage(string message) {
|
|
this.ctxt.OnXsltMessageEncountered(message);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Throw an Xml exception having the specified message text.
|
|
/// </summary>
|
|
public void ThrowException(string text) {
|
|
throw new XslTransformException(text);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Position navThis to the same location as navThat.
|
|
/// </summary>
|
|
internal static XPathNavigator SyncToNavigator(XPathNavigator navigatorThis, XPathNavigator navigatorThat) {
|
|
if (navigatorThis == null || !navigatorThis.MoveTo(navigatorThat))
|
|
return navigatorThat.Clone();
|
|
|
|
return navigatorThis;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Function is called in Debug mode on each time context node change.
|
|
/// </summary>
|
|
public static int OnCurrentNodeChanged(XPathNavigator currentNode) {
|
|
IXmlLineInfo lineInfo = currentNode as IXmlLineInfo;
|
|
|
|
// In case of a namespace node, check whether it is inherited or locally defined
|
|
if (lineInfo != null && ! (currentNode.NodeType == XPathNodeType.Namespace && IsInheritedNamespace(currentNode))) {
|
|
OnCurrentNodeChanged2(currentNode.BaseURI, lineInfo.LineNumber, lineInfo.LinePosition);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// 'true' if current Namespace "inherited" from it's parent. Not defined localy.
|
|
private static bool IsInheritedNamespace(XPathNavigator node) {
|
|
Debug.Assert(node.NodeType == XPathNodeType.Namespace);
|
|
XPathNavigator nav = node.Clone();
|
|
if (nav.MoveToParent()) {
|
|
if (nav.MoveToFirstNamespace(XPathNamespaceScope.Local)) {
|
|
do {
|
|
if ((object)nav.LocalName == (object)node.LocalName) {
|
|
return false;
|
|
}
|
|
} while (nav.MoveToNextNamespace(XPathNamespaceScope.Local));
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
private static void OnCurrentNodeChanged2(string baseUri, int lineNumber, int linePosition) {}
|
|
}
|
|
}
|