• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (C) 2019 The Android Open Source Project
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //      http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include <gtest/gtest.h>
16 
17 #include <vector>
18 
19 #include "src/StatsLogProcessor.h"
20 #include "src/state/StateTracker.h"
21 #include "src/stats_log_util.h"
22 #include "tests/statsd_test_util.h"
23 
24 namespace android {
25 namespace os {
26 namespace statsd {
27 
28 #ifdef __ANDROID__
29 
TEST(DurationMetricE2eTest,TestOneBucket)30 TEST(DurationMetricE2eTest, TestOneBucket) {
31     StatsdConfig config;
32     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
33 
34     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
35     auto screenOffMatcher = CreateScreenTurnedOffAtomMatcher();
36     *config.add_atom_matcher() = screenOnMatcher;
37     *config.add_atom_matcher() = screenOffMatcher;
38 
39     auto durationPredicate = CreateScreenIsOnPredicate();
40     *config.add_predicate() = durationPredicate;
41 
42     int64_t metricId = 123456;
43     auto durationMetric = config.add_duration_metric();
44     durationMetric->set_id(metricId);
45     durationMetric->set_what(durationPredicate.id());
46     durationMetric->set_bucket(FIVE_MINUTES);
47     durationMetric->set_aggregation_type(DurationMetric_AggregationType_SUM);
48 
49     const int64_t baseTimeNs = 0;                                   // 0:00
50     const int64_t configAddedTimeNs = baseTimeNs + 1 * NS_PER_SEC;  // 0:01
51     const int64_t bucketSizeNs =
52             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000LL * 1000LL;
53 
54     int uid = 12345;
55     int64_t cfgId = 98765;
56     ConfigKey cfgKey(uid, cfgId);
57 
58     auto processor = CreateStatsLogProcessor(baseTimeNs, configAddedTimeNs, config, cfgKey);
59 
60     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
61     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
62     EXPECT_TRUE(metricsManager->isConfigValid());
63     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
64     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
65     EXPECT_TRUE(metricsManager->isActive());
66     EXPECT_TRUE(metricProducer->mIsActive);
67 
68     std::unique_ptr<LogEvent> event;
69 
70     // Screen is off at start of bucket.
71     event = CreateScreenStateChangedEvent(configAddedTimeNs,
72                                           android::view::DISPLAY_STATE_OFF);  // 0:01
73     processor->OnLogEvent(event.get());
74 
75     // Turn screen on.
76     const int64_t durationStartNs = configAddedTimeNs + 10 * NS_PER_SEC;  // 0:11
77     event = CreateScreenStateChangedEvent(durationStartNs, android::view::DISPLAY_STATE_ON);
78     processor->OnLogEvent(event.get());
79 
80     // Turn off screen 30 seconds after turning on.
81     const int64_t durationEndNs = durationStartNs + 30 * NS_PER_SEC;  // 0:41
82     event = CreateScreenStateChangedEvent(durationEndNs, android::view::DISPLAY_STATE_OFF);
83     processor->OnLogEvent(event.get());
84 
85     event = CreateScreenBrightnessChangedEvent(durationEndNs + 1 * NS_PER_SEC, 64);  // 0:42
86     processor->OnLogEvent(event.get());
87 
88     ConfigMetricsReportList reports;
89     vector<uint8_t> buffer;
90     processor->onDumpReport(cfgKey, configAddedTimeNs + bucketSizeNs + 1 * NS_PER_SEC, false, true,
91                             ADB_DUMP, FAST, &buffer);  // 5:01
92     EXPECT_TRUE(buffer.size() > 0);
93     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
94     backfillDimensionPath(&reports);
95     backfillStartEndTimestamp(&reports);
96     ASSERT_EQ(1, reports.reports_size());
97     ASSERT_EQ(1, reports.reports(0).metrics_size());
98     EXPECT_EQ(metricId, reports.reports(0).metrics(0).metric_id());
99     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
100 
101     StatsLogReport::DurationMetricDataWrapper durationMetrics;
102     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
103                                     &durationMetrics);
104     ASSERT_EQ(1, durationMetrics.data_size());
105 
106     DurationMetricData data = durationMetrics.data(0);
107     ASSERT_EQ(1, data.bucket_info_size());
108     EXPECT_EQ(durationEndNs - durationStartNs, data.bucket_info(0).duration_nanos());
109     EXPECT_EQ(configAddedTimeNs, data.bucket_info(0).start_bucket_elapsed_nanos());
110     EXPECT_EQ(baseTimeNs + bucketSizeNs, data.bucket_info(0).end_bucket_elapsed_nanos());
111 }
112 
TEST(DurationMetricE2eTest,TestTwoBuckets)113 TEST(DurationMetricE2eTest, TestTwoBuckets) {
114     StatsdConfig config;
115     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
116 
117     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
118     auto screenOffMatcher = CreateScreenTurnedOffAtomMatcher();
119     *config.add_atom_matcher() = screenOnMatcher;
120     *config.add_atom_matcher() = screenOffMatcher;
121 
122     auto durationPredicate = CreateScreenIsOnPredicate();
123     *config.add_predicate() = durationPredicate;
124 
125     int64_t metricId = 123456;
126     auto durationMetric = config.add_duration_metric();
127     durationMetric->set_id(metricId);
128     durationMetric->set_what(durationPredicate.id());
129     durationMetric->set_bucket(FIVE_MINUTES);
130     durationMetric->set_aggregation_type(DurationMetric_AggregationType_SUM);
131 
132     const int64_t baseTimeNs = 0;                                   // 0:00
133     const int64_t configAddedTimeNs = baseTimeNs + 1 * NS_PER_SEC;  // 0:01
134     const int64_t bucketSizeNs =
135             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000LL * 1000LL;
136 
137     int uid = 12345;
138     int64_t cfgId = 98765;
139     ConfigKey cfgKey(uid, cfgId);
140 
141     auto processor = CreateStatsLogProcessor(baseTimeNs, configAddedTimeNs, config, cfgKey);
142 
143     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
144     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
145     EXPECT_TRUE(metricsManager->isConfigValid());
146     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
147     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
148     EXPECT_TRUE(metricsManager->isActive());
149     EXPECT_TRUE(metricProducer->mIsActive);
150 
151     std::unique_ptr<LogEvent> event;
152 
153     // Screen is off at start of bucket.
154     event = CreateScreenStateChangedEvent(configAddedTimeNs,
155                                           android::view::DISPLAY_STATE_OFF);  // 0:01
156     processor->OnLogEvent(event.get());
157 
158     // Turn screen on.
159     const int64_t durationStartNs = configAddedTimeNs + 10 * NS_PER_SEC;  // 0:11
160     event = CreateScreenStateChangedEvent(durationStartNs, android::view::DISPLAY_STATE_ON);
161     processor->OnLogEvent(event.get());
162 
163     // Turn off screen 30 seconds after turning on.
164     const int64_t durationEndNs = durationStartNs + 30 * NS_PER_SEC;  // 0:41
165     event = CreateScreenStateChangedEvent(durationEndNs, android::view::DISPLAY_STATE_OFF);
166     processor->OnLogEvent(event.get());
167 
168     event = CreateScreenBrightnessChangedEvent(durationEndNs + 1 * NS_PER_SEC, 64);  // 0:42
169     processor->OnLogEvent(event.get());
170 
171     ConfigMetricsReportList reports;
172     vector<uint8_t> buffer;
173     processor->onDumpReport(cfgKey, configAddedTimeNs + 2 * bucketSizeNs + 1 * NS_PER_SEC, false,
174                             true, ADB_DUMP, FAST, &buffer);  // 10:01
175     EXPECT_TRUE(buffer.size() > 0);
176     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
177     backfillDimensionPath(&reports);
178     backfillStartEndTimestamp(&reports);
179     ASSERT_EQ(1, reports.reports_size());
180     ASSERT_EQ(1, reports.reports(0).metrics_size());
181     EXPECT_EQ(metricId, reports.reports(0).metrics(0).metric_id());
182     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
183 
184     StatsLogReport::DurationMetricDataWrapper durationMetrics;
185     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
186                                     &durationMetrics);
187     ASSERT_EQ(1, durationMetrics.data_size());
188 
189     DurationMetricData data = durationMetrics.data(0);
190     ASSERT_EQ(1, data.bucket_info_size());
191 
192     auto bucketInfo = data.bucket_info(0);
193     EXPECT_EQ(0, bucketInfo.bucket_num());
194     EXPECT_EQ(durationEndNs - durationStartNs, bucketInfo.duration_nanos());
195     EXPECT_EQ(configAddedTimeNs, bucketInfo.start_bucket_elapsed_nanos());
196     EXPECT_EQ(baseTimeNs + bucketSizeNs, bucketInfo.end_bucket_elapsed_nanos());
197 }
198 
TEST(DurationMetricE2eTest,TestWithActivation)199 TEST(DurationMetricE2eTest, TestWithActivation) {
200     StatsdConfig config;
201     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
202 
203     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
204     auto screenOffMatcher = CreateScreenTurnedOffAtomMatcher();
205     auto crashMatcher = CreateProcessCrashAtomMatcher();
206     *config.add_atom_matcher() = screenOnMatcher;
207     *config.add_atom_matcher() = screenOffMatcher;
208     *config.add_atom_matcher() = crashMatcher;
209 
210     auto durationPredicate = CreateScreenIsOnPredicate();
211     *config.add_predicate() = durationPredicate;
212 
213     int64_t metricId = 123456;
214     auto durationMetric = config.add_duration_metric();
215     durationMetric->set_id(metricId);
216     durationMetric->set_what(durationPredicate.id());
217     durationMetric->set_bucket(FIVE_MINUTES);
218     durationMetric->set_aggregation_type(DurationMetric_AggregationType_SUM);
219 
220     auto metric_activation1 = config.add_metric_activation();
221     metric_activation1->set_metric_id(metricId);
222     auto event_activation1 = metric_activation1->add_event_activation();
223     event_activation1->set_atom_matcher_id(crashMatcher.id());
224     event_activation1->set_ttl_seconds(30);  // 30 secs.
225 
226     const int64_t bucketStartTimeNs = 10000000000;
227     const int64_t bucketSizeNs =
228             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000LL * 1000LL;
229 
230     int uid = 12345;
231     int64_t cfgId = 98765;
232     ConfigKey cfgKey(uid, cfgId);
233 
234     sp<UidMap> m = new UidMap();
235     sp<StatsPullerManager> pullerManager = new StatsPullerManager();
236     sp<AlarmMonitor> anomalyAlarmMonitor;
237     sp<AlarmMonitor> subscriberAlarmMonitor;
238     vector<int64_t> activeConfigsBroadcast;
239 
240     int broadcastCount = 0;
241     StatsLogProcessor processor(
242             m, pullerManager, anomalyAlarmMonitor, subscriberAlarmMonitor, bucketStartTimeNs,
243             [](const ConfigKey& key) { return true; },
244             [&uid, &broadcastCount, &activeConfigsBroadcast](const int& broadcastUid,
245                                                              const vector<int64_t>& activeConfigs) {
246                 broadcastCount++;
247                 EXPECT_EQ(broadcastUid, uid);
248                 activeConfigsBroadcast.clear();
249                 activeConfigsBroadcast.insert(activeConfigsBroadcast.end(), activeConfigs.begin(),
250                                               activeConfigs.end());
251                 return true;
252             });
253 
254     processor.OnConfigUpdated(bucketStartTimeNs, cfgKey, config);  // 0:00
255 
256     ASSERT_EQ(processor.mMetricsManagers.size(), 1u);
257     sp<MetricsManager> metricsManager = processor.mMetricsManagers.begin()->second;
258     EXPECT_TRUE(metricsManager->isConfigValid());
259     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
260     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
261     auto& eventActivationMap = metricProducer->mEventActivationMap;
262 
263     EXPECT_FALSE(metricsManager->isActive());
264     EXPECT_FALSE(metricProducer->mIsActive);
265     ASSERT_EQ(eventActivationMap.size(), 1u);
266     EXPECT_TRUE(eventActivationMap.find(2) != eventActivationMap.end());
267     EXPECT_EQ(eventActivationMap[2]->state, ActivationState::kNotActive);
268     EXPECT_EQ(eventActivationMap[2]->start_ns, 0);
269     EXPECT_EQ(eventActivationMap[2]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
270 
271     std::unique_ptr<LogEvent> event;
272 
273     // Turn screen off.
274     event = CreateScreenStateChangedEvent(bucketStartTimeNs + 2 * NS_PER_SEC,
275                                           android::view::DISPLAY_STATE_OFF);  // 0:02
276     processor.OnLogEvent(event.get(), bucketStartTimeNs + 2 * NS_PER_SEC);
277 
278     // Turn screen on.
279     const int64_t durationStartNs = bucketStartTimeNs + 5 * NS_PER_SEC;  // 0:05
280     event = CreateScreenStateChangedEvent(durationStartNs, android::view::DISPLAY_STATE_ON);
281     processor.OnLogEvent(event.get(), durationStartNs);
282 
283     // Activate metric.
284     const int64_t activationStartNs = bucketStartTimeNs + 5 * NS_PER_SEC;  // 0:10
285     const int64_t activationEndNs =
286             activationStartNs + event_activation1->ttl_seconds() * NS_PER_SEC;  // 0:40
287     event = CreateAppCrashEvent(activationStartNs, 111);
288     processor.OnLogEvent(event.get(), activationStartNs);
289     EXPECT_TRUE(metricsManager->isActive());
290     EXPECT_TRUE(metricProducer->mIsActive);
291     EXPECT_EQ(broadcastCount, 1);
292     ASSERT_EQ(activeConfigsBroadcast.size(), 1);
293     EXPECT_EQ(activeConfigsBroadcast[0], cfgId);
294     EXPECT_EQ(eventActivationMap[2]->state, ActivationState::kActive);
295     EXPECT_EQ(eventActivationMap[2]->start_ns, activationStartNs);
296     EXPECT_EQ(eventActivationMap[2]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
297 
298     // Expire activation.
299     const int64_t expirationNs = activationEndNs + 7 * NS_PER_SEC;
300     event = CreateScreenBrightnessChangedEvent(expirationNs, 64);  // 0:47
301     processor.OnLogEvent(event.get(), expirationNs);
302     EXPECT_FALSE(metricsManager->isActive());
303     EXPECT_FALSE(metricProducer->mIsActive);
304     EXPECT_EQ(broadcastCount, 2);
305     ASSERT_EQ(activeConfigsBroadcast.size(), 0);
306     ASSERT_EQ(eventActivationMap.size(), 1u);
307     EXPECT_TRUE(eventActivationMap.find(2) != eventActivationMap.end());
308     EXPECT_EQ(eventActivationMap[2]->state, ActivationState::kNotActive);
309     EXPECT_EQ(eventActivationMap[2]->start_ns, activationStartNs);
310     EXPECT_EQ(eventActivationMap[2]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
311 
312     // Turn off screen 10 seconds after activation expiration.
313     const int64_t durationEndNs = activationEndNs + 10 * NS_PER_SEC;  // 0:50
314     event = CreateScreenStateChangedEvent(durationEndNs, android::view::DISPLAY_STATE_OFF);
315     processor.OnLogEvent(event.get(), durationEndNs);
316 
317     // Turn screen on.
318     const int64_t duration2StartNs = durationEndNs + 5 * NS_PER_SEC;  // 0:55
319     event = CreateScreenStateChangedEvent(duration2StartNs, android::view::DISPLAY_STATE_ON);
320     processor.OnLogEvent(event.get(), duration2StartNs);
321 
322     // Turn off screen.
323     const int64_t duration2EndNs = duration2StartNs + 10 * NS_PER_SEC;  // 1:05
324     event = CreateScreenStateChangedEvent(duration2EndNs, android::view::DISPLAY_STATE_OFF);
325     processor.OnLogEvent(event.get(), duration2EndNs);
326 
327     // Activate metric.
328     const int64_t activation2StartNs = duration2EndNs + 5 * NS_PER_SEC;  // 1:10
329     const int64_t activation2EndNs =
330             activation2StartNs + event_activation1->ttl_seconds() * NS_PER_SEC;  // 1:40
331     event = CreateAppCrashEvent(activation2StartNs, 211);
332     processor.OnLogEvent(event.get(), activation2StartNs);
333     EXPECT_TRUE(metricsManager->isActive());
334     EXPECT_TRUE(metricProducer->mIsActive);
335     EXPECT_EQ(broadcastCount, 3);
336     ASSERT_EQ(activeConfigsBroadcast.size(), 1);
337     EXPECT_EQ(activeConfigsBroadcast[0], cfgId);
338     EXPECT_EQ(eventActivationMap[2]->state, ActivationState::kActive);
339     EXPECT_EQ(eventActivationMap[2]->start_ns, activation2StartNs);
340     EXPECT_EQ(eventActivationMap[2]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
341 
342     ConfigMetricsReportList reports;
343     vector<uint8_t> buffer;
344     processor.onDumpReport(cfgKey, bucketStartTimeNs + bucketSizeNs + 1 * NS_PER_SEC, false, true,
345                            ADB_DUMP, FAST, &buffer);  // 5:01
346     EXPECT_TRUE(buffer.size() > 0);
347     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
348     backfillDimensionPath(&reports);
349     backfillStartEndTimestamp(&reports);
350     ASSERT_EQ(1, reports.reports_size());
351     ASSERT_EQ(1, reports.reports(0).metrics_size());
352     EXPECT_EQ(metricId, reports.reports(0).metrics(0).metric_id());
353     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
354 
355     StatsLogReport::DurationMetricDataWrapper durationMetrics;
356     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
357                                     &durationMetrics);
358     ASSERT_EQ(1, durationMetrics.data_size());
359 
360     DurationMetricData data = durationMetrics.data(0);
361     ASSERT_EQ(1, data.bucket_info_size());
362 
363     auto bucketInfo = data.bucket_info(0);
364     EXPECT_EQ(0, bucketInfo.bucket_num());
365     EXPECT_EQ(bucketStartTimeNs, bucketInfo.start_bucket_elapsed_nanos());
366     EXPECT_EQ(expirationNs, bucketInfo.end_bucket_elapsed_nanos());
367     EXPECT_EQ(expirationNs - durationStartNs, bucketInfo.duration_nanos());
368 }
369 
TEST(DurationMetricE2eTest,TestWithCondition)370 TEST(DurationMetricE2eTest, TestWithCondition) {
371     StatsdConfig config;
372     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
373     *config.add_atom_matcher() = CreateAcquireWakelockAtomMatcher();
374     *config.add_atom_matcher() = CreateReleaseWakelockAtomMatcher();
375     *config.add_atom_matcher() = CreateMoveToBackgroundAtomMatcher();
376     *config.add_atom_matcher() = CreateMoveToForegroundAtomMatcher();
377 
378     auto holdingWakelockPredicate = CreateHoldingWakelockPredicate();
379     *config.add_predicate() = holdingWakelockPredicate;
380 
381     auto isInBackgroundPredicate = CreateIsInBackgroundPredicate();
382     *config.add_predicate() = isInBackgroundPredicate;
383 
384     auto durationMetric = config.add_duration_metric();
385     durationMetric->set_id(StringToId("WakelockDuration"));
386     durationMetric->set_what(holdingWakelockPredicate.id());
387     durationMetric->set_condition(isInBackgroundPredicate.id());
388     durationMetric->set_aggregation_type(DurationMetric::SUM);
389     durationMetric->set_bucket(FIVE_MINUTES);
390 
391     ConfigKey cfgKey;
392     uint64_t bucketStartTimeNs = 10000000000;
393     uint64_t bucketSizeNs =
394             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
395     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
396     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
397     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
398     EXPECT_TRUE(metricsManager->isConfigValid());
399     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
400     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
401     auto& eventActivationMap = metricProducer->mEventActivationMap;
402     EXPECT_TRUE(metricsManager->isActive());
403     EXPECT_TRUE(metricProducer->mIsActive);
404     EXPECT_TRUE(eventActivationMap.empty());
405 
406     int appUid = 123;
407     vector<int> attributionUids1 = {appUid};
408     vector<string> attributionTags1 = {"App1"};
409 
410     auto event = CreateAcquireWakelockEvent(bucketStartTimeNs + 10 * NS_PER_SEC, attributionUids1,
411                                             attributionTags1,
412                                             "wl1");  // 0:10
413     processor->OnLogEvent(event.get());
414 
415     event = CreateMoveToBackgroundEvent(bucketStartTimeNs + 22 * NS_PER_SEC, appUid);  // 0:22
416     processor->OnLogEvent(event.get());
417 
418     event = CreateMoveToForegroundEvent(bucketStartTimeNs + (3 * 60 + 15) * NS_PER_SEC,
419                                         appUid);  // 3:15
420     processor->OnLogEvent(event.get());
421 
422     event = CreateReleaseWakelockEvent(bucketStartTimeNs + 4 * 60 * NS_PER_SEC, attributionUids1,
423                                        attributionTags1,
424                                        "wl1");  // 4:00
425     processor->OnLogEvent(event.get());
426 
427     vector<uint8_t> buffer;
428     ConfigMetricsReportList reports;
429     processor->onDumpReport(cfgKey, bucketStartTimeNs + bucketSizeNs + 1, false, true, ADB_DUMP,
430                             FAST, &buffer);
431     ASSERT_GT(buffer.size(), 0);
432     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
433     backfillDimensionPath(&reports);
434     backfillStringInReport(&reports);
435     backfillStartEndTimestamp(&reports);
436 
437     ASSERT_EQ(1, reports.reports_size());
438     ASSERT_EQ(1, reports.reports(0).metrics_size());
439     StatsLogReport::DurationMetricDataWrapper durationMetrics;
440     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
441                                     &durationMetrics);
442     ASSERT_EQ(1, durationMetrics.data_size());
443 
444     DurationMetricData data = durationMetrics.data(0);
445 
446     // Validate bucket info.
447     ASSERT_EQ(1, data.bucket_info_size());
448 
449     auto bucketInfo = data.bucket_info(0);
450     EXPECT_EQ(bucketStartTimeNs, bucketInfo.start_bucket_elapsed_nanos());
451     EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, bucketInfo.end_bucket_elapsed_nanos());
452     EXPECT_EQ((2 * 60 + 53) * NS_PER_SEC, bucketInfo.duration_nanos());
453 }
454 
TEST(DurationMetricE2eTest,TestWithSlicedCondition)455 TEST(DurationMetricE2eTest, TestWithSlicedCondition) {
456     StatsdConfig config;
457     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
458     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
459     *config.add_atom_matcher() = CreateAcquireWakelockAtomMatcher();
460     *config.add_atom_matcher() = CreateReleaseWakelockAtomMatcher();
461     *config.add_atom_matcher() = CreateMoveToBackgroundAtomMatcher();
462     *config.add_atom_matcher() = CreateMoveToForegroundAtomMatcher();
463 
464     auto holdingWakelockPredicate = CreateHoldingWakelockPredicate();
465     // The predicate is dimensioning by first attribution node by uid.
466     FieldMatcher dimensions = CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED,
467                                                              {Position::FIRST});
468     *holdingWakelockPredicate.mutable_simple_predicate()->mutable_dimensions() = dimensions;
469     *config.add_predicate() = holdingWakelockPredicate;
470 
471     auto isInBackgroundPredicate = CreateIsInBackgroundPredicate();
472     *isInBackgroundPredicate.mutable_simple_predicate()->mutable_dimensions() =
473             CreateDimensions(util::ACTIVITY_FOREGROUND_STATE_CHANGED, {Position::FIRST});
474     *config.add_predicate() = isInBackgroundPredicate;
475 
476     auto durationMetric = config.add_duration_metric();
477     durationMetric->set_id(StringToId("WakelockDuration"));
478     durationMetric->set_what(holdingWakelockPredicate.id());
479     durationMetric->set_condition(isInBackgroundPredicate.id());
480     durationMetric->set_aggregation_type(DurationMetric::SUM);
481     // The metric is dimensioning by first attribution node and only by uid.
482     *durationMetric->mutable_dimensions_in_what() = CreateAttributionUidDimensions(
483             util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
484     durationMetric->set_bucket(FIVE_MINUTES);
485 
486     // Links between wakelock state atom and condition of app is in background.
487     auto links = durationMetric->add_links();
488     links->set_condition(isInBackgroundPredicate.id());
489     *links->mutable_fields_in_what() =
490             CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
491     auto dimensionCondition = links->mutable_fields_in_condition();
492     dimensionCondition->set_field(util::ACTIVITY_FOREGROUND_STATE_CHANGED);
493     dimensionCondition->add_child()->set_field(1);  // uid field.
494 
495     ConfigKey cfgKey;
496     uint64_t bucketStartTimeNs = 10000000000;
497     uint64_t bucketSizeNs =
498             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
499     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
500     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
501     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
502     EXPECT_TRUE(metricsManager->isConfigValid());
503     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
504     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
505     auto& eventActivationMap = metricProducer->mEventActivationMap;
506     EXPECT_TRUE(metricsManager->isActive());
507     EXPECT_TRUE(metricProducer->mIsActive);
508     EXPECT_TRUE(eventActivationMap.empty());
509 
510     int appUid = 123;
511     std::vector<int> attributionUids1 = {appUid};
512     std::vector<string> attributionTags1 = {"App1"};
513 
514     auto event = CreateAcquireWakelockEvent(bucketStartTimeNs + 10 * NS_PER_SEC, attributionUids1,
515                                             attributionTags1, "wl1");  // 0:10
516     processor->OnLogEvent(event.get());
517 
518     event = CreateMoveToBackgroundEvent(bucketStartTimeNs + 22 * NS_PER_SEC, appUid);  // 0:22
519     processor->OnLogEvent(event.get());
520 
521     event = CreateReleaseWakelockEvent(bucketStartTimeNs + 60 * NS_PER_SEC, attributionUids1,
522                                        attributionTags1, "wl1");  // 1:00
523     processor->OnLogEvent(event.get());
524 
525     event = CreateMoveToForegroundEvent(bucketStartTimeNs + (3 * 60 + 15) * NS_PER_SEC,
526                                         appUid);  // 3:15
527     processor->OnLogEvent(event.get());
528 
529     vector<uint8_t> buffer;
530     ConfigMetricsReportList reports;
531     processor->onDumpReport(cfgKey, bucketStartTimeNs + bucketSizeNs + 1, false, true, ADB_DUMP,
532                             FAST, &buffer);
533     ASSERT_GT(buffer.size(), 0);
534     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
535     backfillDimensionPath(&reports);
536     backfillStringInReport(&reports);
537     backfillStartEndTimestamp(&reports);
538 
539     ASSERT_EQ(1, reports.reports_size());
540     ASSERT_EQ(1, reports.reports(0).metrics_size());
541     StatsLogReport::DurationMetricDataWrapper durationMetrics;
542     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
543                                     &durationMetrics);
544     ASSERT_EQ(1, durationMetrics.data_size());
545 
546     DurationMetricData data = durationMetrics.data(0);
547     // Validate dimension value.
548     ValidateAttributionUidDimension(data.dimensions_in_what(),
549                                     util::WAKELOCK_STATE_CHANGED, appUid);
550     // Validate bucket info.
551     ASSERT_EQ(1, data.bucket_info_size());
552 
553     auto bucketInfo = data.bucket_info(0);
554     EXPECT_EQ(bucketStartTimeNs, bucketInfo.start_bucket_elapsed_nanos());
555     EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, bucketInfo.end_bucket_elapsed_nanos());
556     EXPECT_EQ(38 * NS_PER_SEC, bucketInfo.duration_nanos());
557 }
558 
TEST(DurationMetricE2eTest,TestWithActivationAndSlicedCondition)559 TEST(DurationMetricE2eTest, TestWithActivationAndSlicedCondition) {
560     StatsdConfig config;
561     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
562     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
563     *config.add_atom_matcher() = CreateAcquireWakelockAtomMatcher();
564     *config.add_atom_matcher() = CreateReleaseWakelockAtomMatcher();
565     *config.add_atom_matcher() = CreateMoveToBackgroundAtomMatcher();
566     *config.add_atom_matcher() = CreateMoveToForegroundAtomMatcher();
567     *config.add_atom_matcher() = screenOnMatcher;
568 
569     auto holdingWakelockPredicate = CreateHoldingWakelockPredicate();
570     // The predicate is dimensioning by first attribution node by uid.
571     FieldMatcher dimensions = CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED,
572                                                              {Position::FIRST});
573     *holdingWakelockPredicate.mutable_simple_predicate()->mutable_dimensions() = dimensions;
574     *config.add_predicate() = holdingWakelockPredicate;
575 
576     auto isInBackgroundPredicate = CreateIsInBackgroundPredicate();
577     *isInBackgroundPredicate.mutable_simple_predicate()->mutable_dimensions() =
578             CreateDimensions(util::ACTIVITY_FOREGROUND_STATE_CHANGED, {Position::FIRST});
579     *config.add_predicate() = isInBackgroundPredicate;
580 
581     auto durationMetric = config.add_duration_metric();
582     durationMetric->set_id(StringToId("WakelockDuration"));
583     durationMetric->set_what(holdingWakelockPredicate.id());
584     durationMetric->set_condition(isInBackgroundPredicate.id());
585     durationMetric->set_aggregation_type(DurationMetric::SUM);
586     // The metric is dimensioning by first attribution node and only by uid.
587     *durationMetric->mutable_dimensions_in_what() = CreateAttributionUidDimensions(
588             util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
589     durationMetric->set_bucket(FIVE_MINUTES);
590 
591     // Links between wakelock state atom and condition of app is in background.
592     auto links = durationMetric->add_links();
593     links->set_condition(isInBackgroundPredicate.id());
594     *links->mutable_fields_in_what() =
595             CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
596     auto dimensionCondition = links->mutable_fields_in_condition();
597     dimensionCondition->set_field(util::ACTIVITY_FOREGROUND_STATE_CHANGED);
598     dimensionCondition->add_child()->set_field(1);  // uid field.
599 
600     auto metric_activation1 = config.add_metric_activation();
601     metric_activation1->set_metric_id(durationMetric->id());
602     auto event_activation1 = metric_activation1->add_event_activation();
603     event_activation1->set_atom_matcher_id(screenOnMatcher.id());
604     event_activation1->set_ttl_seconds(60 * 2);  // 2 minutes.
605 
606     ConfigKey cfgKey;
607     uint64_t bucketStartTimeNs = 10000000000;
608     uint64_t bucketSizeNs =
609             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
610     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
611     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
612     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
613     EXPECT_TRUE(metricsManager->isConfigValid());
614     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
615     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
616     auto& eventActivationMap = metricProducer->mEventActivationMap;
617     EXPECT_FALSE(metricsManager->isActive());
618     EXPECT_FALSE(metricProducer->mIsActive);
619     ASSERT_EQ(eventActivationMap.size(), 1u);
620     EXPECT_TRUE(eventActivationMap.find(4) != eventActivationMap.end());
621     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kNotActive);
622     EXPECT_EQ(eventActivationMap[4]->start_ns, 0);
623     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
624 
625     int appUid = 123;
626     std::vector<int> attributionUids1 = {appUid};
627     std::vector<string> attributionTags1 = {"App1"};
628 
629     auto event = CreateAcquireWakelockEvent(bucketStartTimeNs + 10 * NS_PER_SEC, attributionUids1,
630                                             attributionTags1, "wl1");  // 0:10
631     processor->OnLogEvent(event.get());
632     EXPECT_FALSE(metricsManager->isActive());
633     EXPECT_FALSE(metricProducer->mIsActive);
634     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kNotActive);
635     EXPECT_EQ(eventActivationMap[4]->start_ns, 0);
636     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
637 
638     event = CreateMoveToBackgroundEvent(bucketStartTimeNs + 22 * NS_PER_SEC, appUid);  // 0:22
639     processor->OnLogEvent(event.get());
640     EXPECT_FALSE(metricsManager->isActive());
641     EXPECT_FALSE(metricProducer->mIsActive);
642     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kNotActive);
643     EXPECT_EQ(eventActivationMap[4]->start_ns, 0);
644     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
645 
646     const int64_t durationStartNs = bucketStartTimeNs + 30 * NS_PER_SEC;  // 0:30
647     event = CreateScreenStateChangedEvent(durationStartNs, android::view::DISPLAY_STATE_ON);
648     processor->OnLogEvent(event.get());
649     EXPECT_TRUE(metricsManager->isActive());
650     EXPECT_TRUE(metricProducer->mIsActive);
651     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kActive);
652     EXPECT_EQ(eventActivationMap[4]->start_ns, durationStartNs);
653     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
654 
655     const int64_t durationEndNs =
656             durationStartNs + (event_activation1->ttl_seconds() + 30) * NS_PER_SEC;  // 3:00
657     event = CreateAppCrashEvent(durationEndNs, 333);
658     processor->OnLogEvent(event.get());
659     EXPECT_FALSE(metricsManager->isActive());
660     EXPECT_FALSE(metricProducer->mIsActive);
661     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kNotActive);
662     EXPECT_EQ(eventActivationMap[4]->start_ns, durationStartNs);
663     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
664 
665     event = CreateMoveToForegroundEvent(bucketStartTimeNs + (3 * 60 + 15) * NS_PER_SEC,
666                                         appUid);  // 3:15
667     processor->OnLogEvent(event.get());
668 
669     event = CreateReleaseWakelockEvent(bucketStartTimeNs + (4 * 60 + 17) * NS_PER_SEC,
670                                        attributionUids1, attributionTags1, "wl1");  // 4:17
671     processor->OnLogEvent(event.get());
672 
673     event = CreateMoveToBackgroundEvent(bucketStartTimeNs + (4 * 60 + 20) * NS_PER_SEC,
674                                         appUid);  // 4:20
675     processor->OnLogEvent(event.get());
676 
677     event = CreateAcquireWakelockEvent(bucketStartTimeNs + (4 * 60 + 25) * NS_PER_SEC,
678                                        attributionUids1, attributionTags1, "wl1");  // 4:25
679     processor->OnLogEvent(event.get());
680 
681     const int64_t duration2StartNs = bucketStartTimeNs + (4 * 60 + 30) * NS_PER_SEC;  // 4:30
682     event = CreateScreenStateChangedEvent(duration2StartNs, android::view::DISPLAY_STATE_ON);
683     processor->OnLogEvent(event.get());
684     EXPECT_TRUE(metricsManager->isActive());
685     EXPECT_TRUE(metricProducer->mIsActive);
686     EXPECT_EQ(eventActivationMap[4]->state, ActivationState::kActive);
687     EXPECT_EQ(eventActivationMap[4]->start_ns, duration2StartNs);
688     EXPECT_EQ(eventActivationMap[4]->ttl_ns, event_activation1->ttl_seconds() * NS_PER_SEC);
689 
690     vector<uint8_t> buffer;
691     ConfigMetricsReportList reports;
692     processor->onDumpReport(cfgKey, bucketStartTimeNs + bucketSizeNs + 1, false, true, ADB_DUMP,
693                             FAST, &buffer);
694     ASSERT_GT(buffer.size(), 0);
695     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
696     backfillDimensionPath(&reports);
697     backfillStringInReport(&reports);
698     backfillStartEndTimestamp(&reports);
699 
700     ASSERT_EQ(1, reports.reports_size());
701     ASSERT_EQ(1, reports.reports(0).metrics_size());
702     StatsLogReport::DurationMetricDataWrapper durationMetrics;
703     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
704                                     &durationMetrics);
705     ASSERT_EQ(1, durationMetrics.data_size());
706 
707     DurationMetricData data = durationMetrics.data(0);
708     // Validate dimension value.
709     ValidateAttributionUidDimension(data.dimensions_in_what(),
710                                     util::WAKELOCK_STATE_CHANGED, appUid);
711     // Validate bucket info.
712     ASSERT_EQ(2, data.bucket_info_size());
713 
714     auto bucketInfo = data.bucket_info(0);
715     EXPECT_EQ(bucketStartTimeNs, bucketInfo.start_bucket_elapsed_nanos());
716     EXPECT_EQ(durationEndNs, bucketInfo.end_bucket_elapsed_nanos());
717     EXPECT_EQ(durationEndNs - durationStartNs, bucketInfo.duration_nanos());
718 
719     bucketInfo = data.bucket_info(1);
720     EXPECT_EQ(durationEndNs, bucketInfo.start_bucket_elapsed_nanos());
721     EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, bucketInfo.end_bucket_elapsed_nanos());
722     EXPECT_EQ(bucketStartTimeNs + bucketSizeNs - duration2StartNs, bucketInfo.duration_nanos());
723 }
724 
TEST(DurationMetricE2eTest,TestWithSlicedState)725 TEST(DurationMetricE2eTest, TestWithSlicedState) {
726     // Initialize config.
727     StatsdConfig config;
728     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
729 
730     *config.add_atom_matcher() = CreateBatterySaverModeStartAtomMatcher();
731     *config.add_atom_matcher() = CreateBatterySaverModeStopAtomMatcher();
732 
733     auto batterySaverModePredicate = CreateBatterySaverModePredicate();
734     *config.add_predicate() = batterySaverModePredicate;
735 
736     auto screenState = CreateScreenState();
737     *config.add_state() = screenState;
738 
739     // Create duration metric that slices by screen state.
740     auto durationMetric = config.add_duration_metric();
741     durationMetric->set_id(StringToId("DurationBatterySaverModeSliceScreen"));
742     durationMetric->set_what(batterySaverModePredicate.id());
743     durationMetric->add_slice_by_state(screenState.id());
744     durationMetric->set_aggregation_type(DurationMetric::SUM);
745     durationMetric->set_bucket(FIVE_MINUTES);
746 
747     // Initialize StatsLogProcessor.
748     int uid = 12345;
749     int64_t cfgId = 98765;
750     ConfigKey cfgKey(uid, cfgId);
751     uint64_t bucketStartTimeNs = 10000000000;  // 0:10
752     uint64_t bucketSizeNs =
753             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
754     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
755 
756     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
757     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
758     EXPECT_TRUE(metricsManager->isConfigValid());
759     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
760     EXPECT_TRUE(metricsManager->isActive());
761     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
762     EXPECT_TRUE(metricProducer->mIsActive);
763     ASSERT_EQ(metricProducer->mSlicedStateAtoms.size(), 1);
764     EXPECT_EQ(metricProducer->mSlicedStateAtoms.at(0), SCREEN_STATE_ATOM_ID);
765     ASSERT_EQ(metricProducer->mStateGroupMap.size(), 0);
766 
767     // Check that StateTrackers were initialized correctly.
768     EXPECT_EQ(1, StateManager::getInstance().getStateTrackersCount());
769     EXPECT_EQ(1, StateManager::getInstance().getListenersCount(SCREEN_STATE_ATOM_ID));
770 
771     /*
772                bucket #1                      bucket #2
773     |     1     2     3     4     5     6     7     8     9     10 (minutes)
774     |-----------------------------|-----------------------------|--
775         ON              OFF     ON                                  (BatterySaverMode)
776       |          |                   |                              (ScreenIsOnEvent)
777            |                  |                                     (ScreenIsOffEvent)
778               |                                                     (ScreenDozeEvent)
779     */
780     // Initialize log events.
781     std::vector<std::unique_ptr<LogEvent>> events;
782     events.push_back(CreateScreenStateChangedEvent(
783             bucketStartTimeNs + 10 * NS_PER_SEC,
784             android::view::DisplayStateEnum::DISPLAY_STATE_ON));                       // 0:20
785     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 20 * NS_PER_SEC));  // 0:30
786     events.push_back(CreateScreenStateChangedEvent(
787             bucketStartTimeNs + 50 * NS_PER_SEC,
788             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));  // 1:00
789     events.push_back(CreateScreenStateChangedEvent(
790             bucketStartTimeNs + 80 * NS_PER_SEC,
791             android::view::DisplayStateEnum::DISPLAY_STATE_DOZE));  // 1:30
792     events.push_back(CreateScreenStateChangedEvent(
793             bucketStartTimeNs + 120 * NS_PER_SEC,
794             android::view::DisplayStateEnum::DISPLAY_STATE_ON));                         // 2:10
795     events.push_back(CreateBatterySaverOffEvent(bucketStartTimeNs + 200 * NS_PER_SEC));  // 3:30
796     events.push_back(CreateScreenStateChangedEvent(
797             bucketStartTimeNs + 250 * NS_PER_SEC,
798             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));                       // 4:20
799     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 280 * NS_PER_SEC));  // 4:50
800 
801     // Bucket boundary.
802     events.push_back(CreateScreenStateChangedEvent(
803             bucketStartTimeNs + 310 * NS_PER_SEC,
804             android::view::DisplayStateEnum::DISPLAY_STATE_ON));  // 5:20
805 
806     // Send log events to StatsLogProcessor.
807     for (auto& event : events) {
808         processor->OnLogEvent(event.get());
809     }
810 
811     // Check dump report.
812     vector<uint8_t> buffer;
813     ConfigMetricsReportList reports;
814     processor->onDumpReport(cfgKey, bucketStartTimeNs + 360 * NS_PER_SEC,
815                             true /* include current partial bucket */, true, ADB_DUMP, FAST,
816                             &buffer);  // 6:10
817     ASSERT_GT(buffer.size(), 0);
818     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
819     backfillDimensionPath(&reports);
820     backfillStringInReport(&reports);
821     backfillStartEndTimestamp(&reports);
822 
823     ASSERT_EQ(1, reports.reports_size());
824     ASSERT_EQ(1, reports.reports(0).metrics_size());
825     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
826     StatsLogReport::DurationMetricDataWrapper durationMetrics;
827     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
828                                     &durationMetrics);
829     ASSERT_EQ(3, durationMetrics.data_size());
830 
831     DurationMetricData data = durationMetrics.data(0);
832     ASSERT_EQ(1, data.slice_by_state_size());
833     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
834     EXPECT_TRUE(data.slice_by_state(0).has_value());
835     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_OFF, data.slice_by_state(0).value());
836     ASSERT_EQ(2, data.bucket_info_size());
837     EXPECT_EQ(50 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
838     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
839     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
840     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
841     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
842     EXPECT_EQ(370 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
843 
844     data = durationMetrics.data(1);
845     ASSERT_EQ(1, data.slice_by_state_size());
846     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
847     EXPECT_TRUE(data.slice_by_state(0).has_value());
848     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_ON, data.slice_by_state(0).value());
849     ASSERT_EQ(2, data.bucket_info_size());
850     EXPECT_EQ(110 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
851     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
852     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
853     EXPECT_EQ(50 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
854     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
855     EXPECT_EQ(370 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
856 
857     data = durationMetrics.data(2);
858     ASSERT_EQ(1, data.slice_by_state_size());
859     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
860     EXPECT_TRUE(data.slice_by_state(0).has_value());
861     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_DOZE, data.slice_by_state(0).value());
862     ASSERT_EQ(1, data.bucket_info_size());
863     EXPECT_EQ(40 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
864     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
865     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
866 }
867 
TEST(DurationMetricE2eTest,TestWithConditionAndSlicedState)868 TEST(DurationMetricE2eTest, TestWithConditionAndSlicedState) {
869     // Initialize config.
870     StatsdConfig config;
871     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
872 
873     *config.add_atom_matcher() = CreateBatterySaverModeStartAtomMatcher();
874     *config.add_atom_matcher() = CreateBatterySaverModeStopAtomMatcher();
875     *config.add_atom_matcher() = CreateBatteryStateNoneMatcher();
876     *config.add_atom_matcher() = CreateBatteryStateUsbMatcher();
877 
878     auto batterySaverModePredicate = CreateBatterySaverModePredicate();
879     *config.add_predicate() = batterySaverModePredicate;
880 
881     auto deviceUnpluggedPredicate = CreateDeviceUnpluggedPredicate();
882     *config.add_predicate() = deviceUnpluggedPredicate;
883 
884     auto screenState = CreateScreenState();
885     *config.add_state() = screenState;
886 
887     // Create duration metric that has a condition and slices by screen state.
888     auto durationMetric = config.add_duration_metric();
889     durationMetric->set_id(StringToId("DurationBatterySaverModeOnBatterySliceScreen"));
890     durationMetric->set_what(batterySaverModePredicate.id());
891     durationMetric->set_condition(deviceUnpluggedPredicate.id());
892     durationMetric->add_slice_by_state(screenState.id());
893     durationMetric->set_aggregation_type(DurationMetric::SUM);
894     durationMetric->set_bucket(FIVE_MINUTES);
895 
896     // Initialize StatsLogProcessor.
897     int uid = 12345;
898     int64_t cfgId = 98765;
899     ConfigKey cfgKey(uid, cfgId);
900     uint64_t bucketStartTimeNs = 10000000000;  // 0:10
901     uint64_t bucketSizeNs =
902             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
903     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
904 
905     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
906     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
907     EXPECT_TRUE(metricsManager->isConfigValid());
908     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
909     EXPECT_TRUE(metricsManager->isActive());
910     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
911     EXPECT_TRUE(metricProducer->mIsActive);
912     ASSERT_EQ(metricProducer->mSlicedStateAtoms.size(), 1);
913     EXPECT_EQ(metricProducer->mSlicedStateAtoms.at(0), SCREEN_STATE_ATOM_ID);
914     ASSERT_EQ(metricProducer->mStateGroupMap.size(), 0);
915 
916     // Check that StateTrackers were initialized correctly.
917     EXPECT_EQ(1, StateManager::getInstance().getStateTrackersCount());
918     EXPECT_EQ(1, StateManager::getInstance().getListenersCount(SCREEN_STATE_ATOM_ID));
919 
920     /*
921                bucket #1                      bucket #2
922     |       1       2       3       4       5     6     7     8  (minutes)
923     |---------------------------------------|------------------
924              ON                          OFF    ON             (BatterySaverMode)
925                   T            F    T                          (DeviceUnpluggedPredicate)
926          |              |              |                       (ScreenIsOnEvent)
927                 |           |                       |          (ScreenIsOffEvent)
928                                 |                              (ScreenDozeEvent)
929     */
930     // Initialize log events.
931     std::vector<std::unique_ptr<LogEvent>> events;
932     events.push_back(CreateScreenStateChangedEvent(
933             bucketStartTimeNs + 20 * NS_PER_SEC,
934             android::view::DisplayStateEnum::DISPLAY_STATE_ON));                       // 0:30
935     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 60 * NS_PER_SEC));  // 1:10
936     events.push_back(CreateScreenStateChangedEvent(
937             bucketStartTimeNs + 80 * NS_PER_SEC,
938             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));  // 1:30
939     events.push_back(
940             CreateBatteryStateChangedEvent(bucketStartTimeNs + 110 * NS_PER_SEC,
941                                            BatteryPluggedStateEnum::BATTERY_PLUGGED_NONE));  // 2:00
942     events.push_back(CreateScreenStateChangedEvent(
943             bucketStartTimeNs + 145 * NS_PER_SEC,
944             android::view::DisplayStateEnum::DISPLAY_STATE_ON));  // 2:35
945     events.push_back(CreateScreenStateChangedEvent(
946             bucketStartTimeNs + 170 * NS_PER_SEC,
947             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));  // 3:00
948     events.push_back(
949             CreateBatteryStateChangedEvent(bucketStartTimeNs + 180 * NS_PER_SEC,
950                                            BatteryPluggedStateEnum::BATTERY_PLUGGED_USB));  // 3:10
951     events.push_back(CreateScreenStateChangedEvent(
952             bucketStartTimeNs + 200 * NS_PER_SEC,
953             android::view::DisplayStateEnum::DISPLAY_STATE_DOZE));  // 3:30
954     events.push_back(
955             CreateBatteryStateChangedEvent(bucketStartTimeNs + 230 * NS_PER_SEC,
956                                            BatteryPluggedStateEnum::BATTERY_PLUGGED_NONE));  // 4:00
957     events.push_back(CreateScreenStateChangedEvent(
958             bucketStartTimeNs + 260 * NS_PER_SEC,
959             android::view::DisplayStateEnum::DISPLAY_STATE_ON));                         // 4:30
960     events.push_back(CreateBatterySaverOffEvent(bucketStartTimeNs + 280 * NS_PER_SEC));  // 4:50
961 
962     // Bucket boundary.
963     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 320 * NS_PER_SEC));  // 5:30
964     events.push_back(CreateScreenStateChangedEvent(
965             bucketStartTimeNs + 380 * NS_PER_SEC,
966             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));  // 6:30
967 
968     // Send log events to StatsLogProcessor.
969     for (auto& event : events) {
970         processor->OnLogEvent(event.get());
971     }
972 
973     // Check dump report.
974     vector<uint8_t> buffer;
975     ConfigMetricsReportList reports;
976     processor->onDumpReport(cfgKey, bucketStartTimeNs + 410 * NS_PER_SEC,
977                             true /* include current partial bucket */, true, ADB_DUMP, FAST,
978                             &buffer);
979     ASSERT_GT(buffer.size(), 0);
980     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
981     backfillDimensionPath(&reports);
982     backfillStringInReport(&reports);
983     backfillStartEndTimestamp(&reports);
984 
985     ASSERT_EQ(1, reports.reports_size());
986     ASSERT_EQ(1, reports.reports(0).metrics_size());
987     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
988     StatsLogReport::DurationMetricDataWrapper durationMetrics;
989     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
990                                     &durationMetrics);
991     ASSERT_EQ(3, durationMetrics.data_size());
992 
993     DurationMetricData data = durationMetrics.data(0);
994     ASSERT_EQ(1, data.slice_by_state_size());
995     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
996     EXPECT_TRUE(data.slice_by_state(0).has_value());
997     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_OFF, data.slice_by_state(0).value());
998     ASSERT_EQ(2, data.bucket_info_size());
999     EXPECT_EQ(45 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1000     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1001     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1002     EXPECT_EQ(30 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1003     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1004     EXPECT_EQ(420 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1005 
1006     data = durationMetrics.data(1);
1007     ASSERT_EQ(1, data.slice_by_state_size());
1008     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1009     EXPECT_TRUE(data.slice_by_state(0).has_value());
1010     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_ON, data.slice_by_state(0).value());
1011     ASSERT_EQ(2, data.bucket_info_size());
1012     EXPECT_EQ(45 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1013     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1014     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1015     EXPECT_EQ(60 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1016     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1017     EXPECT_EQ(420 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1018 
1019     data = durationMetrics.data(2);
1020     ASSERT_EQ(1, data.slice_by_state_size());
1021     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1022     EXPECT_TRUE(data.slice_by_state(0).has_value());
1023     EXPECT_EQ(android::view::DisplayStateEnum::DISPLAY_STATE_DOZE, data.slice_by_state(0).value());
1024     ASSERT_EQ(1, data.bucket_info_size());
1025     EXPECT_EQ(30 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1026     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1027     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1028 }
1029 
TEST(DurationMetricE2eTest,TestWithSlicedStateMapped)1030 TEST(DurationMetricE2eTest, TestWithSlicedStateMapped) {
1031     // Initialize config.
1032     StatsdConfig config;
1033     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
1034 
1035     *config.add_atom_matcher() = CreateBatterySaverModeStartAtomMatcher();
1036     *config.add_atom_matcher() = CreateBatterySaverModeStopAtomMatcher();
1037 
1038     auto batterySaverModePredicate = CreateBatterySaverModePredicate();
1039     *config.add_predicate() = batterySaverModePredicate;
1040 
1041     int64_t screenOnId = 4444;
1042     int64_t screenOffId = 9876;
1043     auto screenStateWithMap = CreateScreenStateWithOnOffMap(screenOnId, screenOffId);
1044     *config.add_state() = screenStateWithMap;
1045 
1046     // Create duration metric that slices by mapped screen state.
1047     auto durationMetric = config.add_duration_metric();
1048     durationMetric->set_id(StringToId("DurationBatterySaverModeSliceScreenMapped"));
1049     durationMetric->set_what(batterySaverModePredicate.id());
1050     durationMetric->add_slice_by_state(screenStateWithMap.id());
1051     durationMetric->set_aggregation_type(DurationMetric::SUM);
1052     durationMetric->set_bucket(FIVE_MINUTES);
1053 
1054     // Initialize StatsLogProcessor.
1055     int uid = 12345;
1056     int64_t cfgId = 98765;
1057     ConfigKey cfgKey(uid, cfgId);
1058     uint64_t bucketStartTimeNs = 10000000000;  // 0:10
1059     uint64_t bucketSizeNs =
1060             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
1061     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
1062 
1063     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
1064     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
1065     EXPECT_TRUE(metricsManager->isConfigValid());
1066     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
1067     EXPECT_TRUE(metricsManager->isActive());
1068     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
1069     EXPECT_TRUE(metricProducer->mIsActive);
1070     ASSERT_EQ(metricProducer->mSlicedStateAtoms.size(), 1);
1071     EXPECT_EQ(metricProducer->mSlicedStateAtoms.at(0), SCREEN_STATE_ATOM_ID);
1072     ASSERT_EQ(metricProducer->mStateGroupMap.size(), 1);
1073 
1074     // Check that StateTrackers were initialized correctly.
1075     EXPECT_EQ(1, StateManager::getInstance().getStateTrackersCount());
1076     EXPECT_EQ(1, StateManager::getInstance().getListenersCount(SCREEN_STATE_ATOM_ID));
1077 
1078     /*
1079                bucket #1                      bucket #2
1080     |     1     2     3     4     5     6     7     8     9     10 (minutes)
1081     |-----------------------------|-----------------------------|--
1082         ON              OFF     ON                                  (BatterySaverMode)
1083      ---------------------------------------------------------SCREEN_OFF events
1084            |                  |                                  (ScreenStateOffEvent = 1)
1085               |                                                  (ScreenStateDozeEvent = 3)
1086                                                 |                (ScreenStateDozeSuspendEvent = 4)
1087      ---------------------------------------------------------SCREEN_ON events
1088       |          |                   |                           (ScreenStateOnEvent = 2)
1089                       |                                          (ScreenStateVrEvent = 5)
1090                                             |                    (ScreenStateOnSuspendEvent = 6)
1091     */
1092     // Initialize log events.
1093     std::vector<std::unique_ptr<LogEvent>> events;
1094     events.push_back(CreateScreenStateChangedEvent(
1095             bucketStartTimeNs + 10 * NS_PER_SEC,
1096             android::view::DisplayStateEnum::DISPLAY_STATE_ON));                       // 0:20
1097     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 20 * NS_PER_SEC));  // 0:30
1098     events.push_back(CreateScreenStateChangedEvent(
1099             bucketStartTimeNs + 70 * NS_PER_SEC,
1100             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));  // 1:20
1101     events.push_back(CreateScreenStateChangedEvent(
1102             bucketStartTimeNs + 100 * NS_PER_SEC,
1103             android::view::DisplayStateEnum::DISPLAY_STATE_DOZE));  // 1:50
1104     events.push_back(CreateScreenStateChangedEvent(
1105             bucketStartTimeNs + 120 * NS_PER_SEC,
1106             android::view::DisplayStateEnum::DISPLAY_STATE_ON));  // 2:10
1107     events.push_back(CreateScreenStateChangedEvent(
1108             bucketStartTimeNs + 170 * NS_PER_SEC,
1109             android::view::DisplayStateEnum::DISPLAY_STATE_VR));                         // 3:00
1110     events.push_back(CreateBatterySaverOffEvent(bucketStartTimeNs + 200 * NS_PER_SEC));  // 3:30
1111     events.push_back(CreateScreenStateChangedEvent(
1112             bucketStartTimeNs + 250 * NS_PER_SEC,
1113             android::view::DisplayStateEnum::DISPLAY_STATE_OFF));                       // 4:20
1114     events.push_back(CreateBatterySaverOnEvent(bucketStartTimeNs + 280 * NS_PER_SEC));  // 4:50
1115 
1116     // Bucket boundary 5:10.
1117     events.push_back(CreateScreenStateChangedEvent(
1118             bucketStartTimeNs + 320 * NS_PER_SEC,
1119             android::view::DisplayStateEnum::DISPLAY_STATE_ON));  // 5:30
1120     events.push_back(CreateScreenStateChangedEvent(
1121             bucketStartTimeNs + 390 * NS_PER_SEC,
1122             android::view::DisplayStateEnum::DISPLAY_STATE_ON_SUSPEND));  // 6:40
1123     events.push_back(CreateScreenStateChangedEvent(
1124             bucketStartTimeNs + 430 * NS_PER_SEC,
1125             android::view::DisplayStateEnum::DISPLAY_STATE_DOZE_SUSPEND));  // 7:20
1126     // Send log events to StatsLogProcessor.
1127     for (auto& event : events) {
1128         processor->OnLogEvent(event.get());
1129     }
1130 
1131     // Check dump report.
1132     vector<uint8_t> buffer;
1133     ConfigMetricsReportList reports;
1134     processor->onDumpReport(cfgKey, bucketStartTimeNs + 490 * NS_PER_SEC,
1135                             true /* include current partial bucket */, true, ADB_DUMP, FAST,
1136                             &buffer);
1137     ASSERT_GT(buffer.size(), 0);
1138     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
1139     backfillDimensionPath(&reports);
1140     backfillStringInReport(&reports);
1141     backfillStartEndTimestamp(&reports);
1142 
1143     ASSERT_EQ(1, reports.reports_size());
1144     ASSERT_EQ(1, reports.reports(0).metrics_size());
1145     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
1146     StatsLogReport::DurationMetricDataWrapper durationMetrics;
1147     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
1148                                     &durationMetrics);
1149     ASSERT_EQ(2, durationMetrics.data_size());
1150 
1151     DurationMetricData data = durationMetrics.data(0);
1152     ASSERT_EQ(1, data.slice_by_state_size());
1153     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1154     EXPECT_TRUE(data.slice_by_state(0).has_group_id());
1155     EXPECT_EQ(screenOnId, data.slice_by_state(0).group_id());
1156     ASSERT_EQ(2, data.bucket_info_size());
1157     EXPECT_EQ(130 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1158     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1159     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1160     EXPECT_EQ(110 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1161     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1162     EXPECT_EQ(500 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1163 
1164     data = durationMetrics.data(1);
1165     ASSERT_EQ(1, data.slice_by_state_size());
1166     EXPECT_EQ(SCREEN_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1167     EXPECT_TRUE(data.slice_by_state(0).has_group_id());
1168     EXPECT_EQ(screenOffId, data.slice_by_state(0).group_id());
1169     ASSERT_EQ(2, data.bucket_info_size());
1170     EXPECT_EQ(70 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1171     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1172     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1173     EXPECT_EQ(80 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1174     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1175     EXPECT_EQ(500 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1176 }
1177 
TEST(DurationMetricE2eTest,TestSlicedStatePrimaryFieldsNotSubsetDimInWhat)1178 TEST(DurationMetricE2eTest, TestSlicedStatePrimaryFieldsNotSubsetDimInWhat) {
1179     // Initialize config.
1180     StatsdConfig config;
1181     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
1182 
1183     *config.add_atom_matcher() = CreateAcquireWakelockAtomMatcher();
1184     *config.add_atom_matcher() = CreateReleaseWakelockAtomMatcher();
1185 
1186     auto holdingWakelockPredicate = CreateHoldingWakelockPredicate();
1187     *config.add_predicate() = holdingWakelockPredicate;
1188 
1189     auto uidProcessState = CreateUidProcessState();
1190     *config.add_state() = uidProcessState;
1191 
1192     // Create duration metric that slices by uid process state.
1193     auto durationMetric = config.add_duration_metric();
1194     durationMetric->set_id(StringToId("DurationHoldingWakelockSliceUidProcessState"));
1195     durationMetric->set_what(holdingWakelockPredicate.id());
1196     durationMetric->add_slice_by_state(uidProcessState.id());
1197     durationMetric->set_aggregation_type(DurationMetric::SUM);
1198     durationMetric->set_bucket(FIVE_MINUTES);
1199 
1200     // The state has only one primary field (uid).
1201     auto stateLink = durationMetric->add_state_link();
1202     stateLink->set_state_atom_id(UID_PROCESS_STATE_ATOM_ID);
1203     auto fieldsInWhat = stateLink->mutable_fields_in_what();
1204     *fieldsInWhat = CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
1205     auto fieldsInState = stateLink->mutable_fields_in_state();
1206     *fieldsInState = CreateDimensions(UID_PROCESS_STATE_ATOM_ID, {1 /* uid */});
1207 
1208     // Initialize StatsLogProcessor.
1209     int uid = 12345;
1210     int64_t cfgId = 98765;
1211     ConfigKey cfgKey(uid, cfgId);
1212     uint64_t bucketStartTimeNs = 10000000000;  // 0:10
1213     uint64_t bucketSizeNs =
1214             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
1215     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
1216 
1217     // This config is rejected because the dimension in what fields are not a superset of the sliced
1218     // state primary fields.
1219     ASSERT_EQ(processor->mMetricsManagers.size(), 0);
1220 }
1221 
TEST(DurationMetricE2eTest,TestWithSlicedStatePrimaryFieldsSubset)1222 TEST(DurationMetricE2eTest, TestWithSlicedStatePrimaryFieldsSubset) {
1223     // Initialize config.
1224     StatsdConfig config;
1225     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
1226 
1227     *config.add_atom_matcher() = CreateAcquireWakelockAtomMatcher();
1228     *config.add_atom_matcher() = CreateReleaseWakelockAtomMatcher();
1229 
1230     auto holdingWakelockPredicate = CreateHoldingWakelockPredicate();
1231     *(holdingWakelockPredicate.mutable_simple_predicate()->mutable_dimensions()) =
1232             CreateAttributionUidAndOtherDimensions(util::WAKELOCK_STATE_CHANGED, {Position::FIRST},
1233                                                    {3 /* tag */});
1234     *config.add_predicate() = holdingWakelockPredicate;
1235 
1236     auto uidProcessState = CreateUidProcessState();
1237     *config.add_state() = uidProcessState;
1238 
1239     // Create duration metric that slices by uid process state.
1240     auto durationMetric = config.add_duration_metric();
1241     durationMetric->set_id(StringToId("DurationPartialWakelockPerTagUidSliceProcessState"));
1242     durationMetric->set_what(holdingWakelockPredicate.id());
1243     durationMetric->add_slice_by_state(uidProcessState.id());
1244     durationMetric->set_aggregation_type(DurationMetric::SUM);
1245     durationMetric->set_bucket(FIVE_MINUTES);
1246 
1247     // The metric is dimensioning by first uid of attribution node and tag.
1248     *durationMetric->mutable_dimensions_in_what() = CreateAttributionUidAndOtherDimensions(
1249             util::WAKELOCK_STATE_CHANGED, {Position::FIRST}, {3 /* tag */});
1250     // The state has only one primary field (uid).
1251     auto stateLink = durationMetric->add_state_link();
1252     stateLink->set_state_atom_id(UID_PROCESS_STATE_ATOM_ID);
1253     auto fieldsInWhat = stateLink->mutable_fields_in_what();
1254     *fieldsInWhat = CreateAttributionUidDimensions(util::WAKELOCK_STATE_CHANGED, {Position::FIRST});
1255     auto fieldsInState = stateLink->mutable_fields_in_state();
1256     *fieldsInState = CreateDimensions(UID_PROCESS_STATE_ATOM_ID, {1 /* uid */});
1257 
1258     // Initialize StatsLogProcessor.
1259     int uid = 12345;
1260     int64_t cfgId = 98765;
1261     ConfigKey cfgKey(uid, cfgId);
1262     uint64_t bucketStartTimeNs = 10000000000;  // 0:10
1263     uint64_t bucketSizeNs =
1264             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000000LL;
1265     auto processor = CreateStatsLogProcessor(bucketStartTimeNs, bucketStartTimeNs, config, cfgKey);
1266 
1267     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
1268     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
1269     EXPECT_TRUE(metricsManager->isConfigValid());
1270     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
1271     EXPECT_TRUE(metricsManager->isActive());
1272     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
1273     EXPECT_TRUE(metricProducer->mIsActive);
1274     ASSERT_EQ(metricProducer->mSlicedStateAtoms.size(), 1);
1275     EXPECT_EQ(metricProducer->mSlicedStateAtoms.at(0), UID_PROCESS_STATE_ATOM_ID);
1276     ASSERT_EQ(metricProducer->mStateGroupMap.size(), 0);
1277 
1278     // Check that StateTrackers were initialized correctly.
1279     EXPECT_EQ(1, StateManager::getInstance().getStateTrackersCount());
1280     EXPECT_EQ(1, StateManager::getInstance().getListenersCount(UID_PROCESS_STATE_ATOM_ID));
1281 
1282     // Initialize log events.
1283     int appUid1 = 1001;
1284     int appUid2 = 1002;
1285     std::vector<int> attributionUids1 = {appUid1};
1286     std::vector<string> attributionTags1 = {"App1"};
1287 
1288     std::vector<int> attributionUids2 = {appUid2};
1289     std::vector<string> attributionTags2 = {"App2"};
1290 
1291     std::vector<std::unique_ptr<LogEvent>> events;
1292     events.push_back(CreateUidProcessStateChangedEvent(
1293             bucketStartTimeNs + 10 * NS_PER_SEC, appUid1,
1294             android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_FOREGROUND));  // 0:20
1295     events.push_back(CreateAcquireWakelockEvent(bucketStartTimeNs + 20 * NS_PER_SEC,
1296                                                 attributionUids1, attributionTags1,
1297                                                 "wakelock1"));  // 0:30
1298     events.push_back(CreateAcquireWakelockEvent(bucketStartTimeNs + 25 * NS_PER_SEC,
1299                                                 attributionUids1, attributionTags1,
1300                                                 "wakelock2"));  // 0:35
1301     events.push_back(CreateAcquireWakelockEvent(bucketStartTimeNs + 30 * NS_PER_SEC,
1302                                                 attributionUids2, attributionTags2,
1303                                                 "wakelock1"));  // 0:40
1304     events.push_back(CreateAcquireWakelockEvent(bucketStartTimeNs + 35 * NS_PER_SEC,
1305                                                 attributionUids2, attributionTags2,
1306                                                 "wakelock2"));  // 0:45
1307     events.push_back(CreateUidProcessStateChangedEvent(
1308             bucketStartTimeNs + 50 * NS_PER_SEC, appUid2,
1309             android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND));  // 1:00
1310     events.push_back(CreateUidProcessStateChangedEvent(
1311             bucketStartTimeNs + 60 * NS_PER_SEC, appUid1,
1312             android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND));  // 1:10
1313     events.push_back(CreateReleaseWakelockEvent(bucketStartTimeNs + 100 * NS_PER_SEC,
1314                                                 attributionUids2, attributionTags2,
1315                                                 "wakelock1"));  // 1:50
1316     events.push_back(CreateUidProcessStateChangedEvent(
1317             bucketStartTimeNs + 120 * NS_PER_SEC, appUid2,
1318             android::app::ProcessStateEnum::PROCESS_STATE_FOREGROUND_SERVICE));  // 2:10
1319     events.push_back(CreateReleaseWakelockEvent(bucketStartTimeNs + 200 * NS_PER_SEC,
1320                                                 attributionUids1, attributionTags1,
1321                                                 "wakelock2"));  // 3:30
1322 
1323     // Send log events to StatsLogProcessor.
1324     for (auto& event : events) {
1325         processor->OnLogEvent(event.get());
1326     }
1327 
1328     // Check dump report.
1329     vector<uint8_t> buffer;
1330     ConfigMetricsReportList reports;
1331     processor->onDumpReport(cfgKey, bucketStartTimeNs + 320 * NS_PER_SEC,
1332                             true /* include current partial bucket */, true, ADB_DUMP, FAST,
1333                             &buffer);
1334     ASSERT_GT(buffer.size(), 0);
1335     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
1336     backfillDimensionPath(&reports);
1337     backfillStringInReport(&reports);
1338     backfillStartEndTimestamp(&reports);
1339 
1340     ASSERT_EQ(1, reports.reports_size());
1341     ASSERT_EQ(1, reports.reports(0).metrics_size());
1342     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
1343     StatsLogReport::DurationMetricDataWrapper durationMetrics;
1344     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
1345                                     &durationMetrics);
1346     ASSERT_EQ(9, durationMetrics.data_size());
1347 
1348     DurationMetricData data = durationMetrics.data(0);
1349     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid1,
1350                                                   "wakelock1");
1351     ASSERT_EQ(1, data.slice_by_state_size());
1352     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1353     EXPECT_TRUE(data.slice_by_state(0).has_value());
1354     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_FOREGROUND,
1355               data.slice_by_state(0).value());
1356     ASSERT_EQ(1, data.bucket_info_size());
1357     EXPECT_EQ(40 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1358     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1359     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1360 
1361     data = durationMetrics.data(1);
1362     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid1,
1363                                                   "wakelock1");
1364     ASSERT_EQ(1, data.slice_by_state_size());
1365     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1366     EXPECT_TRUE(data.slice_by_state(0).has_value());
1367     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND,
1368               data.slice_by_state(0).value());
1369     ASSERT_EQ(2, data.bucket_info_size());
1370     EXPECT_EQ(240 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1371     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1372     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1373     EXPECT_EQ(20 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1374     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1375     EXPECT_EQ(330 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1376 
1377     data = durationMetrics.data(2);
1378     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid1,
1379                                                   "wakelock2");
1380     ASSERT_EQ(1, data.slice_by_state_size());
1381     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1382     EXPECT_TRUE(data.slice_by_state(0).has_value());
1383     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_FOREGROUND,
1384               data.slice_by_state(0).value());
1385     ASSERT_EQ(1, data.bucket_info_size());
1386     EXPECT_EQ(35 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1387     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1388     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1389 
1390     data = durationMetrics.data(3);
1391     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid1,
1392                                                   "wakelock2");
1393     ASSERT_EQ(1, data.slice_by_state_size());
1394     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1395     EXPECT_TRUE(data.slice_by_state(0).has_value());
1396     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND,
1397               data.slice_by_state(0).value());
1398     ASSERT_EQ(1, data.bucket_info_size());
1399     EXPECT_EQ(140 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1400     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1401     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1402 
1403     data = durationMetrics.data(4);
1404     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid2,
1405                                                   "wakelock1");
1406     ASSERT_EQ(1, data.slice_by_state_size());
1407     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1408     EXPECT_TRUE(data.slice_by_state(0).has_value());
1409     EXPECT_EQ(-1 /* StateTracker:: kStateUnknown */, data.slice_by_state(0).value());
1410     ASSERT_EQ(1, data.bucket_info_size());
1411     EXPECT_EQ(20 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1412     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1413     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1414 
1415     data = durationMetrics.data(5);
1416     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid2,
1417                                                   "wakelock1");
1418     ASSERT_EQ(1, data.slice_by_state_size());
1419     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1420     EXPECT_TRUE(data.slice_by_state(0).has_value());
1421     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND,
1422               data.slice_by_state(0).value());
1423     ASSERT_EQ(1, data.bucket_info_size());
1424     EXPECT_EQ(50 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1425     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1426     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1427 
1428     data = durationMetrics.data(6);
1429     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid2,
1430                                                   "wakelock2");
1431     ASSERT_EQ(1, data.slice_by_state_size());
1432     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1433     EXPECT_TRUE(data.slice_by_state(0).has_value());
1434     EXPECT_EQ(-1 /* StateTracker:: kStateUnknown */, data.slice_by_state(0).value());
1435     ASSERT_EQ(1, data.bucket_info_size());
1436     EXPECT_EQ(15 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1437     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1438     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1439 
1440     data = durationMetrics.data(7);
1441     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid2,
1442                                                   "wakelock2");
1443     ASSERT_EQ(1, data.slice_by_state_size());
1444     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1445     EXPECT_TRUE(data.slice_by_state(0).has_value());
1446     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_FOREGROUND_SERVICE,
1447               data.slice_by_state(0).value());
1448     ASSERT_EQ(2, data.bucket_info_size());
1449     EXPECT_EQ(180 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1450     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1451     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1452     EXPECT_EQ(20 * NS_PER_SEC, data.bucket_info(1).duration_nanos());
1453     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(1).start_bucket_elapsed_nanos());
1454     EXPECT_EQ(330 * NS_PER_SEC, data.bucket_info(1).end_bucket_elapsed_nanos());
1455 
1456     data = durationMetrics.data(8);
1457     ValidateWakelockAttributionUidAndTagDimension(data.dimensions_in_what(), 10, appUid2,
1458                                                   "wakelock2");
1459     ASSERT_EQ(1, data.slice_by_state_size());
1460     EXPECT_EQ(UID_PROCESS_STATE_ATOM_ID, data.slice_by_state(0).atom_id());
1461     EXPECT_TRUE(data.slice_by_state(0).has_value());
1462     EXPECT_EQ(android::app::ProcessStateEnum::PROCESS_STATE_IMPORTANT_BACKGROUND,
1463               data.slice_by_state(0).value());
1464     ASSERT_EQ(1, data.bucket_info_size());
1465     EXPECT_EQ(70 * NS_PER_SEC, data.bucket_info(0).duration_nanos());
1466     EXPECT_EQ(10 * NS_PER_SEC, data.bucket_info(0).start_bucket_elapsed_nanos());
1467     EXPECT_EQ(310 * NS_PER_SEC, data.bucket_info(0).end_bucket_elapsed_nanos());
1468 }
1469 
TEST(DurationMetricE2eTest,TestUploadThreshold)1470 TEST(DurationMetricE2eTest, TestUploadThreshold) {
1471     StatsdConfig config;
1472     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
1473 
1474     auto screenOnMatcher = CreateScreenTurnedOnAtomMatcher();
1475     auto screenOffMatcher = CreateScreenTurnedOffAtomMatcher();
1476     *config.add_atom_matcher() = screenOnMatcher;
1477     *config.add_atom_matcher() = screenOffMatcher;
1478 
1479     auto durationPredicate = CreateScreenIsOnPredicate();
1480     *config.add_predicate() = durationPredicate;
1481 
1482     int64_t thresholdDurationNs = 30 * 1000 * 1000 * 1000LL;  // 30 seconds
1483     UploadThreshold threshold;
1484     threshold.set_gt_int(thresholdDurationNs);
1485 
1486     int64_t metricId = 123456;
1487     auto durationMetric = config.add_duration_metric();
1488     durationMetric->set_id(metricId);
1489     durationMetric->set_what(durationPredicate.id());
1490     durationMetric->set_bucket(FIVE_MINUTES);
1491     durationMetric->set_aggregation_type(DurationMetric_AggregationType_SUM);
1492     *durationMetric->mutable_threshold() = threshold;
1493 
1494     const int64_t baseTimeNs = 0;                                   // 0:00
1495     const int64_t configAddedTimeNs = baseTimeNs + 1 * NS_PER_SEC;  // 0:01
1496     const int64_t bucketSizeNs =
1497             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000LL * 1000LL;
1498 
1499     int uid = 12345;
1500     int64_t cfgId = 98765;
1501     ConfigKey cfgKey(uid, cfgId);
1502 
1503     auto processor = CreateStatsLogProcessor(baseTimeNs, configAddedTimeNs, config, cfgKey);
1504 
1505     ASSERT_EQ(processor->mMetricsManagers.size(), 1u);
1506     sp<MetricsManager> metricsManager = processor->mMetricsManagers.begin()->second;
1507     EXPECT_TRUE(metricsManager->isConfigValid());
1508     ASSERT_EQ(metricsManager->mAllMetricProducers.size(), 1);
1509     sp<MetricProducer> metricProducer = metricsManager->mAllMetricProducers[0];
1510     EXPECT_TRUE(metricsManager->isActive());
1511     EXPECT_TRUE(metricProducer->mIsActive);
1512 
1513     std::unique_ptr<LogEvent> event;
1514 
1515     // Screen is off at start of first bucket.
1516     event = CreateScreenStateChangedEvent(configAddedTimeNs,
1517                                           android::view::DISPLAY_STATE_OFF);  // 0:01
1518     processor->OnLogEvent(event.get());
1519 
1520     // Turn screen on.
1521     const int64_t durationStartNs = configAddedTimeNs + 10 * NS_PER_SEC;  // 0:11
1522     event = CreateScreenStateChangedEvent(durationStartNs, android::view::DISPLAY_STATE_ON);
1523     processor->OnLogEvent(event.get());
1524 
1525     // Turn off screen 30 seconds after turning on.
1526     const int64_t durationEndNs = durationStartNs + 30 * NS_PER_SEC;  // 0:41
1527     event = CreateScreenStateChangedEvent(durationEndNs, android::view::DISPLAY_STATE_OFF);
1528     processor->OnLogEvent(event.get());
1529 
1530     // Turn screen on in second bucket.
1531     const int64_t duration2StartNs = configAddedTimeNs + bucketSizeNs + 10 * NS_PER_SEC;  // 5:11
1532     event = CreateScreenStateChangedEvent(duration2StartNs, android::view::DISPLAY_STATE_ON);
1533     processor->OnLogEvent(event.get());
1534 
1535     // Turn off screen 31 seconds after turning on.
1536     const int64_t duration2EndNs = duration2StartNs + 31 * NS_PER_SEC;  // 5:42
1537     event = CreateScreenStateChangedEvent(duration2EndNs, android::view::DISPLAY_STATE_OFF);
1538     processor->OnLogEvent(event.get());
1539 
1540     ConfigMetricsReportList reports;
1541     vector<uint8_t> buffer;
1542     processor->onDumpReport(cfgKey, configAddedTimeNs + bucketSizeNs * 2 + 1 * NS_PER_SEC, false,
1543                             true, ADB_DUMP, FAST, &buffer);  // 10:01
1544     EXPECT_TRUE(buffer.size() > 0);
1545     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
1546     backfillDimensionPath(&reports);
1547     backfillStartEndTimestamp(&reports);
1548     ASSERT_EQ(1, reports.reports_size());
1549     ASSERT_EQ(1, reports.reports(0).metrics_size());
1550     EXPECT_EQ(metricId, reports.reports(0).metrics(0).metric_id());
1551     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
1552 
1553     StatsLogReport::DurationMetricDataWrapper durationMetrics;
1554     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
1555                                     &durationMetrics);
1556     ASSERT_EQ(1, durationMetrics.data_size());
1557 
1558     DurationMetricData data = durationMetrics.data(0);
1559     ASSERT_EQ(1, data.bucket_info_size());
1560     EXPECT_EQ(duration2EndNs - duration2StartNs, data.bucket_info(0).duration_nanos());
1561     EXPECT_EQ(baseTimeNs + bucketSizeNs, data.bucket_info(0).start_bucket_elapsed_nanos());
1562     EXPECT_EQ(baseTimeNs + bucketSizeNs * 2, data.bucket_info(0).end_bucket_elapsed_nanos());
1563 }
1564 
TEST(DurationMetricE2eTest,TestConditionOnRepeatedEnumField)1565 TEST(DurationMetricE2eTest, TestConditionOnRepeatedEnumField) {
1566     StatsdConfig config;
1567     config.add_allowed_log_source("AID_ROOT");  // LogEvent defaults to UID of root.
1568 
1569     AtomMatcher repeatedStateFirstOffAtomMatcher = CreateTestAtomRepeatedStateFirstOffAtomMatcher();
1570     AtomMatcher repeatedStateFirstOnAtomMatcher = CreateTestAtomRepeatedStateFirstOnAtomMatcher();
1571     *config.add_atom_matcher() = repeatedStateFirstOffAtomMatcher;
1572     *config.add_atom_matcher() = repeatedStateFirstOnAtomMatcher;
1573 
1574     Predicate durationPredicate = CreateTestAtomRepeatedStateFirstOffPredicate();
1575     *config.add_predicate() = durationPredicate;
1576 
1577     int64_t metricId = 123456;
1578     DurationMetric* durationMetric = config.add_duration_metric();
1579     durationMetric->set_id(metricId);
1580     durationMetric->set_what(durationPredicate.id());
1581     durationMetric->set_bucket(FIVE_MINUTES);
1582     durationMetric->set_aggregation_type(DurationMetric_AggregationType_SUM);
1583 
1584     const int64_t baseTimeNs = 0;                                   // 0:00
1585     const int64_t configAddedTimeNs = baseTimeNs + 1 * NS_PER_SEC;  // 0:01
1586     const int64_t bucketSizeNs =
1587             TimeUnitToBucketSizeInMillis(config.duration_metric(0).bucket()) * 1000LL * 1000LL;
1588 
1589     int uid = 12345;
1590     int64_t cfgId = 98765;
1591     ConfigKey cfgKey(uid, cfgId);
1592 
1593     sp<StatsLogProcessor> processor =
1594             CreateStatsLogProcessor(baseTimeNs, configAddedTimeNs, config, cfgKey);
1595 
1596     vector<int> intArray = {3, 6};
1597     vector<int64_t> longArray = {1000L, 10002L};
1598     vector<float> floatArray = {0.3f, 0.09f};
1599     vector<string> stringArray = {"str1", "str2"};
1600     int boolArrayLength = 2;
1601     bool boolArray[boolArrayLength];
1602     boolArray[0] = 1;
1603     boolArray[1] = 0;
1604     vector<int> enumArrayOff = {TestAtomReported::OFF, TestAtomReported::ON};
1605     vector<int> enumArrayOn = {TestAtomReported::ON, TestAtomReported::OFF};
1606 
1607     std::vector<std::unique_ptr<LogEvent>> events;
1608     uint64_t falseDurationStartNs = configAddedTimeNs + 10 * NS_PER_SEC;
1609     uint64_t durationStartNs = configAddedTimeNs + 20 * NS_PER_SEC;
1610     uint64_t durationEndNs = durationStartNs + 50 * NS_PER_SEC;
1611 
1612     // Condition false
1613     events.push_back(CreateTestAtomReportedEventVariableRepeatedFields(falseDurationStartNs, {}, {},
1614                                                                        {}, {}, {}, 0, enumArrayOn));
1615     // Condition true - start collecting duration.
1616     events.push_back(CreateTestAtomReportedEventVariableRepeatedFields(durationStartNs, {}, {}, {},
1617                                                                        {}, {}, 0, enumArrayOff));
1618     // Condition false - stop collecting duration.
1619     events.push_back(CreateTestAtomReportedEventVariableRepeatedFields(durationEndNs, {}, {}, {},
1620                                                                        {}, {}, 0, enumArrayOn));
1621 
1622     // Send log events to StatsLogProcessor.
1623     for (auto& event : events) {
1624         processor->OnLogEvent(event.get());
1625     }
1626 
1627     ConfigMetricsReportList reports;
1628     vector<uint8_t> buffer;
1629     processor->onDumpReport(cfgKey, configAddedTimeNs + bucketSizeNs + 1 * NS_PER_SEC, false, true,
1630                             ADB_DUMP, FAST, &buffer);  // 10:01
1631     EXPECT_TRUE(buffer.size() > 0);
1632     EXPECT_TRUE(reports.ParseFromArray(&buffer[0], buffer.size()));
1633     backfillDimensionPath(&reports);
1634     backfillStartEndTimestamp(&reports);
1635     ASSERT_EQ(1, reports.reports_size());
1636     ASSERT_EQ(1, reports.reports(0).metrics_size());
1637     EXPECT_EQ(metricId, reports.reports(0).metrics(0).metric_id());
1638     EXPECT_TRUE(reports.reports(0).metrics(0).has_duration_metrics());
1639 
1640     StatsLogReport::DurationMetricDataWrapper durationMetrics;
1641     sortMetricDataByDimensionsValue(reports.reports(0).metrics(0).duration_metrics(),
1642                                     &durationMetrics);
1643     ASSERT_EQ(1, durationMetrics.data_size());
1644 
1645     DurationMetricData data = durationMetrics.data(0);
1646     ASSERT_EQ(1, data.bucket_info_size());
1647     EXPECT_EQ(durationEndNs - durationStartNs, data.bucket_info(0).duration_nanos());
1648     EXPECT_EQ(configAddedTimeNs, data.bucket_info(0).start_bucket_elapsed_nanos());
1649     EXPECT_EQ(baseTimeNs + bucketSizeNs, data.bucket_info(0).end_bucket_elapsed_nanos());
1650 }
1651 
1652 #else
1653 GTEST_LOG_(INFO) << "This test does nothing.\n";
1654 #endif
1655 
1656 }  // namespace statsd
1657 }  // namespace os
1658 }  // namespace android
1659