/* * ws.cpp * Author: Benjamin Sergeant * Copyright (c) 2019 Machine Zone, Inc. All rights reserved. */ // // Main driver for websocket utilities // #include "ws.h" // // Main drive for websocket utilities // #include #include #include #include // #include #include #include #include #include #include int main(int argc, char** argv) { ix::initNetSystem(); ix::IXCoreLogger::LogFunc logFunc = [](const char* msg) { spdlog::info(msg); }; ix::IXCoreLogger::setLogFunction(logFunc); // Display command. if (getenv("DEBUG")) { std::cout << "Command: "; for (int i = 0; i < argc; ++i) { std::cout << argv[i] << " "; } std::cout << std::endl; } CLI::App app{"ws is a websocket tool"}; app.require_subcommand(); std::string url("ws://127.0.0.1:8008"); std::string path; std::string user; std::string data; std::string headers; std::string output; std::string hostname("127.0.0.1"); std::string pidfile; std::string channel; std::string message; std::string password; std::string appkey; std::string endpoint; std::string rolename; std::string rolesecret; std::string prefix("ws.test.v0"); std::string fields; std::string dsn; std::string redisHosts("127.0.0.1"); std::string redisPassword; std::string appsConfigPath("appsConfig.json"); bool headersOnly = false; bool followRedirects = false; bool verbose = false; bool save = false; bool compress = false; bool strict = false; bool stress = false; bool disableAutomaticReconnection = false; bool disablePerMessageDeflate = false; bool greetings = false; int port = 8008; int redisPort = 6379; int statsdPort = 8125; int connectTimeOut = 60; int transferTimeout = 1800; int maxRedirects = 5; int delayMs = -1; int count = 1; int jobs = 4; CLI::App* sendApp = app.add_subcommand("send", "Send a file"); sendApp->add_option("url", url, "Connection url")->required(); sendApp->add_option("path", path, "Path to the file to send") ->required()->check(CLI::ExistingPath); sendApp->add_option("--pidfile", pidfile, "Pid file"); CLI::App* receiveApp = app.add_subcommand("receive", "Receive a file"); receiveApp->add_option("url", url, "Connection url")->required(); receiveApp->add_option("--delay", delayMs, "Delay (ms) to wait after receiving a fragment" " to artificially slow down the receiver"); receiveApp->add_option("--pidfile", pidfile, "Pid file"); CLI::App* transferApp = app.add_subcommand("transfer", "Broadcasting server"); transferApp->add_option("--port", port, "Connection url"); transferApp->add_option("--host", hostname, "Hostname"); transferApp->add_option("--pidfile", pidfile, "Pid file"); CLI::App* connectApp = app.add_subcommand("connect", "Connect to a remote server"); connectApp->add_option("url", url, "Connection url")->required(); connectApp->add_flag("-d", disableAutomaticReconnection, "Disable Automatic Reconnection"); connectApp->add_flag("-x", disablePerMessageDeflate, "Disable per message deflate"); CLI::App* chatApp = app.add_subcommand("chat", "Group chat"); chatApp->add_option("url", url, "Connection url")->required(); chatApp->add_option("user", user, "User name")->required(); CLI::App* echoServerApp = app.add_subcommand("echo_server", "Echo server"); echoServerApp->add_option("--port", port, "Port"); echoServerApp->add_option("--host", hostname, "Hostname"); echoServerApp->add_flag("-g", greetings, "Verbose"); CLI::App* broadcastServerApp = app.add_subcommand("broadcast_server", "Broadcasting server"); broadcastServerApp->add_option("--port", port, "Port"); broadcastServerApp->add_option("--host", hostname, "Hostname"); CLI::App* pingPongApp = app.add_subcommand("ping", "Ping pong"); pingPongApp->add_option("url", url, "Connection url")->required(); CLI::App* httpClientApp = app.add_subcommand("curl", "HTTP Client"); httpClientApp->add_option("url", url, "Connection url")->required(); httpClientApp->add_option("-d", data, "Form data")->join(); httpClientApp->add_option("-F", data, "Form data")->join(); httpClientApp->add_option("-H", headers, "Header")->join(); httpClientApp->add_option("--output", output, "Output file"); httpClientApp->add_flag("-I", headersOnly, "Send a HEAD request"); httpClientApp->add_flag("-L", followRedirects, "Follow redirects"); httpClientApp->add_option("--max-redirects", maxRedirects, "Max Redirects"); httpClientApp->add_flag("-v", verbose, "Verbose"); httpClientApp->add_flag("-O", save, "Save output to disk"); httpClientApp->add_flag("--compress", compress, "Enable gzip compression"); httpClientApp->add_option("--connect-timeout", connectTimeOut, "Connection timeout"); httpClientApp->add_option("--transfer-timeout", transferTimeout, "Transfer timeout"); CLI::App* redisPublishApp = app.add_subcommand("redis_publish", "Redis publisher"); redisPublishApp->add_option("--port", redisPort, "Port"); redisPublishApp->add_option("--host", hostname, "Hostname"); redisPublishApp->add_option("--password", password, "Password"); redisPublishApp->add_option("channel", channel, "Channel")->required(); redisPublishApp->add_option("message", message, "Message")->required(); redisPublishApp->add_option("-c", count, "Count"); CLI::App* redisSubscribeApp = app.add_subcommand("redis_subscribe", "Redis subscriber"); redisSubscribeApp->add_option("--port", redisPort, "Port"); redisSubscribeApp->add_option("--host", hostname, "Hostname"); redisSubscribeApp->add_option("--password", password, "Password"); redisSubscribeApp->add_option("channel", channel, "Channel")->required(); redisSubscribeApp->add_flag("-v", verbose, "Verbose"); redisSubscribeApp->add_option("--pidfile", pidfile, "Pid file"); CLI::App* cobraSubscribeApp = app.add_subcommand("cobra_subscribe", "Cobra subscriber"); cobraSubscribeApp->add_option("--appkey", appkey, "Appkey"); cobraSubscribeApp->add_option("--endpoint", endpoint, "Endpoint"); cobraSubscribeApp->add_option("--rolename", rolename, "Role name"); cobraSubscribeApp->add_option("--rolesecret", rolesecret, "Role secret"); cobraSubscribeApp->add_option("channel", channel, "Channel")->required(); cobraSubscribeApp->add_option("--pidfile", pidfile, "Pid file"); CLI::App* cobraPublish = app.add_subcommand("cobra_publish", "Cobra publisher"); cobraPublish->add_option("--appkey", appkey, "Appkey"); cobraPublish->add_option("--endpoint", endpoint, "Endpoint"); cobraPublish->add_option("--rolename", rolename, "Role name"); cobraPublish->add_option("--rolesecret", rolesecret, "Role secret"); cobraPublish->add_option("channel", channel, "Channel")->required(); cobraPublish->add_option("--pidfile", pidfile, "Pid file"); cobraPublish->add_option("path", path, "Path to the file to send") ->required()->check(CLI::ExistingPath); cobraPublish->add_flag("--stress", stress, "Stress mode"); CLI::App* cobra2statsd = app.add_subcommand("cobra_to_statsd", "Cobra to statsd"); cobra2statsd->add_option("--appkey", appkey, "Appkey"); cobra2statsd->add_option("--endpoint", endpoint, "Endpoint"); cobra2statsd->add_option("--rolename", rolename, "Role name"); cobra2statsd->add_option("--rolesecret", rolesecret, "Role secret"); cobra2statsd->add_option("--host", hostname, "Statsd host"); cobra2statsd->add_option("--port", statsdPort, "Statsd port"); cobra2statsd->add_option("--prefix", prefix, "Statsd prefix"); cobra2statsd->add_option("--fields", fields, "Extract fields for naming the event")->join(); cobra2statsd->add_option("channel", channel, "Channel")->required(); cobra2statsd->add_flag("-v", verbose, "Verbose"); cobra2statsd->add_option("--pidfile", pidfile, "Pid file"); CLI::App* cobra2sentry = app.add_subcommand("cobra_to_sentry", "Cobra to sentry"); cobra2sentry->add_option("--appkey", appkey, "Appkey"); cobra2sentry->add_option("--endpoint", endpoint, "Endpoint"); cobra2sentry->add_option("--rolename", rolename, "Role name"); cobra2sentry->add_option("--rolesecret", rolesecret, "Role secret"); cobra2sentry->add_option("--dsn", dsn, "Sentry DSN"); cobra2sentry->add_option("--jobs", jobs, "Number of thread sending events to Sentry"); cobra2sentry->add_option("channel", channel, "Channel")->required(); cobra2sentry->add_flag("-v", verbose, "Verbose"); cobra2sentry->add_flag("-s", strict, "Strict mode. Error out when sending to sentry fails"); cobra2sentry->add_option("--pidfile", pidfile, "Pid file"); CLI::App* runApp = app.add_subcommand("snake", "Snake server"); runApp->add_option("--port", port, "Connection url"); runApp->add_option("--host", hostname, "Hostname"); runApp->add_option("--pidfile", pidfile, "Pid file"); runApp->add_option("--redis_hosts", redisHosts, "Redis hosts"); runApp->add_option("--redis_port", redisPort, "Redis hosts"); runApp->add_option("--redis_password", redisPassword, "Redis password"); runApp->add_option("--apps_config_path", appsConfigPath, "Path to auth data") ->check(CLI::ExistingPath); runApp->add_flag("-v", verbose, "Verbose"); CLI11_PARSE(app, argc, argv); // pid file handling if (!pidfile.empty()) { unlink(pidfile.c_str()); std::ofstream f; f.open(pidfile); f << getpid(); f.close(); } int ret = 1; if (app.got_subcommand("transfer")) { ret = ix::ws_transfer_main(port, hostname); } else if (app.got_subcommand("send")) { ret = ix::ws_send_main(url, path); } else if (app.got_subcommand("receive")) { bool enablePerMessageDeflate = false; ret = ix::ws_receive_main(url, enablePerMessageDeflate, delayMs); } else if (app.got_subcommand("connect")) { ret = ix::ws_connect_main(url, disableAutomaticReconnection, disablePerMessageDeflate); } else if (app.got_subcommand("chat")) { ret = ix::ws_chat_main(url, user); } else if (app.got_subcommand("echo_server")) { ret = ix::ws_echo_server_main(port, greetings, hostname); } else if (app.got_subcommand("broadcast_server")) { ret = ix::ws_broadcast_server_main(port, hostname); } else if (app.got_subcommand("ping")) { ret = ix::ws_ping_pong_main(url); } else if (app.got_subcommand("curl")) { ret = ix::ws_http_client_main(url, headers, data, headersOnly, connectTimeOut, transferTimeout, followRedirects, maxRedirects, verbose, save, output, compress); } else if (app.got_subcommand("redis_publish")) { ret = ix::ws_redis_publish_main(hostname, redisPort, password, channel, message, count); } else if (app.got_subcommand("redis_subscribe")) { ret = ix::ws_redis_subscribe_main(hostname, redisPort, password, channel, verbose); } else if (app.got_subcommand("cobra_subscribe")) { ret = ix::ws_cobra_subscribe_main(appkey, endpoint, rolename, rolesecret, channel); } else if (app.got_subcommand("cobra_publish")) { ret = ix::ws_cobra_publish_main(appkey, endpoint, rolename, rolesecret, channel, path, stress); } else if (app.got_subcommand("cobra_to_statsd")) { ret = ix::ws_cobra_to_statsd_main(appkey, endpoint, rolename, rolesecret, channel, hostname, statsdPort, prefix, fields, verbose); } else if (app.got_subcommand("cobra_to_sentry")) { ret = ix::ws_cobra_to_sentry_main(appkey, endpoint, rolename, rolesecret, channel, dsn, verbose, strict, jobs); } else if (app.got_subcommand("snake")) { ret = ix::ws_snake_main(port, hostname, redisHosts, redisPort, redisPassword, verbose, appsConfigPath); } ix::uninitNetSystem(); return ret; }