blob: a07683e5b0455fa7c39371ee9f911586710a4acd [file] [log] [blame]
Yao Chen93fe3a32017-11-02 13:52:59 -07001// Copyright (C) 2017 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
Yao Chen93fe3a32017-11-02 13:52:59 -070015#include "src/metrics/CountMetricProducer.h"
Yangster-macb8144812018-01-04 10:56:23 -080016#include "src/stats_log_util.h"
Yao Chend5aa01b32017-12-19 16:46:36 -080017#include "metrics_test_helper.h"
Yangster-mac94e197c2018-01-02 16:03:03 -080018#include "tests/statsd_test_util.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070019
20#include <gmock/gmock.h>
21#include <gtest/gtest.h>
22#include <stdio.h>
23#include <vector>
24
25using namespace testing;
26using android::sp;
27using std::set;
28using std::unordered_map;
29using std::vector;
30
31#ifdef __ANDROID__
32
33namespace android {
34namespace os {
35namespace statsd {
36
Yangster-mac94e197c2018-01-02 16:03:03 -080037const ConfigKey kConfigKey(0, 12345);
Yao Chenb3561512017-11-21 18:07:17 -080038
Yao Chen93fe3a32017-11-02 13:52:59 -070039TEST(CountMetricProducerTest, TestNonDimensionalEvents) {
40 int64_t bucketStartTimeNs = 10000000000;
Yangster-macb8144812018-01-04 10:56:23 -080041 int64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
Yao Chen93fe3a32017-11-02 13:52:59 -070042 int64_t bucket2StartTimeNs = bucketStartTimeNs + bucketSizeNs;
43 int64_t bucket3StartTimeNs = bucketStartTimeNs + 2 * bucketSizeNs;
44 int tagId = 1;
45
46 CountMetric metric;
Yangster-mac94e197c2018-01-02 16:03:03 -080047 metric.set_id(1);
Yangster-macb8144812018-01-04 10:56:23 -080048 metric.set_bucket(ONE_MINUTE);
Yao Chen93fe3a32017-11-02 13:52:59 -070049
50 LogEvent event1(tagId, bucketStartTimeNs + 1);
Yangster-mac330af582018-02-08 15:24:38 -080051 event1.init();
Yao Chen93fe3a32017-11-02 13:52:59 -070052 LogEvent event2(tagId, bucketStartTimeNs + 2);
Yangster-mac330af582018-02-08 15:24:38 -080053 event2.init();
Yao Chen93fe3a32017-11-02 13:52:59 -070054
55 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
56
Yao Chenb3561512017-11-21 18:07:17 -080057 CountMetricProducer countProducer(kConfigKey, metric, -1 /*-1 meaning no condition*/, wizard,
Yao Chen93fe3a32017-11-02 13:52:59 -070058 bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -080059 countProducer.setBucketSize(60 * NS_PER_SEC);
Yao Chen93fe3a32017-11-02 13:52:59 -070060
61 // 2 events in bucket 1.
Chenjie Yua7259ab2017-12-10 08:31:05 -080062 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event1);
63 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event2);
Yangster-mace2cd6d52017-11-09 20:38:30 -080064
65 // Flushes at event #2.
Yangsterf2bee6f2017-11-29 12:01:05 -080066 countProducer.flushIfNeededLocked(bucketStartTimeNs + 2);
Yangster-mace2cd6d52017-11-09 20:38:30 -080067 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
68
69 // Flushes.
Yangsterf2bee6f2017-11-29 12:01:05 -080070 countProducer.flushIfNeededLocked(bucketStartTimeNs + bucketSizeNs + 1);
Yao Chen93fe3a32017-11-02 13:52:59 -070071 EXPECT_EQ(1UL, countProducer.mPastBuckets.size());
Yangster-mac93694462018-01-22 20:49:31 -080072 EXPECT_TRUE(countProducer.mPastBuckets.find(DEFAULT_METRIC_DIMENSION_KEY) !=
Yao Chen93fe3a32017-11-02 13:52:59 -070073 countProducer.mPastBuckets.end());
Yangster-mac93694462018-01-22 20:49:31 -080074 const auto& buckets = countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY];
Yao Chen93fe3a32017-11-02 13:52:59 -070075 EXPECT_EQ(1UL, buckets.size());
Yangster-mace2cd6d52017-11-09 20:38:30 -080076 EXPECT_EQ(bucketStartTimeNs, buckets[0].mBucketStartNs);
77 EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, buckets[0].mBucketEndNs);
78 EXPECT_EQ(2LL, buckets[0].mCount);
Yao Chen93fe3a32017-11-02 13:52:59 -070079
80 // 1 matched event happens in bucket 2.
81 LogEvent event3(tagId, bucketStartTimeNs + bucketSizeNs + 2);
Yangster-mac330af582018-02-08 15:24:38 -080082 event3.init();
83
Chenjie Yua7259ab2017-12-10 08:31:05 -080084 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event3);
Yangsterf2bee6f2017-11-29 12:01:05 -080085 countProducer.flushIfNeededLocked(bucketStartTimeNs + 2 * bucketSizeNs + 1);
Yao Chen93fe3a32017-11-02 13:52:59 -070086 EXPECT_EQ(1UL, countProducer.mPastBuckets.size());
Yangster-mac93694462018-01-22 20:49:31 -080087 EXPECT_TRUE(countProducer.mPastBuckets.find(DEFAULT_METRIC_DIMENSION_KEY) !=
Yao Chen93fe3a32017-11-02 13:52:59 -070088 countProducer.mPastBuckets.end());
Yangster-mac93694462018-01-22 20:49:31 -080089 EXPECT_EQ(2UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
90 const auto& bucketInfo2 = countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][1];
Yao Chen93fe3a32017-11-02 13:52:59 -070091 EXPECT_EQ(bucket2StartTimeNs, bucketInfo2.mBucketStartNs);
92 EXPECT_EQ(bucket2StartTimeNs + bucketSizeNs, bucketInfo2.mBucketEndNs);
93 EXPECT_EQ(1LL, bucketInfo2.mCount);
94
95 // nothing happens in bucket 3. we should not record anything for bucket 3.
Yangsterf2bee6f2017-11-29 12:01:05 -080096 countProducer.flushIfNeededLocked(bucketStartTimeNs + 3 * bucketSizeNs + 1);
Yao Chen93fe3a32017-11-02 13:52:59 -070097 EXPECT_EQ(1UL, countProducer.mPastBuckets.size());
Yangster-mac93694462018-01-22 20:49:31 -080098 EXPECT_TRUE(countProducer.mPastBuckets.find(DEFAULT_METRIC_DIMENSION_KEY) !=
Yao Chen93fe3a32017-11-02 13:52:59 -070099 countProducer.mPastBuckets.end());
Yangster-mac93694462018-01-22 20:49:31 -0800100 const auto& buckets3 = countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY];
Yao Chen93fe3a32017-11-02 13:52:59 -0700101 EXPECT_EQ(2UL, buckets3.size());
102}
103
104TEST(CountMetricProducerTest, TestEventsWithNonSlicedCondition) {
105 int64_t bucketStartTimeNs = 10000000000;
Yangster-macb8144812018-01-04 10:56:23 -0800106 int64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
Yao Chen93fe3a32017-11-02 13:52:59 -0700107
108 CountMetric metric;
Yangster-mac94e197c2018-01-02 16:03:03 -0800109 metric.set_id(1);
Yangster-macb8144812018-01-04 10:56:23 -0800110 metric.set_bucket(ONE_MINUTE);
Yangster-mac94e197c2018-01-02 16:03:03 -0800111 metric.set_condition(StringToId("SCREEN_ON"));
Yao Chen93fe3a32017-11-02 13:52:59 -0700112
113 LogEvent event1(1, bucketStartTimeNs + 1);
Yangster-mac330af582018-02-08 15:24:38 -0800114 event1.init();
115
Yao Chen93fe3a32017-11-02 13:52:59 -0700116 LogEvent event2(1, bucketStartTimeNs + 10);
Yangster-mac330af582018-02-08 15:24:38 -0800117 event2.init();
Yao Chen93fe3a32017-11-02 13:52:59 -0700118
119 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
120
Yao Chenb3561512017-11-21 18:07:17 -0800121 CountMetricProducer countProducer(kConfigKey, metric, 1, wizard, bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -0800122 countProducer.setBucketSize(60 * NS_PER_SEC);
Yao Chen93fe3a32017-11-02 13:52:59 -0700123
124 countProducer.onConditionChanged(true, bucketStartTimeNs);
Chenjie Yua7259ab2017-12-10 08:31:05 -0800125 countProducer.onMatchedLogEvent(1 /*matcher index*/, event1);
Yao Chen93fe3a32017-11-02 13:52:59 -0700126 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
127
128 countProducer.onConditionChanged(false /*new condition*/, bucketStartTimeNs + 2);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800129 // Upon this match event, the matched event1 is flushed.
Chenjie Yua7259ab2017-12-10 08:31:05 -0800130 countProducer.onMatchedLogEvent(1 /*matcher index*/, event2);
Yao Chen93fe3a32017-11-02 13:52:59 -0700131 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
132
Yangsterf2bee6f2017-11-29 12:01:05 -0800133 countProducer.flushIfNeededLocked(bucketStartTimeNs + bucketSizeNs + 1);
Yao Chen93fe3a32017-11-02 13:52:59 -0700134 EXPECT_EQ(1UL, countProducer.mPastBuckets.size());
Yangster-mac93694462018-01-22 20:49:31 -0800135 EXPECT_TRUE(countProducer.mPastBuckets.find(DEFAULT_METRIC_DIMENSION_KEY) !=
Yao Chen93fe3a32017-11-02 13:52:59 -0700136 countProducer.mPastBuckets.end());
Yangster-mace2cd6d52017-11-09 20:38:30 -0800137 {
Yangster-mac93694462018-01-22 20:49:31 -0800138 const auto& buckets = countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY];
Yangster-mace2cd6d52017-11-09 20:38:30 -0800139 EXPECT_EQ(1UL, buckets.size());
140 const auto& bucketInfo = buckets[0];
141 EXPECT_EQ(bucketStartTimeNs, bucketInfo.mBucketStartNs);
142 EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, bucketInfo.mBucketEndNs);
143 EXPECT_EQ(1LL, bucketInfo.mCount);
144 }
Yao Chen93fe3a32017-11-02 13:52:59 -0700145}
146
147TEST(CountMetricProducerTest, TestEventsWithSlicedCondition) {
148 int64_t bucketStartTimeNs = 10000000000;
Yangster-macb8144812018-01-04 10:56:23 -0800149 int64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
Yao Chen93fe3a32017-11-02 13:52:59 -0700150
Yangster-mac20877162017-12-22 17:19:39 -0800151 int tagId = 1;
152 int conditionTagId = 2;
153
Yao Chen93fe3a32017-11-02 13:52:59 -0700154 CountMetric metric;
Yangster-mac94e197c2018-01-02 16:03:03 -0800155 metric.set_id(1);
Yangster-macb8144812018-01-04 10:56:23 -0800156 metric.set_bucket(ONE_MINUTE);
Yangster-mac94e197c2018-01-02 16:03:03 -0800157 metric.set_condition(StringToId("APP_IN_BACKGROUND_PER_UID_AND_SCREEN_ON"));
Stefan Lafona5b51912017-12-05 21:43:52 -0800158 MetricConditionLink* link = metric.add_links();
Yangster-mac94e197c2018-01-02 16:03:03 -0800159 link->set_condition(StringToId("APP_IN_BACKGROUND_PER_UID"));
Yangster-mac7ba8fc32018-01-24 16:16:46 -0800160 buildSimpleAtomFieldMatcher(tagId, 1, link->mutable_fields_in_what());
161 buildSimpleAtomFieldMatcher(conditionTagId, 2, link->mutable_fields_in_condition());
Yao Chen93fe3a32017-11-02 13:52:59 -0700162
Yangster-mac20877162017-12-22 17:19:39 -0800163 LogEvent event1(tagId, bucketStartTimeNs + 1);
Yao Chen80235402017-11-13 20:42:25 -0800164 event1.write("111"); // uid
Yao Chen93fe3a32017-11-02 13:52:59 -0700165 event1.init();
166 ConditionKey key1;
Yangster-mac94e197c2018-01-02 16:03:03 -0800167 key1[StringToId("APP_IN_BACKGROUND_PER_UID")] =
168 {getMockedDimensionKey(conditionTagId, 2, "111")};
Yao Chen93fe3a32017-11-02 13:52:59 -0700169
Yangster-mac20877162017-12-22 17:19:39 -0800170 LogEvent event2(tagId, bucketStartTimeNs + 10);
Yao Chen80235402017-11-13 20:42:25 -0800171 event2.write("222"); // uid
Yao Chen93fe3a32017-11-02 13:52:59 -0700172 event2.init();
173 ConditionKey key2;
Yangster-mac94e197c2018-01-02 16:03:03 -0800174 key2[StringToId("APP_IN_BACKGROUND_PER_UID")] =
175 {getMockedDimensionKey(conditionTagId, 2, "222")};
Yao Chen93fe3a32017-11-02 13:52:59 -0700176
177 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
Yangster13fb7e42018-03-07 17:30:49 -0800178 EXPECT_CALL(*wizard, query(_, key1, _, _, _, _)).WillOnce(Return(ConditionState::kFalse));
Yao Chen93fe3a32017-11-02 13:52:59 -0700179
Yangster13fb7e42018-03-07 17:30:49 -0800180 EXPECT_CALL(*wizard, query(_, key2, _, _, _, _)).WillOnce(Return(ConditionState::kTrue));
Yao Chen93fe3a32017-11-02 13:52:59 -0700181
Yao Chenb3561512017-11-21 18:07:17 -0800182 CountMetricProducer countProducer(kConfigKey, metric, 1 /*condition tracker index*/, wizard,
Yao Chen93fe3a32017-11-02 13:52:59 -0700183 bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -0800184 countProducer.setBucketSize(60 * NS_PER_SEC);
Yao Chen93fe3a32017-11-02 13:52:59 -0700185
Chenjie Yua7259ab2017-12-10 08:31:05 -0800186 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event1);
Yangsterf2bee6f2017-11-29 12:01:05 -0800187 countProducer.flushIfNeededLocked(bucketStartTimeNs + 1);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800188 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
189
Chenjie Yua7259ab2017-12-10 08:31:05 -0800190 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event2);
Yangsterf2bee6f2017-11-29 12:01:05 -0800191 countProducer.flushIfNeededLocked(bucketStartTimeNs + bucketSizeNs + 1);
Yao Chen93fe3a32017-11-02 13:52:59 -0700192 EXPECT_EQ(1UL, countProducer.mPastBuckets.size());
Yangster-mac93694462018-01-22 20:49:31 -0800193 EXPECT_TRUE(countProducer.mPastBuckets.find(DEFAULT_METRIC_DIMENSION_KEY) !=
Yao Chen93fe3a32017-11-02 13:52:59 -0700194 countProducer.mPastBuckets.end());
Yangster-mac93694462018-01-22 20:49:31 -0800195 const auto& buckets = countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY];
Yao Chen93fe3a32017-11-02 13:52:59 -0700196 EXPECT_EQ(1UL, buckets.size());
197 const auto& bucketInfo = buckets[0];
198 EXPECT_EQ(bucketStartTimeNs, bucketInfo.mBucketStartNs);
199 EXPECT_EQ(bucketStartTimeNs + bucketSizeNs, bucketInfo.mBucketEndNs);
200 EXPECT_EQ(1LL, bucketInfo.mCount);
201}
202
David Chen27785a82018-01-19 17:06:45 -0800203TEST(CountMetricProducerTest, TestEventWithAppUpgrade) {
Yangster-mac932ecec2018-02-01 10:23:52 -0800204 sp<AlarmMonitor> alarmMonitor;
David Chen27785a82018-01-19 17:06:45 -0800205 uint64_t bucketStartTimeNs = 10000000000;
206 uint64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
207 uint64_t eventUpgradeTimeNs = bucketStartTimeNs + 15 * NS_PER_SEC;
208
209 int tagId = 1;
210 int conditionTagId = 2;
211
212 CountMetric metric;
213 metric.set_id(1);
214 metric.set_bucket(ONE_MINUTE);
215 Alert alert;
216 alert.set_num_buckets(3);
217 alert.set_trigger_if_sum_gt(2);
218 LogEvent event1(tagId, bucketStartTimeNs + 1);
219 event1.write("111"); // uid
220 event1.init();
221 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
222 CountMetricProducer countProducer(kConfigKey, metric, -1 /* no condition */, wizard,
223 bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -0800224 countProducer.setBucketSize(60 * NS_PER_SEC);
225
Yangster-mac932ecec2018-02-01 10:23:52 -0800226 sp<AnomalyTracker> anomalyTracker = countProducer.addAnomalyTracker(alert, alarmMonitor);
David Chen27785a82018-01-19 17:06:45 -0800227 EXPECT_TRUE(anomalyTracker != nullptr);
228
229 // Bucket is flushed yet.
230 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event1);
231 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
232 EXPECT_EQ(0, anomalyTracker->getSumOverPastBuckets(DEFAULT_METRIC_DIMENSION_KEY));
233
234 // App upgrade forces bucket flush.
235 // Check that there's a past bucket and the bucket end is not adjusted.
236 countProducer.notifyAppUpgrade(eventUpgradeTimeNs, "ANY.APP", 1, 1);
237 EXPECT_EQ(1UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
238 EXPECT_EQ((long long)bucketStartTimeNs,
239 countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][0].mBucketStartNs);
240 EXPECT_EQ((long long)eventUpgradeTimeNs,
241 countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][0].mBucketEndNs);
242 EXPECT_EQ(eventUpgradeTimeNs, countProducer.mCurrentBucketStartTimeNs);
243 // Anomaly tracker only contains full buckets.
244 EXPECT_EQ(0, anomalyTracker->getSumOverPastBuckets(DEFAULT_METRIC_DIMENSION_KEY));
245
246 uint64_t lastEndTimeNs = countProducer.getCurrentBucketEndTimeNs();
247 // Next event occurs in same bucket as partial bucket created.
248 LogEvent event2(tagId, bucketStartTimeNs + 59 * NS_PER_SEC + 10);
249 event2.write("222"); // uid
250 event2.init();
251 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event2);
252 EXPECT_EQ(1UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
253 EXPECT_EQ(eventUpgradeTimeNs, countProducer.mCurrentBucketStartTimeNs);
254 EXPECT_EQ(0, anomalyTracker->getSumOverPastBuckets(DEFAULT_METRIC_DIMENSION_KEY));
255
256 // Third event in following bucket.
257 LogEvent event3(tagId, bucketStartTimeNs + 62 * NS_PER_SEC + 10);
258 event3.write("333"); // uid
259 event3.init();
260 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event3);
261 EXPECT_EQ(2UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
262 EXPECT_EQ(lastEndTimeNs, countProducer.mCurrentBucketStartTimeNs);
263 EXPECT_EQ(2, anomalyTracker->getSumOverPastBuckets(DEFAULT_METRIC_DIMENSION_KEY));
264}
265
266TEST(CountMetricProducerTest, TestEventWithAppUpgradeInNextBucket) {
267 uint64_t bucketStartTimeNs = 10000000000;
268 uint64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
269 uint64_t eventUpgradeTimeNs = bucketStartTimeNs + 65 * NS_PER_SEC;
270
271 int tagId = 1;
272 int conditionTagId = 2;
273
274 CountMetric metric;
275 metric.set_id(1);
276 metric.set_bucket(ONE_MINUTE);
277 LogEvent event1(tagId, bucketStartTimeNs + 1);
278 event1.write("111"); // uid
279 event1.init();
280 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
281 CountMetricProducer countProducer(kConfigKey, metric, -1 /* no condition */, wizard,
282 bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -0800283 countProducer.setBucketSize(60 * NS_PER_SEC);
David Chen27785a82018-01-19 17:06:45 -0800284
285 // Bucket is flushed yet.
286 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event1);
287 EXPECT_EQ(0UL, countProducer.mPastBuckets.size());
288
289 // App upgrade forces bucket flush.
290 // Check that there's a past bucket and the bucket end is not adjusted.
291 countProducer.notifyAppUpgrade(eventUpgradeTimeNs, "ANY.APP", 1, 1);
292 EXPECT_EQ(1UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
293 EXPECT_EQ((int64_t)bucketStartTimeNs,
294 countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][0].mBucketStartNs);
295 EXPECT_EQ(bucketStartTimeNs + bucketSizeNs,
296 (uint64_t)countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][0].mBucketEndNs);
297 EXPECT_EQ(eventUpgradeTimeNs, countProducer.mCurrentBucketStartTimeNs);
298
299 // Next event occurs in same bucket as partial bucket created.
300 LogEvent event2(tagId, bucketStartTimeNs + 70 * NS_PER_SEC + 10);
301 event2.write("222"); // uid
302 event2.init();
303 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event2);
304 EXPECT_EQ(1UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
305
306 // Third event in following bucket.
307 LogEvent event3(tagId, bucketStartTimeNs + 121 * NS_PER_SEC + 10);
308 event3.write("333"); // uid
309 event3.init();
310 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event3);
311 EXPECT_EQ(2UL, countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY].size());
312 EXPECT_EQ((int64_t)eventUpgradeTimeNs,
313 countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][1].mBucketStartNs);
314 EXPECT_EQ(bucketStartTimeNs + 2 * bucketSizeNs,
315 (uint64_t)countProducer.mPastBuckets[DEFAULT_METRIC_DIMENSION_KEY][1].mBucketEndNs);
316}
317
Bookatz1bf94382018-01-04 11:43:20 -0800318TEST(CountMetricProducerTest, TestAnomalyDetectionUnSliced) {
Yangster-mac932ecec2018-02-01 10:23:52 -0800319 sp<AlarmMonitor> alarmMonitor;
Yangster-mace2cd6d52017-11-09 20:38:30 -0800320 Alert alert;
Yangster-mac94e197c2018-01-02 16:03:03 -0800321 alert.set_id(11);
322 alert.set_metric_id(1);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800323 alert.set_trigger_if_sum_gt(2);
Yangster-maca7fb12d2018-01-03 17:17:20 -0800324 alert.set_num_buckets(2);
Bookatz1bf94382018-01-04 11:43:20 -0800325 const int32_t refPeriodSec = 1;
326 alert.set_refractory_period_secs(refPeriodSec);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800327
328 int64_t bucketStartTimeNs = 10000000000;
Yangster-macb8144812018-01-04 10:56:23 -0800329 int64_t bucketSizeNs = TimeUnitToBucketSizeInMillis(ONE_MINUTE) * 1000000LL;
Yangster-mace2cd6d52017-11-09 20:38:30 -0800330 int64_t bucket2StartTimeNs = bucketStartTimeNs + bucketSizeNs;
331 int64_t bucket3StartTimeNs = bucketStartTimeNs + 2 * bucketSizeNs;
332
Yangster-mace2cd6d52017-11-09 20:38:30 -0800333 CountMetric metric;
Yangster-mac94e197c2018-01-02 16:03:03 -0800334 metric.set_id(1);
Yangster-macb8144812018-01-04 10:56:23 -0800335 metric.set_bucket(ONE_MINUTE);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800336
337 sp<MockConditionWizard> wizard = new NaggyMock<MockConditionWizard>();
Yao Chenb3561512017-11-21 18:07:17 -0800338 CountMetricProducer countProducer(kConfigKey, metric, -1 /*-1 meaning no condition*/, wizard,
Yangster-mace2cd6d52017-11-09 20:38:30 -0800339 bucketStartTimeNs);
yro59cc24d2018-02-13 20:17:32 -0800340 countProducer.setBucketSize(60 * NS_PER_SEC);
341
Yangster-mac932ecec2018-02-01 10:23:52 -0800342 sp<AnomalyTracker> anomalyTracker = countProducer.addAnomalyTracker(alert, alarmMonitor);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800343
344 int tagId = 1;
345 LogEvent event1(tagId, bucketStartTimeNs + 1);
Yangster-mac330af582018-02-08 15:24:38 -0800346 event1.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800347 LogEvent event2(tagId, bucketStartTimeNs + 2);
Yangster-mac330af582018-02-08 15:24:38 -0800348 event2.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800349 LogEvent event3(tagId, bucketStartTimeNs + 2 * bucketSizeNs + 1);
Yangster-mac330af582018-02-08 15:24:38 -0800350 event3.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800351 LogEvent event4(tagId, bucketStartTimeNs + 3 * bucketSizeNs + 1);
Yangster-mac330af582018-02-08 15:24:38 -0800352 event4.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800353 LogEvent event5(tagId, bucketStartTimeNs + 3 * bucketSizeNs + 2);
Yangster-mac330af582018-02-08 15:24:38 -0800354 event5.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800355 LogEvent event6(tagId, bucketStartTimeNs + 3 * bucketSizeNs + 3);
Yangster-mac330af582018-02-08 15:24:38 -0800356 event6.init();
Bookatz1bf94382018-01-04 11:43:20 -0800357 LogEvent event7(tagId, bucketStartTimeNs + 3 * bucketSizeNs + 2 * NS_PER_SEC);
Yangster-mac330af582018-02-08 15:24:38 -0800358 event7.init();
Yangster-mace2cd6d52017-11-09 20:38:30 -0800359
360 // Two events in bucket #0.
Chenjie Yua7259ab2017-12-10 08:31:05 -0800361 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event1);
362 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event2);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800363
364 EXPECT_EQ(1UL, countProducer.mCurrentSlicedCounter->size());
365 EXPECT_EQ(2L, countProducer.mCurrentSlicedCounter->begin()->second);
Yangster-mac93694462018-01-22 20:49:31 -0800366 EXPECT_EQ(anomalyTracker->getRefractoryPeriodEndsSec(DEFAULT_METRIC_DIMENSION_KEY), 0U);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800367
368 // One event in bucket #2. No alarm as bucket #0 is trashed out.
Chenjie Yua7259ab2017-12-10 08:31:05 -0800369 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event3);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800370 EXPECT_EQ(1UL, countProducer.mCurrentSlicedCounter->size());
371 EXPECT_EQ(1L, countProducer.mCurrentSlicedCounter->begin()->second);
Yangster-mac93694462018-01-22 20:49:31 -0800372 EXPECT_EQ(anomalyTracker->getRefractoryPeriodEndsSec(DEFAULT_METRIC_DIMENSION_KEY), 0U);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800373
374 // Two events in bucket #3.
Chenjie Yua7259ab2017-12-10 08:31:05 -0800375 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event4);
376 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event5);
377 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event6);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800378 EXPECT_EQ(1UL, countProducer.mCurrentSlicedCounter->size());
379 EXPECT_EQ(3L, countProducer.mCurrentSlicedCounter->begin()->second);
380 // Anomaly at event 6 is within refractory period. The alarm is at event 5 timestamp not event 6
Yangster-mac93694462018-01-22 20:49:31 -0800381 EXPECT_EQ(anomalyTracker->getRefractoryPeriodEndsSec(DEFAULT_METRIC_DIMENSION_KEY),
Yangster-mac330af582018-02-08 15:24:38 -0800382 event5.GetElapsedTimestampNs() / NS_PER_SEC + refPeriodSec);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800383
Chenjie Yua7259ab2017-12-10 08:31:05 -0800384 countProducer.onMatchedLogEvent(1 /*log matcher index*/, event7);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800385 EXPECT_EQ(1UL, countProducer.mCurrentSlicedCounter->size());
386 EXPECT_EQ(4L, countProducer.mCurrentSlicedCounter->begin()->second);
Yangster-mac93694462018-01-22 20:49:31 -0800387 EXPECT_EQ(anomalyTracker->getRefractoryPeriodEndsSec(DEFAULT_METRIC_DIMENSION_KEY),
Yangster-mac330af582018-02-08 15:24:38 -0800388 event7.GetElapsedTimestampNs() / NS_PER_SEC + refPeriodSec);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800389}
390
Yao Chen93fe3a32017-11-02 13:52:59 -0700391} // namespace statsd
392} // namespace os
393} // namespace android
394#else
395GTEST_LOG_(INFO) << "This test does nothing.\n";
396#endif