MessageEncoder.cs 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) Microsoft Corporation. All rights reserved.
  3. //-----------------------------------------------------------------------------
  4. namespace System.ServiceModel.Channels
  5. {
  6. using System;
  7. using System.IO;
  8. using System.Net.Mime;
  9. using System.Runtime.Serialization;
  10. using System.Runtime.Diagnostics;
  11. using System.ServiceModel.Diagnostics;
  12. using System.Runtime;
  13. using System.Threading;
  14. using System.ServiceModel.Diagnostics.Application;
  15. public abstract class MessageEncoder
  16. {
  17. private string traceSourceString;
  18. public abstract string ContentType { get; }
  19. public abstract string MediaType { get; }
  20. public abstract MessageVersion MessageVersion { get; }
  21. public virtual T GetProperty<T>() where T : class
  22. {
  23. if (typeof(T) == typeof(FaultConverter))
  24. {
  25. return (T)(object)FaultConverter.GetDefaultFaultConverter(this.MessageVersion);
  26. }
  27. return null;
  28. }
  29. public Message ReadMessage(Stream stream, int maxSizeOfHeaders)
  30. {
  31. return ReadMessage(stream, maxSizeOfHeaders, null);
  32. }
  33. public abstract Message ReadMessage(Stream stream, int maxSizeOfHeaders, string contentType);
  34. public Message ReadMessage(ArraySegment<byte> buffer, BufferManager bufferManager)
  35. {
  36. Message message = ReadMessage(buffer, bufferManager, null);
  37. return message;
  38. }
  39. public abstract Message ReadMessage(ArraySegment<byte> buffer, BufferManager bufferManager, string contentType);
  40. // used for buffered streaming
  41. internal ArraySegment<byte> BufferMessageStream(Stream stream, BufferManager bufferManager, int maxBufferSize)
  42. {
  43. byte[] buffer = bufferManager.TakeBuffer(ConnectionOrientedTransportDefaults.ConnectionBufferSize);
  44. int offset = 0;
  45. int currentBufferSize = Math.Min(buffer.Length, maxBufferSize);
  46. while (offset < currentBufferSize)
  47. {
  48. int count = stream.Read(buffer, offset, currentBufferSize - offset);
  49. if (count == 0)
  50. {
  51. stream.Close();
  52. break;
  53. }
  54. offset += count;
  55. if (offset == currentBufferSize)
  56. {
  57. if (currentBufferSize >= maxBufferSize)
  58. {
  59. throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(MaxMessageSizeStream.CreateMaxReceivedMessageSizeExceededException(maxBufferSize));
  60. }
  61. currentBufferSize = Math.Min(currentBufferSize * 2, maxBufferSize);
  62. byte[] temp = bufferManager.TakeBuffer(currentBufferSize);
  63. Buffer.BlockCopy(buffer, 0, temp, 0, offset);
  64. bufferManager.ReturnBuffer(buffer);
  65. buffer = temp;
  66. }
  67. }
  68. return new ArraySegment<byte>(buffer, 0, offset);
  69. }
  70. // used for buffered streaming
  71. internal virtual Message ReadMessage(Stream stream, BufferManager bufferManager, int maxBufferSize, string contentType)
  72. {
  73. return ReadMessage(BufferMessageStream(stream, bufferManager, maxBufferSize), bufferManager, contentType);
  74. }
  75. public override string ToString()
  76. {
  77. return ContentType;
  78. }
  79. public abstract void WriteMessage(Message message, Stream stream);
  80. public virtual IAsyncResult BeginWriteMessage(Message message, Stream stream, AsyncCallback callback, object state)
  81. {
  82. return new WriteMessageAsyncResult(message, stream, this, callback, state);
  83. }
  84. public virtual void EndWriteMessage(IAsyncResult result)
  85. {
  86. WriteMessageAsyncResult.End(result);
  87. }
  88. public ArraySegment<byte> WriteMessage(Message message, int maxMessageSize, BufferManager bufferManager)
  89. {
  90. ArraySegment<byte> arraySegment = WriteMessage(message, maxMessageSize, bufferManager, 0);
  91. return arraySegment;
  92. }
  93. public abstract ArraySegment<byte> WriteMessage(Message message, int maxMessageSize,
  94. BufferManager bufferManager, int messageOffset);
  95. public virtual bool IsContentTypeSupported(string contentType)
  96. {
  97. if (contentType == null)
  98. throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("contentType"));
  99. return IsContentTypeSupported(contentType, this.ContentType, this.MediaType);
  100. }
  101. internal bool IsContentTypeSupported(string contentType, string supportedContentType, string supportedMediaType)
  102. {
  103. if (supportedContentType == contentType)
  104. return true;
  105. if (contentType.Length > supportedContentType.Length &&
  106. contentType.StartsWith(supportedContentType, StringComparison.Ordinal) &&
  107. contentType[supportedContentType.Length] == ';')
  108. return true;
  109. // now check case-insensitively
  110. if (contentType.StartsWith(supportedContentType, StringComparison.OrdinalIgnoreCase))
  111. {
  112. if (contentType.Length == supportedContentType.Length)
  113. {
  114. return true;
  115. }
  116. else if (contentType.Length > supportedContentType.Length)
  117. {
  118. char ch = contentType[supportedContentType.Length];
  119. // Linear Whitespace is allowed to appear between the end of one property and the semicolon.
  120. // LWS = [CRLF]? (SP | HT)+
  121. if (ch == ';')
  122. {
  123. return true;
  124. }
  125. // Consume the [CRLF]?
  126. int i = supportedContentType.Length;
  127. if (ch == '\r' && contentType.Length > supportedContentType.Length + 1 && contentType[i + 1] == '\n')
  128. {
  129. i += 2;
  130. ch = contentType[i];
  131. }
  132. // Look for a ';' or nothing after (SP | HT)+
  133. if (ch == ' ' || ch == '\t')
  134. {
  135. i++;
  136. while (i < contentType.Length)
  137. {
  138. ch = contentType[i];
  139. if (ch != ' ' && ch != '\t')
  140. break;
  141. ++i;
  142. }
  143. }
  144. if (ch == ';' || i == contentType.Length)
  145. return true;
  146. }
  147. }
  148. // sometimes we get a contentType that has parameters, but our encoders
  149. // merely expose the base content-type, so we will check a stripped version
  150. try
  151. {
  152. ContentType parsedContentType = new ContentType(contentType);
  153. if (supportedMediaType.Length > 0 && !supportedMediaType.Equals(parsedContentType.MediaType, StringComparison.OrdinalIgnoreCase))
  154. return false;
  155. if (!IsCharSetSupported(parsedContentType.CharSet))
  156. return false;
  157. }
  158. catch (FormatException)
  159. {
  160. // bad content type, so we definitely don't support it!
  161. return false;
  162. }
  163. return true;
  164. }
  165. internal virtual bool IsCharSetSupported(string charset)
  166. {
  167. return false;
  168. }
  169. internal void ThrowIfMismatchedMessageVersion(Message message)
  170. {
  171. if (message.Version != MessageVersion)
  172. {
  173. throw TraceUtility.ThrowHelperError(
  174. new ProtocolException(SR.GetString(SR.EncoderMessageVersionMismatch, message.Version, MessageVersion)),
  175. message);
  176. }
  177. }
  178. internal string GetTraceSourceString()
  179. {
  180. if (this.traceSourceString == null)
  181. {
  182. this.traceSourceString = DiagnosticTraceBase.CreateDefaultSourceString(this);
  183. }
  184. return this.traceSourceString;
  185. }
  186. class WriteMessageAsyncResult : ScheduleActionItemAsyncResult
  187. {
  188. MessageEncoder encoder;
  189. Message message;
  190. Stream stream;
  191. public WriteMessageAsyncResult(Message message, Stream stream, MessageEncoder encoder, AsyncCallback callback, object state)
  192. : base(callback, state)
  193. {
  194. Fx.Assert(encoder != null, "encoder should never be null");
  195. this.encoder = encoder;
  196. this.message = message;
  197. this.stream = stream;
  198. Schedule();
  199. }
  200. protected override void OnDoWork()
  201. {
  202. this.encoder.WriteMessage(this.message, this.stream);
  203. }
  204. }
  205. }
  206. }