using System; using System.Diagnostics; using CommandLine; using RabbitMQ.Client; using Tapeti.Cmd.Commands; using Tapeti.Cmd.Serialization; namespace Tapeti.Cmd { public class Program { public class CommonOptions { [Option('h', "host", HelpText = "Hostname of the RabbitMQ server.", Default = "localhost")] public string Host { get; set; } [Option('p', "port", HelpText = "AMQP port of the RabbitMQ server.", Default = 5672)] public int Port { get; set; } [Option('v', "virtualhost", HelpText = "Virtual host used for the RabbitMQ connection.", Default = "/")] public string VirtualHost { get; set; } [Option('u', "username", HelpText = "Username used to connect to the RabbitMQ server.", Default = "guest")] public string Username { get; set; } [Option('p', "password", HelpText = "Password used to connect to the RabbitMQ server.", Default = "guest")] public string Password { get; set; } } public enum SerializationMethod { SingleFileJSON, EasyNetQHosepipe } public class MessageSerializerOptions : CommonOptions { [Option('s', "serialization", HelpText = "The method used to serialize the message for import or export. Valid options: SingleFileJSON, EasyNetQHosepipe.", Default = SerializationMethod.SingleFileJSON)] public SerializationMethod SerializationMethod { get; set; } } [Verb("export", HelpText = "Fetch messages from a queue and write it to disk.")] public class ExportOptions : MessageSerializerOptions { [Option('q', "queue", Required = true, HelpText = "The queue to read the messages from.")] public string QueueName { get; set; } [Option('o', "output", Required = true, HelpText = "Path or filename (depending on the chosen serialization method) where the messages will be output to.")] public string OutputPath { get; set; } [Option('r', "remove", HelpText = "If specified messages are acknowledged and removed from the queue. If not messages are kept.")] public bool RemoveMessages { get; set; } [Option('n', "maxcount", HelpText = "(Default: all) Maximum number of messages to retrieve from the queue.")] public int? MaxCount { get; set; } } [Verb("import", HelpText = "Read messages from disk as previously exported and publish them to a queue.")] public class ImportOptions : MessageSerializerOptions { [Option('i', "input", Required = true, HelpText = "Path or filename (depending on the chosen serialization method) where the messages will be read from.")] public string Input { get; set; } [Option('e', "exchange", HelpText = "If specified publishes to the originating exchange using the original routing key. By default these are ignored and the message is published directly to the originating queue.")] public bool PublishToExchange { get; set; } } [Verb("shovel", HelpText = "Reads messages from a queue and publishes them to another queue, optionally to another RabbitMQ server.")] public class ShovelOptions : CommonOptions { [Option('q', "queue", Required = true, HelpText = "The queue to read the messages from.")] public string QueueName { get; set; } [Option('t', "targetqueue", HelpText = "The target queue to publish the messages to. Defaults to the source queue if a different target host, port or virtualhost is specified. Otherwise it must be different from the source queue.")] public string TargetQueueName { get; set; } [Option('r', "remove", HelpText = "If specified messages are acknowledged and removed from the source queue. If not messages are kept.")] public bool RemoveMessages { get; set; } [Option('n', "maxcount", HelpText = "(Default: all) Maximum number of messages to retrieve from the queue.")] public int? MaxCount { get; set; } [Option("targethost", HelpText = "Hostname of the target RabbitMQ server. Defaults to the source host. Note that you may still specify a different targetusername for example.")] public string TargetHost { get; set; } [Option("targetport", HelpText = "AMQP port of the target RabbitMQ server. Defaults to the source port.")] public int? TargetPort { get; set; } [Option("targetvirtualhost", HelpText = "Virtual host used for the target RabbitMQ connection. Defaults to the source virtualhost.")] public string TargetVirtualHost { get; set; } [Option("targetusername", HelpText = "Username used to connect to the target RabbitMQ server. Defaults to the source username.")] public string TargetUsername { get; set; } [Option("targetpassword", HelpText = "Password used to connect to the target RabbitMQ server. Defaults to the source password.")] public string TargetPassword { get; set; } } public static int Main(string[] args) { return Parser.Default.ParseArguments(args) .MapResult( (ExportOptions o) => ExecuteVerb(o, RunExport), (ImportOptions o) => ExecuteVerb(o, RunImport), (ShovelOptions o) => ExecuteVerb(o, RunShovel), errs => { if (!Debugger.IsAttached) return 1; Console.WriteLine("Press any Enter key to continue..."); Console.ReadLine(); return 1; } ); } private static int ExecuteVerb(T options, Action execute) where T : class { try { execute(options); return 0; } catch (Exception e) { Console.WriteLine(e.Message); return 1; } } private static IConnection GetConnection(CommonOptions options) { var factory = new ConnectionFactory { HostName = options.Host, Port = options.Port, VirtualHost = options.VirtualHost, UserName = options.Username, Password = options.Password }; return factory.CreateConnection(); } private static IMessageSerializer GetMessageSerializer(MessageSerializerOptions options, string path) { switch (options.SerializationMethod) { case SerializationMethod.SingleFileJSON: return new SingleFileJSONMessageSerializer(path); case SerializationMethod.EasyNetQHosepipe: return new EasyNetQMessageSerializer(path); default: throw new ArgumentOutOfRangeException(nameof(options.SerializationMethod), options.SerializationMethod, "Invalid SerializationMethod"); } } private static void RunExport(ExportOptions options) { int messageCount; using (var messageSerializer = GetMessageSerializer(options, options.OutputPath)) using (var connection = GetConnection(options)) using (var channel = connection.CreateModel()) { messageCount = new ExportCommand { MessageSerializer = messageSerializer, QueueName = options.QueueName, RemoveMessages = options.RemoveMessages, MaxCount = options.MaxCount }.Execute(channel); } Console.WriteLine($"{messageCount} message{(messageCount != 1 ? "s" : "")} exported."); } private static void RunImport(ImportOptions options) { int messageCount; using (var messageSerializer = GetMessageSerializer(options, options.Input)) using (var connection = GetConnection(options)) using (var channel = connection.CreateModel()) { messageCount = new ImportCommand { MessageSerializer = messageSerializer, DirectToQueue = !options.PublishToExchange }.Execute(channel); } Console.WriteLine($"{messageCount} message{(messageCount != 1 ? "s" : "")} published."); } private static void RunShovel(ShovelOptions options) { int messageCount; using (var sourceConnection = GetConnection(options)) using (var sourceChannel = sourceConnection.CreateModel()) { var shovelCommand = new ShovelCommand { QueueName = options.QueueName, TargetQueueName = !string.IsNullOrEmpty(options.TargetQueueName) ? options.TargetQueueName : options.QueueName, RemoveMessages = options.RemoveMessages, MaxCount = options.MaxCount }; if (RequiresSecondConnection(options)) { using (var targetConnection = GetTargetConnection(options)) using (var targetChannel = targetConnection.CreateModel()) { messageCount = shovelCommand.Execute(sourceChannel, targetChannel); } } else messageCount = shovelCommand.Execute(sourceChannel, sourceChannel); } Console.WriteLine($"{messageCount} message{(messageCount != 1 ? "s" : "")} shoveled."); } private static bool RequiresSecondConnection(ShovelOptions options) { if (!string.IsNullOrEmpty(options.TargetHost) && options.TargetHost != options.Host) return true; if (options.TargetPort.HasValue && options.TargetPort.Value != options.Port) return true; if (!string.IsNullOrEmpty(options.TargetVirtualHost) && options.TargetVirtualHost != options.VirtualHost) return true; // All relevant target host parameters are either omitted or the same. This means the queue must be different // to prevent an infinite loop. if (string.IsNullOrEmpty(options.TargetQueueName) || options.TargetQueueName == options.QueueName) throw new ArgumentException("Target queue must be different from the source queue when shoveling within the same (virtual) host"); if (!string.IsNullOrEmpty(options.TargetUsername) && options.TargetUsername != options.Username) return true; // ReSharper disable once ConvertIfStatementToReturnStatement if (!string.IsNullOrEmpty(options.TargetPassword) && options.TargetPassword != options.Password) return true; // Everything's the same, we can use the same channel return false; } private static IConnection GetTargetConnection(ShovelOptions options) { var factory = new ConnectionFactory { HostName = !string.IsNullOrEmpty(options.TargetHost) ? options.TargetHost : options.Host, Port = options.TargetPort ?? options.Port, VirtualHost = !string.IsNullOrEmpty(options.TargetVirtualHost) ? options.TargetVirtualHost : options.VirtualHost, UserName = !string.IsNullOrEmpty(options.TargetUsername) ? options.TargetUsername : options.Username, Password = !string.IsNullOrEmpty(options.TargetPassword) ? options.TargetPassword : options.Password, }; return factory.CreateConnection(); } } }