-
Notifications
You must be signed in to change notification settings - Fork 1
/
metrics_utils.h
154 lines (134 loc) · 5.04 KB
/
metrics_utils.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
//
// Copyright (C) 2015 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#ifndef UPDATE_ENGINE_METRICS_UTILS_H_
#define UPDATE_ENGINE_METRICS_UTILS_H_
#include <chrono>
#include <string>
#include <string_view>
#include <type_traits>
#include <utility>
#include <base/time/time.h>
#include "update_engine/common/clock_interface.h"
#include "update_engine/common/connection_utils.h"
#include "update_engine/common/error_code.h"
#include "update_engine/common/metrics_constants.h"
#include "update_engine/common/metrics_reporter_interface.h"
#include "update_engine/common/prefs_interface.h"
namespace chromeos_update_engine {
namespace metrics_utils {
// Transforms a ErrorCode value into a metrics::DownloadErrorCode.
// This obviously only works for errors related to downloading so if |code|
// is e.g. |ErrorCode::kFilesystemCopierError| then
// |kDownloadErrorCodeInputMalformed| is returned.
metrics::DownloadErrorCode GetDownloadErrorCode(ErrorCode code);
// Transforms a ErrorCode value into a metrics::AttemptResult.
//
// If metrics::AttemptResult::kPayloadDownloadError is returned, you
// can use utils::GetDownloadError() to get more detail.
metrics::AttemptResult GetAttemptResult(ErrorCode code);
// Calculates the internet connection type given |type| and |tethering|.
metrics::ConnectionType GetConnectionType(ConnectionType type,
ConnectionTethering tethering);
// Returns the persisted value from prefs for the given key. It also
// validates that the value returned is non-negative.
int64_t GetPersistedValue(std::string_view key, PrefsInterface* prefs);
// Persists the reboot count of the update attempt to |kPrefsNumReboots|.
void SetNumReboots(int64_t num_reboots, PrefsInterface* prefs);
// Persists the payload attempt number to |kPrefsPayloadAttemptNumber|.
void SetPayloadAttemptNumber(int64_t payload_attempt_number,
PrefsInterface* prefs);
// Persists the finished time of an update to the |kPrefsSystemUpdatedMarker|.
void SetSystemUpdatedMarker(ClockInterface* clock, PrefsInterface* prefs);
// Persists the start monotonic time of an update to
// |kPrefsUpdateTimestampStart|.
void SetUpdateTimestampStart(const base::Time& update_start_time,
PrefsInterface* prefs);
// Persists the start boot time of an update to
// |kPrefsUpdateBootTimestampStart|.
void SetUpdateBootTimestampStart(const base::Time& update_start_boot_time,
PrefsInterface* prefs);
// Called at program startup if the device booted into a new update.
// The |time_to_reboot| parameter contains the (monotonic-clock) duration
// from when the update successfully completed (the value in
// |kPrefsSystemUpdatedMarker|) until the device was booted into the update
// (current monotonic-clock time).
bool LoadAndReportTimeToReboot(MetricsReporterInterface* metrics_reporter,
PrefsInterface* prefs,
ClockInterface* clock);
template <typename T>
class PersistedValue {
public:
PersistedValue(std::string_view key, PrefsInterface* prefs)
: key_(key), prefs_(prefs) {
val_ = metrics_utils::GetPersistedValue(key, prefs);
}
~PersistedValue() { Flush(true); }
void Delete() {
val_ = {};
prefs_->Delete(key_);
}
T get() const { return val_; }
using clock = std::chrono::system_clock;
using time_point = clock::time_point;
// prefix increment
PersistedValue<T>& operator++() {
++val_;
Flush();
return *this;
}
PersistedValue<T>& operator--() {
--val_;
Flush();
return *this;
}
PersistedValue<T>& operator+=(T&& t) {
val_ += std::forward<T>(t);
Flush();
return *this;
}
PersistedValue<T>& operator-=(T&& t) {
val_ -= std::forward<T>(t);
Flush();
return *this;
}
PersistedValue<T>& operator=(T&& t) {
val_ = std::forward<T>(t);
Flush();
return *this;
}
void Flush(bool force = false) {
auto now = clock::now();
if (now - last_save_ > metrics::kMetricFlushInterval || force) {
last_save_ = now;
if (std::is_integral_v<T>) {
prefs_->SetInt64(key_, val_);
} else if (std::is_same_v<T, bool>) {
prefs_->SetBoolean(key_, val_);
} else {
auto value = std::to_string(val_);
prefs_->SetString(key_, value);
}
}
}
private:
const std::string_view key_;
PrefsInterface* prefs_;
T val_;
time_point last_save_{};
};
} // namespace metrics_utils
} // namespace chromeos_update_engine
#endif // UPDATE_ENGINE_METRICS_UTILS_H_