File: media_sink_service_status.cc

package info (click to toggle)
chromium 139.0.7258.127-1~deb13u1
  • links: PTS, VCS
  • area: main
  • in suites:
  • size: 6,122,096 kB
  • sloc: cpp: 35,100,771; ansic: 7,163,530; javascript: 4,103,002; python: 1,436,920; asm: 946,517; xml: 746,709; pascal: 187,653; perl: 88,691; sh: 88,436; objc: 79,953; sql: 51,488; cs: 44,583; fortran: 24,137; makefile: 22,147; tcl: 15,277; php: 13,980; yacc: 8,984; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (125 lines) | stat: -rw-r--r-- 4,487 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "chrome/browser/media/router/mojo/media_sink_service_status.h"

#include "base/json/json_string_value_serializer.h"
#include "base/strings/strcat.h"
#include "components/media_router/browser/log_util.h"

namespace media_router {

namespace {

constexpr size_t kMaxAvailableSinksSize = 100;

// Helper function to convert `value` to JSON string.
std::string ToJSONString(const base::Value::Dict& value) {
  std::string json;
  JSONStringValueSerializer serializer(&json);
  serializer.set_pretty_print(true);
  return serializer.Serialize(value) ? json : "";
}

// Helper function to convert `sink_internal` to JSON format represented by
// base::Value::Dict.
base::Value::Dict ToValue(const MediaSinkInternal& sink_internal) {
  base::Value::Dict dict;
  const MediaSink& sink = sink_internal.sink();
  dict.Set("id", log_util::TruncateId(sink.id()));
  dict.Set("name", sink.name());
  dict.Set("icon_type", static_cast<int>(sink.icon_type()));

  if (sink_internal.is_dial_sink()) {
    DialSinkExtraData extra_data = sink_internal.dial_data();
    dict.Set("ip_address", extra_data.ip_address.ToString());
    dict.Set("model_name", extra_data.model_name);
    dict.Set("app_url", extra_data.app_url.spec());
  }

  if (sink_internal.is_cast_sink()) {
    CastSinkExtraData extra_data = sink_internal.cast_data();
    dict.Set("ip_endpoint", extra_data.ip_endpoint.ToString());
    dict.Set("model_name", extra_data.model_name);
    dict.Set("capabilities",
             base::checked_cast<int>(extra_data.capabilities.ToEnumBitmask()));
    dict.Set("channel_id", extra_data.cast_channel_id);
    dict.Set("discovered_by_dial",
             extra_data.discovery_type == CastDiscoveryType::kDial);
  }
  return dict;
}

// Helper function to convert `sinks` to JSON format represented by
// base::Value::Dict. The key is the source id and the value is a list of
// sinks represented by base::Value::Dict.
base::Value::Dict ConvertDiscoveredSinksToValues(
    const base::flat_map<std::string, std::vector<MediaSinkInternal>>& sinks) {
  base::Value::Dict dict;
  for (const auto& sinks_it : sinks) {
    base::Value::List list;
    for (const auto& inner_sink : sinks_it.second) {
      list.Append(ToValue(inner_sink));
    }
    dict.Set(sinks_it.first, std::move(list));
  }
  return dict;
}

// Helper function to convert `available_sinks` to a dictionary of availability
// strings in JSON format represented by base::Value::Dict. The key is the
// source id and the value is a list of sink ids.
base::Value::Dict ConvertAvailableSinksToValues(
    const base::LRUCache<std::string, std::vector<MediaSinkInternal>>&
        available_sinks) {
  base::Value::Dict dict;
  for (const auto& sinks_it : available_sinks) {
    base::Value::List list;
    for (const auto& inner_sink : sinks_it.second) {
      std::string sink_id = inner_sink.sink().id();
      list.Append(log_util::TruncateId(sink_id));
    }
    dict.Set(sinks_it.first, std::move(list));
  }
  return dict;
}

}  // namespace

MediaSinkServiceStatus::MediaSinkServiceStatus()
    : available_sinks_(kMaxAvailableSinksSize) {}
MediaSinkServiceStatus::~MediaSinkServiceStatus() = default;

void MediaSinkServiceStatus::UpdateDiscoveredSinks(
    const std::string& provider_name,
    const std::vector<MediaSinkInternal>& discovered_sinks) {
  discovered_sinks_[provider_name] = discovered_sinks;
}

void MediaSinkServiceStatus::UpdateAvailableSinks(
    mojom::MediaRouteProviderId provider_id,
    const std::string& media_source,
    const std::vector<MediaSinkInternal>& available_sinks) {
  // TODO(takumif): It'd be safer and more efficient to make use
  // pair<MediaRouteProviderId, string> than a serialized "id:name" string.
  std::string key =
      base::StrCat({ProviderIdToString(provider_id), ":", media_source});
  available_sinks_.Put(key, available_sinks);
}

base::Value::Dict MediaSinkServiceStatus::GetStatusAsValue() const {
  base::Value::Dict status_dict;
  status_dict.Set("discovered_sinks",
                  ConvertDiscoveredSinksToValues(discovered_sinks_));
  status_dict.Set("available_sinks",
                  ConvertAvailableSinksToValues(available_sinks_));

  return status_dict;
}

std::string MediaSinkServiceStatus::GetStatusAsJSONString() const {
  return ToJSONString(GetStatusAsValue());
}

}  // namespace media_router