/* * Copyright (c) 2011, Peter Thorson. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the name of the WebSocket++ Project nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL PETER THORSON BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #ifndef WSPERF_STRESS_HANDLER_HPP #define WSPERF_STRESS_HANDLER_HPP #include "wscmd.hpp" #include "../../src/roles/client.hpp" #include "../../src/websocketpp.hpp" #include #include #include using websocketpp::client; namespace wsperf { namespace con_lifetime { enum value { FIXED = 0, RANDOM = 1, UNLIMITED = 2 }; } namespace msg_mode { enum value { NONE = 0, FIXED = 1, UNLIMITED = 2 }; } struct msg_data { typedef boost::chrono::steady_clock::time_point time_point; size_t msg_id; time_point send_time; time_point recv_time; //size_t payload_len; }; struct con_data { typedef boost::chrono::steady_clock::time_point time_point; con_data() {} con_data(size_t id,time_point init) : id(id) , init(init) , start(init) , tcp_established(init) , on_open(init) , on_fail(init) , close_sent(init) , on_close(init) , status("Connecting") { } std::string print() const { std::stringstream o; o << "{"; o << "\"id\":" << id; o << ",\"status\":\"" << status << "\""; o << ",\"start\":" << get_rel_microseconds(start); o << ",\"tcp\":" << get_rel_microseconds(tcp_established); o << ",\"open\":" << get_rel_microseconds(on_open); o << ",\"fail\":" << get_rel_microseconds(on_fail); o << ",\"close_sent\":" << get_rel_microseconds(close_sent); o << ",\"close\":" << get_rel_microseconds(on_close); o << ",\"messages\":["; std::string sep = ""; std::vector::const_iterator it; for (it = messages.begin(); it != messages.end(); it++) { o << sep << "[" << get_rel_microseconds((*it).send_time) << "," << get_rel_microseconds((*it).recv_time) << "]"; sep = ","; } o << "]"; o << "}"; return o.str(); } double get_rel_microseconds(time_point t) const { boost::chrono::nanoseconds dur = t - init; return static_cast (dur.count()) / 1000.; } size_t id; time_point init; time_point start; time_point tcp_established; time_point on_open; time_point on_fail; time_point close_sent; time_point on_close; std::string status; std::vector messages; //stress_handler::message_ptr msg; }; class stress_handler : public client::handler { public: typedef stress_handler type; typedef boost::chrono::steady_clock::time_point time_point; typedef std::map time_map; /// Construct a stress test from a wscmd command explicit stress_handler(wscmd::cmd& cmd); void on_connect(connection_ptr con); void on_message(connection_ptr con,websocketpp::message::data_ptr msg); void on_handshake_init(connection_ptr con); void on_open(connection_ptr con); void on_close(connection_ptr con); void on_fail(connection_ptr con); void start(connection_ptr con); void close(connection_ptr con); void end(); std::string get_data() const; virtual bool maintenance(); void start_message_test(); protected: size_t m_current_connections; size_t m_max_connections; size_t m_total_connections; size_t m_failed_connections; size_t m_next_con_id; time_point m_init; // connection related timestamps std::map m_con_data; mutable std::list m_dirty; // Stats update timer size_t m_timeout; boost::shared_ptr m_timer; size_t m_next_msg_id; boost::shared_ptr m_msg; // test settings pulled from the command con_lifetime::value m_con_lifetime; size_t m_con_duration; bool m_con_sync; msg_mode::value m_msg_mode; size_t m_msg_count; size_t m_msg_size; mutable boost::mutex m_lock; }; typedef boost::shared_ptr stress_handler_ptr; } // namespace wsperf #endif // WSPERF_STRESS_HANDLER_HPP