OLD | NEW |
(Empty) | |
| 1 // Copyright 2017 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "chromecast/base/cast_features.h" |
| 6 |
| 7 #include "base/feature_list.h" |
| 8 #include "base/macros.h" |
| 9 #include "base/memory/ptr_util.h" |
| 10 #include "base/metrics/field_trial.h" |
| 11 #include "base/metrics/field_trial_params.h" |
| 12 #include "base/values.h" |
| 13 #include "testing/gmock/include/gmock/gmock.h" |
| 14 #include "testing/gtest/include/gtest/gtest.h" |
| 15 |
| 16 namespace chromecast { |
| 17 namespace { |
| 18 |
| 19 const char kTestBooleanFeatureName[] = "test_boolean_feature"; |
| 20 const char kTestBooleanFeatureName2[] = "test_boolean_feature_2"; |
| 21 const char kTestBooleanFeatureName3[] = "test_boolean_feature_3"; |
| 22 const char kTestBooleanFeatureName4[] = "test_boolean_feature_4"; |
| 23 |
| 24 const char kTestParamsFeatureName[] = "test_params_feature"; |
| 25 |
| 26 } // namespace |
| 27 |
| 28 class CastFeaturesTest : public testing::Test { |
| 29 public: |
| 30 CastFeaturesTest() : field_trial_list_(nullptr) {} |
| 31 ~CastFeaturesTest() override {} |
| 32 |
| 33 // testing::Test implementation: |
| 34 void SetUp() override { ResetCastFeaturesForTesting(); } |
| 35 |
| 36 private: |
| 37 // A field trial list must be created before attempting to create FieldTrials. |
| 38 // In production, this instance lives in CastBrowserMainParts. |
| 39 base::FieldTrialList field_trial_list_; |
| 40 |
| 41 DISALLOW_COPY_AND_ASSIGN(CastFeaturesTest); |
| 42 }; |
| 43 |
| 44 TEST_F(CastFeaturesTest, EnableDisableMultipleBooleanFeatures) { |
| 45 // Declare several boolean features. |
| 46 base::Feature bool_feature(kTestBooleanFeatureName, |
| 47 base::FEATURE_DISABLED_BY_DEFAULT); |
| 48 base::Feature bool_feature_2(kTestBooleanFeatureName2, |
| 49 base::FEATURE_ENABLED_BY_DEFAULT); |
| 50 base::Feature bool_feature_3(kTestBooleanFeatureName3, |
| 51 base::FEATURE_DISABLED_BY_DEFAULT); |
| 52 base::Feature bool_feature_4(kTestBooleanFeatureName4, |
| 53 base::FEATURE_ENABLED_BY_DEFAULT); |
| 54 |
| 55 // Override those features with DCS configs. |
| 56 auto experiments = base::MakeUnique<base::ListValue>(); |
| 57 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 58 features->SetBoolean(kTestBooleanFeatureName, false); |
| 59 features->SetBoolean(kTestBooleanFeatureName2, false); |
| 60 features->SetBoolean(kTestBooleanFeatureName3, true); |
| 61 features->SetBoolean(kTestBooleanFeatureName4, true); |
| 62 |
| 63 InitializeFeatureList(*features, *experiments, "", ""); |
| 64 |
| 65 // Test that features are properly enabled (they should match the |
| 66 // DCS config). |
| 67 ASSERT_FALSE(base::FeatureList::IsEnabled(bool_feature)); |
| 68 ASSERT_FALSE(base::FeatureList::IsEnabled(bool_feature_2)); |
| 69 ASSERT_TRUE(base::FeatureList::IsEnabled(bool_feature_3)); |
| 70 ASSERT_TRUE(base::FeatureList::IsEnabled(bool_feature_4)); |
| 71 } |
| 72 |
| 73 TEST_F(CastFeaturesTest, EnableSingleFeatureWithParams) { |
| 74 // Define a feature with params. |
| 75 base::Feature test_feature(kTestParamsFeatureName, |
| 76 base::FEATURE_DISABLED_BY_DEFAULT); |
| 77 |
| 78 // Pass params via DCS. |
| 79 auto experiments = base::MakeUnique<base::ListValue>(); |
| 80 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 81 auto params = base::MakeUnique<base::DictionaryValue>(); |
| 82 params->SetString("foo_key", "foo"); |
| 83 params->SetString("bar_key", "bar"); |
| 84 params->SetString("doub_key", "3.14159"); |
| 85 params->SetString("long_doub_key", "1.23459999999999999"); |
| 86 params->SetString("int_key", "4242"); |
| 87 params->SetString("bool_key", "true"); |
| 88 features->Set(kTestParamsFeatureName, std::move(params)); |
| 89 |
| 90 InitializeFeatureList(*features, *experiments, "", ""); |
| 91 |
| 92 // Test that this feature is enabled, and params are correct. |
| 93 ASSERT_TRUE(base::FeatureList::IsEnabled(test_feature)); |
| 94 ASSERT_EQ("foo", |
| 95 base::GetFieldTrialParamValueByFeature(test_feature, "foo_key")); |
| 96 ASSERT_EQ("bar", |
| 97 base::GetFieldTrialParamValueByFeature(test_feature, "bar_key")); |
| 98 ASSERT_EQ(3.14159, base::GetFieldTrialParamByFeatureAsDouble( |
| 99 test_feature, "doub_key", 0.000)); |
| 100 ASSERT_EQ(1.23459999999999999, base::GetFieldTrialParamByFeatureAsDouble( |
| 101 test_feature, "long_doub_key", 0.000)); |
| 102 ASSERT_EQ(4242, base::GetFieldTrialParamByFeatureAsInt(test_feature, |
| 103 "int_key", -1)); |
| 104 ASSERT_EQ(true, base::GetFieldTrialParamByFeatureAsBool(test_feature, |
| 105 "bool_key", false)); |
| 106 } |
| 107 |
| 108 TEST_F(CastFeaturesTest, CommandLineOverridesDcsAndDefault) { |
| 109 // Declare several boolean features. |
| 110 base::Feature bool_feature(kTestBooleanFeatureName, |
| 111 base::FEATURE_DISABLED_BY_DEFAULT); |
| 112 base::Feature bool_feature_2(kTestBooleanFeatureName2, |
| 113 base::FEATURE_ENABLED_BY_DEFAULT); |
| 114 base::Feature bool_feature_3(kTestBooleanFeatureName3, |
| 115 base::FEATURE_DISABLED_BY_DEFAULT); |
| 116 base::Feature bool_feature_4(kTestBooleanFeatureName4, |
| 117 base::FEATURE_ENABLED_BY_DEFAULT); |
| 118 |
| 119 // Override those features with DCS configs. |
| 120 auto experiments = base::MakeUnique<base::ListValue>(); |
| 121 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 122 features->SetBoolean(kTestBooleanFeatureName, false); |
| 123 features->SetBoolean(kTestBooleanFeatureName2, false); |
| 124 features->SetBoolean(kTestBooleanFeatureName3, true); |
| 125 features->SetBoolean(kTestBooleanFeatureName4, true); |
| 126 |
| 127 // Also override a param feature with DCS config. |
| 128 base::Feature params_feature(kTestParamsFeatureName, |
| 129 base::FEATURE_ENABLED_BY_DEFAULT); |
| 130 auto params = base::MakeUnique<base::DictionaryValue>(); |
| 131 params->SetString("foo_key", "foo"); |
| 132 features->Set(kTestParamsFeatureName, std::move(params)); |
| 133 |
| 134 // Now override with command line flags. Command line flags should have the |
| 135 // final say. |
| 136 std::string enabled_features = std::string(kTestBooleanFeatureName) |
| 137 .append(",") |
| 138 .append(kTestBooleanFeatureName2); |
| 139 |
| 140 std::string disabled_features = std::string(kTestBooleanFeatureName4) |
| 141 .append(",") |
| 142 .append(kTestParamsFeatureName); |
| 143 |
| 144 InitializeFeatureList(*features, *experiments, enabled_features, |
| 145 disabled_features); |
| 146 |
| 147 // Test that features are properly enabled (they should match the |
| 148 // DCS config). |
| 149 ASSERT_TRUE(base::FeatureList::IsEnabled(bool_feature)); |
| 150 ASSERT_TRUE(base::FeatureList::IsEnabled(bool_feature_2)); |
| 151 ASSERT_TRUE(base::FeatureList::IsEnabled(bool_feature_3)); |
| 152 ASSERT_FALSE(base::FeatureList::IsEnabled(bool_feature_4)); |
| 153 |
| 154 // Test that the params feature is disabled, and params are not set. |
| 155 ASSERT_FALSE(base::FeatureList::IsEnabled(params_feature)); |
| 156 ASSERT_EQ("", |
| 157 base::GetFieldTrialParamValueByFeature(params_feature, "foo_key")); |
| 158 } |
| 159 |
| 160 TEST_F(CastFeaturesTest, SetEmptyExperiments) { |
| 161 // Override those features with DCS configs. |
| 162 auto experiments = base::MakeUnique<base::ListValue>(); |
| 163 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 164 |
| 165 InitializeFeatureList(*features, *experiments, "", ""); |
| 166 ASSERT_EQ(0u, GetDCSExperimentIds().size()); |
| 167 } |
| 168 |
| 169 TEST_F(CastFeaturesTest, SetGoodExperiments) { |
| 170 // Override those features with DCS configs. |
| 171 auto experiments = base::MakeUnique<base::ListValue>(); |
| 172 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 173 |
| 174 int32_t ids[] = {12345678, 123, 0, -1}; |
| 175 std::unordered_set<int32_t> expected; |
| 176 for (int32_t id : ids) { |
| 177 experiments->AppendInteger(id); |
| 178 expected.insert(id); |
| 179 } |
| 180 |
| 181 InitializeFeatureList(*features, *experiments, "", ""); |
| 182 ASSERT_EQ(expected, GetDCSExperimentIds()); |
| 183 } |
| 184 |
| 185 TEST_F(CastFeaturesTest, SetSomeGoodExperiments) { |
| 186 // Override those features with DCS configs. |
| 187 auto experiments = base::MakeUnique<base::ListValue>(); |
| 188 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 189 experiments->AppendInteger(1234); |
| 190 experiments->AppendString("foobar"); |
| 191 experiments->AppendBoolean(true); |
| 192 experiments->AppendInteger(1); |
| 193 experiments->AppendDouble(1.23456); |
| 194 |
| 195 std::unordered_set<int32_t> expected; |
| 196 expected.insert(1234); |
| 197 expected.insert(1); |
| 198 |
| 199 InitializeFeatureList(*features, *experiments, "", ""); |
| 200 ASSERT_EQ(expected, GetDCSExperimentIds()); |
| 201 } |
| 202 |
| 203 TEST_F(CastFeaturesTest, SetAllBadExperiments) { |
| 204 // Override those features with DCS configs. |
| 205 auto experiments = base::MakeUnique<base::ListValue>(); |
| 206 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 207 experiments->AppendString("foobar"); |
| 208 experiments->AppendBoolean(true); |
| 209 experiments->AppendDouble(1.23456); |
| 210 |
| 211 std::unordered_set<int32_t> expected; |
| 212 |
| 213 InitializeFeatureList(*features, *experiments, "", ""); |
| 214 ASSERT_EQ(expected, GetDCSExperimentIds()); |
| 215 } |
| 216 |
| 217 TEST_F(CastFeaturesTest, GetOverriddenFeaturesForStorage) { |
| 218 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 219 features->SetBoolean("bool_key", false); |
| 220 features->SetBoolean("bool_key_2", true); |
| 221 |
| 222 auto params = base::MakeUnique<base::DictionaryValue>(); |
| 223 params->SetString("foo_key", "foo"); |
| 224 params->SetString("bar_key", "bar"); |
| 225 params->SetDouble("doub_key", 3.14159); |
| 226 params->SetDouble("long_doub_key", 1.234599999999999); |
| 227 params->SetInteger("int_key", 4242); |
| 228 params->SetInteger("negint_key", -273); |
| 229 params->SetBoolean("bool_key", true); |
| 230 features->Set("params_key", std::move(params)); |
| 231 |
| 232 auto dict = GetOverriddenFeaturesForStorage(*features); |
| 233 bool bval; |
| 234 ASSERT_EQ(3u, dict.size()); |
| 235 ASSERT_TRUE(dict.GetBoolean("bool_key", &bval)); |
| 236 ASSERT_EQ(false, bval); |
| 237 ASSERT_TRUE(dict.GetBoolean("bool_key_2", &bval)); |
| 238 ASSERT_EQ(true, bval); |
| 239 |
| 240 const base::DictionaryValue* dval; |
| 241 std::string sval; |
| 242 ASSERT_TRUE(dict.GetDictionary("params_key", &dval)); |
| 243 ASSERT_EQ(7u, dval->size()); |
| 244 ASSERT_TRUE(dval->GetString("foo_key", &sval)); |
| 245 ASSERT_EQ("foo", sval); |
| 246 ASSERT_TRUE(dval->GetString("bar_key", &sval)); |
| 247 ASSERT_EQ("bar", sval); |
| 248 ASSERT_TRUE(dval->GetString("doub_key", &sval)); |
| 249 ASSERT_EQ("3.14159", sval); |
| 250 ASSERT_TRUE(dval->GetString("long_doub_key", &sval)); |
| 251 ASSERT_EQ("1.234599999999999", sval); |
| 252 ASSERT_TRUE(dval->GetString("int_key", &sval)); |
| 253 ASSERT_EQ("4242", sval); |
| 254 ASSERT_TRUE(dval->GetString("negint_key", &sval)); |
| 255 ASSERT_EQ("-273", sval); |
| 256 ASSERT_TRUE(dval->GetString("bool_key", &sval)); |
| 257 ASSERT_EQ("true", sval); |
| 258 } |
| 259 |
| 260 TEST_F(CastFeaturesTest, GetOverriddenFeaturesForStorage_BadParams) { |
| 261 auto features = base::MakeUnique<base::DictionaryValue>(); |
| 262 features->SetBoolean("bool_key", false); |
| 263 features->SetString("str_key", "foobar"); |
| 264 features->SetInteger("int_key", 12345); |
| 265 features->SetDouble("doub_key", 4.5678); |
| 266 |
| 267 auto params = base::MakeUnique<base::DictionaryValue>(); |
| 268 params->SetString("foo_key", "foo"); |
| 269 features->Set("params_key", std::move(params)); |
| 270 |
| 271 auto dict = GetOverriddenFeaturesForStorage(*features); |
| 272 bool bval; |
| 273 ASSERT_EQ(2u, dict.size()); |
| 274 ASSERT_TRUE(dict.GetBoolean("bool_key", &bval)); |
| 275 ASSERT_EQ(false, bval); |
| 276 |
| 277 const base::DictionaryValue* dval; |
| 278 std::string sval; |
| 279 ASSERT_TRUE(dict.GetDictionary("params_key", &dval)); |
| 280 ASSERT_EQ(1u, dval->size()); |
| 281 ASSERT_TRUE(dval->GetString("foo_key", &sval)); |
| 282 ASSERT_EQ("foo", sval); |
| 283 } |
| 284 |
| 285 } // namespace chromecast |
OLD | NEW |