| src/examples/cpp03/chat/chat_server.cpp | src/examples/cpp11/chat/chat_server.cpp | 
|---|
| ⋮ | ⋮ | 
| 1 | // | 1 | // | 
| 2 | //·chat_server.cpp | 2 | //·chat_server.cpp | 
| 3 | //·~~~~~~~~~~~~~~~ | 3 | //·~~~~~~~~~~~~~~~ | 
| 4 | // | 4 | // | 
| 5 | //·Copyright·(c)·2003-2021·Christopher·M.·Kohlhoff·(chris·at·kohlhoff·dot·com) | 5 | //·Copyright·(c)·2003-2021·Christopher·M.·Kohlhoff·(chris·at·kohlhoff·dot·com) | 
| 6 | // | 6 | // | 
| 7 | //·Distributed·under·the·Boost·Software·License,·Version·1.0.·(See·accompanying | 7 | //·Distributed·under·the·Boost·Software·License,·Version·1.0.·(See·accompanying | 
| 8 | //·file·LICENSE_1_0.txt·or·copy·at·http://www.boost.org/LICENSE_1_0.txt) | 8 | //·file·LICENSE_1_0.txt·or·copy·at·http://www.boost.org/LICENSE_1_0.txt) | 
| 9 | // | 9 | // | 
| 10 |  | 10 |  | 
| 11 | #include·<algorithm> |  | 
| 12 | #include·<cstdlib> | 11 | #include·<cstdlib> | 
| 13 | #include·<deque> | 12 | #include·<deque> | 
| 14 | #include·<iostream> | 13 | #include·<iostream> | 
| 15 | #include·<list> | 14 | #include·<list> | 
|  | 15 | #include·<memory> | 
| 16 | #include·<set> | 16 | #include·<set> | 
| 17 | #include·<boost/bind/bind.hpp> | 17 | #include·<utility> | 
| 18 | #include·<boost/shared_ptr.hpp> |  | 
| 19 | #include·<boost/enable_shared_from_this.hpp> |  | 
| 20 | #include·"asio.hpp" | 18 | #include·"asio.hpp" | 
| 21 | #include·"chat_message.hpp" | 19 | #include·"chat_message.hpp" | 
| 22 |  | 20 |  | 
| 23 | using·asio::ip::tcp; | 21 | using·asio::ip::tcp; | 
| 24 |  | 22 |  | 
| 25 | //---------------------------------------------------------------------- | 23 | //---------------------------------------------------------------------- | 
| 26 |  | 24 |  | 
| 27 | typedef·std::deque<chat_message>·chat_message_queue; | 25 | typedef·std::deque<chat_message>·chat_message_queue; | 
| 28 |  | 26 |  | 
| 29 | //---------------------------------------------------------------------- | 27 | //---------------------------------------------------------------------- | 
| 30 |  | 28 |  | 
| 31 | class·chat_participant | 29 | class·chat_participant | 
| 32 | { | 30 | { | 
| 33 | public: | 31 | public: | 
| 34 | ··virtual·~chat_participant()·{} | 32 | ··virtual·~chat_participant()·{} | 
| 35 | ··virtual·void·deliver(const·chat_message&·msg)·=·0; | 33 | ··virtual·void·deliver(const·chat_message&·msg)·=·0; | 
| 36 | }; | 34 | }; | 
| 37 |  | 35 |  | 
| 38 | typedef·boost::shared_ptr<chat_participant>·chat_participant_ptr; | 36 | typedef·std::shared_ptr<chat_participant>·chat_participant_ptr; | 
| 39 |  | 37 |  | 
| 40 | //---------------------------------------------------------------------- | 38 | //---------------------------------------------------------------------- | 
| 41 |  | 39 |  | 
| 42 | class·chat_room | 40 | class·chat_room | 
| 43 | { | 41 | { | 
| 44 | public: | 42 | public: | 
| 45 | ··void·join(chat_participant_ptr·participant) | 43 | ··void·join(chat_participant_ptr·participant) | 
| 46 | ··{ | 44 | ··{ | 
| 47 | ····participants_.insert(participant); | 45 | ····participants_.insert(participant); | 
| 48 | ····std::for_each(recent_msgs_.begin(),·recent_msgs_.end(), | 46 | ····for·(auto·msg:·recent_msgs_) | 
| 49 | ········boost::bind(&chat_participant::deliver, | 47 | ······participant->deliver(msg); | 
| 50 | ··········participant,·boost::placeholders::_1)); |  | 
| 51 | ··} | 48 | ··} | 
| 52 |  | 49 |  | 
| 53 | ··void·leave(chat_participant_ptr·participant) | 50 | ··void·leave(chat_participant_ptr·participant) | 
| 54 | ··{ | 51 | ··{ | 
| 55 | ····participants_.erase(participant); | 52 | ····participants_.erase(participant); | 
| 56 | ··} | 53 | ··} | 
| 57 |  | 54 |  | 
| 58 | ··void·deliver(const·chat_message&·msg) | 55 | ··void·deliver(const·chat_message&·msg) | 
| 59 | ··{ | 56 | ··{ | 
| 60 | ····recent_msgs_.push_back(msg); | 57 | ····recent_msgs_.push_back(msg); | 
| 61 | ····while·(recent_msgs_.size()·>·max_recent_msgs) | 58 | ····while·(recent_msgs_.size()·>·max_recent_msgs) | 
| 62 | ······recent_msgs_.pop_front(); | 59 | ······recent_msgs_.pop_front(); | 
| 63 |  | 60 |  | 
| 64 | ····std::for_each(participants_.begin(),·participants_.end(), | 61 | ····for·(auto·participant:·participants_) | 
| 65 | ········boost::bind(&chat_participant::deliver, | 62 | ······participant->deliver(msg); | 
| 66 | ··········boost::placeholders::_1,·boost::ref(msg))); |  | 
| 67 | ··} | 63 | ··} | 
| 68 |  | 64 |  | 
| 69 | private: | 65 | private: | 
| 70 | ··std::set<chat_participant_ptr>·participants_; | 66 | ··std::set<chat_participant_ptr>·participants_; | 
| 71 | ··enum·{·max_recent_msgs·=·100·}; | 67 | ··enum·{·max_recent_msgs·=·100·}; | 
| 72 | ··chat_message_queue·recent_msgs_; | 68 | ··chat_message_queue·recent_msgs_; | 
| 73 | }; | 69 | }; | 
| 74 |  | 70 |  | 
| 75 | //---------------------------------------------------------------------- | 71 | //---------------------------------------------------------------------- | 
| 76 |  | 72 |  | 
| 77 | class·chat_session | 73 | class·chat_session | 
| 78 | ··:·public·chat_participant, | 74 | ··:·public·chat_participant, | 
| 79 | ····public·boost::enable_shared_from_this<chat_session> | 75 | ····public·std::enable_shared_from_this<chat_session> | 
| 80 | { | 76 | { | 
| 81 | public: | 77 | public: | 
| 82 | ··chat_session(asio::io_context&·io_context,·chat_room&·room) | 78 | ··chat_session(tcp::socket·socket,·chat_room&·room) | 
| 83 | ····:·socket_(io_context), | 79 | ····:·socket_(std::move(socket)), | 
| 84 | ······room_(room) | 80 | ······room_(room) | 
| 85 | ··{ | 81 | ··{ | 
| 86 | ··} | 82 | ··} | 
| 87 |  | 83 |  | 
| 88 | ··tcp::socket&·socket() |  | 
| 89 | ··{ |  | 
| 90 | ····return·socket_; |  | 
| 91 | ··} |  | 
| 92 |  |  | 
| 93 | ··void·start() | 84 | ··void·start() | 
| 94 | ··{ | 85 | ··{ | 
| 95 | ····room_.join(shared_from_this()); | 86 | ····room_.join(shared_from_this()); | 
| 96 | ····asio::async_read(socket_, | 87 | ····do_read_header(); | 
| 97 | ········asio::buffer(read_msg_.data(),·chat_message::header_length), |  | 
| 98 | ········boost::bind( |  | 
| 99 | ··········&chat_session::handle_read_header,·shared_from_this(), |  | 
| 100 | ··········asio::placeholders::error)); |  | 
| 101 | ··} | 88 | ··} | 
| 102 |  | 89 |  | 
| 103 | ··void·deliver(const·chat_message&·msg) | 90 | ··void·deliver(const·chat_message&·msg) | 
| 104 | ··{ | 91 | ··{ | 
| 105 | ····bool·write_in_progress·=·!write_msgs_.empty(); | 92 | ····bool·write_in_progress·=·!write_msgs_.empty(); | 
| 106 | ····write_msgs_.push_back(msg); | 93 | ····write_msgs_.push_back(msg); | 
| 107 | ····if·(!write_in_progress) | 94 | ····if·(!write_in_progress) | 
| 108 | ····{ | 95 | ····{ | 
| 109 | ······asio::async_write(socket_, | 96 | ······do_write(); | 
| 110 | ··········asio::buffer(write_msgs_.front().data(), |  | 
| 111 | ············write_msgs_.front().length()), |  | 
| 112 | ··········boost::bind(&chat_session::handle_write,·shared_from_this(), |  | 
| 113 | ············asio::placeholders::error)); |  | 
| 114 | ····} | 97 | ····} | 
| 115 | ··} | 98 | ··} | 
| 116 |  | 99 |  | 
| 117 | ··void·handle_read_header(const·asio::error_code&·error) | 100 | private: | 
| 118 | ··{ | 101 | ··void·do_read_header() | 
| 119 | ····if·(!error·&&·read_msg_.decode_header()) |  | 
| 120 | ····{ |  | 
| 121 | ······asio::async_read(socket_, |  | 
| 122 | ··········asio::buffer(read_msg_.body(),·read_msg_.body_length()), |  | 
| 123 | ··········boost::bind(&chat_session::handle_read_body,·shared_from_this(), |  | 
| 124 | ············asio::placeholders::error)); |  | 
| 125 | ····} |  | 
| 126 | ····else |  | 
| 127 | ····{ |  | 
| 128 | ······room_.leave(shared_from_this()); |  | 
| 129 | ····} |  | 
| 130 | ··} |  | 
| 131 |  |  | 
| 132 | ··void·handle_read_body(const·asio::error_code&·error) |  | 
| 133 | ··{ | 102 | ··{ | 
| 134 | ····if·(!error) | 103 | ····auto·self(shared_from_this()); | 
| 135 | ····{ | 104 | ····asio::async_read(socket_, | 
| 136 | ······room_.deliver(read_msg_); | 105 | ········asio::buffer(read_msg_.data(),·chat_message::header_length), | 
| 137 | ······asio::async_read(socket_, | 106 | ········[this,·self](std::error_code·ec,·std::size_t·/*length*/) | 
| 138 | ··········asio::buffer(read_msg_.data(),·chat_message::header_length), | 107 | ········{ | 
| 139 | ··········boost::bind(&chat_session::handle_read_header,·shared_from_this(), | 108 | ··········if·(!ec·&&·read_msg_.decode_header()) | 
| 140 | ············asio::placeholders::error)); | 109 | ··········{ | 
| 141 | ····} | 110 | ············do_read_body(); | 
| 142 | ····else | 111 | ··········} | 
| 143 | ····{ | 112 | ··········else | 
| 144 | ······room_.leave(shared_from_this()); | 113 | ··········{ | 
| 145 | ····} | 114 | ············room_.leave(shared_from_this()); | 
|  | 115 | ··········} | 
|  | 116 | ········}); | 
| 146 | ··} | 117 | ··} | 
| 147 |  | 118 |  | 
| 148 | ··void·handle_write(const·asio::error_code&·error) | 119 | ··void·do_read_body() | 
| 149 | ··{ | 120 | ··{ | 
| 150 | ····if·(!error) | 121 | ····auto·self(shared_from_this()); | 
| 151 | ····{ | 122 | ····asio::async_read(socket_, | 
| 152 | ······write_msgs_.pop_front(); | 123 | ········asio::buffer(read_msg_.body(),·read_msg_.body_length()), | 
| 153 | ······if·(!write_msgs_.empty()) | 124 | ········[this,·self](std::error_code·ec,·std::size_t·/*length*/) | 
| 154 | ······{ | 125 | ········{ | 
| 155 | ········asio::async_write(socket_, | 126 | ··········if·(!ec) | 
| 156 | ············asio::buffer(write_msgs_.front().data(), | 127 | ··········{ | 
| 157 | ··············write_msgs_.front().length()), | 128 | ············room_.deliver(read_msg_); | 
| 158 | ············boost::bind(&chat_session::handle_write,·shared_from_this(), | 129 | ············do_read_header(); | 
| 159 | ··············asio::placeholders::error)); | 130 | ··········} | 
| 160 | ······} | 131 | ··········else | 
| 161 | ····} | 132 | ··········{ | 
| 162 | ····else | 133 | ············room_.leave(shared_from_this()); | 
| 163 | ····{ | 134 | ··········} | 
| 164 | ······room_.leave(shared_from_this()); | 135 | ········}); | 
| 165 | ····} | 136 | ··} | 
|  | 137 |  | 
|  | 138 | ··void·do_write() | 
|  | 139 | ··{ | 
|  | 140 | ····auto·self(shared_from_this()); | 
|  | 141 | ····asio::async_write(socket_, | 
|  | 142 | ········asio::buffer(write_msgs_.front().data(), | 
|  | 143 | ··········write_msgs_.front().length()), | 
|  | 144 | ········[this,·self](std::error_code·ec,·std::size_t·/*length*/) | 
|  | 145 | ········{ | 
|  | 146 | ··········if·(!ec) | 
|  | 147 | ··········{ | 
|  | 148 | ············write_msgs_.pop_front(); | 
|  | 149 | ············if·(!write_msgs_.empty()) | 
|  | 150 | ············{ | 
|  | 151 | ··············do_write(); | 
|  | 152 | ············} | 
|  | 153 | ··········} | 
|  | 154 | ··········else | 
|  | 155 | ··········{ | 
|  | 156 | ············room_.leave(shared_from_this()); | 
|  | 157 | ··········} | 
|  | 158 | ········}); | 
| 166 | ··} | 159 | ··} | 
| 167 |  | 160 |  | 
| 168 | private: |  | 
| 169 | ··tcp::socket·socket_; | 161 | ··tcp::socket·socket_; | 
| 170 | ··chat_room&·room_; | 162 | ··chat_room&·room_; | 
| 171 | ··chat_message·read_msg_; | 163 | ··chat_message·read_msg_; | 
| 172 | ··chat_message_queue·write_msgs_; | 164 | ··chat_message_queue·write_msgs_; | 
| 173 | }; | 165 | }; | 
| 174 |  | 166 |  | 
| 175 | typedef·boost::shared_ptr<chat_session>·chat_session_ptr; |  | 
| 176 |  |  | 
| 177 | //---------------------------------------------------------------------- | 167 | //---------------------------------------------------------------------- | 
| 178 |  | 168 |  | 
| 179 | class·chat_server | 169 | class·chat_server | 
| 180 | { | 170 | { | 
| 181 | public: | 171 | public: | 
| 182 | ··chat_server(asio::io_context&·io_context, | 172 | ··chat_server(asio::io_context&·io_context, | 
| 183 | ······const·tcp::endpoint&·endpoint) | 173 | ······const·tcp::endpoint&·endpoint) | 
| 184 | ····:·io_context_(io_context), | 174 | ····:·acceptor_(io_context,·endpoint) | 
| 185 | ······acceptor_(io_context,·endpoint) |  | 
| 186 | ··{ | 175 | ··{ | 
| 187 | ····start_accept(); | 176 | ····do_accept(); | 
| 188 | ··} | 177 | ··} | 
| 189 |  | 178 |  | 
| 190 | ··void·start_accept() | 179 | private: | 
| 191 | ··{ | 180 | ··void·do_accept() | 
| 192 | ····chat_session_ptr·new_session(new·chat_session(io_context_,·room_)); |  | 
| 193 | ····acceptor_.async_accept(new_session->socket(), |  | 
| 194 | ········boost::bind(&chat_server::handle_accept,·this,·new_session, |  | 
| 195 | ··········asio::placeholders::error)); |  | 
| 196 | ··} |  | 
| 197 |  |  | 
| 198 | ··void·handle_accept(chat_session_ptr·session, |  | 
| 199 | ······const·asio::error_code&·error) |  | 
| 200 | ··{ | 181 | ··{ | 
| 201 | ····if·(!error) | 182 | ····acceptor_.async_accept( | 
| 202 | ····{ | 183 | ········[this](std::error_code·ec,·tcp::socket·socket) | 
| 203 | ······session->start(); | 184 | ········{ | 
| 204 | ····} | 185 | ··········if·(!ec) | 
|  | 186 | ··········{ | 
|  | 187 | ············std::make_shared<chat_session>(std::move(socket),·room_)->start(); | 
|  | 188 | ··········} | 
| 205 |  | 189 |  | 
| 206 | ····start_accept(); | 190 | ··········do_accept(); | 
|  | 191 | ········}); | 
| 207 | ··} | 192 | ··} | 
| 208 |  | 193 |  | 
| 209 | private: |  | 
| 210 | ··asio::io_context&·io_context_; |  | 
| 211 | ··tcp::acceptor·acceptor_; | 194 | ··tcp::acceptor·acceptor_; | 
| 212 | ··chat_room·room_; | 195 | ··chat_room·room_; | 
| 213 | }; | 196 | }; | 
| 214 |  | 197 |  | 
| 215 | typedef·boost::shared_ptr<chat_server>·chat_server_ptr; |  | 
| 216 | typedef·std::list<chat_server_ptr>·chat_server_list; |  | 
| 217 |  |  | 
| 218 | //---------------------------------------------------------------------- | 198 | //---------------------------------------------------------------------- | 
| 219 |  | 199 |  | 
| 220 | int·main(int·argc,·char*·argv[]) | 200 | int·main(int·argc,·char*·argv[]) | 
| 221 | { | 201 | { | 
| 222 | ··try | 202 | ··try | 
| 223 | ··{ | 203 | ··{ | 
| 224 | ····if·(argc·<·2) | 204 | ····if·(argc·<·2) | 
| 225 | ····{ | 205 | ····{ | 
| 226 | ······std::cerr·<<·"Usage:·chat_server·<port>·[<port>·...]\n"; | 206 | ······std::cerr·<<·"Usage:·chat_server·<port>·[<port>·...]\n"; | 
| 227 | ······return·1; | 207 | ······return·1; | 
| 228 | ····} | 208 | ····} | 
| 229 |  | 209 |  | 
| 230 | ····asio::io_context·io_context; | 210 | ····asio::io_context·io_context; | 
| 231 |  | 211 |  | 
| 232 | ····chat_server_list·servers; | 212 | ····std::list<chat_server>·servers; | 
| 233 | ····for·(int·i·=·1;·i·<·argc;·++i) | 213 | ····for·(int·i·=·1;·i·<·argc;·++i) | 
| 234 | ····{ | 214 | ····{ | 
| 235 | ······using·namespace·std;·//·For·atoi. | 215 | ······tcp::endpoint·endpoint(tcp::v4(),·std::atoi(argv[i])); | 
| 236 | ······tcp::endpoint·endpoint(tcp::v4(),·atoi(argv[i])); | 216 | ······servers.emplace_back(io_context,·endpoint); | 
| 237 | ······chat_server_ptr·server(new·chat_server(io_context,·endpoint)); |  | 
| 238 | ······servers.push_back(server); |  | 
| 239 | ····} | 217 | ····} | 
| 240 |  | 218 |  | 
| 241 | ····io_context.run(); | 219 | ····io_context.run(); | 
| 242 | ··} | 220 | ··} | 
| 243 | ··catch·(std::exception&·e) | 221 | ··catch·(std::exception&·e) | 
| 244 | ··{ | 222 | ··{ | 
| 245 | ····std::cerr·<<·"Exception:·"·<<·e.what()·<<·"\n"; | 223 | ····std::cerr·<<·"Exception:·"·<<·e.what()·<<·"\n"; | 
| 246 | ··} | 224 | ··} | 
| 247 |  | 225 |  | 
| 248 | ··return·0; | 226 | ··return·0; | 
| 249 | } | 227 | } |