| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113 |
- #ifndef HV_WEBSOCKET_CHANNEL_H_
- #define HV_WEBSOCKET_CHANNEL_H_
- #include <mutex>
- #include "Channel.h"
- #include "wsdef.h"
- #include "hmath.h"
- namespace hv {
- class WebSocketChannel : public SocketChannel {
- public:
- ws_session_type type;
- WebSocketChannel(hio_t* io, ws_session_type type = WS_CLIENT)
- : SocketChannel(io)
- , type(type)
- {}
- ~WebSocketChannel() {}
- // isConnected, send, close
- int send(const std::string& msg, enum ws_opcode opcode = WS_OPCODE_TEXT, bool fin = true) {
- return send(msg.c_str(), msg.size(), opcode, fin);
- }
- int send(const char* buf, int len, enum ws_opcode opcode = WS_OPCODE_BINARY, bool fin = true) {
- int fragment = 0xFFFF; // 65535
- if (len > fragment) {
- return send(buf, len, fragment, opcode);
- }
- std::lock_guard<std::mutex> locker(mutex_);
- return sendFrame(buf, len, opcode, fin);
- }
- // websocket fragment
- // lock ->
- // send(p, fragment, opcode, false) ->
- // send(p, fragment, WS_OPCODE_CONTINUE, false) ->
- // ... ->
- // send(p, remain, WS_OPCODE_CONTINUE, true)
- // unlock
- int send(const char* buf, int len, int fragment, enum ws_opcode opcode = WS_OPCODE_BINARY) {
- std::lock_guard<std::mutex> locker(mutex_);
- if (len <= fragment) {
- return sendFrame(buf, len, opcode, true);
- }
- // first fragment
- int nsend = sendFrame(buf, fragment, opcode, false);
- if (nsend < 0) return nsend;
- const char* p = buf + fragment;
- int remain = len - fragment;
- while (remain > fragment) {
- nsend = sendFrame(p, fragment, WS_OPCODE_CONTINUE, false);
- if (nsend < 0) return nsend;
- p += fragment;
- remain -= fragment;
- }
- // last fragment
- nsend = sendFrame(p, remain, WS_OPCODE_CONTINUE, true);
- if (nsend < 0) return nsend;
- return len;
- }
- int sendPing() {
- std::lock_guard<std::mutex> locker(mutex_);
- if (type == WS_CLIENT) {
- return write(WS_CLIENT_PING_FRAME, WS_CLIENT_MIN_FRAME_SIZE);
- }
- return write(WS_SERVER_PING_FRAME, WS_SERVER_MIN_FRAME_SIZE);
- }
- int sendPong() {
- std::lock_guard<std::mutex> locker(mutex_);
- if (type == WS_CLIENT) {
- return write(WS_CLIENT_PONG_FRAME, WS_CLIENT_MIN_FRAME_SIZE);
- }
- return write(WS_SERVER_PONG_FRAME, WS_SERVER_MIN_FRAME_SIZE);
- }
- protected:
- int sendFrame(const char* buf, int len, enum ws_opcode opcode = WS_OPCODE_BINARY, bool fin = true) {
- bool has_mask = false;
- char mask[4] = {0};
- if (type == WS_CLIENT) {
- *(int*)mask = rand();
- has_mask = true;
- }
- int frame_size = ws_calc_frame_size(len, has_mask);
- if (sendbuf_.len < frame_size) {
- sendbuf_.resize(ceil2e(frame_size));
- }
- ws_build_frame(sendbuf_.base, buf, len, mask, has_mask, opcode, fin);
- return write(sendbuf_.base, frame_size);
- }
- public:
- enum ws_opcode opcode;
- private:
- Buffer sendbuf_;
- std::mutex mutex_;
- };
- }
- typedef std::shared_ptr<hv::WebSocketChannel> WebSocketChannelPtr;
- #endif // HV_WEBSOCKET_CHANNEL_H_
|