/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ namespace Apache.Qpid.Test.Channel.WcfPerftest { using System; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Configuration; using System.Diagnostics; using System.IO; using System.ServiceModel; using System.ServiceModel.Channels; using System.ServiceModel.Description; using System.Threading; using System.Transactions; using System.Text; using System.Xml; using Apache.Qpid.AmqpTypes; using Apache.Qpid.Channel; // this program implements a subset of the functionality in qpid\cpp\src\tests\perftest.cpp // for a given broker, create reader and writer channels to queues/exchanges // lazilly creates binding and channel factories public class QueueChannelFactory { private static AmqpBinding brokerBinding; private static BindingParameterCollection bindingParameters; private static IChannelFactory readerFactory; private static IChannelFactory writerFactory; private static string brokerAddr = "127.0.0.1"; private static int brokerPort = 5672; private static string userName; private static string password; private static bool ssl = false; public static void SetBroker(string addr, int port) { brokerAddr = addr; brokerPort = port; } public static void SetSecurity(bool sslMode, string name, string pass) { ssl = sslMode; if (name != null) { userName = name; password = pass; } } private static void InitializeBinding() { AmqpBinaryBinding binding = new AmqpBinaryBinding(); bindingParameters = new BindingParameterCollection(); binding.BrokerHost = brokerAddr; binding.BrokerPort = brokerPort; binding.TransferMode = TransferMode.Streamed; binding.PrefetchLimit = 5000; binding.Shared = true; if (ssl || (userName != null)) { binding.Security.Mode = AmqpSecurityMode.Transport; binding.Security.Transport.UseSSL = ssl; if (userName != null) { binding.Security.Transport.CredentialType = AmqpCredentialType.Plain; ClientCredentials credentials = new ClientCredentials(); credentials.UserName.UserName = userName; credentials.UserName.Password = password; bindingParameters.Add(credentials); } } brokerBinding = binding; } public static IInputChannel CreateReaderChannel(string queueName) { lock (typeof(QueueChannelFactory)) { if (brokerBinding == null) { InitializeBinding(); } if (readerFactory == null) { readerFactory = brokerBinding.BuildChannelFactory(bindingParameters); readerFactory.Open(); } IInputChannel channel = readerFactory.CreateChannel(new EndpointAddress( new Uri("amqp:" + queueName))); channel.Open(); return channel; } } public static IOutputChannel CreateWriterChannel(string exchangeName, string routingKey) { lock (typeof(QueueChannelFactory)) { if (brokerBinding == null) { InitializeBinding(); } if (writerFactory == null) { writerFactory = brokerBinding.BuildChannelFactory(bindingParameters); writerFactory.Open(); } IOutputChannel channel = writerFactory.CreateChannel(new EndpointAddress( "amqp:" + exchangeName + "?routingkey=" + routingKey)); channel.Open(); return channel; } } } public enum ClientType { Publisher, Subscriber, InteropDemo } public enum SaslMechanism { None, Plain } public class Options { public string broker; public int port; public UInt64 messageCount; public int messageSize; public ClientType type; public string baseName; public int subTxSize; public int pubTxSize; public bool durable; public bool ssl; public string username; public string password; public SaslMechanism saslMechanism; public Options() { this.broker = "127.0.0.1"; this.port = 5672; this.messageCount = 500000; this.messageSize = 1024; this.type = ClientType.InteropDemo; // default: once as pub and once as sub this.baseName = "qpid-perftest"; this.pubTxSize = 0; this.subTxSize = 0; this.durable = false; this.ssl = false; this.username = null; this.password = null; this.saslMechanism = SaslMechanism.None; } public void Parse(string[] args) { int argCount = args.Length; int current = 0; bool typeSelected = false; while (current < argCount) { string arg = args[current]; if (arg == "--publish") { if (typeSelected) throw new ArgumentException("too many roles"); this.type = ClientType.Publisher; typeSelected = true; } else if (arg == "--subscribe") { if (typeSelected) throw new ArgumentException("too many roles"); this.type = ClientType.Subscriber; typeSelected = true; } else if (arg == "--size") { arg = args[++current]; int i = int.Parse(arg); if (i > 0) { this.messageSize = i; } } else if (arg == "--count") { arg = args[++current]; UInt64 i = UInt64.Parse(arg); if (i > 0) { this.messageCount = i; } } else if (arg == "--broker") { this.broker = args[++current]; } else if (arg == "--port") { arg = args[++current]; int i = int.Parse(arg); if (i > 0) { this.port = i; } } else if (arg == "--base-name") { this.baseName = args[++current]; } else if (arg == "--tx") { arg = args[++current]; int i = int.Parse(arg); if (i > 0) { this.subTxSize = i; this.pubTxSize = i; } } else if (arg == "--durable") { arg = args[++current]; if (arg.Equals("yes")) { this.durable = true; } } else if (arg == "--protocol") { arg = args[++current]; if (arg.Equals("ssl")) { this.ssl = true; } } else if (arg == "--username") { this.username = args[++current]; } else if (arg == "--password") { this.password = args[++current]; } else if (arg == "--mechanism") { arg = args[++current]; if (arg.Equals("PLAIN", StringComparison.OrdinalIgnoreCase)) { this.saslMechanism = SaslMechanism.Plain; } } else { throw new ArgumentException(String.Format("unknown argument \"{0}\"", arg)); } current++; } if (this.saslMechanism == SaslMechanism.Plain) { // use guest/guest as defaults if neither is specified if ((this.username == null) && (this.password == null)) { this.username = "guest"; this.password = "guest"; } else { if (this.username == null) { this.username = ""; } if (this.password == null) { this.password = ""; } } } } } public class Client { protected Options opts; public static void Expect(string actual, string expect) { if (expect != actual) { throw new Exception("Expecting " + expect + " but received " + actual); } } public static void Close(IChannel channel) { if (channel == null) { return; } try { channel.Close(); } catch (Exception e) { Console.WriteLine("channel close exception {0}", e); } } public string Fqn(string name) { return opts.baseName + '_' + name; } } public class WcfPerftest { static void WarmUpTransactionSubsystem(Options opts) { // see if any use of transactions is expected if ((opts.type == ClientType.Publisher) && (opts.pubTxSize == 0)) return; if ((opts.type == ClientType.Subscriber) && (opts.subTxSize == 0)) return; if (opts.type == ClientType.InteropDemo) { if ((opts.subTxSize == 0) && (opts.pubTxSize == 0)) return; } Console.WriteLine("Initializing transactions"); IRawBodyUtility bodyUtil = new RawEncoderUtility(); // Send a transacted message to nowhere to force the initial registration with MSDTC. // MSDTC insists on verifying it can contact the resource in the manner expected for // recovery. This requires setting up and finishing a separate connection to the // broker by a thread owned by the DTC. Excluding this time allows the existing // reporting mechanisms to better reflect the cost per transaction without requiring // long test runs. IOutputChannel channel = QueueChannelFactory.CreateWriterChannel("amq.direct", Guid.NewGuid().ToString()); Message msg = bodyUtil.CreateMessage("sacrificial transacted message from WcfPerftest"); using (TransactionScope ts = new TransactionScope()) { channel.Send(msg); // abort/rollback ts.Dispose(); } channel.Close(); Console.WriteLine("transaction resource manager ready"); } // demonstrate message exchange between WcfPerftest.exe and native // C++ perftest.exe static void InteropDemo(Options opts) { string perftest_cpp_exe = "qpid-perftest.exe"; string commonArgs = String.Format(" --count {0} --size {1} --broker {2} --port {3}", opts.messageCount, opts.messageSize, opts.broker, opts.port); if (opts.durable) { commonArgs += " --durable yes"; } if (opts.ssl) { commonArgs += " --protocol ssl"; } if (opts.saslMechanism == SaslMechanism.Plain) { commonArgs += String.Format(" --username {0} --password {1} --mechanism PLAIN", opts.username, opts.password); } Console.WriteLine("===== WCF Subscriber and C++ Publisher ====="); Process setup = new Process(); setup.StartInfo.FileName = perftest_cpp_exe; setup.StartInfo.UseShellExecute = false; setup.StartInfo.Arguments = "--setup" + commonArgs; try { setup.Start(); } catch (Win32Exception win32e) { Console.WriteLine("Cannot execute {0}: PATH not set?", perftest_cpp_exe); Console.WriteLine(" Error: {0}", win32e.Message); return; } setup.WaitForExit(); Process control = new Process(); control.StartInfo.FileName = perftest_cpp_exe; control.StartInfo.UseShellExecute = false; control.StartInfo.Arguments = "--control" + commonArgs; control.Start(); Process publish = new Process(); publish.StartInfo.FileName = perftest_cpp_exe; publish.StartInfo.UseShellExecute = false; publish.StartInfo.Arguments = "--publish" + commonArgs; publish.Start(); SubscribeThread subscribeWcf = new SubscribeThread(opts.baseName + "0", opts); Thread subThread = new Thread(subscribeWcf.Run); subThread.Start(); subThread.Join(); publish.WaitForExit(); control.WaitForExit(); Console.WriteLine(); Console.WriteLine("===== WCF Publisher and C++ Subscriber ====="); setup = new Process(); setup.StartInfo.FileName = perftest_cpp_exe; setup.StartInfo.UseShellExecute = false; setup.StartInfo.Arguments = "--setup" + commonArgs; setup.Start(); setup.WaitForExit(); control = new Process(); control.StartInfo.FileName = perftest_cpp_exe; control.StartInfo.UseShellExecute = false; control.StartInfo.Arguments = "--control" + commonArgs; control.Start(); PublishThread pub = new PublishThread(opts.baseName + "0", "", opts); Thread pubThread = new Thread(pub.Run); pubThread.Start(); Process subscribeCpp = new Process(); subscribeCpp.StartInfo.FileName = perftest_cpp_exe; subscribeCpp.StartInfo.UseShellExecute = false; subscribeCpp.StartInfo.Arguments = "--subscribe" + commonArgs; subscribeCpp.Start(); subscribeCpp.WaitForExit(); pubThread.Join(); control.WaitForExit(); } static void Main(string[] mainArgs) { Options opts = new Options(); opts.Parse(mainArgs); QueueChannelFactory.SetBroker(opts.broker, opts.port); QueueChannelFactory.SetSecurity(opts.ssl, opts.username, opts.password); WarmUpTransactionSubsystem(opts); if (opts.type == ClientType.Publisher) { PublishThread pub = new PublishThread(opts.baseName + "0", "", opts); Thread pubThread = new Thread(pub.Run); pubThread.Start(); pubThread.Join(); } else if (opts.type == ClientType.Subscriber) { SubscribeThread sub = new SubscribeThread(opts.baseName + "0", opts); Thread subThread = new Thread(sub.Run); subThread.Start(); subThread.Join(); } else { InteropDemo(opts); } if (System.Diagnostics.Debugger.IsAttached) { Console.WriteLine("Hit return to continue..."); Console.ReadLine(); } } } public class PublishThread : Client { string destination; // exchange/queue string routingKey; int msgSize; UInt64 msgCount; IOutputChannel publishQueue; public PublishThread(string key, string q, Options opts) { this.routingKey = key; this.destination = q; this.msgSize = opts.messageSize; this.msgCount = opts.messageCount; this.opts = opts; } static void StampSequenceNo(byte[] data, UInt64 n) { int wordLen = IntPtr.Size; // mimic size_t in C++ if (data.Length < wordLen) throw new ArgumentException("message size"); for (int i = 0; i < wordLen; i++) { data[i] = (byte) (n & 0xff); n >>= 8; } } public void Run() { IRawBodyUtility bodyUtil = new RawEncoderUtility(); IInputChannel startQueue = null; IOutputChannel doneQueue = null; UInt64 batchSize = (UInt64)opts.pubTxSize; bool txPending = false; AmqpProperties amqpProperties = null; if (opts.durable) { amqpProperties = new AmqpProperties(); amqpProperties.Durable = true; } try { publishQueue = QueueChannelFactory.CreateWriterChannel(this.destination, this.routingKey); doneQueue = QueueChannelFactory.CreateWriterChannel("", this.Fqn("pub_done")); startQueue = QueueChannelFactory.CreateReaderChannel(this.Fqn("pub_start")); // wait for our start signal Message msg; msg = startQueue.Receive(TimeSpan.MaxValue); Expect(bodyUtil.GetText(msg), "start"); msg.Close(); Stopwatch stopwatch = new Stopwatch(); AsyncCallback sendCallback = new AsyncCallback(this.AsyncSendCB); byte[] data = new byte[this.msgSize]; IAsyncResult sendResult = null; Console.WriteLine("sending {0}", this.msgCount); stopwatch.Start(); if (batchSize > 0) { Transaction.Current = new CommittableTransaction(); } for (UInt64 i = 0; i < this.msgCount; i++) { StampSequenceNo(data, i); msg = bodyUtil.CreateMessage(data); if (amqpProperties != null) { msg.Properties.Add("AmqpProperties", amqpProperties); } sendResult = publishQueue.BeginSend(msg, TimeSpan.MaxValue, sendCallback, msg); if (batchSize > 0) { txPending = true; if (((i + 1) % batchSize) == 0) { ((CommittableTransaction)Transaction.Current).Commit(); txPending = false; Transaction.Current = new CommittableTransaction(); } } } if (txPending) { ((CommittableTransaction)Transaction.Current).Commit(); } Transaction.Current = null; sendResult.AsyncWaitHandle.WaitOne(); stopwatch.Stop(); double mps = (msgCount / stopwatch.Elapsed.TotalSeconds); msg = bodyUtil.CreateMessage(String.Format("{0:0.##}", mps)); doneQueue.Send(msg, TimeSpan.MaxValue); msg.Close(); } finally { Close((IChannel)doneQueue); Close((IChannel)publishQueue); Close(startQueue); } } void AsyncSendCB(IAsyncResult result) { publishQueue.EndSend(result); ((Message)result.AsyncState).Close(); } } public class SubscribeThread : Client { string queue; int msgSize; UInt64 msgCount; IInputChannel subscribeQueue; public SubscribeThread(string q, Options opts) { this.queue = q; this.msgSize = opts.messageSize; this.msgCount = opts.messageCount; this.opts = opts; } static UInt64 GetSequenceNumber(byte[] data) { int wordLen = IntPtr.Size; // mimic size_t in C++ if (data.Length < wordLen) throw new ArgumentException("message size"); UInt64 n = 0; for (int i = (wordLen - 1); i >= 0; i--) { n = (256 * n) + data[i]; } return n; } public void Run() { IRawBodyUtility bodyUtil = new RawEncoderUtility(); IOutputChannel readyQueue = null; IOutputChannel doneQueue = null; UInt64 batchSize = (UInt64)opts.subTxSize; bool txPending = false; byte[] data = null; try { this.subscribeQueue = QueueChannelFactory.CreateReaderChannel(this.queue); readyQueue = QueueChannelFactory.CreateWriterChannel("", this.Fqn("sub_ready")); doneQueue = QueueChannelFactory.CreateWriterChannel("", this.Fqn("sub_done")); Message msg = bodyUtil.CreateMessage("ready"); readyQueue.Send(msg, TimeSpan.MaxValue); msg.Close(); Stopwatch stopwatch = new Stopwatch(); stopwatch.Start(); Console.WriteLine("receiving {0}", this.msgCount); UInt64 expect = 0; if (batchSize > 0) { Transaction.Current = new CommittableTransaction(); } for (UInt64 i = 0; i < this.msgCount; i++) { msg = subscribeQueue.Receive(TimeSpan.MaxValue); data = bodyUtil.GetBytes(msg, data); msg.Close(); if (data.Length != this.msgSize) { throw new Exception("subscribe message size mismatch"); } UInt64 n = GetSequenceNumber(data); if (n != expect) { throw new Exception(String.Format("message sequence error. expected {0} got {1}", expect, n)); } expect = n + 1; if (batchSize > 0) { txPending = true; if (((i + 1) % batchSize) == 0) { ((CommittableTransaction)Transaction.Current).Commit(); txPending = false; Transaction.Current = new CommittableTransaction(); } } } if (txPending) { ((CommittableTransaction)Transaction.Current).Commit(); } Transaction.Current = null; stopwatch.Stop(); double mps = (msgCount / stopwatch.Elapsed.TotalSeconds); msg = bodyUtil.CreateMessage(String.Format("{0:0.##}", mps)); doneQueue.Send(msg, TimeSpan.MaxValue); msg.Close(); subscribeQueue.Close(); } finally { Close((IChannel)doneQueue); Close((IChannel)this.subscribeQueue); Close(readyQueue); } } } }