diff options
| author | Mike Buland <mbuland@penny-arcade.com> | 2017-10-18 07:49:59 -0700 |
|---|---|---|
| committer | Mike Buland <mbuland@penny-arcade.com> | 2017-10-18 07:49:59 -0700 |
| commit | ea96e4decaa23fc8ddfb528d4851751ec9496490 (patch) | |
| tree | 585ed55a2fda09b388e02f1b6812893ac234f30a | |
| parent | 317a740f0198085937bf4fdbf032cc481d20f4ed (diff) | |
| download | libbu++-ea96e4decaa23fc8ddfb528d4851751ec9496490.tar.gz libbu++-ea96e4decaa23fc8ddfb528d4851751ec9496490.tar.bz2 libbu++-ea96e4decaa23fc8ddfb528d4851751ec9496490.tar.xz libbu++-ea96e4decaa23fc8ddfb528d4851751ec9496490.zip | |
I think this version works.
It would be really cool to add more features, like pre-websocket
negotiation and callbacks for serving web content, etc.
Diffstat (limited to '')
| -rw-r--r-- | src/unstable/protocolwebsocket.cpp | 300 | ||||
| -rw-r--r-- | src/unstable/protocolwebsocket.h | 64 |
2 files changed, 364 insertions, 0 deletions
diff --git a/src/unstable/protocolwebsocket.cpp b/src/unstable/protocolwebsocket.cpp new file mode 100644 index 0000000..9200904 --- /dev/null +++ b/src/unstable/protocolwebsocket.cpp | |||
| @@ -0,0 +1,300 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2014 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | #include "bu/config.h" | ||
| 9 | #include "bu/protocolwebsocket.h" | ||
| 10 | |||
| 11 | #include "bu/sio.h" | ||
| 12 | #include "bu/client.h" | ||
| 13 | |||
| 14 | #include "bu/membuf.h" | ||
| 15 | #include "bu/base64.h" | ||
| 16 | #include "bu/sha1.h" | ||
| 17 | #include "bu/json.h" | ||
| 18 | |||
| 19 | #include <stdlib.h> | ||
| 20 | |||
| 21 | Bu::ProtocolWebSocket::ProtocolWebSocket() : | ||
| 22 | eStatus( stProtoId ) | ||
| 23 | { | ||
| 24 | } | ||
| 25 | |||
| 26 | Bu::ProtocolWebSocket::~ProtocolWebSocket() | ||
| 27 | { | ||
| 28 | } | ||
| 29 | |||
| 30 | void Bu::ProtocolWebSocket::onNewConnection( Bu::Client *pClient ) | ||
| 31 | { | ||
| 32 | this->pClient = pClient; | ||
| 33 | } | ||
| 34 | |||
| 35 | void Bu::ProtocolWebSocket::onNewData( Bu::Client * /*pClient*/ ) | ||
| 36 | { | ||
| 37 | for(;;) | ||
| 38 | { | ||
| 39 | switch( eStatus ) | ||
| 40 | { | ||
| 41 | case stProtoId: | ||
| 42 | if( !stateProtoId() ) | ||
| 43 | return; | ||
| 44 | break; | ||
| 45 | |||
| 46 | case stHandshake: | ||
| 47 | if( !stateHandshake() ) | ||
| 48 | return; | ||
| 49 | break; | ||
| 50 | |||
| 51 | case stReady: | ||
| 52 | if( !parseMessage() ) | ||
| 53 | return; | ||
| 54 | break; | ||
| 55 | } | ||
| 56 | } | ||
| 57 | } | ||
| 58 | |||
| 59 | void Bu::ProtocolWebSocket::writeMessage( const Bu::String &sData, | ||
| 60 | Bu::ProtocolWebSocket::Operation eOp ) | ||
| 61 | { | ||
| 62 | uint8_t cHeader[32]; | ||
| 63 | //uint8_t *cMask; | ||
| 64 | memset( cHeader, 0, 32 ); | ||
| 65 | |||
| 66 | int idx = 2; | ||
| 67 | |||
| 68 | cHeader[0] = (((uint8_t)(eOp&0x0f)))|0x80; | ||
| 69 | |||
| 70 | uint64_t iLen = sData.getSize(); | ||
| 71 | if( iLen < 126 ) | ||
| 72 | { | ||
| 73 | cHeader[1] = ((uint8_t)iLen); | ||
| 74 | } | ||
| 75 | else if( iLen < 2147483648 ) | ||
| 76 | { | ||
| 77 | cHeader[1] = ((uint8_t)126); | ||
| 78 | uint16_t uLen = iLen; | ||
| 79 | uLen = htobe16( uLen ); | ||
| 80 | memcpy( cHeader+idx, &uLen, 2 ); | ||
| 81 | idx += 2; | ||
| 82 | } | ||
| 83 | else | ||
| 84 | { | ||
| 85 | cHeader[1] = ((uint8_t)127); | ||
| 86 | uint64_t iTmp = htobe64( iLen ); | ||
| 87 | memcpy( cHeader+idx, &iTmp, 8 ); | ||
| 88 | idx += 8; | ||
| 89 | } | ||
| 90 | /* | ||
| 91 | Bu::println("Message size: %1 (%2)").arg( iLen ).arg( iLen, Bu::Fmt::bin(4) ); | ||
| 92 | for( int j = 0; j < idx; j++ ) | ||
| 93 | { | ||
| 94 | Bu::print(" %1").arg( cHeader[j], Bu::Fmt::bin(8) ); | ||
| 95 | } | ||
| 96 | Bu::println(""); | ||
| 97 | */ | ||
| 98 | pClient->write( cHeader, idx ); | ||
| 99 | pClient->write( sData ); | ||
| 100 | } | ||
| 101 | |||
| 102 | bool Bu::ProtocolWebSocket::stateProtoId() | ||
| 103 | { | ||
| 104 | Bu::String sLine; | ||
| 105 | if( !readHttpHdrLine( sLine ) ) | ||
| 106 | return false; | ||
| 107 | |||
| 108 | Bu::StringList lChunks = sLine.split(' '); | ||
| 109 | if( lChunks.getSize() != 3 ) | ||
| 110 | { | ||
| 111 | pClient->disconnect(); | ||
| 112 | return false; | ||
| 113 | } | ||
| 114 | Bu::StringList::iterator i = lChunks.begin(); | ||
| 115 | if( *i != "GET" ) | ||
| 116 | { | ||
| 117 | pClient->disconnect(); | ||
| 118 | return false; | ||
| 119 | } | ||
| 120 | sPath = *(++i); | ||
| 121 | if( *(++i) != "HTTP/1.1" ) | ||
| 122 | { | ||
| 123 | pClient->disconnect(); | ||
| 124 | return false; | ||
| 125 | } | ||
| 126 | |||
| 127 | eStatus = stHandshake; | ||
| 128 | |||
| 129 | return true; | ||
| 130 | } | ||
| 131 | |||
| 132 | bool Bu::ProtocolWebSocket::stateHandshake() | ||
| 133 | { | ||
| 134 | Bu::String sLine; | ||
| 135 | if( !readHttpHdrLine( sLine ) ) | ||
| 136 | return false; | ||
| 137 | |||
| 138 | if( sLine.getSize() == 0 ) | ||
| 139 | { | ||
| 140 | if( !processHeaders() ) | ||
| 141 | return false; | ||
| 142 | onHandshakeComplete(); | ||
| 143 | eStatus = stReady; | ||
| 144 | return true; | ||
| 145 | } | ||
| 146 | |||
| 147 | int iPos = sLine.findIdx(':'); | ||
| 148 | if( iPos < 0 ) | ||
| 149 | { | ||
| 150 | pClient->disconnect(); | ||
| 151 | return false; | ||
| 152 | } | ||
| 153 | Bu::String sKey( sLine, iPos ); | ||
| 154 | Bu::String sValue( sLine.getSubStrIdx( iPos+2 ) ); | ||
| 155 | |||
| 156 | if( !hHeader.has( sKey ) ) | ||
| 157 | { | ||
| 158 | hHeader.insert( sKey, Bu::StringList() ); | ||
| 159 | } | ||
| 160 | hHeader.get( sKey ).append( sValue ); | ||
| 161 | |||
| 162 | // Bu::println("Hdr: >>%1<<").arg( sLine ); | ||
| 163 | // Bu::println("%1 = %2").arg( sKey ).arg( sValue ); | ||
| 164 | |||
| 165 | return true; | ||
| 166 | } | ||
| 167 | |||
| 168 | bool Bu::ProtocolWebSocket::readHttpHdrLine( Bu::String &sLine ) | ||
| 169 | { | ||
| 170 | char buf[1024]; | ||
| 171 | int iSize = pClient->peek( buf, 1024 ); | ||
| 172 | for( int j = 0; j < iSize-1; j++ ) | ||
| 173 | { | ||
| 174 | if( buf[j] == '\r' && buf[j+1] == '\n' ) | ||
| 175 | { | ||
| 176 | pClient->seek(j+2); | ||
| 177 | sLine.set( buf, j ); | ||
| 178 | return true; | ||
| 179 | } | ||
| 180 | } | ||
| 181 | return false; | ||
| 182 | } | ||
| 183 | |||
| 184 | bool Bu::ProtocolWebSocket::processHeaders() | ||
| 185 | { | ||
| 186 | if( !headerMatch("Connection", "Upgrade") || | ||
| 187 | !headerMatch("Upgrade", "websocket") || | ||
| 188 | !headerMatch("Sec-WebSocket-Version", "13") ) | ||
| 189 | { | ||
| 190 | pClient->disconnect(); | ||
| 191 | return false; | ||
| 192 | } | ||
| 193 | |||
| 194 | Bu::String sNonce; | ||
| 195 | if( !hHeader.has("Sec-WebSocket-Key") ) | ||
| 196 | { | ||
| 197 | pClient->disconnect(); | ||
| 198 | return false; | ||
| 199 | } | ||
| 200 | |||
| 201 | sNonce = hHeader.get("Sec-WebSocket-Key").first(); | ||
| 202 | |||
| 203 | Bu::String sGuid = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; | ||
| 204 | |||
| 205 | Bu::Sha1 sum; | ||
| 206 | sum.addData( sNonce+sGuid ); | ||
| 207 | Bu::MemBuf mbOut; | ||
| 208 | Bu::Base64 bOut( mbOut ); | ||
| 209 | sum.writeResult( bOut ); | ||
| 210 | bOut.stop(); | ||
| 211 | |||
| 212 | Bu::println("accept: %1").arg( mbOut.getString() ); | ||
| 213 | |||
| 214 | pClient->write("HTTP/1.1 101 Switching Protocols\r\n" | ||
| 215 | "Upgrade: websocket\r\n" | ||
| 216 | "Connection: Upgrade\r\n" | ||
| 217 | "Sec-WebSocket-Accept: " + mbOut.getString() + "\r\n" | ||
| 218 | "\r\n" | ||
| 219 | ); | ||
| 220 | |||
| 221 | return true; | ||
| 222 | } | ||
| 223 | |||
| 224 | bool Bu::ProtocolWebSocket::headerMatch( const Bu::String &sKey, const Bu::String &sValue ) | ||
| 225 | { | ||
| 226 | if( !hHeader.has( sKey ) ) | ||
| 227 | return false; | ||
| 228 | |||
| 229 | for( Bu::StringList::iterator i = hHeader.get( sKey ).begin(); i; i++ ) | ||
| 230 | { | ||
| 231 | if( *i == sValue ) | ||
| 232 | return true; | ||
| 233 | } | ||
| 234 | |||
| 235 | return false; | ||
| 236 | } | ||
| 237 | |||
| 238 | bool Bu::ProtocolWebSocket::parseMessage() | ||
| 239 | { | ||
| 240 | if( pClient->getInputSize() < 2 ) | ||
| 241 | return false; | ||
| 242 | |||
| 243 | uint8_t buf[32]; | ||
| 244 | int64_t iTgtLength = 2; | ||
| 245 | pClient->peek( buf, 2, 0 ); | ||
| 246 | Operation eOp = (Operation)(buf[0]&0x0f); | ||
| 247 | int64_t iLen = buf[1]&(~0x80); | ||
| 248 | bool bMasked = (buf[1]&0x80) == 0x80; | ||
| 249 | if( iLen == 126 ) | ||
| 250 | { | ||
| 251 | uint16_t iLenBuf; | ||
| 252 | if( pClient->getInputSize() < iTgtLength+2 ) | ||
| 253 | return false; | ||
| 254 | pClient->peek( &iLenBuf, 2, iTgtLength ); | ||
| 255 | iTgtLength += 2; | ||
| 256 | iLen = be16toh( iLenBuf ); | ||
| 257 | } | ||
| 258 | else if( iLen == 127 ) | ||
| 259 | { | ||
| 260 | if( pClient->getInputSize() < iTgtLength+8 ) | ||
| 261 | return false; | ||
| 262 | pClient->peek( &iLen, 8, iTgtLength ); | ||
| 263 | iTgtLength += 8; | ||
| 264 | iLen = be64toh( iLen ); | ||
| 265 | } | ||
| 266 | char cMask[4]; | ||
| 267 | if( bMasked ) | ||
| 268 | { | ||
| 269 | if( pClient->getInputSize() < iTgtLength+4 ) | ||
| 270 | return false; | ||
| 271 | pClient->peek( cMask, 4, iTgtLength ); | ||
| 272 | iTgtLength += 4; | ||
| 273 | } | ||
| 274 | if( pClient->getInputSize() < iTgtLength+iLen ) | ||
| 275 | return false; | ||
| 276 | |||
| 277 | pClient->seek( iTgtLength ); | ||
| 278 | |||
| 279 | Bu::String sData( iLen ); | ||
| 280 | int iRead = 0; | ||
| 281 | do | ||
| 282 | { | ||
| 283 | iRead += pClient->read( sData.getStr()+iRead, iLen-iRead ); | ||
| 284 | } while( iRead < iLen ); | ||
| 285 | if( bMasked ) | ||
| 286 | { | ||
| 287 | for( int j = 0; j < iLen; j++ ) | ||
| 288 | { | ||
| 289 | sData[j] = sData[j]^cMask[j%4]; | ||
| 290 | } | ||
| 291 | } | ||
| 292 | |||
| 293 | Bu::println(""); | ||
| 294 | Bu::println("Data: >>%1<<").arg( sData ); | ||
| 295 | |||
| 296 | onNewMessage( sData, eOp ); | ||
| 297 | |||
| 298 | return true; | ||
| 299 | } | ||
| 300 | |||
diff --git a/src/unstable/protocolwebsocket.h b/src/unstable/protocolwebsocket.h new file mode 100644 index 0000000..cf00d34 --- /dev/null +++ b/src/unstable/protocolwebsocket.h | |||
| @@ -0,0 +1,64 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2014 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | #ifndef BU_PROTOCOL_WEB_SOCKET_H | ||
| 9 | #define BU_PROTOCOL_WEB_SOCKET_H | ||
| 10 | |||
| 11 | #include "bu/protocol.h" | ||
| 12 | #include "bu/hash.h" | ||
| 13 | |||
| 14 | namespace Bu | ||
| 15 | { | ||
| 16 | class ProtocolWebSocket : public Bu::Protocol | ||
| 17 | { | ||
| 18 | public: | ||
| 19 | enum Operation | ||
| 20 | { | ||
| 21 | Continuation = 0, | ||
| 22 | Text = 1, | ||
| 23 | Binary = 2, | ||
| 24 | ConClose = 8, | ||
| 25 | Ping = 9, | ||
| 26 | Pong = 10 | ||
| 27 | }; | ||
| 28 | |||
| 29 | public: | ||
| 30 | ProtocolWebSocket(); | ||
| 31 | virtual ~ProtocolWebSocket(); | ||
| 32 | |||
| 33 | virtual void onNewConnection( Bu::Client *pClient ); | ||
| 34 | virtual void onNewData( Bu::Client *pClient ); | ||
| 35 | |||
| 36 | virtual void onHandshakeComplete()=0; | ||
| 37 | virtual void onNewMessage( const Bu::String &sData, Operation eOp )=0; | ||
| 38 | void writeMessage( const Bu::String &sData, Operation eOp=Text ); | ||
| 39 | |||
| 40 | private: | ||
| 41 | bool stateProtoId(); | ||
| 42 | bool stateHandshake(); | ||
| 43 | bool processHeaders(); | ||
| 44 | bool headerMatch( const Bu::String &sKey, const Bu::String &sValue ); | ||
| 45 | bool readHttpHdrLine( Bu::String &sLine ); | ||
| 46 | bool parseMessage(); | ||
| 47 | |||
| 48 | private: | ||
| 49 | enum Status | ||
| 50 | { | ||
| 51 | stProtoId = 0, | ||
| 52 | stHandshake = 1, | ||
| 53 | stReady = 10 | ||
| 54 | }; | ||
| 55 | |||
| 56 | protected: | ||
| 57 | Bu::Client *pClient; | ||
| 58 | Status eStatus; | ||
| 59 | Bu::String sPath; | ||
| 60 | Bu::Hash<Bu::String, Bu::StringList> hHeader; | ||
| 61 | }; | ||
| 62 | } | ||
| 63 | |||
| 64 | #endif | ||
