MessageQueryTable.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. //------------------------------------------------------------
  2. // Copyright (c) Microsoft Corporation. All rights reserved.
  3. //------------------------------------------------------------
  4. namespace System.ServiceModel.Dispatcher
  5. {
  6. using System;
  7. using System.Collections;
  8. using System.Collections.Generic;
  9. using System.Collections.ObjectModel;
  10. using System.Diagnostics.CodeAnalysis;
  11. using System.Runtime;
  12. using System.ServiceModel.Channels;
  13. public class MessageQueryTable<TItem> : IDictionary<MessageQuery, TItem>
  14. {
  15. Dictionary<Type, MessageQueryCollection> collectionsByType;
  16. Dictionary<MessageQuery, TItem> dictionary;
  17. public MessageQueryTable()
  18. {
  19. this.dictionary = new Dictionary<MessageQuery, TItem>();
  20. this.collectionsByType = new Dictionary<Type, MessageQueryCollection>();
  21. }
  22. public int Count
  23. {
  24. get { return this.dictionary.Count; }
  25. }
  26. public bool IsReadOnly
  27. {
  28. get { return false; }
  29. }
  30. public ICollection<MessageQuery> Keys
  31. {
  32. get { return this.dictionary.Keys; }
  33. }
  34. public ICollection<TItem> Values
  35. {
  36. get { return this.dictionary.Values; }
  37. }
  38. public TItem this[MessageQuery key]
  39. {
  40. get
  41. {
  42. return this.dictionary[key];
  43. }
  44. set
  45. {
  46. this.Add(key, value);
  47. }
  48. }
  49. [SuppressMessage(FxCop.Category.Usage, "CA2301:EmbeddableTypesInContainersRule", MessageId = "collectionsByType", Justification = "No need to support type equivalence here.")]
  50. public void Add(MessageQuery key, TItem value)
  51. {
  52. if (key == null)
  53. {
  54. throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("key");
  55. }
  56. Type queryType = key.GetType();
  57. MessageQueryCollection collection;
  58. if (!this.collectionsByType.TryGetValue(queryType, out collection))
  59. {
  60. collection = key.CreateMessageQueryCollection();
  61. if (collection == null)
  62. {
  63. collection = new SequentialMessageQueryCollection();
  64. }
  65. this.collectionsByType.Add(queryType, collection);
  66. }
  67. collection.Add(key);
  68. this.dictionary.Add(key, value);
  69. }
  70. public void Add(KeyValuePair<MessageQuery, TItem> item)
  71. {
  72. this.Add(item.Key, item.Value);
  73. }
  74. public void Clear()
  75. {
  76. this.collectionsByType.Clear();
  77. this.dictionary.Clear();
  78. }
  79. public bool Contains(KeyValuePair<MessageQuery, TItem> item)
  80. {
  81. return ((ICollection<KeyValuePair<MessageQuery, TItem>>) this.dictionary).Contains(item);
  82. }
  83. public bool ContainsKey(MessageQuery key)
  84. {
  85. return this.dictionary.ContainsKey(key);
  86. }
  87. public void CopyTo(KeyValuePair<MessageQuery, TItem>[] array, int arrayIndex)
  88. {
  89. ((ICollection<KeyValuePair<MessageQuery, TItem>>) this.dictionary).CopyTo(array, arrayIndex);
  90. }
  91. public IEnumerable<KeyValuePair<MessageQuery, TResult>> Evaluate<TResult>(Message message)
  92. {
  93. if (message == null)
  94. {
  95. throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("message");
  96. }
  97. return new MessageEnumerable<TResult>(this, message);
  98. }
  99. public IEnumerable<KeyValuePair<MessageQuery, TResult>> Evaluate<TResult>(MessageBuffer buffer)
  100. {
  101. if (buffer == null)
  102. {
  103. throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("buffer");
  104. }
  105. return new MessageBufferEnumerable<TResult>(this, buffer);
  106. }
  107. public IEnumerator<KeyValuePair<MessageQuery, TItem>> GetEnumerator()
  108. {
  109. return ((ICollection<KeyValuePair<MessageQuery, TItem>>) this.dictionary).GetEnumerator();
  110. }
  111. IEnumerator IEnumerable.GetEnumerator()
  112. {
  113. return this.GetEnumerator();
  114. }
  115. [SuppressMessage(FxCop.Category.Usage, "CA2301:EmbeddableTypesInContainersRule", MessageId = "collectionsByType", Justification = "No need to support type equivalence here.")]
  116. public bool Remove(MessageQuery key)
  117. {
  118. if (this.dictionary.Remove(key))
  119. {
  120. MessageQueryCollection collection;
  121. Type queryType = key.GetType();
  122. collection = this.collectionsByType[queryType];
  123. collection.Remove(key);
  124. if (collection.Count == 0)
  125. {
  126. this.collectionsByType.Remove(queryType);
  127. }
  128. return true;
  129. }
  130. else
  131. {
  132. return false;
  133. }
  134. }
  135. public bool Remove(KeyValuePair<MessageQuery, TItem> item)
  136. {
  137. return this.Remove(item.Key);
  138. }
  139. public bool TryGetValue(MessageQuery key, out TItem value)
  140. {
  141. return this.dictionary.TryGetValue(key, out value);
  142. }
  143. class SequentialMessageQueryCollection : MessageQueryCollection
  144. {
  145. public override IEnumerable<KeyValuePair<MessageQuery, TResult>> Evaluate<TResult>(Message message)
  146. {
  147. return new MessageSequentialResultEnumerable<TResult>(this, message);
  148. }
  149. public override IEnumerable<KeyValuePair<MessageQuery, TResult>> Evaluate<TResult>(MessageBuffer buffer)
  150. {
  151. return new MessageBufferSequentialResultEnumerable<TResult>(this, buffer);
  152. }
  153. abstract class SequentialResultEnumerable<TSource, TResult> : IEnumerable<KeyValuePair<MessageQuery, TResult>>
  154. {
  155. SequentialMessageQueryCollection collection;
  156. TSource source;
  157. public SequentialResultEnumerable(SequentialMessageQueryCollection collection, TSource source)
  158. {
  159. this.collection = collection;
  160. this.source = source;
  161. }
  162. SequentialMessageQueryCollection Collection
  163. {
  164. get
  165. {
  166. return this.collection;
  167. }
  168. }
  169. protected TSource Source
  170. {
  171. get
  172. {
  173. return this.source;
  174. }
  175. }
  176. public IEnumerator<KeyValuePair<MessageQuery, TResult>> GetEnumerator()
  177. {
  178. return new SequentialResultEnumerator(this);
  179. }
  180. IEnumerator IEnumerable.GetEnumerator()
  181. {
  182. return this.GetEnumerator();
  183. }
  184. protected abstract TResult Evaluate(MessageQuery query);
  185. class SequentialResultEnumerator : IEnumerator<KeyValuePair<MessageQuery, TResult>>
  186. {
  187. SequentialResultEnumerable<TSource, TResult> enumerable;
  188. IEnumerator<MessageQuery> queries;
  189. public SequentialResultEnumerator(SequentialResultEnumerable<TSource, TResult> enumerable)
  190. {
  191. this.enumerable = enumerable;
  192. this.queries = enumerable.Collection.GetEnumerator();
  193. }
  194. public KeyValuePair<MessageQuery, TResult> Current
  195. {
  196. get
  197. {
  198. MessageQuery query = queries.Current;
  199. TResult result = enumerable.Evaluate(query);
  200. return new KeyValuePair<MessageQuery, TResult>(query, result);
  201. }
  202. }
  203. object IEnumerator.Current
  204. {
  205. get
  206. {
  207. return this.Current;
  208. }
  209. }
  210. public void Dispose()
  211. {
  212. }
  213. public bool MoveNext()
  214. {
  215. return this.queries.MoveNext();
  216. }
  217. public void Reset()
  218. {
  219. throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotSupportedException());
  220. }
  221. }
  222. }
  223. class MessageSequentialResultEnumerable<TResult> : SequentialResultEnumerable<Message, TResult>
  224. {
  225. public MessageSequentialResultEnumerable(
  226. SequentialMessageQueryCollection collection, Message message)
  227. : base(collection, message)
  228. {
  229. }
  230. protected override TResult Evaluate(MessageQuery query)
  231. {
  232. return query.Evaluate<TResult>(this.Source);
  233. }
  234. }
  235. class MessageBufferSequentialResultEnumerable<TResult> : SequentialResultEnumerable<MessageBuffer, TResult>
  236. {
  237. public MessageBufferSequentialResultEnumerable(
  238. SequentialMessageQueryCollection collection, MessageBuffer buffer)
  239. : base(collection, buffer)
  240. {
  241. }
  242. protected override TResult Evaluate(MessageQuery query)
  243. {
  244. return query.Evaluate<TResult>(this.Source);
  245. }
  246. }
  247. }
  248. abstract class Enumerable<TSource, TResult> : IEnumerable<KeyValuePair<MessageQuery, TResult>>
  249. {
  250. TSource source;
  251. MessageQueryTable<TItem> table;
  252. public Enumerable(MessageQueryTable<TItem> table, TSource source)
  253. {
  254. this.table = table;
  255. this.source = source;
  256. }
  257. protected TSource Source
  258. {
  259. get
  260. {
  261. return this.source;
  262. }
  263. }
  264. public IEnumerator<KeyValuePair<MessageQuery, TResult>> GetEnumerator()
  265. {
  266. return new Enumerator(this);
  267. }
  268. protected abstract IEnumerator<KeyValuePair<MessageQuery, TResult>> GetInnerEnumerator(MessageQueryCollection collection);
  269. IEnumerator IEnumerable.GetEnumerator()
  270. {
  271. return this.GetEnumerator();
  272. }
  273. class Enumerator : IEnumerator<KeyValuePair<MessageQuery, TResult>>
  274. {
  275. Enumerable<TSource, TResult> enumerable;
  276. IEnumerator<KeyValuePair<MessageQuery, TResult>> innerEnumerator;
  277. IEnumerator<MessageQueryCollection> outerEnumerator;
  278. public Enumerator(Enumerable<TSource, TResult> enumerable)
  279. {
  280. this.outerEnumerator = enumerable.table.collectionsByType.Values.GetEnumerator();
  281. this.enumerable = enumerable;
  282. }
  283. public KeyValuePair<MessageQuery, TResult> Current
  284. {
  285. get { return this.innerEnumerator.Current; }
  286. }
  287. object IEnumerator.Current
  288. {
  289. get
  290. {
  291. return this.Current;
  292. }
  293. }
  294. public void Dispose()
  295. {
  296. }
  297. public bool MoveNext()
  298. {
  299. if (innerEnumerator == null || !this.innerEnumerator.MoveNext())
  300. {
  301. if (!this.outerEnumerator.MoveNext())
  302. {
  303. return false;
  304. }
  305. else
  306. {
  307. MessageQueryCollection collection = this.outerEnumerator.Current;
  308. this.innerEnumerator = this.enumerable.GetInnerEnumerator(collection);
  309. return this.innerEnumerator.MoveNext();
  310. }
  311. }
  312. else
  313. {
  314. return true;
  315. }
  316. }
  317. public void Reset()
  318. {
  319. throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotSupportedException());
  320. }
  321. }
  322. }
  323. class MessageBufferEnumerable<TResult> : Enumerable<MessageBuffer, TResult>
  324. {
  325. public MessageBufferEnumerable(MessageQueryTable<TItem> table, MessageBuffer buffer)
  326. : base(table, buffer)
  327. {
  328. }
  329. protected override IEnumerator<KeyValuePair<MessageQuery, TResult>> GetInnerEnumerator(
  330. MessageQueryCollection collection)
  331. {
  332. return collection.Evaluate<TResult>(this.Source).GetEnumerator();
  333. }
  334. }
  335. class MessageEnumerable<TResult> : Enumerable<Message, TResult>
  336. {
  337. public MessageEnumerable(MessageQueryTable<TItem> table, Message message)
  338. : base(table, message)
  339. {
  340. }
  341. protected override IEnumerator<KeyValuePair<MessageQuery, TResult>> GetInnerEnumerator(
  342. MessageQueryCollection collection)
  343. {
  344. return collection.Evaluate<TResult>(this.Source).GetEnumerator();
  345. }
  346. }
  347. }
  348. }