Я нашел ошибку в реализации Microsoft Message.ToString ().
Тогда я выяснил причину и нашел решение.
Message.ToString () может содержать содержимое тела в виде "... stream ...".
Это означает, что Сообщение было создано с использованием XmlRead или XmlDictionaryReader, созданного из потока, который еще не был прочитан.
ToString задокументировано как НЕ изменяющее состояние сообщения.
Таким образом, они не читают поток, просто вставляют маркер, который включен.
Поскольку моей целью было (1) получить строку, (2) изменить строку и (3) создать новое Сообщение из измененной строки, мне нужно было сделать немного больше.
Вот что я придумал:
/// <summary>
/// Get the XML of a Message even if it contains an unread Stream as its Body.
/// <para>message.ToString() would contain "... stream ..." as
/// the Body contents.</para>
/// </summary>
/// <param name="m">A reference to the <c>Message</c>. </param>
/// <returns>A String of the XML after the Message has been fully
/// read and parsed.</returns>
/// <remarks>The Message <paramref cref="m"/> is re-created
/// in its original state.</remarks>
String MessageString(ref Message m)
{
// copy the message into a working buffer.
MessageBuffer mb = m.CreateBufferedCopy(int.MaxValue);
// re-create the original message, because "copy" changes its state.
m = mb.CreateMessage();
Stream s = new MemoryStream();
XmlWriter xw = XmlWriter.Create(s);
mb.CreateMessage().WriteMessage(xw);
xw.Flush();
s.Position = 0;
byte[] bXML = new byte[s.Length];
s.Read(bXML, 0, s.Length);
// sometimes bXML[] starts with a BOM
if (bXML[0] != (byte)'<')
{
return Encoding.UTF8.GetString(bXML,3,bXML.Length-3);
}
else
{
return Encoding.UTF8.GetString(bXML,0,bXML.Length);
}
}
/// <summary>
/// Create an XmlReader from the String containing the XML.
/// </summary>
/// <param name="xml">The XML string o fhe entire SOAP Message.</param>
/// <returns>
/// An XmlReader to a MemoryStream to the <paramref cref="xml"/> string.
/// </returns>
XmlReader XmlReaderFromString(String xml)
{
var stream = new System.IO.MemoryStream();
// NOTE: don't use using(var writer ...){...}
// because the end of the StreamWriter's using closes the Stream itself.
//
var writer = new System.IO.StreamWriter(stream);
writer.Write(xml);
writer.Flush();
stream.Position = 0;
return XmlReader.Create(stream);
}
/// <summary>
/// Creates a Message object from the XML of the entire SOAP message.
/// </summary>
/// <param name="xml">The XML string of the entire SOAP message.</param>
/// <param name="">The MessageVersion constant to pass in
/// to Message.CreateMessage.</param>
/// <returns>
/// A Message that is built from the SOAP <paramref cref="xml"/>.
/// </returns>
Message CreateMessageFromString(String xml, MessageVersion ver)
{
return Message.CreateMessage(XmlReaderFromString(xml), ver);
}
-Jesse