| /* |
| * Copyright 2018 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. |
| */ |
| #include "test/scenario/scenario.h" |
| |
| #include <algorithm> |
| |
| #include "absl/memory/memory.h" |
| #include "api/audio_codecs/builtin_audio_decoder_factory.h" |
| #include "api/audio_codecs/builtin_audio_encoder_factory.h" |
| #include "rtc_base/flags.h" |
| #include "rtc_base/socket_address.h" |
| #include "test/logging/file_log_writer.h" |
| #include "test/scenario/network/network_emulation.h" |
| #include "test/testsupport/file_utils.h" |
| #include "test/time_controller/real_time_controller.h" |
| #include "test/time_controller/simulated_time_controller.h" |
| |
| WEBRTC_DEFINE_bool(scenario_logs, false, "Save logs from scenario framework."); |
| WEBRTC_DEFINE_string(scenario_logs_root, |
| "", |
| "Output root path, based on project root if unset."); |
| |
| namespace webrtc { |
| namespace test { |
| namespace { |
| const Timestamp kSimulatedStartTime = Timestamp::seconds(100000); |
| |
| std::unique_ptr<FileLogWriterFactory> GetScenarioLogManager( |
| std::string file_name) { |
| if (FLAG_scenario_logs && !file_name.empty()) { |
| std::string output_root = FLAG_scenario_logs_root; |
| if (output_root.empty()) |
| output_root = OutputPath() + "output_data/"; |
| |
| auto base_filename = output_root + file_name + "."; |
| RTC_LOG(LS_INFO) << "Saving scenario logs to: " << base_filename; |
| return absl::make_unique<FileLogWriterFactory>(base_filename); |
| } |
| return nullptr; |
| } |
| std::unique_ptr<TimeController> CreateTimeController(bool real_time) { |
| if (real_time) { |
| return absl::make_unique<RealTimeController>(); |
| } else { |
| return absl::make_unique<GlobalSimulatedTimeController>( |
| kSimulatedStartTime); |
| } |
| } |
| } |
| |
| Scenario::Scenario() |
| : Scenario(std::unique_ptr<LogWriterFactoryInterface>(), true) {} |
| |
| Scenario::Scenario(std::string file_name) : Scenario(file_name, true) {} |
| |
| Scenario::Scenario(std::string file_name, bool real_time) |
| : Scenario(GetScenarioLogManager(file_name), real_time) {} |
| |
| Scenario::Scenario( |
| std::unique_ptr<LogWriterFactoryInterface> log_writer_factory, |
| bool real_time) |
| : log_writer_factory_(std::move(log_writer_factory)), |
| time_controller_(CreateTimeController(real_time)), |
| clock_(time_controller_->GetClock()), |
| audio_decoder_factory_(CreateBuiltinAudioDecoderFactory()), |
| audio_encoder_factory_(CreateBuiltinAudioEncoderFactory()), |
| task_queue_(time_controller_->GetTaskQueueFactory()->CreateTaskQueue( |
| "Scenario", |
| TaskQueueFactory::Priority::NORMAL)) {} |
| |
| Scenario::~Scenario() { |
| if (start_time_.IsFinite()) |
| Stop(); |
| for (auto& call_client : clients_) |
| call_client->transport_->Disconnect(); |
| } |
| |
| ColumnPrinter Scenario::TimePrinter() { |
| return ColumnPrinter::Lambda("time", |
| [this](rtc::SimpleStringBuilder& sb) { |
| sb.AppendFormat("%.3lf", |
| Now().seconds<double>()); |
| }, |
| 32); |
| } |
| |
| StatesPrinter* Scenario::CreatePrinter(std::string name, |
| TimeDelta interval, |
| std::vector<ColumnPrinter> printers) { |
| std::vector<ColumnPrinter> all_printers{TimePrinter()}; |
| for (auto& printer : printers) |
| all_printers.push_back(printer); |
| StatesPrinter* printer = new StatesPrinter(GetLogWriter(name), all_printers); |
| printers_.emplace_back(printer); |
| printer->PrintHeaders(); |
| if (interval.IsFinite()) |
| Every(interval, [printer] { printer->PrintRow(); }); |
| return printer; |
| } |
| |
| CallClient* Scenario::CreateClient(std::string name, CallClientConfig config) { |
| CallClient* client = |
| new CallClient(time_controller_.get(), GetLogWriterFactory(name), config); |
| if (config.transport.state_log_interval.IsFinite()) { |
| Every(config.transport.state_log_interval, [this, client]() { |
| client->network_controller_factory_.LogCongestionControllerStats(Now()); |
| }); |
| } |
| clients_.emplace_back(client); |
| return client; |
| } |
| |
| CallClient* Scenario::CreateClient( |
| std::string name, |
| std::function<void(CallClientConfig*)> config_modifier) { |
| CallClientConfig config; |
| config_modifier(&config); |
| return CreateClient(name, config); |
| } |
| |
| CallClientPair* Scenario::CreateRoutes( |
| CallClient* first, |
| std::vector<EmulatedNetworkNode*> send_link, |
| CallClient* second, |
| std::vector<EmulatedNetworkNode*> return_link) { |
| return CreateRoutes(first, send_link, |
| DataSize::bytes(PacketOverhead::kDefault), second, |
| return_link, DataSize::bytes(PacketOverhead::kDefault)); |
| } |
| |
| CallClientPair* Scenario::CreateRoutes( |
| CallClient* first, |
| std::vector<EmulatedNetworkNode*> send_link, |
| DataSize first_overhead, |
| CallClient* second, |
| std::vector<EmulatedNetworkNode*> return_link, |
| DataSize second_overhead) { |
| CallClientPair* client_pair = new CallClientPair(first, second); |
| ChangeRoute(client_pair->forward(), send_link, first_overhead); |
| ChangeRoute(client_pair->reverse(), return_link, second_overhead); |
| client_pairs_.emplace_back(client_pair); |
| return client_pair; |
| } |
| |
| void Scenario::ChangeRoute(std::pair<CallClient*, CallClient*> clients, |
| std::vector<EmulatedNetworkNode*> over_nodes) { |
| ChangeRoute(clients, over_nodes, DataSize::bytes(PacketOverhead::kDefault)); |
| } |
| |
| void Scenario::ChangeRoute(std::pair<CallClient*, CallClient*> clients, |
| std::vector<EmulatedNetworkNode*> over_nodes, |
| DataSize overhead) { |
| rtc::IPAddress route_ip(next_route_id_++); |
| clients.second->route_overhead_.insert({route_ip, overhead}); |
| EmulatedNetworkNode::CreateRoute(route_ip, over_nodes, clients.second); |
| clients.first->transport_->Connect(over_nodes.front(), route_ip, overhead); |
| } |
| |
| SimulatedTimeClient* Scenario::CreateSimulatedTimeClient( |
| std::string name, |
| SimulatedTimeClientConfig config, |
| std::vector<PacketStreamConfig> stream_configs, |
| std::vector<EmulatedNetworkNode*> send_link, |
| std::vector<EmulatedNetworkNode*> return_link) { |
| rtc::IPAddress send_ip(next_route_id_++); |
| rtc::IPAddress return_ip(next_route_id_++); |
| SimulatedTimeClient* client = new SimulatedTimeClient( |
| time_controller_.get(), GetLogWriterFactory(name), config, stream_configs, |
| send_link, return_link, send_ip, return_ip, Now()); |
| if (log_writer_factory_ && !name.empty() && |
| config.transport.state_log_interval.IsFinite()) { |
| Every(config.transport.state_log_interval, [this, client]() { |
| client->network_controller_factory_.LogCongestionControllerStats(Now()); |
| }); |
| } |
| if (client->GetNetworkControllerProcessInterval().IsFinite()) { |
| Every(client->GetNetworkControllerProcessInterval(), |
| [this, client] { client->CongestionProcess(Now()); }); |
| } else { |
| task_queue_.PostTask([this, client] { client->CongestionProcess(Now()); }); |
| } |
| Every(TimeDelta::ms(5), [this, client] { client->PacerProcess(Now()); }); |
| simulated_time_clients_.emplace_back(client); |
| return client; |
| } |
| |
| SimulationNode* Scenario::CreateSimulationNode( |
| std::function<void(NetworkNodeConfig*)> config_modifier) { |
| NetworkNodeConfig config; |
| config_modifier(&config); |
| return CreateSimulationNode(config); |
| } |
| |
| SimulationNode* Scenario::CreateSimulationNode(NetworkNodeConfig config) { |
| RTC_DCHECK(config.mode == NetworkNodeConfig::TrafficMode::kSimulation); |
| auto network_node = SimulationNode::Create(clock_, &task_queue_, config); |
| SimulationNode* sim_node = network_node.get(); |
| network_nodes_.emplace_back(std::move(network_node)); |
| return sim_node; |
| } |
| |
| EmulatedNetworkNode* Scenario::CreateNetworkNode( |
| std::unique_ptr<NetworkBehaviorInterface> behavior) { |
| network_nodes_.emplace_back( |
| new EmulatedNetworkNode(clock_, &task_queue_, std::move(behavior))); |
| EmulatedNetworkNode* network_node = network_nodes_.back().get(); |
| return network_node; |
| } |
| |
| void Scenario::TriggerPacketBurst(std::vector<EmulatedNetworkNode*> over_nodes, |
| size_t num_packets, |
| size_t packet_size) { |
| rtc::IPAddress route_ip(next_route_id_++); |
| EmulatedNetworkNode::CreateRoute(route_ip, over_nodes, &null_receiver_); |
| for (size_t i = 0; i < num_packets; ++i) |
| over_nodes[0]->OnPacketReceived(EmulatedIpPacket( |
| /*from=*/rtc::SocketAddress(), /*to=*/rtc::SocketAddress(route_ip, 0), |
| rtc::CopyOnWriteBuffer(packet_size), Now())); |
| } |
| |
| void Scenario::NetworkDelayedAction( |
| std::vector<EmulatedNetworkNode*> over_nodes, |
| size_t packet_size, |
| std::function<void()> action) { |
| rtc::IPAddress route_ip(next_route_id_++); |
| action_receivers_.emplace_back(new ActionReceiver(action)); |
| EmulatedNetworkNode::CreateRoute(route_ip, over_nodes, |
| action_receivers_.back().get()); |
| over_nodes[0]->OnPacketReceived(EmulatedIpPacket( |
| /*from=*/rtc::SocketAddress(), /*to=*/rtc::SocketAddress(route_ip, 0), |
| rtc::CopyOnWriteBuffer(packet_size), Now())); |
| } |
| |
| CrossTrafficSource* Scenario::CreateCrossTraffic( |
| std::vector<EmulatedNetworkNode*> over_nodes, |
| std::function<void(CrossTrafficConfig*)> config_modifier) { |
| CrossTrafficConfig cross_config; |
| config_modifier(&cross_config); |
| return CreateCrossTraffic(over_nodes, cross_config); |
| } |
| |
| CrossTrafficSource* Scenario::CreateCrossTraffic( |
| std::vector<EmulatedNetworkNode*> over_nodes, |
| CrossTrafficConfig config) { |
| rtc::IPAddress route_ip(next_route_id_++); |
| cross_traffic_sources_.emplace_back( |
| new CrossTrafficSource(over_nodes.front(), route_ip, config)); |
| CrossTrafficSource* node = cross_traffic_sources_.back().get(); |
| EmulatedNetworkNode::CreateRoute(route_ip, over_nodes, &null_receiver_); |
| Every(config.min_packet_interval, |
| [this, node](TimeDelta delta) { node->Process(Now(), delta); }); |
| return node; |
| } |
| |
| VideoStreamPair* Scenario::CreateVideoStream( |
| std::pair<CallClient*, CallClient*> clients, |
| std::function<void(VideoStreamConfig*)> config_modifier) { |
| VideoStreamConfig config; |
| config_modifier(&config); |
| return CreateVideoStream(clients, config); |
| } |
| |
| VideoStreamPair* Scenario::CreateVideoStream( |
| std::pair<CallClient*, CallClient*> clients, |
| VideoStreamConfig config) { |
| std::unique_ptr<RtcEventLogOutput> quality_logger; |
| if (config.analyzer.log_to_file) |
| quality_logger = clients.first->GetLogWriter(".video_quality.txt"); |
| video_streams_.emplace_back(new VideoStreamPair( |
| clients.first, clients.second, config, std::move(quality_logger))); |
| return video_streams_.back().get(); |
| } |
| |
| AudioStreamPair* Scenario::CreateAudioStream( |
| std::pair<CallClient*, CallClient*> clients, |
| std::function<void(AudioStreamConfig*)> config_modifier) { |
| AudioStreamConfig config; |
| config_modifier(&config); |
| return CreateAudioStream(clients, config); |
| } |
| |
| AudioStreamPair* Scenario::CreateAudioStream( |
| std::pair<CallClient*, CallClient*> clients, |
| AudioStreamConfig config) { |
| audio_streams_.emplace_back( |
| new AudioStreamPair(clients.first, audio_encoder_factory_, clients.second, |
| audio_decoder_factory_, config)); |
| return audio_streams_.back().get(); |
| } |
| |
| void Scenario::Every(TimeDelta interval, |
| std::function<void(TimeDelta)> function) { |
| RepeatingTaskHandle::DelayedStart(task_queue_.Get(), interval, |
| [interval, function] { |
| function(interval); |
| return interval; |
| }); |
| } |
| |
| void Scenario::Every(TimeDelta interval, std::function<void()> function) { |
| RepeatingTaskHandle::DelayedStart(task_queue_.Get(), interval, |
| [interval, function] { |
| function(); |
| return interval; |
| }); |
| } |
| |
| void Scenario::At(TimeDelta offset, std::function<void()> function) { |
| RTC_DCHECK_GT(offset.ms(), TimeSinceStart().ms()); |
| task_queue_.PostDelayedTask(function, TimeUntilTarget(offset).ms()); |
| } |
| |
| void Scenario::RunFor(TimeDelta duration) { |
| if (start_time_.IsInfinite()) |
| Start(); |
| time_controller_->Sleep(duration); |
| } |
| |
| void Scenario::RunUntil(TimeDelta target_time_since_start) { |
| RunFor(TimeUntilTarget(target_time_since_start)); |
| } |
| |
| void Scenario::RunUntil(TimeDelta target_time_since_start, |
| TimeDelta check_interval, |
| std::function<bool()> exit_function) { |
| if (start_time_.IsInfinite()) |
| Start(); |
| while (check_interval >= TimeUntilTarget(target_time_since_start)) { |
| time_controller_->Sleep(check_interval); |
| if (exit_function()) |
| return; |
| } |
| time_controller_->Sleep(TimeUntilTarget(target_time_since_start)); |
| } |
| |
| void Scenario::Start() { |
| start_time_ = Timestamp::us(clock_->TimeInMicroseconds()); |
| for (auto& stream_pair : video_streams_) |
| stream_pair->receive()->Start(); |
| for (auto& stream_pair : audio_streams_) |
| stream_pair->receive()->Start(); |
| for (auto& stream_pair : video_streams_) { |
| if (stream_pair->config_.autostart) { |
| stream_pair->send()->Start(); |
| } |
| } |
| for (auto& stream_pair : audio_streams_) { |
| if (stream_pair->config_.autostart) { |
| stream_pair->send()->Start(); |
| } |
| } |
| } |
| |
| void Scenario::Stop() { |
| RTC_DCHECK(start_time_.IsFinite()); |
| for (auto& stream_pair : video_streams_) { |
| stream_pair->send()->Stop(); |
| } |
| for (auto& stream_pair : audio_streams_) |
| stream_pair->send()->Stop(); |
| for (auto& stream_pair : video_streams_) |
| stream_pair->receive()->Stop(); |
| for (auto& stream_pair : audio_streams_) |
| stream_pair->receive()->Stop(); |
| start_time_ = Timestamp::PlusInfinity(); |
| } |
| |
| Timestamp Scenario::Now() { |
| return Timestamp::us(clock_->TimeInMicroseconds()); |
| } |
| |
| TimeDelta Scenario::TimeSinceStart() { |
| if (start_time_.IsInfinite()) |
| return TimeDelta::Zero(); |
| return Now() - start_time_; |
| } |
| |
| TimeDelta Scenario::TimeUntilTarget(TimeDelta target_time_offset) { |
| return target_time_offset - TimeSinceStart(); |
| } |
| |
| } // namespace test |
| } // namespace webrtc |