This repository was archived by the owner on Oct 12, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathwsendpoint.hpp
193 lines (167 loc) · 5.88 KB
/
wsendpoint.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
/* vim: set et ts=4 sw=4 cindent:
*
* FreeRDP-WebConnect,
* A gateway for seamless access to your RDP-Sessions in any HTML5-compliant browser.
*
* Copyright 2012 Fritz Elfert <wsgate@fritz-elfert.de>
* This file has been partially derived from the WebSockets++ project at
* https://github.com/zaphoyd/websocketpp which is licensed under a BSD-license.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef WSENDPOINT_H
#define WSENDPOINT_H
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <vector>
#include <sstream>
#include <iostream>
#include <string>
#include <boost/thread.hpp>
#include "wsframe.hpp"
#include "wsgate.hpp"
#include "wshandler.hpp"
#ifndef HAVE_BOOST_LOCK_GUARD
#include <pthread.h>
/**
* Automatically unlocks a mutex if destroyed.
*/
class MutexHelper {
public:
/**
* Constructs a new instance.
* The associated mutex is locked by default.
* @param mutex The mutex to use.
* @param locknow false, if the mutex should not be locked initially
*/
MutexHelper(pthread_mutex_t *mutex, bool locknow = true);
/**
* Unlocks the associated mutex.
*/
~MutexHelper();
/**
* Locks the associated mutex.
*/
void Lock();
/**
* Unlocks the associated mutex.
*/
void Unlock();
private:
pthread_mutex_t *m_pMutex;
bool m_bLocked;
MutexHelper(const MutexHelper &);
MutexHelper & operator=(const MutexHelper &);
};
#endif
namespace wspp {
using wsgate::log;
/**
* This class implements a server-side WebSockets endpoint.
*/
class wsendpoint {
private:
// Non-copyable
wsendpoint(const wspp::wsendpoint&);
wsendpoint& operator=(const wspp::wsendpoint&);
public:
/**
* Constructor
* @param h The corresponding wshandler instance.
*/
wsendpoint(wshandler *h);
#ifndef HAVE_BOOST_LOCK_GUARD
~wsendpoint();
#endif
/**
* Processes incoming data from the client.
* The incoming data is decoded, according to RFC6455.
* If any message is completely assembled, the on_message
* method of the corresponding wshandler is invoked. For
* internal replys (e.g. PONG responses) the do_response
* method of the corresponding wshandler is used. All other
* on_xxx methods are called when the corresponding events occur.
*
* @param data the raw data, received from the client.
*/
void AddRxData(std::string data);
/**
* Send a data message.
* This method is invoked from the corresponding wshandler
* in order to send TEXT and BINARY payloads.
* @param payload The payload data.
* @param op The opcode according to RFC6455
*/
void send(const std::string& payload, frame::opcode::value op);
private:
void process_data();
/// Ends the connection by cleaning up based on current state
/** Terminate will review the outstanding resources and close each
* appropriately. Attached handlers will recieve an on_fail or on_close call
*
* TODO: should we protect against long running handlers?
*
* Visibility: protected
* State: Valid from any state except CLOSED.
* Concurrency: Must be called from within m_strand
*
*/
void shutdown();
/**
* Send Pong
* Initiates a pong with the given payload.
*
* There is no feedback from pong.
*
* Visibility: public
* State: Valid from OPEN, ignored otherwise
* Concurrency: callable from any thread
*
* @param payload Payload to be used for the pong
*/
void pong(const std::vector<unsigned char> & payload);
/// Send a close frame
/**
* Initiates a close handshake by sending a close frame with the given code
* and reason.
*
* Visibility: protected
* State: Valid for OPEN, ignored otherwise.
* Concurrency: Must be called within m_strand
*
* @param code The code to send
* @param reason The reason to send
*/
void send_close(close::status::value code, const std::string& reason);
/// send an acknowledgement close frame
/**
* Visibility: private
* State: no state checking, should only be called within process_control
* Concurrency: Must be called within m_stranded method
*/
void send_close_ack(close::status::value remote_close_code, std::string remote_close_reason);
void process_control();
private:
simple_rng m_rng;
frame::parser<simple_rng> m_parser;
session::state::value m_state;
#ifdef HAVE_BOOST_LOCK_GUARD
mutable boost::recursive_mutex m_lock;
#else
mutable pthread_mutex_t m_lock;
#endif
wshandler *m_handler;
};
}
#endif