|  | /* | 
|  | *  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_ |