blob: cdaca1bf85d7b697242f3faa3dc6ace33abcd09c [file] [log] [blame]
Yao Chenab273e22017-09-06 12:53:50 -07001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Joe Onorato9fc9edf2017-10-15 20:08:52 -070017#include "Log.h"
Yao Chenab273e22017-09-06 12:53:50 -070018
Joe Onorato9fc9edf2017-10-15 20:08:52 -070019#include "StatsLogProcessor.h"
20#include "frameworks/base/cmds/statsd/src/stats_log.pb.h"
21#include "metrics/CountMetricProducer.h"
22#include "stats_util.h"
23
yro00698da2017-09-15 10:06:40 -070024#include <log/log_event_list.h>
Yao Chenef99c4f2017-09-22 16:26:54 -070025#include <utils/Errors.h>
Yao Chenab273e22017-09-06 12:53:50 -070026
27using namespace android;
Yao Chen44cf27c2017-09-14 22:32:50 -070028using std::make_unique;
29using std::unique_ptr;
30using std::vector;
Bookatz906a35c2017-09-20 15:26:44 -070031
32namespace android {
33namespace os {
34namespace statsd {
Yao Chenab273e22017-09-06 12:53:50 -070035
yro31eb67b2017-10-24 13:33:21 -070036StatsLogProcessor::StatsLogProcessor(const sp<UidMap>& uidMap,
37 const std::function<void(const vector<uint8_t>&)>& pushLog)
yrofd05a4e2017-10-26 14:08:26 -070038 : mUidMap(uidMap), mPushLog(pushLog) {
Yao Chenab273e22017-09-06 12:53:50 -070039}
40
Yao Chenef99c4f2017-09-22 16:26:54 -070041StatsLogProcessor::~StatsLogProcessor() {
Yao Chenab273e22017-09-06 12:53:50 -070042}
43
Yao Chen44cf27c2017-09-14 22:32:50 -070044// TODO: what if statsd service restarts? How do we know what logs are already processed before?
Joe Onoratoc4dfae52017-10-17 23:38:21 -070045void StatsLogProcessor::OnLogEvent(const LogEvent& msg) {
Yao Chen44cf27c2017-09-14 22:32:50 -070046 // pass the event to metrics managers.
47 for (auto& pair : mMetricsManagers) {
48 pair.second->onLogEvent(msg);
yro69007c82017-10-26 20:42:57 -070049 flushIfNecessary(msg.GetTimestampNs(), pair.first, pair.second);
Yao Chenab273e22017-09-06 12:53:50 -070050 }
51}
52
Joe Onorato9fc9edf2017-10-15 20:08:52 -070053void StatsLogProcessor::OnConfigUpdated(const ConfigKey& key, const StatsdConfig& config) {
54 auto it = mMetricsManagers.find(key);
Yao Chen44cf27c2017-09-14 22:32:50 -070055 if (it != mMetricsManagers.end()) {
56 it->second->finish();
57 }
58
Joe Onorato9fc9edf2017-10-15 20:08:52 -070059 ALOGD("Updated configuration for key %s", key.ToString().c_str());
Yao Chen44cf27c2017-09-14 22:32:50 -070060
Yao Chencaf339d2017-10-06 16:01:10 -070061 unique_ptr<MetricsManager> newMetricsManager = std::make_unique<MetricsManager>(config);
62 if (newMetricsManager->isConfigValid()) {
Joe Onorato9fc9edf2017-10-15 20:08:52 -070063 mMetricsManagers[key] = std::move(newMetricsManager);
64 // Why doesn't this work? mMetricsManagers.insert({key, std::move(newMetricsManager)});
Yao Chencaf339d2017-10-06 16:01:10 -070065 ALOGD("StatsdConfig valid");
66 } else {
67 // If there is any error in the config, don't use it.
68 ALOGD("StatsdConfig NOT valid");
69 }
yro00698da2017-09-15 10:06:40 -070070}
Bookatz906a35c2017-09-20 15:26:44 -070071
Yao Chen729093d2017-10-16 10:33:26 -070072vector<StatsLogReport> StatsLogProcessor::onDumpReport(const ConfigKey& key) {
73 auto it = mMetricsManagers.find(key);
74 if (it == mMetricsManagers.end()) {
75 ALOGW("Config source %s does not exist", key.ToString().c_str());
76 return vector<StatsLogReport>();
77 }
78
79 return it->second->onDumpReport();
80}
81
Joe Onorato9fc9edf2017-10-15 20:08:52 -070082void StatsLogProcessor::OnConfigRemoved(const ConfigKey& key) {
83 auto it = mMetricsManagers.find(key);
84 if (it != mMetricsManagers.end()) {
85 it->second->finish();
86 mMetricsManagers.erase(it);
87 }
yro69007c82017-10-26 20:42:57 -070088 auto flushTime = mLastFlushTimes.find(key);
89 if (flushTime != mLastFlushTimes.end()) {
90 mLastFlushTimes.erase(flushTime);
91 }
Joe Onorato9fc9edf2017-10-15 20:08:52 -070092}
93
yro69007c82017-10-26 20:42:57 -070094void StatsLogProcessor::flushIfNecessary(uint64_t timestampNs,
95 const ConfigKey& key,
96 const unique_ptr<MetricsManager>& metricsManager) {
97 auto lastFlushNs = mLastFlushTimes.find(key);
98 if (lastFlushNs != mLastFlushTimes.end()) {
99 if (timestampNs - lastFlushNs->second < kMinFlushPeriod) {
100 return;
101 }
102 }
yro31eb67b2017-10-24 13:33:21 -0700103
yro69007c82017-10-26 20:42:57 -0700104 size_t totalBytes = metricsManager->byteSize();
105 if (totalBytes > kMaxSerializedBytes) {
106 flush();
107 mLastFlushTimes[key] = std::move(timestampNs);
yro31eb67b2017-10-24 13:33:21 -0700108 }
109}
110
111void StatsLogProcessor::flush() {
yro69007c82017-10-26 20:42:57 -0700112 // TODO: Take ConfigKey as an argument and flush metrics related to the
113 // ConfigKey. Also, create a wrapper that holds a repeated field of
114 // StatsLogReport's.
115 /*
yro31eb67b2017-10-24 13:33:21 -0700116 StatsLogReport logReport;
yro31eb67b2017-10-24 13:33:21 -0700117 const int numBytes = logReport.ByteSize();
118 vector<uint8_t> logReportBuffer(numBytes);
119 logReport.SerializeToArray(&logReportBuffer[0], numBytes);
120 mPushLog(logReportBuffer);
yro69007c82017-10-26 20:42:57 -0700121 */
yro31eb67b2017-10-24 13:33:21 -0700122}
123
Yao Chenef99c4f2017-09-22 16:26:54 -0700124} // namespace statsd
125} // namespace os
126} // namespace android