Home | History | Annotate | Download | only in anomaly
      1 /*
      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 
     17 #define DEBUG false  // STOPSHIP if true
     18 #include "Log.h"
     19 
     20 #include "DurationAnomalyTracker.h"
     21 #include "guardrail/StatsdStats.h"
     22 
     23 namespace android {
     24 namespace os {
     25 namespace statsd {
     26 
     27 DurationAnomalyTracker::DurationAnomalyTracker(const Alert& alert, const ConfigKey& configKey,
     28                                                const sp<AlarmMonitor>& alarmMonitor)
     29         : AnomalyTracker(alert, configKey), mAlarmMonitor(alarmMonitor) {
     30     VLOG("DurationAnomalyTracker() called");
     31 }
     32 
     33 DurationAnomalyTracker::~DurationAnomalyTracker() {
     34     VLOG("~DurationAnomalyTracker() called");
     35     cancelAllAlarms();
     36 }
     37 
     38 void DurationAnomalyTracker::startAlarm(const MetricDimensionKey& dimensionKey,
     39                                         const int64_t& timestampNs) {
     40     // Alarms are stored in secs. Must round up, since if it fires early, it is ignored completely.
     41     uint32_t timestampSec = static_cast<uint32_t>((timestampNs -1) / NS_PER_SEC) + 1; // round up
     42     if (isInRefractoryPeriod(timestampNs, dimensionKey)) {
     43         VLOG("Not setting anomaly alarm since it would fall in the refractory period.");
     44         return;
     45     }
     46 
     47     auto itr = mAlarms.find(dimensionKey);
     48     if (itr != mAlarms.end() && mAlarmMonitor != nullptr) {
     49         mAlarmMonitor->remove(itr->second);
     50     }
     51 
     52     sp<const InternalAlarm> alarm = new InternalAlarm{timestampSec};
     53     mAlarms[dimensionKey] = alarm;
     54     if (mAlarmMonitor != nullptr) {
     55         mAlarmMonitor->add(alarm);
     56     }
     57 }
     58 
     59 void DurationAnomalyTracker::stopAlarm(const MetricDimensionKey& dimensionKey,
     60                                        const int64_t& timestampNs) {
     61     const auto itr = mAlarms.find(dimensionKey);
     62     if (itr == mAlarms.end()) {
     63         return;
     64     }
     65 
     66     // If the alarm is set in the past but hasn't fired yet (due to lag), catch it now.
     67     if (itr->second != nullptr && timestampNs >= (int64_t)NS_PER_SEC * itr->second->timestampSec) {
     68         declareAnomaly(timestampNs, mAlert.metric_id(), dimensionKey,
     69                        mAlert.trigger_if_sum_gt() + (timestampNs / NS_PER_SEC) -
     70                                itr->second->timestampSec);
     71     }
     72     if (mAlarmMonitor != nullptr) {
     73         mAlarmMonitor->remove(itr->second);
     74     }
     75     mAlarms.erase(dimensionKey);
     76 }
     77 
     78 void DurationAnomalyTracker::cancelAllAlarms() {
     79     if (mAlarmMonitor != nullptr) {
     80         for (const auto& itr : mAlarms) {
     81             mAlarmMonitor->remove(itr.second);
     82         }
     83     }
     84     mAlarms.clear();
     85 }
     86 
     87 void DurationAnomalyTracker::informAlarmsFired(const int64_t& timestampNs,
     88         unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>>& firedAlarms) {
     89 
     90     if (firedAlarms.empty() || mAlarms.empty()) return;
     91     // Find the intersection of firedAlarms and mAlarms.
     92     // The for loop is inefficient, since it loops over all keys, but that's okay since it is very
     93     // seldomly called. The alternative would be having InternalAlarms store information about the
     94     // DurationAnomalyTracker and key, but that's a lot of data overhead to speed up something that
     95     // is rarely ever called.
     96     unordered_map<MetricDimensionKey, sp<const InternalAlarm>> matchedAlarms;
     97     for (const auto& kv : mAlarms) {
     98         if (firedAlarms.count(kv.second) > 0) {
     99             matchedAlarms.insert({kv.first, kv.second});
    100         }
    101     }
    102 
    103     // Now declare each of these alarms to have fired.
    104     for (const auto& kv : matchedAlarms) {
    105         declareAnomaly(
    106                 timestampNs, mAlert.metric_id(), kv.first,
    107                 mAlert.trigger_if_sum_gt() + (timestampNs / NS_PER_SEC) - kv.second->timestampSec);
    108         mAlarms.erase(kv.first);
    109         firedAlarms.erase(kv.second);  // No one else can also own it, so we're done with it.
    110     }
    111 }
    112 
    113 }  // namespace statsd
    114 }  // namespace os
    115 }  // namespace android
    116