94b2861243
Former-commit-id: 5f9c6ae75f295e057a7d2971f3a6df4656fa8850
976 lines
33 KiB
C#
976 lines
33 KiB
C#
//------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//------------------------------------------------------------
|
|
namespace System.ServiceModel.Dispatcher
|
|
{
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Runtime;
|
|
using System.ServiceModel.Channels;
|
|
using System.ServiceModel.Diagnostics;
|
|
using System.Xml;
|
|
using System.Xml.XPath;
|
|
using System.Xml.Xsl;
|
|
|
|
internal enum QueryCompilerFlags
|
|
{
|
|
None = 0x00000000,
|
|
InverseQuery = 0x00000001
|
|
}
|
|
|
|
internal struct FilterResult
|
|
{
|
|
QueryProcessor processor;
|
|
bool result;
|
|
|
|
internal FilterResult(QueryProcessor processor)
|
|
{
|
|
this.processor = processor;
|
|
this.result = this.processor.Result;
|
|
}
|
|
|
|
internal FilterResult(bool result)
|
|
{
|
|
this.processor = null;
|
|
this.result = result;
|
|
}
|
|
|
|
#if NO
|
|
internal ICollection<MessageFilter> Matches
|
|
{
|
|
get
|
|
{
|
|
return this.processor.ResultSet;
|
|
}
|
|
}
|
|
#endif
|
|
internal QueryProcessor Processor
|
|
{
|
|
get
|
|
{
|
|
return this.processor;
|
|
}
|
|
}
|
|
|
|
internal bool Result
|
|
{
|
|
get
|
|
{
|
|
return this.result;
|
|
}
|
|
}
|
|
|
|
internal MessageFilter GetSingleMatch()
|
|
{
|
|
Collection<MessageFilter> matches = processor.MatchList;
|
|
MessageFilter match;
|
|
switch (matches.Count)
|
|
{
|
|
default:
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new MultipleFilterMatchesException(SR.GetString(SR.FilterMultipleMatches), null, matches));
|
|
|
|
case 0:
|
|
match = null;
|
|
break;
|
|
|
|
case 1:
|
|
match = matches[0];
|
|
break;
|
|
}
|
|
|
|
return match;
|
|
}
|
|
}
|
|
|
|
// XPathResult.GetResultAsString and XPathResult.GetResultAsBoolean,
|
|
// drive knowledge of TResult into the engine.
|
|
internal class QueryResult<TResult> : IEnumerable<KeyValuePair<MessageQuery, TResult>>
|
|
{
|
|
bool evalBody;
|
|
QueryMatcher matcher;
|
|
Message message;
|
|
|
|
internal QueryResult(QueryMatcher matcher, Message message, bool evalBody)
|
|
{
|
|
this.matcher = matcher;
|
|
this.message = message;
|
|
this.evalBody = evalBody;
|
|
}
|
|
|
|
public TResult GetSingleResult()
|
|
{
|
|
QueryProcessor processor = this.matcher.CreateProcessor();
|
|
XPathResult result;
|
|
|
|
try
|
|
{
|
|
processor.Eval(this.matcher.RootOpcode, this.message, this.evalBody);
|
|
}
|
|
catch (XPathNavigatorException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.matcher.RootOpcode), this.message);
|
|
}
|
|
catch (NavigatorInvalidBodyAccessException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.matcher.RootOpcode), this.message);
|
|
}
|
|
finally
|
|
{
|
|
if (this.evalBody)
|
|
{
|
|
this.message.Close();
|
|
}
|
|
|
|
result = processor.QueryResult;
|
|
this.matcher.ReleaseProcessor(processor);
|
|
}
|
|
|
|
if (typeof(TResult) == typeof(XPathResult) || typeof(TResult) == typeof(object))
|
|
{
|
|
return (TResult)(object)result;
|
|
}
|
|
else if (typeof(TResult) == typeof(string))
|
|
{
|
|
return (TResult)(object)result.GetResultAsString();
|
|
}
|
|
else if (typeof(TResult) == typeof(bool))
|
|
{
|
|
return (TResult)(object)result.GetResultAsBoolean();
|
|
}
|
|
else
|
|
{
|
|
throw Fx.AssertAndThrowFatal("unsupported type");
|
|
}
|
|
}
|
|
|
|
public IEnumerator<KeyValuePair<MessageQuery, TResult>> GetEnumerator()
|
|
{
|
|
QueryProcessor processor = this.matcher.CreateProcessor();
|
|
Collection<KeyValuePair<MessageQuery, XPathResult>> results =
|
|
new Collection<KeyValuePair<MessageQuery, XPathResult>>();
|
|
processor.ResultSet = results;
|
|
|
|
try
|
|
{
|
|
processor.Eval(this.matcher.RootOpcode, this.message, this.evalBody);
|
|
|
|
if (typeof(TResult) == typeof(XPathResult))
|
|
{
|
|
return (IEnumerator<KeyValuePair<MessageQuery, TResult>>)(object)results.GetEnumerator();
|
|
}
|
|
else if (typeof(TResult) == typeof(string) ||
|
|
typeof(TResult) == typeof(bool) ||
|
|
typeof(TResult) == typeof(object))
|
|
{
|
|
Collection<KeyValuePair<MessageQuery, TResult>> typedResults =
|
|
new Collection<KeyValuePair<MessageQuery, TResult>>();
|
|
|
|
foreach (var result in results)
|
|
{
|
|
if (typeof(TResult) == typeof(string))
|
|
{
|
|
typedResults.Add(
|
|
new KeyValuePair<MessageQuery, TResult>(
|
|
result.Key, (TResult)(object)result.Value.GetResultAsString()));
|
|
}
|
|
else if (typeof(TResult) == typeof(bool))
|
|
{
|
|
typedResults.Add(
|
|
new KeyValuePair<MessageQuery, TResult>(
|
|
result.Key, (TResult)(object)result.Value.GetResultAsBoolean()));
|
|
}
|
|
else
|
|
{
|
|
typedResults.Add(new KeyValuePair<MessageQuery, TResult>(
|
|
result.Key, (TResult)(object)result.Value));
|
|
}
|
|
}
|
|
|
|
return (IEnumerator<KeyValuePair<MessageQuery, TResult>>)typedResults.GetEnumerator();
|
|
}
|
|
else
|
|
{
|
|
throw Fx.AssertAndThrowFatal("unsupported type");
|
|
}
|
|
}
|
|
catch (XPathNavigatorException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.matcher.RootOpcode), this.message);
|
|
}
|
|
catch (NavigatorInvalidBodyAccessException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.matcher.RootOpcode), this.message);
|
|
}
|
|
finally
|
|
{
|
|
if (this.evalBody)
|
|
{
|
|
this.message.Close();
|
|
}
|
|
|
|
this.matcher.ReleaseProcessor(processor);
|
|
}
|
|
}
|
|
|
|
IEnumerator IEnumerable.GetEnumerator()
|
|
{
|
|
return this.GetEnumerator();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
///
|
|
/// </summary>
|
|
internal abstract class QueryMatcher
|
|
{
|
|
static IFunctionLibrary[] defaultFunctionLibs; // The set of function libraries that our XPath compiler will link to
|
|
static XPathNavigator fxCompiler; // fx compiler
|
|
|
|
protected int maxNodes; // Maximum # of nodes that we will process while performing any individual match
|
|
protected Opcode query; // root opcode - this is where query evaluation starts
|
|
protected int subExprVars; // the number of subexpr node sequences the processing context must hold
|
|
|
|
// Processor Pool
|
|
protected WeakReference processorPool;
|
|
|
|
internal class QueryProcessorPool
|
|
{
|
|
QueryProcessor processor;
|
|
|
|
internal QueryProcessorPool()
|
|
{
|
|
}
|
|
|
|
internal QueryProcessor Pop()
|
|
{
|
|
QueryProcessor p = this.processor;
|
|
if (null != p)
|
|
{
|
|
this.processor = (QueryProcessor)p.next;
|
|
p.next = null;
|
|
return p;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
internal void Push(QueryProcessor p)
|
|
{
|
|
p.next = this.processor;
|
|
this.processor = p;
|
|
}
|
|
}
|
|
|
|
[SuppressMessage("Microsoft.Security.Xml", "CA3057:DoNotUseLoadXml")]
|
|
static QueryMatcher()
|
|
{
|
|
QueryMatcher.defaultFunctionLibs = new IFunctionLibrary[] { new XPathFunctionLibrary() };
|
|
|
|
// For some incomprehensible reason, the Framework XPath compiler requires an instance of an XPath navigator
|
|
// to compile an xpath. This compiler uses a dummy xml document to create a navigator
|
|
XmlDocument doc = new XmlDocument();
|
|
doc.LoadXml("<a/>");
|
|
QueryMatcher.fxCompiler = doc.CreateNavigator();
|
|
}
|
|
|
|
internal QueryMatcher()
|
|
{
|
|
this.maxNodes = int.MaxValue;
|
|
this.query = null;
|
|
this.processorPool = new WeakReference(null);
|
|
this.subExprVars = 0;
|
|
}
|
|
#if NO
|
|
internal QueryMatcher(QueryMatcher matcher)
|
|
{
|
|
this.processorPool = new WeakReference(null);
|
|
this.maxNodes = matcher.maxNodes;
|
|
this.query = matcher.query;
|
|
this.subExprVars = matcher.subExprVars;
|
|
}
|
|
#endif
|
|
internal bool IsCompiled
|
|
{
|
|
get
|
|
{
|
|
return (null != this.query);
|
|
}
|
|
}
|
|
|
|
internal int NodeQuota
|
|
{
|
|
get
|
|
{
|
|
return this.maxNodes;
|
|
}
|
|
set
|
|
{
|
|
Fx.Assert(value > 0, "");
|
|
this.maxNodes = value;
|
|
}
|
|
}
|
|
|
|
internal Opcode RootOpcode
|
|
{
|
|
get
|
|
{
|
|
return this.query;
|
|
}
|
|
}
|
|
|
|
internal int SubExprVarCount
|
|
{
|
|
get
|
|
{
|
|
return this.subExprVars;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compile the given filter to run on an external (fx) xpath engine
|
|
/// </summary>
|
|
internal static OpcodeBlock CompileForExternalEngine(string expression, XmlNamespaceManager namespaces, object item, bool match)
|
|
{
|
|
// Compile...
|
|
XPathExpression xpathExpr = QueryMatcher.fxCompiler.Compile(expression);
|
|
|
|
// Fx will bind prefixes and functions here.
|
|
if (namespaces != null)
|
|
{
|
|
// There's a bug in System.Xml.XPath. If we pass an XsltContext to SetContext it won't throw if there's
|
|
// an undefined prefix.
|
|
if (namespaces is XsltContext)
|
|
{
|
|
// Lex the xpath to find all prefixes used
|
|
XPathLexer lexer = new XPathLexer(expression, false);
|
|
while (lexer.MoveNext())
|
|
{
|
|
string prefix = lexer.Token.Prefix;
|
|
|
|
if (prefix.Length > 0 && namespaces.LookupNamespace(prefix) == null)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new XsltException(SR.GetString(SR.FilterUndefinedPrefix, prefix)));
|
|
}
|
|
}
|
|
}
|
|
|
|
xpathExpr.SetContext(namespaces);
|
|
}
|
|
|
|
//
|
|
// FORCE the function to COMPILE - they won't bind namespaces unless we check the return type
|
|
//
|
|
if (XPathResultType.Error == xpathExpr.ReturnType)
|
|
{
|
|
// This should never be reached. The above property should throw if there's an error
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new XPathException(SR.GetString(SR.FilterCouldNotCompile, expression)));
|
|
}
|
|
|
|
OpcodeBlock codeBlock = new OpcodeBlock();
|
|
SingleFxEngineResultOpcode op;
|
|
|
|
if (!match)
|
|
{
|
|
op = new QuerySingleFxEngineResultOpcode();
|
|
}
|
|
else
|
|
{
|
|
op = new MatchSingleFxEngineResultOpcode();
|
|
}
|
|
|
|
op.XPath = xpathExpr;
|
|
op.Item = item;
|
|
|
|
codeBlock.Append(op);
|
|
return codeBlock;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compile the given filter for evaluation using the internal engine.
|
|
/// </summary>
|
|
/// <param name="flags">Caller customizes optimizations via the flags parameter</param>
|
|
/// <param name="returnType">Every xpath expression has a return type</param>
|
|
/// <returns>The opcode block we execute to evaluate</returns>
|
|
internal static OpcodeBlock CompileForInternalEngine(XPathMessageFilter filter, QueryCompilerFlags flags, IFunctionLibrary[] functionLibs, out ValueDataType returnType)
|
|
{
|
|
return QueryMatcher.CompileForInternalEngine(filter.XPath.Trim(), filter.namespaces, flags, functionLibs, out returnType);
|
|
}
|
|
|
|
internal static OpcodeBlock CompileForInternalEngine(string xpath, XmlNamespaceManager nsManager, QueryCompilerFlags flags, IFunctionLibrary[] functionLibs, out ValueDataType returnType)
|
|
{
|
|
OpcodeBlock codeBlock;
|
|
|
|
returnType = ValueDataType.None;
|
|
if (0 == xpath.Length)
|
|
{
|
|
// 0 length XPaths always match
|
|
codeBlock = new OpcodeBlock();
|
|
codeBlock.Append(new PushBooleanOpcode(true)); // Always match by pushing true on the eval stack
|
|
}
|
|
else
|
|
{
|
|
// Try to parse the xpath. Bind to default function libraries
|
|
// The parser returns an expression tree
|
|
XPathParser parser = new XPathParser(xpath, nsManager, functionLibs);
|
|
XPathExpr parseTree = parser.Parse();
|
|
|
|
if (null == parseTree)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new QueryCompileException(QueryCompileError.CouldNotParseExpression));
|
|
}
|
|
|
|
returnType = parseTree.ReturnType;
|
|
|
|
// Compile the expression tree
|
|
XPathCompiler compiler = new XPathCompiler(flags);
|
|
|
|
codeBlock = compiler.Compile(parseTree);
|
|
}
|
|
|
|
return codeBlock;
|
|
}
|
|
|
|
internal static OpcodeBlock CompileForInternalEngine(string xpath, XmlNamespaceManager ns, QueryCompilerFlags flags, out ValueDataType returnType)
|
|
{
|
|
return QueryMatcher.CompileForInternalEngine(xpath, ns, flags, QueryMatcher.defaultFunctionLibs, out returnType);
|
|
}
|
|
|
|
internal SeekableXPathNavigator CreateMessageNavigator(Message message, bool matchBody)
|
|
{
|
|
SeekableXPathNavigator nav = message.GetNavigator(matchBody, this.maxNodes);
|
|
|
|
// Position the navigator at the root element
|
|
// This allows a caller to run relative XPaths on message
|
|
nav.MoveToRoot();
|
|
return nav;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks the context pool for a generic navigator first. If none is available, creates a new one
|
|
/// </summary>
|
|
internal SeekableXPathNavigator CreateSeekableNavigator(XPathNavigator navigator)
|
|
{
|
|
return new GenericSeekableNavigator(navigator);
|
|
}
|
|
|
|
internal SeekableXPathNavigator CreateSafeNavigator(SeekableXPathNavigator navigator)
|
|
{
|
|
INodeCounter counter = navigator as INodeCounter;
|
|
if (counter != null)
|
|
{
|
|
counter.CounterMarker = this.maxNodes;
|
|
counter.MaxCounter = this.maxNodes;
|
|
}
|
|
else
|
|
{
|
|
navigator = new SafeSeekableNavigator(navigator, this.maxNodes);
|
|
}
|
|
return navigator;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks the context pool for a processor first. If none is available, creates a new one
|
|
/// </summary>
|
|
internal QueryProcessor CreateProcessor()
|
|
{
|
|
QueryProcessor p = null;
|
|
|
|
lock (this.processorPool)
|
|
{
|
|
QueryProcessorPool pool = this.processorPool.Target as QueryProcessorPool;
|
|
if (null != pool)
|
|
{
|
|
p = pool.Pop();
|
|
}
|
|
}
|
|
|
|
if (null != p)
|
|
{
|
|
p.ClearProcessor();
|
|
}
|
|
else
|
|
{
|
|
p = new QueryProcessor(this);
|
|
}
|
|
|
|
p.AddRef();
|
|
return p;
|
|
}
|
|
|
|
internal FilterResult Match(MessageBuffer messageBuffer, ICollection<MessageFilter> matches)
|
|
{
|
|
Message message = messageBuffer.CreateMessage();
|
|
FilterResult result;
|
|
try
|
|
{
|
|
result = this.Match(message, true, matches);
|
|
}
|
|
finally
|
|
{
|
|
message.Close();
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
internal FilterResult Match(Message message, bool matchBody, ICollection<MessageFilter> matches)
|
|
{
|
|
QueryProcessor processor = this.CreateProcessor();
|
|
processor.MatchSet = matches;
|
|
processor.EnsureFilterCollection();
|
|
try
|
|
{
|
|
processor.Eval(this.query, message, matchBody);
|
|
}
|
|
catch (XPathNavigatorException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.query), message);
|
|
}
|
|
catch (NavigatorInvalidBodyAccessException e)
|
|
{
|
|
throw TraceUtility.ThrowHelperError(e.Process(this.query), message);
|
|
}
|
|
|
|
return new FilterResult(processor);
|
|
}
|
|
|
|
internal QueryResult<TResult> Evaluate<TResult>(MessageBuffer messageBuffer)
|
|
{
|
|
Message message = messageBuffer.CreateMessage();
|
|
return this.Evaluate<TResult>(message, true);
|
|
}
|
|
|
|
internal QueryResult<TResult> Evaluate<TResult>(Message message, bool matchBody)
|
|
{
|
|
return new QueryResult<TResult>(this, message, matchBody);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Execute matches over the given seekable navigator. If the navigator is not safe, wrap it with one that is
|
|
/// </summary>
|
|
internal FilterResult Match(SeekableXPathNavigator navigator, ICollection<MessageFilter> matches)
|
|
{
|
|
// If the matcher places restrictions on the # of nodes we will inspect, and the navigator passed does
|
|
// not do any nodecounting itself, we must make that navigator safe by wrapping it
|
|
if (this.maxNodes < int.MaxValue)
|
|
{
|
|
navigator = this.CreateSafeNavigator(navigator);
|
|
}
|
|
|
|
QueryProcessor processor = this.CreateProcessor();
|
|
processor.MatchSet = matches;
|
|
processor.EnsureFilterCollection();
|
|
try
|
|
{
|
|
processor.Eval(this.query, navigator);
|
|
}
|
|
catch (XPathNavigatorException e)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(e.Process(this.query));
|
|
}
|
|
catch (NavigatorInvalidBodyAccessException e)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(e.Process(this.query));
|
|
}
|
|
|
|
return new FilterResult(processor);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Execute matches over the given navigator by wrapping it with a Seekable Navigator
|
|
/// </summary>
|
|
internal FilterResult Match(XPathNavigator navigator, ICollection<MessageFilter> matches)
|
|
{
|
|
SeekableXPathNavigator nav = this.CreateSeekableNavigator(navigator);
|
|
return this.Match(nav, matches);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Release the given processor and place it back in the context pool
|
|
/// </summary>
|
|
internal void ReleaseProcessor(QueryProcessor processor)
|
|
{
|
|
if (!processor.ReleaseRef())
|
|
{
|
|
return;
|
|
}
|
|
|
|
lock (this.processorPool)
|
|
{
|
|
QueryProcessorPool pool = this.processorPool.Target as QueryProcessorPool;
|
|
if (null == pool)
|
|
{
|
|
pool = new QueryProcessorPool();
|
|
this.processorPool.Target = pool;
|
|
}
|
|
pool.Push(processor);
|
|
}
|
|
}
|
|
|
|
internal void ReleaseResult(FilterResult result)
|
|
{
|
|
if (null != result.Processor)
|
|
{
|
|
result.Processor.MatchSet = null;
|
|
this.ReleaseProcessor(result.Processor);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Trim all pool
|
|
/// </summary>
|
|
internal virtual void Trim()
|
|
{
|
|
if (this.query != null)
|
|
{
|
|
this.query.Trim();
|
|
}
|
|
}
|
|
}
|
|
|
|
internal enum XPathFilterFlags
|
|
{
|
|
None = 0x00,
|
|
AlwaysMatch = 0x01, // filter always matches
|
|
IsFxFilter = 0x02, // filter is matched using the framework engine
|
|
}
|
|
|
|
/// <summary>
|
|
/// A matcher used to evalute single XPath expressions
|
|
/// </summary>
|
|
internal class XPathQueryMatcher : QueryMatcher
|
|
{
|
|
XPathFilterFlags flags;
|
|
bool match;
|
|
static PushBooleanOpcode matchAlwaysFilter; // used for compiling xpaths that always match - i.e. xpath.Length == 0
|
|
static OpcodeBlock rootFilter; // used for compiling "/"
|
|
|
|
static XPathQueryMatcher()
|
|
{
|
|
XPathQueryMatcher.matchAlwaysFilter = new PushBooleanOpcode(true); //dummy
|
|
|
|
ValueDataType returnType;
|
|
XPathQueryMatcher.rootFilter = QueryMatcher.CompileForInternalEngine("/", null, QueryCompilerFlags.None, out returnType);
|
|
XPathQueryMatcher.rootFilter.Append(new MatchResultOpcode());
|
|
}
|
|
|
|
internal XPathQueryMatcher(bool match)
|
|
: base()
|
|
{
|
|
this.flags = XPathFilterFlags.None;
|
|
this.match = match;
|
|
}
|
|
#if NO
|
|
internal XPathFilterMatcher(XPathFilterMatcher matcher)
|
|
: base(matcher)
|
|
{
|
|
this.flags = matcher.flags;
|
|
}
|
|
#endif
|
|
internal bool IsAlwaysMatch
|
|
{
|
|
get
|
|
{
|
|
return (0 != (this.flags & XPathFilterFlags.AlwaysMatch));
|
|
}
|
|
}
|
|
|
|
internal bool IsFxFilter
|
|
{
|
|
get
|
|
{
|
|
return (0 != (this.flags & XPathFilterFlags.IsFxFilter));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// If the xpath is an empty string, there is nothing to compile and the filter always matches
|
|
/// If not, try to compile the filter for execution within the filter engine's own query processor
|
|
/// If that query processor cannot accept the filter (it doesn't fall within the class of xpaths it can handle),
|
|
/// then revert to the fall-back solution - the slower Fx engine
|
|
/// </summary>
|
|
internal void Compile(string expression, XmlNamespaceManager namespaces)
|
|
{
|
|
if (null == this.query)
|
|
{
|
|
// Try to compile for the internal engine first
|
|
try
|
|
{
|
|
this.CompileForInternal(expression, namespaces);
|
|
}
|
|
catch (QueryCompileException)
|
|
{
|
|
}
|
|
if (null == this.query)
|
|
{
|
|
// Try for an external engine that might work..
|
|
this.CompileForExternal(expression, namespaces);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compile this xpath to run on an external (fx) xpath engine
|
|
/// </summary>
|
|
internal void CompileForExternal(string xpath, XmlNamespaceManager names)
|
|
{
|
|
Opcode op = QueryMatcher.CompileForExternalEngine(xpath, names, null, this.match).First;
|
|
this.query = op;
|
|
this.flags |= XPathFilterFlags.IsFxFilter;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compile for the internal engine with default flags
|
|
/// By defalt, we compile an xpath to run stand alone, with standard optimizations
|
|
/// </summary>
|
|
internal void CompileForInternal(string xpath, XmlNamespaceManager names)
|
|
{
|
|
this.query = null;
|
|
xpath = xpath.Trim();
|
|
|
|
if (0 == xpath.Length)
|
|
{
|
|
// Empty xpaths always match. Same for xpaths that refer to the root only
|
|
// We will evaluate such filters with minimal overhead. However, we
|
|
// don't want a null value for this.query, so we stick a dummy value in there
|
|
this.query = XPathQueryMatcher.matchAlwaysFilter;
|
|
this.flags |= (XPathFilterFlags.AlwaysMatch);
|
|
}
|
|
else if (1 == xpath.Length && '/' == xpath[0])
|
|
{
|
|
this.query = XPathQueryMatcher.rootFilter.First;
|
|
this.flags |= (XPathFilterFlags.AlwaysMatch);
|
|
}
|
|
else
|
|
{
|
|
ValueDataType returnType;
|
|
OpcodeBlock codeBlock = QueryMatcher.CompileForInternalEngine(xpath, names, QueryCompilerFlags.None, out returnType);
|
|
// Inject a final opcode that will place the query result on the query context
|
|
// This query is now ready for execution STAND ALONE
|
|
if (this.match)
|
|
{
|
|
codeBlock.Append(new MatchResultOpcode());
|
|
}
|
|
else
|
|
{
|
|
codeBlock.Append(new QueryResultOpcode());
|
|
}
|
|
|
|
this.query = codeBlock.First;
|
|
}
|
|
|
|
this.flags &= ~XPathFilterFlags.IsFxFilter;
|
|
}
|
|
|
|
internal FilterResult Match(MessageBuffer messageBuffer)
|
|
{
|
|
Message message = messageBuffer.CreateMessage();
|
|
FilterResult result;
|
|
|
|
try
|
|
{
|
|
result = this.Match(message, true);
|
|
}
|
|
finally
|
|
{
|
|
message.Close();
|
|
}
|
|
return result;
|
|
}
|
|
|
|
internal FilterResult Match(Message message, bool matchBody)
|
|
{
|
|
if (this.IsAlwaysMatch)
|
|
{
|
|
// No need to do any expensive query evaluation if we know that the query will always match
|
|
return new FilterResult(true);
|
|
}
|
|
|
|
return base.Match(message, matchBody, null);
|
|
}
|
|
|
|
internal FilterResult Match(SeekableXPathNavigator navigator)
|
|
{
|
|
if (this.IsAlwaysMatch)
|
|
{
|
|
// No need to do any expensive query evaluation if we know that the query will always match
|
|
return new FilterResult(true);
|
|
}
|
|
|
|
// Is it a filter that we will evaluate using the framework engine?
|
|
// We can evaluate that without having to allocate a query processor
|
|
if (this.IsFxFilter)
|
|
{
|
|
return new FilterResult(this.MatchFx(navigator));
|
|
}
|
|
|
|
return base.Match(navigator, null);
|
|
}
|
|
|
|
internal FilterResult Match(XPathNavigator navigator)
|
|
{
|
|
Fx.Assert(null != this.query, "");
|
|
if (this.IsAlwaysMatch)
|
|
{
|
|
return new FilterResult(true);
|
|
}
|
|
// Is it a filter that we will evaluate using the framework engine?
|
|
// We can evaluate that without having to allocate a query processor
|
|
if (this.IsFxFilter)
|
|
{
|
|
return new FilterResult(this.MatchFx(navigator));
|
|
}
|
|
|
|
return base.Match(navigator, null);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Evaluates the filter over infosets surfaced via the given navigator by using the Fx engine
|
|
/// We assume that the filter was pre-compiled using the framework engine
|
|
/// </summary>
|
|
internal bool MatchFx(XPathNavigator navigator)
|
|
{
|
|
INodeCounter counter = navigator as INodeCounter;
|
|
if (counter == null)
|
|
{
|
|
navigator = new SafeSeekableNavigator(new GenericSeekableNavigator(navigator), this.NodeQuota);
|
|
}
|
|
else
|
|
{
|
|
counter.CounterMarker = this.NodeQuota;
|
|
counter.MaxCounter = this.NodeQuota;
|
|
}
|
|
Fx.Assert(null != this.query && OpcodeID.MatchSingleFx == this.query.ID, "");
|
|
try
|
|
{
|
|
return ((MatchSingleFxEngineResultOpcode)this.query).Match(navigator);
|
|
}
|
|
catch (XPathNavigatorException e)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(e.Process(this.query));
|
|
}
|
|
catch (NavigatorInvalidBodyAccessException e)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(e.Process(this.query));
|
|
}
|
|
}
|
|
}
|
|
|
|
internal class InverseQueryMatcher : QueryMatcher
|
|
{
|
|
SubExprEliminator elim;
|
|
Dictionary<object, Opcode> lastLookup;
|
|
bool match;
|
|
|
|
internal InverseQueryMatcher(bool match)
|
|
: base()
|
|
{
|
|
this.elim = new SubExprEliminator();
|
|
this.lastLookup = new Dictionary<object, Opcode>();
|
|
this.match = match;
|
|
}
|
|
|
|
internal void Add(string expression, XmlNamespaceManager names, object item, bool forceExternal)
|
|
{
|
|
Fx.Assert(null != item, "");
|
|
|
|
// Compile the new filter
|
|
|
|
bool compiled = false;
|
|
OpcodeBlock codeBlock = new OpcodeBlock();
|
|
|
|
codeBlock.Append(new NoOpOpcode(OpcodeID.QueryTree));
|
|
if (!forceExternal)
|
|
{
|
|
try
|
|
{
|
|
ValueDataType returnType = ValueDataType.None;
|
|
|
|
// Try to compile and merge the compiled query into the query tree
|
|
codeBlock.Append(QueryMatcher.CompileForInternalEngine(expression, names, QueryCompilerFlags.InverseQuery, out returnType));
|
|
|
|
MultipleResultOpcode opcode;
|
|
|
|
if (!this.match)
|
|
{
|
|
opcode = new QueryMultipleResultOpcode();
|
|
}
|
|
else
|
|
{
|
|
opcode = new MatchMultipleResultOpcode();
|
|
}
|
|
|
|
opcode.AddItem(item);
|
|
codeBlock.Append(opcode);
|
|
compiled = true;
|
|
|
|
// Perform SubExpression Elimination
|
|
codeBlock = new OpcodeBlock(this.elim.Add(item, codeBlock.First));
|
|
this.subExprVars = this.elim.VariableCount;
|
|
}
|
|
catch (QueryCompileException)
|
|
{
|
|
// If the filter couldn't be compiled, we drop down to the framework engine
|
|
}
|
|
}
|
|
|
|
if (!compiled)
|
|
{
|
|
codeBlock.Append(QueryMatcher.CompileForExternalEngine(expression, names, item, this.match));
|
|
}
|
|
|
|
// Merge the compiled query into the query tree
|
|
QueryTreeBuilder builder = new QueryTreeBuilder();
|
|
this.query = builder.Build(this.query, codeBlock);
|
|
// To de-merge this filter from the tree, we'll have to walk backwards up the tree... so we
|
|
// have to remember the last opcode that is executed on behalf of this filter
|
|
this.lastLookup[item] = builder.LastOpcode;
|
|
}
|
|
|
|
internal void Clear()
|
|
{
|
|
foreach (object item in this.lastLookup.Keys)
|
|
{
|
|
this.Remove(this.lastLookup[item], item);
|
|
this.elim.Remove(item);
|
|
}
|
|
this.subExprVars = this.elim.VariableCount;
|
|
this.lastLookup.Clear();
|
|
}
|
|
|
|
internal void Remove(object item)
|
|
{
|
|
Fx.Assert(this.lastLookup.ContainsKey(item), "");
|
|
|
|
this.Remove(this.lastLookup[item], item);
|
|
this.lastLookup.Remove(item);
|
|
|
|
// Remove filter from subexpr eliminator
|
|
this.elim.Remove(item);
|
|
this.subExprVars = this.elim.VariableCount;
|
|
}
|
|
|
|
void Remove(Opcode opcode, object item)
|
|
{
|
|
MultipleResultOpcode multiOpcode = opcode as MultipleResultOpcode;
|
|
|
|
if (multiOpcode != null)
|
|
{
|
|
multiOpcode.RemoveItem(item);
|
|
}
|
|
else
|
|
{
|
|
opcode.Remove();
|
|
}
|
|
}
|
|
|
|
internal override void Trim()
|
|
{
|
|
base.Trim();
|
|
elim.Trim();
|
|
}
|
|
}
|
|
}
|