/* * Copyright 2004 The WebRTC Project Authors. All rights reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #ifndef RTC_BASE_TEST_UTILS_H_ #define RTC_BASE_TEST_UTILS_H_ // Utilities for testing rtc infrastructure in unittests #include <map> #include <utility> #include "rtc_base/socket.h" #include "rtc_base/third_party/sigslot/sigslot.h" namespace webrtc { namespace testing { /////////////////////////////////////////////////////////////////////////////// // StreamSink - Monitor asynchronously signalled events from Socket. /////////////////////////////////////////////////////////////////////////////// // Note: Any event that is an error is treated as SSE_ERROR instead of that // event. enum StreamSinkEvent { SSE_OPEN = 1, SSE_READ = 2, SSE_WRITE = 4, SSE_CLOSE = 8, SSE_ERROR = 16 }; class StreamSink : public sigslot::has_slots<> { public: StreamSink(); ~StreamSink() override; void Monitor(rtc::Socket* socket) { socket->SignalConnectEvent.connect(this, &StreamSink::OnConnectEvent); socket->SignalReadEvent.connect(this, &StreamSink::OnReadEvent); socket->SignalWriteEvent.connect(this, &StreamSink::OnWriteEvent); socket->SignalCloseEvent.connect(this, &StreamSink::OnCloseEvent); // In case you forgot to unmonitor a previous object with this address events_.erase(socket); } void Unmonitor(rtc::Socket* socket) { socket->SignalConnectEvent.disconnect(this); socket->SignalReadEvent.disconnect(this); socket->SignalWriteEvent.disconnect(this); socket->SignalCloseEvent.disconnect(this); events_.erase(socket); } bool Check(rtc::Socket* socket, StreamSinkEvent event, bool reset = true) { return DoCheck(socket, event, reset); } private: typedef std::map<rtc::Socket*, int> EventMap; void OnConnectEvent(rtc::Socket* socket) { AddEvents(socket, SSE_OPEN); } void OnReadEvent(rtc::Socket* socket) { AddEvents(socket, SSE_READ); } void OnWriteEvent(rtc::Socket* socket) { AddEvents(socket, SSE_WRITE); } void OnCloseEvent(rtc::Socket* socket, int error) { AddEvents(socket, (0 == error) ? SSE_CLOSE : SSE_ERROR); } void AddEvents(rtc::Socket* obj, int events) { EventMap::iterator it = events_.find(obj); if (events_.end() == it) { events_.insert(EventMap::value_type(obj, events)); } else { it->second |= events; } } bool DoCheck(rtc::Socket* obj, StreamSinkEvent event, bool reset) { EventMap::iterator it = events_.find(obj); if ((events_.end() == it) || (0 == (it->second & event))) { return false; } if (reset) { it->second &= ~event; } return true; } EventMap events_; }; } // namespace testing } // namespace webrtc #endif // RTC_BASE_TEST_UTILS_H_