blob: ea959c943afa3dfa7048a41d345e38a513ffdd89 [file] [log] [blame]
Sebastian Jansson7cbee842019-08-06 15:19:381/*
2 * Copyright (c) 2019 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#include "test/peer_scenario/peer_scenario.h"
11
Sebastian Jansson64672dc2019-10-01 13:23:3912#include "absl/flags/flag.h"
Sebastian Jansson7cbee842019-08-06 15:19:3813#include "absl/memory/memory.h"
Sebastian Janssonfc8279d2020-01-16 10:45:5914#include "rtc_base/null_socket_server.h"
Sebastian Jansson64672dc2019-10-01 13:23:3915#include "rtc_base/string_encode.h"
16#include "rtc_base/strings/string_builder.h"
17#include "test/logging/file_log_writer.h"
18#include "test/testsupport/file_utils.h"
Sebastian Janssonfc8279d2020-01-16 10:45:5919#include "test/time_controller/real_time_controller.h"
20#include "test/time_controller/simulated_time_controller.h"
Sebastian Jansson64672dc2019-10-01 13:23:3921
22ABSL_FLAG(bool, peer_logs, false, "Save logs from peer scenario framework.");
23ABSL_FLAG(std::string,
24 peer_logs_root,
25 "",
26 "Output root path, based on project root if unset.");
Sebastian Jansson7cbee842019-08-06 15:19:3827
28namespace webrtc {
29namespace test {
Sebastian Jansson64672dc2019-10-01 13:23:3930namespace {
31std::unique_ptr<FileLogWriterFactory> GetPeerScenarioLogManager(
32 std::string file_name) {
33 if (absl::GetFlag(FLAGS_peer_logs) && !file_name.empty()) {
34 std::string output_root = absl::GetFlag(FLAGS_peer_logs_root);
35 if (output_root.empty())
36 output_root = OutputPath() + "output_data/";
Sebastian Jansson7cbee842019-08-06 15:19:3837
Sebastian Jansson64672dc2019-10-01 13:23:3938 auto base_filename = output_root + file_name + ".";
39 RTC_LOG(LS_INFO) << "Saving peer scenario logs to: " << base_filename;
40 return std::make_unique<FileLogWriterFactory>(base_filename);
41 }
42 return nullptr;
43}
44} // namespace
45
Sebastian Jansson6ce033a2020-01-22 09:12:5646PeerScenario::PeerScenario(const testing::TestInfo& test_info, TimeMode mode)
Sebastian Janssonfc8279d2020-01-16 10:45:5947 : PeerScenario(
48 std::string(test_info.test_suite_name()) + "/" + test_info.name(),
Sebastian Jansson6ce033a2020-01-22 09:12:5649 mode) {}
Sebastian Jansson64672dc2019-10-01 13:23:3950
Sebastian Jansson6ce033a2020-01-22 09:12:5651PeerScenario::PeerScenario(std::string file_name, TimeMode mode)
52 : PeerScenario(GetPeerScenarioLogManager(file_name), mode) {}
Sebastian Jansson64672dc2019-10-01 13:23:3953
54PeerScenario::PeerScenario(
Sebastian Janssonfc8279d2020-01-16 10:45:5955 std::unique_ptr<LogWriterFactoryInterface> log_writer_manager,
Sebastian Jansson6ce033a2020-01-22 09:12:5656 TimeMode mode)
Sebastian Janssonfc8279d2020-01-16 10:45:5957 : log_writer_manager_(std::move(log_writer_manager)),
Sebastian Jansson6ce033a2020-01-22 09:12:5658 net_(mode),
59 signaling_thread_(net_.time_controller()->GetMainThread()) {}
Sebastian Jansson7cbee842019-08-06 15:19:3860
61PeerScenarioClient* PeerScenario::CreateClient(
62 PeerScenarioClient::Config config) {
Sebastian Jansson64672dc2019-10-01 13:23:3963 return CreateClient(
64 std::string("client_") + rtc::ToString(peer_clients_.size() + 1), config);
65}
66
67PeerScenarioClient* PeerScenario::CreateClient(
68 std::string name,
69 PeerScenarioClient::Config config) {
Sebastian Jansson6ce033a2020-01-22 09:12:5670 peer_clients_.emplace_back(net(), signaling_thread_,
Sebastian Janssonfc8279d2020-01-16 10:45:5971 GetLogWriterFactory(name), config);
Sebastian Jansson7cbee842019-08-06 15:19:3872 return &peer_clients_.back();
73}
74
75SignalingRoute PeerScenario::ConnectSignaling(
76 PeerScenarioClient* caller,
77 PeerScenarioClient* callee,
78 std::vector<EmulatedNetworkNode*> send_link,
79 std::vector<EmulatedNetworkNode*> ret_link) {
Andrey Logvinf9ee0e02021-01-14 09:50:3280 return SignalingRoute(caller, callee, net_.CreateCrossTrafficRoute(send_link),
81 net_.CreateCrossTrafficRoute(ret_link));
Sebastian Jansson7cbee842019-08-06 15:19:3882}
83
84void PeerScenario::SimpleConnection(
85 PeerScenarioClient* caller,
86 PeerScenarioClient* callee,
87 std::vector<EmulatedNetworkNode*> send_link,
88 std::vector<EmulatedNetworkNode*> ret_link) {
89 net()->CreateRoute(caller->endpoint(), send_link, callee->endpoint());
90 net()->CreateRoute(callee->endpoint(), ret_link, caller->endpoint());
91 auto signaling = ConnectSignaling(caller, callee, send_link, ret_link);
92 signaling.StartIceSignaling();
Sebastian Jansson2a92d2b2020-01-13 15:49:3293 std::atomic<bool> done(false);
Sebastian Jansson7cbee842019-08-06 15:19:3894 signaling.NegotiateSdp(
Sebastian Jansson2a92d2b2020-01-13 15:49:3295 [&](const SessionDescriptionInterface&) { done = true; });
Sebastian Jansson7cbee842019-08-06 15:19:3896 RTC_CHECK(WaitAndProcess(&done));
97}
98
99void PeerScenario::AttachVideoQualityAnalyzer(VideoQualityAnalyzer* analyzer,
100 VideoTrackInterface* send_track,
101 PeerScenarioClient* receiver) {
102 video_quality_pairs_.emplace_back(clock(), analyzer);
103 auto pair = &video_quality_pairs_.back();
104 send_track->AddOrUpdateSink(&pair->capture_tap_, rtc::VideoSinkWants());
105 receiver->AddVideoReceiveSink(send_track->id(), &pair->decode_tap_);
106}
107
Sebastian Jansson2a92d2b2020-01-13 15:49:32108bool PeerScenario::WaitAndProcess(std::atomic<bool>* event,
109 TimeDelta max_duration) {
Sebastian Jansson6ce033a2020-01-22 09:12:56110 return net_.time_controller()->Wait([event] { return event->load(); },
111 max_duration);
Sebastian Jansson7cbee842019-08-06 15:19:38112}
113
114void PeerScenario::ProcessMessages(TimeDelta duration) {
Sebastian Jansson6ce033a2020-01-22 09:12:56115 net_.time_controller()->AdvanceTime(duration);
Sebastian Jansson7cbee842019-08-06 15:19:38116}
117
Sebastian Jansson64672dc2019-10-01 13:23:39118std::unique_ptr<LogWriterFactoryInterface> PeerScenario::GetLogWriterFactory(
119 std::string name) {
120 if (!log_writer_manager_ || name.empty())
121 return nullptr;
122 return std::make_unique<LogWriterFactoryAddPrefix>(log_writer_manager_.get(),
123 name);
124}
125
Sebastian Jansson7cbee842019-08-06 15:19:38126} // namespace test
127} // namespace webrtc