1 // Copyright (C) 2018 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 <gmock/gmock.h>
16 #include <gtest/gtest.h>
17 #include <stdio.h>
18
19 #include <chrono>
20 #include <thread>
21 #include <vector>
22
23 #include "../metrics/metrics_test_helper.h"
24 #include "src/stats_log_util.h"
25 #include "stats_event.h"
26 #include "tests/statsd_test_util.h"
27
28 #ifdef __ANDROID__
29
30 namespace android {
31 namespace os {
32 namespace statsd {
33
34 using namespace testing;
35 using std::make_shared;
36 using std::shared_ptr;
37 using std::vector;
38 using std::this_thread::sleep_for;
39 using testing::Contains;
40
41 namespace {
42 int pullTagId = 10014;
43
44 bool pullSuccess;
45 vector<std::shared_ptr<LogEvent>> pullData;
46 long pullDelayNs;
47
48 class FakePuller : public StatsPuller {
49 public:
FakePuller()50 FakePuller()
51 : StatsPuller(pullTagId, /*coolDownNs=*/MillisToNano(10), /*timeoutNs=*/MillisToNano(5)){};
52
53 private:
PullInternal(vector<std::shared_ptr<LogEvent>> * data)54 PullErrorCode PullInternal(vector<std::shared_ptr<LogEvent>>* data) override {
55 (*data) = pullData;
56 sleep_for(std::chrono::nanoseconds(pullDelayNs));
57 return pullSuccess ? PULL_SUCCESS : PULL_FAIL;
58 }
59 };
60
61 FakePuller puller;
62
createSimpleEvent(int64_t eventTimeNs,int64_t value)63 std::unique_ptr<LogEvent> createSimpleEvent(int64_t eventTimeNs, int64_t value) {
64 AStatsEvent* statsEvent = AStatsEvent_obtain();
65 AStatsEvent_setAtomId(statsEvent, pullTagId);
66 AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
67 AStatsEvent_writeInt64(statsEvent, value);
68
69 std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
70 parseStatsEventToLogEvent(statsEvent, logEvent.get());
71 return logEvent;
72 }
73
74 class StatsPullerTest : public ::testing::Test {
75 public:
StatsPullerTest()76 StatsPullerTest() {
77 }
78
SetUp()79 void SetUp() override {
80 puller.ForceClearCache();
81 pullSuccess = false;
82 pullDelayNs = 0;
83 pullData.clear();
84 }
85 };
86
87 } // Anonymous namespace.
88
TEST_F(StatsPullerTest,PullSuccess)89 TEST_F(StatsPullerTest, PullSuccess) {
90 pullData.push_back(createSimpleEvent(1111L, 33));
91
92 pullSuccess = true;
93
94 vector<std::shared_ptr<LogEvent>> dataHolder;
95 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_SUCCESS);
96 ASSERT_EQ(1, dataHolder.size());
97 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
98 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
99 ASSERT_EQ(1, dataHolder[0]->size());
100 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
101
102 sleep_for(std::chrono::milliseconds(11));
103
104 pullData.clear();
105 pullData.push_back(createSimpleEvent(2222L, 44));
106
107 pullSuccess = true;
108
109 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_SUCCESS);
110 ASSERT_EQ(1, dataHolder.size());
111 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
112 EXPECT_EQ(2222L, dataHolder[0]->GetElapsedTimestampNs());
113 ASSERT_EQ(1, dataHolder[0]->size());
114 EXPECT_EQ(44, dataHolder[0]->getValues()[0].mValue.int_value);
115 }
116
TEST_F(StatsPullerTest,PullFailAfterSuccess)117 TEST_F(StatsPullerTest, PullFailAfterSuccess) {
118 pullData.push_back(createSimpleEvent(1111L, 33));
119
120 pullSuccess = true;
121
122 vector<std::shared_ptr<LogEvent>> dataHolder;
123 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_SUCCESS);
124 ASSERT_EQ(1, dataHolder.size());
125 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
126 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
127 ASSERT_EQ(1, dataHolder[0]->size());
128 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
129
130 sleep_for(std::chrono::milliseconds(11));
131
132 pullData.clear();
133 pullData.push_back(createSimpleEvent(2222L, 44));
134
135 pullSuccess = false;
136 dataHolder.clear();
137 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
138 ASSERT_EQ(0, dataHolder.size());
139
140 // Fails due to hitting the cool down.
141 pullSuccess = true;
142 dataHolder.clear();
143 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
144 ASSERT_EQ(0, dataHolder.size());
145 }
146
147 // Test pull takes longer than timeout, 2nd pull happens shorter than cooldown
TEST_F(StatsPullerTest,PullTakeTooLongAndPullFast)148 TEST_F(StatsPullerTest, PullTakeTooLongAndPullFast) {
149 pullData.push_back(createSimpleEvent(1111L, 33));
150 pullSuccess = true;
151 // timeout is 5ms
152 pullDelayNs = MillisToNano(6);
153
154 vector<std::shared_ptr<LogEvent>> dataHolder;
155 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
156 ASSERT_EQ(0, dataHolder.size());
157
158 pullData.clear();
159 pullData.push_back(createSimpleEvent(2222L, 44));
160 pullDelayNs = 0;
161
162 pullSuccess = true;
163 dataHolder.clear();
164 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
165 ASSERT_EQ(0, dataHolder.size());
166 }
167
TEST_F(StatsPullerTest,PullFail)168 TEST_F(StatsPullerTest, PullFail) {
169 pullData.push_back(createSimpleEvent(1111L, 33));
170
171 pullSuccess = false;
172
173 vector<std::shared_ptr<LogEvent>> dataHolder;
174 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
175 ASSERT_EQ(0, dataHolder.size());
176 }
177
TEST_F(StatsPullerTest,PullTakeTooLong)178 TEST_F(StatsPullerTest, PullTakeTooLong) {
179 pullData.push_back(createSimpleEvent(1111L, 33));
180
181 pullSuccess = true;
182 pullDelayNs = MillisToNano(6);
183
184 vector<std::shared_ptr<LogEvent>> dataHolder;
185 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
186 ASSERT_EQ(0, dataHolder.size());
187 }
188
TEST_F(StatsPullerTest,PullTooFast)189 TEST_F(StatsPullerTest, PullTooFast) {
190 pullData.push_back(createSimpleEvent(1111L, 33));
191
192 pullSuccess = true;
193
194 vector<std::shared_ptr<LogEvent>> dataHolder;
195 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_SUCCESS);
196 ASSERT_EQ(1, dataHolder.size());
197 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
198 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
199 ASSERT_EQ(1, dataHolder[0]->size());
200 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
201
202 pullData.clear();
203 pullData.push_back(createSimpleEvent(2222L, 44));
204
205 pullSuccess = true;
206
207 dataHolder.clear();
208 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_SUCCESS);
209 ASSERT_EQ(1, dataHolder.size());
210 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
211 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
212 ASSERT_EQ(1, dataHolder[0]->size());
213 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
214 }
215
TEST_F(StatsPullerTest,PullFailsAndTooFast)216 TEST_F(StatsPullerTest, PullFailsAndTooFast) {
217 pullData.push_back(createSimpleEvent(1111L, 33));
218
219 pullSuccess = false;
220
221 vector<std::shared_ptr<LogEvent>> dataHolder;
222 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
223 ASSERT_EQ(0, dataHolder.size());
224
225 pullData.clear();
226 pullData.push_back(createSimpleEvent(2222L, 44));
227
228 pullSuccess = true;
229
230 EXPECT_EQ(puller.Pull(getElapsedRealtimeNs(), &dataHolder), PULL_FAIL);
231 ASSERT_EQ(0, dataHolder.size());
232 }
233
TEST_F(StatsPullerTest,PullSameEventTime)234 TEST_F(StatsPullerTest, PullSameEventTime) {
235 pullData.push_back(createSimpleEvent(1111L, 33));
236
237 pullSuccess = true;
238 int64_t eventTimeNs = getElapsedRealtimeNs();
239
240 vector<std::shared_ptr<LogEvent>> dataHolder;
241 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_SUCCESS);
242 ASSERT_EQ(1, dataHolder.size());
243 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
244 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
245 ASSERT_EQ(1, dataHolder[0]->size());
246 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
247
248 pullData.clear();
249 pullData.push_back(createSimpleEvent(2222L, 44));
250
251 // Sleep to ensure the cool down expires.
252 sleep_for(std::chrono::milliseconds(11));
253 pullSuccess = true;
254
255 dataHolder.clear();
256 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_SUCCESS);
257 ASSERT_EQ(1, dataHolder.size());
258 EXPECT_EQ(pullTagId, dataHolder[0]->GetTagId());
259 EXPECT_EQ(1111L, dataHolder[0]->GetElapsedTimestampNs());
260 ASSERT_EQ(1, dataHolder[0]->size());
261 EXPECT_EQ(33, dataHolder[0]->getValues()[0].mValue.int_value);
262 }
263
264 // Test pull takes longer than timeout, 2nd pull happens at same event time
TEST_F(StatsPullerTest,PullTakeTooLongAndPullSameEventTime)265 TEST_F(StatsPullerTest, PullTakeTooLongAndPullSameEventTime) {
266 pullData.push_back(createSimpleEvent(1111L, 33));
267 pullSuccess = true;
268 int64_t eventTimeNs = getElapsedRealtimeNs();
269 // timeout is 5ms
270 pullDelayNs = MillisToNano(6);
271
272 vector<std::shared_ptr<LogEvent>> dataHolder;
273 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_FAIL);
274 ASSERT_EQ(0, dataHolder.size());
275
276 // Sleep to ensure the cool down expires. 6ms is taken by the delay, so only 5 is needed here.
277 sleep_for(std::chrono::milliseconds(5));
278
279 pullData.clear();
280 pullData.push_back(createSimpleEvent(2222L, 44));
281 pullDelayNs = 0;
282
283 pullSuccess = true;
284 dataHolder.clear();
285 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_FAIL);
286 ASSERT_EQ(0, dataHolder.size());
287 }
288
TEST_F(StatsPullerTest,PullFailsAndPullSameEventTime)289 TEST_F(StatsPullerTest, PullFailsAndPullSameEventTime) {
290 pullData.push_back(createSimpleEvent(1111L, 33));
291
292 pullSuccess = false;
293 int64_t eventTimeNs = getElapsedRealtimeNs();
294
295 vector<std::shared_ptr<LogEvent>> dataHolder;
296 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_FAIL);
297 ASSERT_EQ(0, dataHolder.size());
298
299 // Sleep to ensure the cool down expires.
300 sleep_for(std::chrono::milliseconds(11));
301
302 pullData.clear();
303 pullData.push_back(createSimpleEvent(2222L, 44));
304
305 pullSuccess = true;
306
307 EXPECT_EQ(puller.Pull(eventTimeNs, &dataHolder), PULL_FAIL);
308 ASSERT_EQ(0, dataHolder.size());
309 }
310
311 } // namespace statsd
312 } // namespace os
313 } // namespace android
314 #else
315 GTEST_LOG_(INFO) << "This test does nothing.\n";
316 #endif
317