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
|
// Copyright 2021 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/privacy_budget/surface_set_equivalence.h"
#include <iterator>
#include <set>
#include <string_view>
#include "base/containers/contains.h"
#include "chrome/common/privacy_budget/field_trial_param_conversions.h"
#include "chrome/common/privacy_budget/privacy_budget_features.h"
SurfaceSetEquivalence::EquivalenceClassIdentifierMap
SurfaceSetEquivalence::DecodeEquivalenceClassSet(std::string_view param_value) {
EquivalenceClassIdentifierMap representative_map;
auto surface_set_set =
DecodeIdentifiabilityFieldTrialParam<SurfaceSetEquivalentClassesList>(
param_value);
// A surface must only appear once as a key _or_ as a value in our mapping.
// Otherwise the mapping will violate the invariants.
std::set<blink::IdentifiableSurface> seen;
for (const auto& surface_set : surface_set_set) {
// A single surface set or an empty one is a no-op. These shouldn't be sent
// as a part of a valid configuration, but the code shouldn't get confused
// upon seeing one either.
if (surface_set.size() <= 1)
continue;
auto first_surface = surface_set.front();
if (base::Contains(seen, first_surface))
continue;
seen.insert(first_surface);
auto representative = RepresentativeSurface(first_surface);
for (const auto& surface : surface_set) {
if (!base::Contains(seen, surface)) {
seen.insert(surface);
representative_map[surface] = representative;
}
}
}
return representative_map;
}
SurfaceSetEquivalence::SurfaceSetEquivalence()
: equivalence_map_(DecodeEquivalenceClassSet(
features::kIdentifiabilityStudySurfaceEquivalenceClasses.Get())) {}
SurfaceSetEquivalence::~SurfaceSetEquivalence() = default;
RepresentativeSurface SurfaceSetEquivalence::GetRepresentative(
blink::IdentifiableSurface surface) const {
auto it = equivalence_map_.find(surface);
if (it == equivalence_map_.end())
return RepresentativeSurface(surface);
return RepresentativeSurface(it->second);
}
RepresentativeSurfaceSet SurfaceSetEquivalence::GetRepresentatives(
const IdentifiableSurfaceSet& source) const {
return base::MakeFlatSet<RepresentativeSurface>(
source, {},
[this](const auto& surface) { return GetRepresentative(surface); });
}
RepresentativeSurfaceList SurfaceSetEquivalence::GetRepresentatives(
const IdentifiableSurfaceList& source) const {
RepresentativeSurfaceList result;
result.reserve(source.size());
std::set<RepresentativeSurface> seen;
for (const auto surface : source) {
auto representative = GetRepresentative(surface);
auto inserted = seen.insert(representative);
if (inserted.second)
result.push_back(representative);
}
result.shrink_to_fit();
return result;
}
bool SurfaceSetEquivalence::IsRepresentative(
blink::IdentifiableSurface surface) const {
return surface == GetRepresentative(surface).value();
}
bool SurfaceSetEquivalence::IsRepresentative(
const IdentifiableSurfaceSet& source) const {
return std::ranges::all_of(source,
[this](auto s) { return IsRepresentative(s); });
}
|