blob: 8966a9a35ddef092470bcafc4efb3fd700e802f9 [file] [log] [blame]
Alex Deymo8c499142014-01-02 19:33:34 -08001// Copyright (c) 2014 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Gilad Arnold2cbb3852014-03-07 12:40:50 -08005#ifndef CHROMEOS_PLATFORM_UPDATE_ENGINE_POLICY_MANAGER_VARIABLE_H_
6#define CHROMEOS_PLATFORM_UPDATE_ENGINE_POLICY_MANAGER_VARIABLE_H_
Alex Deymo8c499142014-01-02 19:33:34 -08007
Alex Deymoa07a1232014-02-25 14:19:50 -08008#include <algorithm>
9#include <list>
Alex Deymo8c499142014-01-02 19:33:34 -080010#include <string>
11
Alex Deymo53556ec2014-03-17 10:05:57 -070012#include <base/bind.h>
13#include <base/logging.h>
Alex Deymo8c499142014-01-02 19:33:34 -080014#include <base/time.h>
Alex Deymoca0aaa62014-01-06 10:39:58 -080015#include <gtest/gtest_prod.h> // for FRIEND_TEST
Alex Deymo8c499142014-01-02 19:33:34 -080016
Alex Deymo53556ec2014-03-17 10:05:57 -070017#include "update_engine/policy_manager/event_loop.h"
18
Alex Deymo8c499142014-01-02 19:33:34 -080019namespace chromeos_policy_manager {
20
Alex Deymo0e433692014-02-20 07:23:03 -080021// The VariableMode specifies important behavior of the variable in terms of
22// whether, how and when the value of the variable changes.
23enum VariableMode {
24 // Const variables never changes during the life of a policy request, so the
25 // EvaluationContext caches the value even between different evaluations of
26 // the same policy request.
27 kVariableModeConst,
28
29 // Poll variables, or synchronous variables, represent a variable with a value
30 // that can be queried at any time, but it is not known when the value
31 // changes on the source of information. In order to detect if the value of
32 // the variable changes, it has to be queried again.
33 kVariableModePoll,
34
35 // Async variables are able to produce a signal or callback whenever the
36 // value changes. This means that it's not required to poll the value to
37 // detect when it changes, instead, you should register an observer to get
38 // a notification when that happens.
39 kVariableModeAsync,
40};
41
Alex Deymo391ad9f2014-01-29 14:36:20 -080042// This class is a base class with the common functionality that doesn't
43// deppend on the variable's type, implemented by all the variables.
44class BaseVariable {
Alex Deymo8c499142014-01-02 19:33:34 -080045 public:
Alex Deymoa07a1232014-02-25 14:19:50 -080046 // Interface for observing changes on variable value.
Alex Deymo53556ec2014-03-17 10:05:57 -070047 class ObserverInterface {
Alex Deymoa07a1232014-02-25 14:19:50 -080048 public:
Alex Deymo53556ec2014-03-17 10:05:57 -070049 virtual ~ObserverInterface() {}
Alex Deymoa07a1232014-02-25 14:19:50 -080050
51 // Called when the value on the variable changes.
52 virtual void ValueChanged(BaseVariable* variable) = 0;
53 };
54
Alex Deymo53556ec2014-03-17 10:05:57 -070055 virtual ~BaseVariable() {
56 if (!observer_list_.empty()) {
57 LOG(WARNING) << "Variable " << name_ << " deleted with "
58 << observer_list_.size() << " observers.";
59 }
60 }
Alex Deymo391ad9f2014-01-29 14:36:20 -080061
62 // Returns the variable name as a string.
Alex Deymo0e433692014-02-20 07:23:03 -080063 const std::string& GetName() const {
Alex Deymo391ad9f2014-01-29 14:36:20 -080064 return name_;
65 }
66
Alex Deymo0e433692014-02-20 07:23:03 -080067 // Returns the variable mode.
68 VariableMode GetMode() const {
69 return mode_;
70 }
71
Alex Deymoa8033932014-02-25 10:33:13 -080072 // For VariableModePoll variables, it returns the polling interval of this
73 // variable. In other case, it returns 0.
74 base::TimeDelta GetPollInterval() const {
75 return poll_interval_;
76 }
77
Alex Deymoa07a1232014-02-25 14:19:50 -080078 // Adds and removes observers for value changes on the variable. This only
79 // works for kVariableAsync variables since the other modes don't track value
80 // changes. Adding the same observer twice has no effect.
Alex Deymo53556ec2014-03-17 10:05:57 -070081 virtual void AddObserver(BaseVariable::ObserverInterface* observer) {
Alex Deymoa07a1232014-02-25 14:19:50 -080082 if (std::find(observer_list_.begin(), observer_list_.end(), observer) ==
83 observer_list_.end()) {
84 observer_list_.push_back(observer);
85 }
86 }
87
Alex Deymo53556ec2014-03-17 10:05:57 -070088 virtual void RemoveObserver(BaseVariable::ObserverInterface* observer) {
Alex Deymoa07a1232014-02-25 14:19:50 -080089 observer_list_.remove(observer);
90 }
91
Alex Deymo0e433692014-02-20 07:23:03 -080092 protected:
Alex Deymoa8033932014-02-25 10:33:13 -080093 // Creates a BaseVariable using the default polling interval (5 minutes).
Alex Deymo0e433692014-02-20 07:23:03 -080094 BaseVariable(const std::string& name, VariableMode mode)
Alex Deymoa8033932014-02-25 10:33:13 -080095 : BaseVariable(name, mode,
96 base::TimeDelta::FromMinutes(kDefaultPollMinutes)) {}
97
98 // Creates a BaseVariable with mode kVariableModePoll and the provided
99 // polling interval.
100 BaseVariable(const std::string& name, base::TimeDelta poll_interval)
101 : BaseVariable(name, kVariableModePoll, poll_interval) {}
Alex Deymo0e433692014-02-20 07:23:03 -0800102
Alex Deymoa07a1232014-02-25 14:19:50 -0800103 // Calls ValueChanged on all the observers.
104 void NotifyValueChanged() {
Alex Deymo53556ec2014-03-17 10:05:57 -0700105 for (auto& observer : observer_list_) {
106 RunFromMainLoop(base::Bind(&BaseVariable::ObserverInterface::ValueChanged,
107 base::Unretained(observer), this));
108 }
Alex Deymoa07a1232014-02-25 14:19:50 -0800109 }
110
Alex Deymo391ad9f2014-01-29 14:36:20 -0800111 private:
Alex Deymo53556ec2014-03-17 10:05:57 -0700112 friend class PmEvaluationContextTest;
Alex Deymoa07a1232014-02-25 14:19:50 -0800113 friend class PmBaseVariableTest;
114 FRIEND_TEST(PmBaseVariableTest, RepeatedObserverTest);
115 FRIEND_TEST(PmBaseVariableTest, NotifyValueChangedTest);
116
Alex Deymoa8033932014-02-25 10:33:13 -0800117 BaseVariable(const std::string& name, VariableMode mode,
118 base::TimeDelta poll_interval)
119 : name_(name), mode_(mode),
120 poll_interval_(mode == kVariableModePoll ?
121 poll_interval : base::TimeDelta()) {}
122
123 // The default PollInterval in minutes.
124 static constexpr int kDefaultPollMinutes = 5;
125
Alex Deymo391ad9f2014-01-29 14:36:20 -0800126 // The variable's name as a string.
127 const std::string name_;
Alex Deymo0e433692014-02-20 07:23:03 -0800128
129 // The variable's mode.
130 const VariableMode mode_;
Alex Deymoa8033932014-02-25 10:33:13 -0800131
132 // The variable's polling interval for VariableModePoll variable and 0 for
133 // other modes.
134 const base::TimeDelta poll_interval_;
Alex Deymoa07a1232014-02-25 14:19:50 -0800135
136 // The list of value changes observers.
Alex Deymo53556ec2014-03-17 10:05:57 -0700137 std::list<BaseVariable::ObserverInterface*> observer_list_;
Alex Deymo231a8512014-03-21 12:56:10 -0700138
139 DISALLOW_COPY_AND_ASSIGN(BaseVariable);
Alex Deymo391ad9f2014-01-29 14:36:20 -0800140};
141
142// Interface to a Policy Manager variable of a given type. Implementation
143// internals are hidden as protected members, since policies should not be
144// using them directly.
145template<typename T>
146class Variable : public BaseVariable {
147 public:
Alex Deymo8c499142014-01-02 19:33:34 -0800148 virtual ~Variable() {}
149
150 protected:
Alex Deymo23949d42014-02-05 15:20:59 -0800151 // Only allow to get values through the EvaluationContext class and not
152 // directly from the variable.
153 friend class EvaluationContext;
154
Alex Deymobb019fe2014-02-03 20:12:17 -0800155 friend class PmRealRandomProviderTest;
156 FRIEND_TEST(PmRealRandomProviderTest, GetRandomValues);
Gilad Arnold55f39b72014-01-28 12:51:45 -0800157 friend class PmRealShillProviderTest;
Gilad Arnold5ef9c482014-03-03 13:51:02 -0800158 FRIEND_TEST(PmRealShillProviderTest, ReadDefaultValues);
159 FRIEND_TEST(PmRealShillProviderTest, ReadChangedValuesConnectedViaEthernet);
160 FRIEND_TEST(PmRealShillProviderTest, ReadChangedValuesConnectedViaVpn);
Gilad Arnoldbeb39e92014-03-11 11:34:50 -0700161 FRIEND_TEST(PmRealShillProviderTest, ReadChangedValuesConnectedTwoSignals);
Gilad Arnold78a78112014-03-13 14:58:06 -0700162 friend class PmRealTimeProviderTest;
163 FRIEND_TEST(PmRealTimeProviderTest, CurrDateValid);
164 FRIEND_TEST(PmRealTimeProviderTest, CurrHourValid);
Alex Deymoca0aaa62014-01-06 10:39:58 -0800165
Alex Deymo0e433692014-02-20 07:23:03 -0800166 Variable(const std::string& name, VariableMode mode)
167 : BaseVariable(name, mode) {}
168
Alex Deymoa8033932014-02-25 10:33:13 -0800169 Variable(const std::string& name, const base::TimeDelta& poll_interval)
170 : BaseVariable(name, poll_interval) {}
171
Alex Deymo8c499142014-01-02 19:33:34 -0800172 // Gets the current value of the variable. The current value is copied to a
173 // new object and returned. The caller of this method owns the object and
174 // should delete it.
175 //
176 // In case of and error getting the current value or the |timeout| timeout is
177 // exceeded, a NULL value is returned and the |errmsg| is set.
178 //
179 // The caller can pass a NULL value for |errmsg|, in which case the error
180 // message won't be set.
181 virtual const T* GetValue(base::TimeDelta timeout, std::string* errmsg) = 0;
Alex Deymo231a8512014-03-21 12:56:10 -0700182
183 private:
184 DISALLOW_COPY_AND_ASSIGN(Variable);
Alex Deymo8c499142014-01-02 19:33:34 -0800185};
186
187} // namespace chromeos_policy_manager
188
Gilad Arnold2cbb3852014-03-07 12:40:50 -0800189#endif // CHROMEOS_PLATFORM_UPDATE_ENGINE_POLICY_MANAGER_VARIABLE_H_