/* * 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 { 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 << "}"; 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; }; 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_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(); 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; mutable boost::mutex m_lock; }; typedef boost::shared_ptr stress_handler_ptr; } // namespace wsperf #endif // WSPERF_STRESS_HANDLER_HPP