blob: 9b372dcca95d61a371e923d6103afa80d097683e [file] [log] [blame]
henrike@webrtc.orgf0488722014-05-13 18:00:26 +00001/*
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef RTC_BASE_TESTUTILS_H_
12#define RTC_BASE_TESTUTILS_H_
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000013
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +020014// Utilities for testing rtc infrastructure in unittests
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000015
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +020016#if defined(WEBRTC_LINUX) && !defined(WEBRTC_ANDROID)
17#include <X11/Xlib.h>
18#include <X11/extensions/Xrandr.h>
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000019
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +020020// X defines a few macros that stomp on types that gunit.h uses.
21#undef None
22#undef Bool
23#endif
24
25#include <algorithm>
26#include <map>
27#include <memory>
28#include <vector>
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020029#include "rtc_base/arraysize.h"
30#include "rtc_base/asyncsocket.h"
31#include "rtc_base/checks.h"
32#include "rtc_base/gunit.h"
33#include "rtc_base/nethelpers.h"
34#include "rtc_base/pathutils.h"
35#include "rtc_base/stream.h"
36#include "rtc_base/stringencode.h"
37#include "rtc_base/stringutils.h"
38#include "rtc_base/thread.h"
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +020039
40namespace webrtc {
41namespace testing {
42
43using namespace rtc;
44
45///////////////////////////////////////////////////////////////////////////////
46// StreamSink - Monitor asynchronously signalled events from StreamInterface
47// or AsyncSocket (which should probably be a StreamInterface.
48///////////////////////////////////////////////////////////////////////////////
49
50// Note: Any event that is an error is treaded as SSE_ERROR instead of that
51// event.
52
53enum StreamSinkEvent {
54 SSE_OPEN = SE_OPEN,
55 SSE_READ = SE_READ,
56 SSE_WRITE = SE_WRITE,
57 SSE_CLOSE = SE_CLOSE,
58 SSE_ERROR = 16
59};
60
61class StreamSink : public sigslot::has_slots<> {
62 public:
Steve Anton9de3aac2017-10-24 10:08:26 -070063 StreamSink();
64 ~StreamSink() override;
65
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +020066 void Monitor(StreamInterface* stream) {
67 stream->SignalEvent.connect(this, &StreamSink::OnEvent);
68 events_.erase(stream);
69 }
70 void Unmonitor(StreamInterface* stream) {
71 stream->SignalEvent.disconnect(this);
72 // In case you forgot to unmonitor a previous object with this address
73 events_.erase(stream);
74 }
75 bool Check(StreamInterface* stream, StreamSinkEvent event, bool reset = true) {
76 return DoCheck(stream, event, reset);
77 }
78 int Events(StreamInterface* stream, bool reset = true) {
79 return DoEvents(stream, reset);
80 }
81
82 void Monitor(AsyncSocket* socket) {
83 socket->SignalConnectEvent.connect(this, &StreamSink::OnConnectEvent);
84 socket->SignalReadEvent.connect(this, &StreamSink::OnReadEvent);
85 socket->SignalWriteEvent.connect(this, &StreamSink::OnWriteEvent);
86 socket->SignalCloseEvent.connect(this, &StreamSink::OnCloseEvent);
87 // In case you forgot to unmonitor a previous object with this address
88 events_.erase(socket);
89 }
90 void Unmonitor(AsyncSocket* socket) {
91 socket->SignalConnectEvent.disconnect(this);
92 socket->SignalReadEvent.disconnect(this);
93 socket->SignalWriteEvent.disconnect(this);
94 socket->SignalCloseEvent.disconnect(this);
95 events_.erase(socket);
96 }
97 bool Check(AsyncSocket* socket, StreamSinkEvent event, bool reset = true) {
98 return DoCheck(socket, event, reset);
99 }
100 int Events(AsyncSocket* socket, bool reset = true) {
101 return DoEvents(socket, reset);
102 }
103
104 private:
105 typedef std::map<void*,int> EventMap;
106
107 void OnEvent(StreamInterface* stream, int events, int error) {
108 if (error) {
109 events = SSE_ERROR;
110 }
111 AddEvents(stream, events);
112 }
113 void OnConnectEvent(AsyncSocket* socket) {
114 AddEvents(socket, SSE_OPEN);
115 }
116 void OnReadEvent(AsyncSocket* socket) {
117 AddEvents(socket, SSE_READ);
118 }
119 void OnWriteEvent(AsyncSocket* socket) {
120 AddEvents(socket, SSE_WRITE);
121 }
122 void OnCloseEvent(AsyncSocket* socket, int error) {
123 AddEvents(socket, (0 == error) ? SSE_CLOSE : SSE_ERROR);
124 }
125
126 void AddEvents(void* obj, int events) {
127 EventMap::iterator it = events_.find(obj);
128 if (events_.end() == it) {
129 events_.insert(EventMap::value_type(obj, events));
130 } else {
131 it->second |= events;
132 }
133 }
134 bool DoCheck(void* obj, StreamSinkEvent event, bool reset) {
135 EventMap::iterator it = events_.find(obj);
136 if ((events_.end() == it) || (0 == (it->second & event))) {
137 return false;
138 }
139 if (reset) {
140 it->second &= ~event;
141 }
142 return true;
143 }
144 int DoEvents(void* obj, bool reset) {
145 EventMap::iterator it = events_.find(obj);
146 if (events_.end() == it)
147 return 0;
148 int events = it->second;
149 if (reset) {
150 it->second = 0;
151 }
152 return events;
153 }
154
155 EventMap events_;
156};
157
158///////////////////////////////////////////////////////////////////////////////
159// StreamSource - Implements stream interface and simulates asynchronous
160// events on the stream, without a network. Also buffers written data.
161///////////////////////////////////////////////////////////////////////////////
162
163class StreamSource : public StreamInterface {
164public:
Steve Anton9de3aac2017-10-24 10:08:26 -0700165 StreamSource();
166 ~StreamSource() override;
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200167
Steve Anton9de3aac2017-10-24 10:08:26 -0700168 void Clear() {
169 readable_data_.clear();
170 written_data_.clear();
171 state_ = SS_CLOSED;
172 read_block_ = 0;
173 write_block_ = SIZE_UNKNOWN;
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200174 }
175 void QueueString(const char* data) {
176 QueueData(data, strlen(data));
177 }
Niels Möller00f934a2017-12-14 13:30:46 +0100178#if defined(__GNUC__)
179 // Note: Implicit |this| argument counts as the first argument.
180 __attribute__((__format__(__printf__, 2, 3)))
181#endif
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200182 void QueueStringF(const char* format, ...) {
183 va_list args;
184 va_start(args, format);
185 char buffer[1024];
186 size_t len = vsprintfn(buffer, sizeof(buffer), format, args);
187 RTC_CHECK(len < sizeof(buffer) - 1);
188 va_end(args);
189 QueueData(buffer, len);
190 }
191 void QueueData(const char* data, size_t len) {
192 readable_data_.insert(readable_data_.end(), data, data + len);
193 if ((SS_OPEN == state_) && (readable_data_.size() == len)) {
194 SignalEvent(this, SE_READ, 0);
195 }
196 }
197 std::string ReadData() {
198 std::string data;
199 // avoid accessing written_data_[0] if it is undefined
200 if (written_data_.size() > 0) {
201 data.insert(0, &written_data_[0], written_data_.size());
202 }
203 written_data_.clear();
204 return data;
205 }
206 void SetState(StreamState state) {
207 int events = 0;
208 if ((SS_OPENING == state_) && (SS_OPEN == state)) {
209 events |= SE_OPEN;
210 if (!readable_data_.empty()) {
211 events |= SE_READ;
212 }
213 } else if ((SS_CLOSED != state_) && (SS_CLOSED == state)) {
214 events |= SE_CLOSE;
215 }
216 state_ = state;
217 if (events) {
218 SignalEvent(this, events, 0);
219 }
220 }
221 // Will cause Read to block when there are pos bytes in the read queue.
222 void SetReadBlock(size_t pos) { read_block_ = pos; }
223 // Will cause Write to block when there are pos bytes in the write queue.
224 void SetWriteBlock(size_t pos) { write_block_ = pos; }
225
Steve Anton9de3aac2017-10-24 10:08:26 -0700226 StreamState GetState() const override;
227 StreamResult Read(void* buffer,
228 size_t buffer_len,
229 size_t* read,
230 int* error) override;
231 StreamResult Write(const void* data,
232 size_t data_len,
233 size_t* written,
234 int* error) override;
235 void Close() override;
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200236
Steve Anton9de3aac2017-10-24 10:08:26 -0700237 private:
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200238 typedef std::vector<char> Buffer;
239 Buffer readable_data_, written_data_;
240 StreamState state_;
241 size_t read_block_, write_block_;
242};
243
244///////////////////////////////////////////////////////////////////////////////
245// SocketTestClient
246// Creates a simulated client for testing. Works on real and virtual networks.
247///////////////////////////////////////////////////////////////////////////////
248
249class SocketTestClient : public sigslot::has_slots<> {
250public:
Steve Anton9de3aac2017-10-24 10:08:26 -0700251 SocketTestClient();
252 SocketTestClient(AsyncSocket* socket);
253 SocketTestClient(const SocketAddress& address);
254 ~SocketTestClient() override;
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200255
Steve Anton9de3aac2017-10-24 10:08:26 -0700256 AsyncSocket* socket() { return socket_.get(); }
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200257
Steve Anton9de3aac2017-10-24 10:08:26 -0700258 void QueueString(const char* data) { QueueData(data, strlen(data)); }
259 void QueueStringF(const char* format, ...) {
260 va_list args;
261 va_start(args, format);
262 char buffer[1024];
263 size_t len = vsprintfn(buffer, sizeof(buffer), format, args);
264 RTC_CHECK(len < sizeof(buffer) - 1);
265 va_end(args);
266 QueueData(buffer, len);
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200267 }
268 void QueueData(const char* data, size_t len) {
269 send_buffer_.insert(send_buffer_.end(), data, data + len);
270 if (Socket::CS_CONNECTED == socket_->GetState()) {
271 Flush();
272 }
273 }
274 std::string ReadData() {
275 std::string data(&recv_buffer_[0], recv_buffer_.size());
276 recv_buffer_.clear();
277 return data;
278 }
279
280 bool IsConnected() const {
281 return (Socket::CS_CONNECTED == socket_->GetState());
282 }
283 bool IsClosed() const {
284 return (Socket::CS_CLOSED == socket_->GetState());
285 }
286
287private:
288 typedef std::vector<char> Buffer;
289
290 void Init(AsyncSocket* socket, int family) {
291 if (!socket) {
292 socket = Thread::Current()->socketserver()
293 ->CreateAsyncSocket(family, SOCK_STREAM);
294 }
295 socket_.reset(socket);
296 socket_->SignalConnectEvent.connect(this,
297 &SocketTestClient::OnConnectEvent);
298 socket_->SignalReadEvent.connect(this, &SocketTestClient::OnReadEvent);
299 socket_->SignalWriteEvent.connect(this, &SocketTestClient::OnWriteEvent);
300 socket_->SignalCloseEvent.connect(this, &SocketTestClient::OnCloseEvent);
301 }
302
303 void Flush() {
304 size_t sent = 0;
305 while (sent < send_buffer_.size()) {
306 int result = socket_->Send(&send_buffer_[sent],
307 send_buffer_.size() - sent);
308 if (result > 0) {
309 sent += result;
310 } else {
311 break;
312 }
313 }
314 size_t new_size = send_buffer_.size() - sent;
315 memmove(&send_buffer_[0], &send_buffer_[sent], new_size);
316 send_buffer_.resize(new_size);
317 }
318
319 void OnConnectEvent(AsyncSocket* socket) {
320 if (!send_buffer_.empty()) {
321 Flush();
322 }
323 }
324 void OnReadEvent(AsyncSocket* socket) {
325 char data[64 * 1024];
326 int result = socket_->Recv(data, arraysize(data), nullptr);
327 if (result > 0) {
328 recv_buffer_.insert(recv_buffer_.end(), data, data + result);
329 }
330 }
331 void OnWriteEvent(AsyncSocket* socket) {
332 if (!send_buffer_.empty()) {
333 Flush();
334 }
335 }
336 void OnCloseEvent(AsyncSocket* socket, int error) {
337 }
338
339 std::unique_ptr<AsyncSocket> socket_;
340 Buffer send_buffer_, recv_buffer_;
341};
342
343///////////////////////////////////////////////////////////////////////////////
344// SocketTestServer
345// Creates a simulated server for testing. Works on real and virtual networks.
346///////////////////////////////////////////////////////////////////////////////
347
348class SocketTestServer : public sigslot::has_slots<> {
349 public:
Steve Anton9de3aac2017-10-24 10:08:26 -0700350 SocketTestServer(const SocketAddress& address);
351 ~SocketTestServer() override;
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200352
353 size_t size() const { return clients_.size(); }
354 SocketTestClient* client(size_t index) const { return clients_[index]; }
355 SocketTestClient* operator[](size_t index) const { return client(index); }
356
357 void clear() {
358 for (size_t i=0; i<clients_.size(); ++i) {
359 delete clients_[i];
360 }
361 clients_.clear();
362 }
363
364 private:
365 void OnReadEvent(AsyncSocket* socket) {
366 AsyncSocket* accepted = static_cast<AsyncSocket*>(socket_->Accept(nullptr));
367 if (!accepted)
368 return;
369 clients_.push_back(new SocketTestClient(accepted));
370 }
371
372 std::unique_ptr<AsyncSocket> socket_;
373 std::vector<SocketTestClient*> clients_;
374};
375
376///////////////////////////////////////////////////////////////////////////////
377// Unittest predicates which are similar to STREQ, but for raw memory
378///////////////////////////////////////////////////////////////////////////////
379
380inline ::testing::AssertionResult CmpHelperMemEq(
381 const char* expected_expression,
382 const char* expected_length_expression,
383 const char* actual_expression,
384 const char* actual_length_expression,
385 const void* expected,
386 size_t expected_length,
387 const void* actual,
388 size_t actual_length) {
389 if ((expected_length == actual_length)
390 && (0 == memcmp(expected, actual, expected_length))) {
391 return ::testing::AssertionSuccess();
392 }
393
394 ::testing::Message msg;
395 msg << "Value of: " << actual_expression
396 << " [" << actual_length_expression << "]";
397 if (true) { //!actual_value.Equals(actual_expression)) {
398 size_t buffer_size = actual_length * 2 + 1;
399 char* buffer = STACK_ARRAY(char, buffer_size);
400 hex_encode(buffer, buffer_size,
401 reinterpret_cast<const char*>(actual), actual_length);
402 msg << "\n Actual: " << buffer << " [" << actual_length << "]";
403 }
404
405 msg << "\nExpected: " << expected_expression
406 << " [" << expected_length_expression << "]";
407 if (true) { //!expected_value.Equals(expected_expression)) {
408 size_t buffer_size = expected_length * 2 + 1;
409 char* buffer = STACK_ARRAY(char, buffer_size);
410 hex_encode(buffer, buffer_size,
411 reinterpret_cast<const char*>(expected), expected_length);
412 msg << "\nWhich is: " << buffer << " [" << expected_length << "]";
413 }
414
415 return AssertionFailure(msg);
416}
417
418#define EXPECT_MEMEQ(expected, expected_length, actual, actual_length) \
419 EXPECT_PRED_FORMAT4(::testing::CmpHelperMemEq, expected, expected_length, \
420 actual, actual_length)
421
422#define ASSERT_MEMEQ(expected, expected_length, actual, actual_length) \
423 ASSERT_PRED_FORMAT4(::testing::CmpHelperMemEq, expected, expected_length, \
424 actual, actual_length)
425
426///////////////////////////////////////////////////////////////////////////////
427// Helpers for initializing constant memory with integers in a particular byte
428// order
429///////////////////////////////////////////////////////////////////////////////
430
431#define BYTE_CAST(x) static_cast<uint8_t>((x)&0xFF)
432
433// Declare a N-bit integer as a little-endian sequence of bytes
434#define LE16(x) BYTE_CAST(((uint16_t)x) >> 0), BYTE_CAST(((uint16_t)x) >> 8)
435
436#define LE32(x) \
437 BYTE_CAST(((uint32_t)x) >> 0), BYTE_CAST(((uint32_t)x) >> 8), \
438 BYTE_CAST(((uint32_t)x) >> 16), BYTE_CAST(((uint32_t)x) >> 24)
439
440#define LE64(x) \
441 BYTE_CAST(((uint64_t)x) >> 0), BYTE_CAST(((uint64_t)x) >> 8), \
442 BYTE_CAST(((uint64_t)x) >> 16), BYTE_CAST(((uint64_t)x) >> 24), \
443 BYTE_CAST(((uint64_t)x) >> 32), BYTE_CAST(((uint64_t)x) >> 40), \
444 BYTE_CAST(((uint64_t)x) >> 48), BYTE_CAST(((uint64_t)x) >> 56)
445
446// Declare a N-bit integer as a big-endian (Internet) sequence of bytes
447#define BE16(x) BYTE_CAST(((uint16_t)x) >> 8), BYTE_CAST(((uint16_t)x) >> 0)
448
449#define BE32(x) \
450 BYTE_CAST(((uint32_t)x) >> 24), BYTE_CAST(((uint32_t)x) >> 16), \
451 BYTE_CAST(((uint32_t)x) >> 8), BYTE_CAST(((uint32_t)x) >> 0)
452
453#define BE64(x) \
454 BYTE_CAST(((uint64_t)x) >> 56), BYTE_CAST(((uint64_t)x) >> 48), \
455 BYTE_CAST(((uint64_t)x) >> 40), BYTE_CAST(((uint64_t)x) >> 32), \
456 BYTE_CAST(((uint64_t)x) >> 24), BYTE_CAST(((uint64_t)x) >> 16), \
457 BYTE_CAST(((uint64_t)x) >> 8), BYTE_CAST(((uint64_t)x) >> 0)
458
459// Declare a N-bit integer as a this-endian (local machine) sequence of bytes
460#ifndef BIG_ENDIAN
461#define BIG_ENDIAN 1
462#endif // BIG_ENDIAN
463
464#if BIG_ENDIAN
465#define TE16 BE16
466#define TE32 BE32
467#define TE64 BE64
468#else // !BIG_ENDIAN
469#define TE16 LE16
470#define TE32 LE32
471#define TE64 LE64
472#endif // !BIG_ENDIAN
473
474///////////////////////////////////////////////////////////////////////////////
475
476// Helpers for determining if X/screencasting is available (on linux).
477
Mirko Bonadei675513b2017-11-09 11:09:25 +0100478#define MAYBE_SKIP_SCREENCAST_TEST() \
479 if (!testing::IsScreencastingAvailable()) { \
480 RTC_LOG(LS_WARNING) \
481 << "Skipping test, since it doesn't have the requisite " \
482 << "X environment for screen capture."; \
483 return; \
484 }
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200485
486#if defined(WEBRTC_LINUX) && !defined(WEBRTC_ANDROID)
487struct XDisplay {
488 XDisplay() : display_(XOpenDisplay(nullptr)) {}
489 ~XDisplay() { if (display_) XCloseDisplay(display_); }
490 bool IsValid() const { return display_ != nullptr; }
491 operator Display*() { return display_; }
492 private:
493 Display* display_;
494};
495#endif
496
497// Returns true if screencasting is available. When false, anything that uses
498// screencasting features may fail.
499inline bool IsScreencastingAvailable() {
500#if defined(WEBRTC_LINUX) && !defined(WEBRTC_ANDROID)
501 XDisplay display;
502 if (!display.IsValid()) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100503 RTC_LOG(LS_WARNING) << "No X Display available.";
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200504 return false;
505 }
506 int ignored_int, major_version, minor_version;
507 if (!XRRQueryExtension(display, &ignored_int, &ignored_int) ||
Steve Anton9de3aac2017-10-24 10:08:26 -0700508 !XRRQueryVersion(display, &major_version, &minor_version)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100509 RTC_LOG(LS_WARNING) << "XRandr is not supported.";
Steve Anton9de3aac2017-10-24 10:08:26 -0700510 return false;
511 }
512 if (major_version < 1 || (major_version < 2 && minor_version < 3)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100513 RTC_LOG(LS_WARNING) << "XRandr is too old (version: " << major_version
514 << "." << minor_version << "). Need 1.3 or later.";
Henrik Kjellanderec78f1c2017-06-29 07:52:50 +0200515 return false;
516 }
517#endif
518 return true;
519}
520
521} // namespace testing
522} // namespace webrtc
523
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200524#endif // RTC_BASE_TESTUTILS_H_