Читать WCF MSMQ сообщение из C#
Я использовал клиент WCF для отправки сообщения msmq в службу WCF, прежде чем служба WCF обработает сообщение msmq, я хочу проверить текст сообщения.
После проверки содержимого тела сообщения msmq я получил результат ниже.
Но мне не удалось получить точную строку.
Ниже приведено определение службы WCF
[ServiceContract]
public interface IMSMQService
{
[OperationContract(IsOneWay = true)]
void ShowMessage(string msg);
}
Я использовал метод ниже, чтобы получить тело сообщения, и он выводит пустой, если я добавлю наблюдение на мс, я мог бы получить
"\0\u0001\0\u0001\u0004\u0002(net.msmq:// VDI-v-tazho / частный / TestQueue \ u0003 \ аУ \u0002\v\u0001s\u0004\v\u0001a\u0006V\ Bd \n\u001e\0 + http://tempuri.org/IMSMQService/ShowMessage@\ u0017VsDebuggerCausalityData \ bAhttp: //schemas.microsoft.com/vstudio/diagnostics/servicemodelsink< ϣ-lN FoJ 0 и \ u0006 \ "\ 0 \ 0 \ 0 \ 0 \ u0017 \ 0i8CI \ 7 ^ QA \ u0012w} \ фА \ u000f \ г ޮ ре \ 0 \ т \ 0 \ 0D \ е \ u001e \ 0 (net.msmq: // VDI-v-tazho / частный / TestQueue \ u0001V \ u000e @ \ vShowMessage \ Ь \ u0013http: //tempuri.org/@ \ u0003msg \ u0004test \ u0001 \ u0001 \ u0001"
//message.Formatter = new XmlMessageFormatter(new String[] { });
//StreamReader sr = new StreamReader(message.BodyStream);
string ms = "";
//while (sr.Peek() >= 0)
//{
// ms += sr.ReadLine();
//}
message.Formatter = new ActiveXMessageFormatter();
string result = System.Text.Encoding.UTF8.GetString(message.Body as byte[]);
StreamReader reader = new StreamReader(message.BodyStream);
ms = reader.ReadToEnd();
MessageBox.Show(ms);
3 ответа
Я получил код ниже, чтобы выполнить мое требование.
private void MSMQStringBody_Click(object sender, EventArgs e)
{
System.Messaging.MessageQueue[] queueList = System.Messaging.MessageQueue.GetPrivateQueuesByMachine(Environment.MachineName);
MessageQueue myQueue = queueList[1];
List<System.Messaging.Message> messages = myQueue.GetAllMessages().ToList();
foreach (System.Messaging.Message message in messages)
{
System.Xml.XmlDocument result = ConvertToXMLDoc(message);
MessageBox.Show(result.InnerText);
}
}
public System.Xml.XmlDocument ConvertToXMLDoc(System.Messaging.Message msg)
{
byte[] buffer = new byte[msg.BodyStream.Length];
msg.BodyStream.Read(buffer, 0, (int)msg.BodyStream.Length);
int envelopeStart = FindEnvolopeStart(buffer);
System.IO.MemoryStream stream = new System.IO.MemoryStream(buffer, envelopeStart, buffer.Length - envelopeStart);
System.ServiceModel.Channels.BinaryMessageEncodingBindingElement elm = new System.ServiceModel.Channels.BinaryMessageEncodingBindingElement();
System.ServiceModel.Channels.Message msg1 = elm.CreateMessageEncoderFactory().Encoder.ReadMessage(stream, Int32.MaxValue);
System.Xml.XmlDocument doc = new System.Xml.XmlDocument();
doc.Load(msg1.GetReaderAtBodyContents());
msg.BodyStream.Position = 0;
return doc;
}
private int FindEnvolopeStart(byte[] stream)
{
int i = 0;
byte prevByte = stream[i];
byte curByte = (byte)0;
for (i = 0; i < stream.Length; i++)
{
curByte = stream[i];
if (curByte == (byte)0x02 &&
prevByte == (byte)0x56)
break;
prevByte = curByte;
}
return i - 1;
}
Когда вы отправляете / извлекаете тело сообщения в / из MSMQ, вы должны использовать тот же "тип". см. универсальный тип "T" ниже. T может быть любым вашим классом.
public T GetQueueMessage()
{
Message message = queue.Receive(new TimeSpan(0, 0, 0, 1, 0));
return (T)message.Body;
}
public void InsertQueueMessage(T message)
{
using (Message msg = new Message((object)message))
{
queue.Send(msg, MessageQueueTransactionType.Single);
}
}
Я знаю, что вопрос заключается в коде C#, но я включаю здесь скрипт Powershell для тех из нас, кому нравится быстро проверять содержимое MSMQ с помощью оболочки. Используя код Эдварда, я также сделал эту работу с Powershell, если кто-то ищет сценарий, который может извлечь эти сообщения WCF, которые отправляются в качестве полезной нагрузки в службу WCF с конечными точками, используя привязку протокола NetMsmqbinding или NetMsmqIntegrationBinding.
Вместо использования Int32.MaxValue я использовал другое большое целочисленное значение, которое, как я знаю, будет достаточным в большинстве случаев. Кстати, было бы неплохо узнать логику, лежащую в основе поиска Soap Envelope.
Вот скрипт Powershell:
[System.Reflection.Assembly]::LoadWithPartialName("System.Messaging") | Out-Null
[System.Reflection.Assembly]::LoadWithPartialName("System.Xml") | Out-Null
[System.Reflection.Assembly]::LoadWithPartialName("System.ServiceModel") | Out-Null
[System.Reflection.Assembly]::LoadWithPartialName("System.IO") | Out-Null
$queuePath = ".\private$\demoqueue4"
Write-Host "Powershell MSMQ queue WCF inspector v0.1. Inspecting queue contents of the queue: $queuePath"
Write-Host ""
Run-MainDemoIterateMsmq $queuePath
Function Get-XmlFromWcfMessage([System.Messaging.Message] $msg) {
$doc = New-Object System.Xml.XmlDocument;
$messageLength = [int] $msg.BodyStream.Length
$buffer = New-Object byte[] $messageLength
$msg.BodyStream.Read($buffer, 0, $messageLength)
$envelopeStart = Find-SoapEnvelopeStart($buffer)
$envelopeStart = $envelopeStart - 0
$envelopeLength = $($buffer.Length - $envelopeStart)
#Write-Host $envelopeStart
$stream = New-Object System.IO.MemoryStream($buffer, $envelopeStart, $envelopeLength)
$elm = New-Object System.ServiceModel.Channels.BinaryMessageEncodingBindingElement
$elm.ReaderQuotas.MaxStringContentLength = 10000000
$elm.ReaderQuotas.MaxBytesPerRead = 10000000
$msg1 = $elm.CreateMessageEncoderFactory().Encoder.ReadMessage($stream, 10000000);
$doc.Load($msg1.GetReaderAtBodyContents());
$msg.BodyStream.Position = 0;
return $doc;
}
Function Find-SoapEnvelopeStart([byte[]] $stream)
{
$i = 0;
$j = 0;
$prevByte = $stream[$i];
$curByte = [byte]$j;
for ($i = 0; $i -lt $stream.Length; $i++)
{
$curByte = $stream[$i];
if ($curByte -eq [byte] 0x02 -and $prevByte -eq [byte] 0x56) {
break;
}
$prevByte = $curByte;
}
return $i - 1;
}
Function Run-MainDemoIterateMsmq([string] $queuePath) {
$queue = New-Object System.Messaging.MessageQueue $queuePath
foreach ($message in $queue.GetAllMessages()){
$xmlDoc = Get-XmlFromWcfMessage $message
Write-Host $xmlDoc.OuterXml
}
}
Вывод может выглядеть так, например:
<SendMessage xmlns="http://tempuri.org/"><message>this is another test!</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>why hello srmp</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>test</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>my message to msmq</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>This is a new message!</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>Another message to MSMQ! </message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>another test here</message></SendMessage>
<SendMessage xmlns="http://tempuri.org/"><message>This is a test message that will be sent using NetMsmqBinding. </message></SendMessage>
<SendMessageDataContract xmlns="http://tempuri.org/"><message xmlns:b="http://schemas.datacontract.org/2004/07/WcfDemoNetMsmqBinding.Host" xmlns:i="http://www.w3.org/2001
/XMLSchema-instance"><b:BoolValue>true</b:BoolValue><b:StringValue>This is a test of a stringvalue for a CompositeType</b:StringValue></message></SendMessageDataContract>