blob: f0a401e257528c1859cc1a31518d46053f4aad5f [file] [log] [blame]
Vitaly Buka4615e0d2015-10-14 15:35:12 -07001// Copyright 2015 The Weave Authors. All rights reserved.
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Vitaly Buka912b6982015-07-06 11:13:03 -07005#ifndef LIBWEAVE_SRC_COMMANDS_PROP_VALUES_H_
6#define LIBWEAVE_SRC_COMMANDS_PROP_VALUES_H_
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -07007
8#include <map>
9#include <memory>
10#include <string>
11
Vitaly Buka0801a1f2015-08-14 10:03:46 -070012#include <weave/error.h>
Alex Vakulenko5f472062014-08-14 17:54:04 -070013
Stefan Sauer2d16dfa2015-09-25 17:08:35 +020014#include "src/commands/schema_utils.h"
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070015
16namespace base {
17class Value;
18class DictionaryValue;
19} // namespace base
20
Vitaly Bukab6f015a2015-07-09 14:59:23 -070021namespace weave {
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070022
23// Enumeration to indicate supported command parameter types.
24enum class ValueType {
25 Int,
26 Double,
27 String,
Alex Vakulenko66ec2922014-06-17 15:30:22 -070028 Boolean,
Alex Vakulenko29e64442015-03-20 13:59:19 -070029 Object,
30 Array,
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070031};
32
Alex Vakulenko66ec2922014-06-17 15:30:22 -070033class PropValue;
34class IntValue;
35class DoubleValue;
36class StringValue;
37class BooleanValue;
38class ObjectValue;
Alex Vakulenko29e64442015-03-20 13:59:19 -070039class ArrayValue;
Alex Vakulenko66ec2922014-06-17 15:30:22 -070040
41class PropType;
42
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070043// Helper methods to get the parameter type enum value for the given
44// native C++ data representation.
45// The generic GetValueType<T>() is undefined, however particular
46// type specializations return the appropriate ValueType.
Vitaly Bukaa647c852015-07-06 14:51:01 -070047template <typename T>
48ValueType GetValueType(); // Undefined.
49template <>
50inline ValueType GetValueType<int>() {
51 return ValueType::Int;
52}
53template <>
54inline ValueType GetValueType<double>() {
55 return ValueType::Double;
56}
57template <>
58inline ValueType GetValueType<std::string>() {
59 return ValueType::String;
60}
61template <>
62inline ValueType GetValueType<bool>() {
63 return ValueType::Boolean;
64}
65template <>
Vitaly Buka774cdf52015-07-21 13:55:00 -070066inline ValueType GetValueType<ValueMap>() {
Alex Vakulenko66ec2922014-06-17 15:30:22 -070067 return ValueType::Object;
68}
Vitaly Bukaa647c852015-07-06 14:51:01 -070069template <>
Vitaly Buka774cdf52015-07-21 13:55:00 -070070inline ValueType GetValueType<ValueVector>() {
Alex Vakulenko29e64442015-03-20 13:59:19 -070071 return ValueType::Array;
72}
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070073
Alex Vakulenko66ec2922014-06-17 15:30:22 -070074// The base class for property values.
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070075// Concrete value classes of various types will be derived from this base.
Alex Vakulenko66ec2922014-06-17 15:30:22 -070076// A property value is the actual command parameter value (or a concrete value
77// that can be used in constraints and presets). The PropValue is mostly
78// just parsed content of base::Value when a command is dispatched, however
79// it does have some additional functionality:
80// - it has a reference to the type definition (PropType) which is used
81// when validating the value, especially for "object" types.
82// - it can be compared with another instances of values of the same type.
83// This is used to validate the values against "enum"/"one of" constraints.
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070084class PropValue {
85 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -070086 // Only CreateDefaultValue should use this constructor.
87 explicit PropValue(const PropType& type);
Alex Vakulenko5ef75792015-03-19 15:50:44 -070088 virtual ~PropValue();
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -070089
90 // Gets the type of the value.
91 virtual ValueType GetType() const = 0;
92
93 // Type conversion methods. Used in lieu of RTTI and dynamic_cast<>.
94 virtual IntValue* GetInt() { return nullptr; }
95 virtual IntValue const* GetInt() const { return nullptr; }
96 virtual DoubleValue* GetDouble() { return nullptr; }
97 virtual DoubleValue const* GetDouble() const { return nullptr; }
98 virtual StringValue* GetString() { return nullptr; }
99 virtual StringValue const* GetString() const { return nullptr; }
100 virtual BooleanValue* GetBoolean() { return nullptr; }
101 virtual BooleanValue const* GetBoolean() const { return nullptr; }
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700102 virtual ObjectValue* GetObject() { return nullptr; }
103 virtual ObjectValue const* GetObject() const { return nullptr; }
Alex Vakulenko29e64442015-03-20 13:59:19 -0700104 virtual ArrayValue* GetArray() { return nullptr; }
105 virtual ArrayValue const* GetArray() const { return nullptr; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700106
107 // Makes a full copy of this value class.
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700108 virtual std::unique_ptr<PropValue> Clone() const = 0;
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700109
Vitaly Buka6942e1f2015-07-28 15:33:55 -0700110 // Saves the value as a JSON object. Never fails.
111 virtual std::unique_ptr<base::Value> ToJson() const = 0;
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700112
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700113 // Return the type definition of this value.
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700114 const PropType* GetPropType() const { return type_.get(); }
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700115 // Compares two values and returns true if they are equal.
116 virtual bool IsEqual(const PropValue* value) const = 0;
117
118 protected:
Vitaly Bukac58a8282015-07-29 01:25:20 -0700119 // Special out-of-line constructor to help implement PropValue::Clone().
120 // That method needs to clone the underlying type but can't do this in this
121 // header file since PropType is just forward-declared (it needs PropValue
122 // fully defined in its own inner workings).
123 explicit PropValue(const PropValue& other);
124
125 private:
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700126 std::unique_ptr<const PropType> type_;
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700127};
128
Alex Vakulenko29e64442015-03-20 13:59:19 -0700129// A helper template base class for implementing value classes.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700130template <typename T>
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700131class TypedValueBase : public PropValue {
132 public:
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700133 using PropValue::PropValue;
134
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700135 // Overrides from PropValue base class.
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700136 ValueType GetType() const override { return GetValueType<T>(); }
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700137
Vitaly Buka6942e1f2015-07-28 15:33:55 -0700138 std::unique_ptr<base::Value> ToJson() const override {
139 return TypedValueToJson(value_);
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700140 }
141
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700142 bool IsEqual(const PropValue* value) const override {
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700143 if (GetType() != value->GetType())
144 return false;
Vitaly Buka79c05e92015-07-29 12:25:37 -0700145 return CompareValue(GetValue(),
146 static_cast<const TypedValueBase*>(value)->GetValue());
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700147 }
148
149 // Helper methods to get and set the C++ representation of the value.
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700150 const T& GetValue() const { return value_; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700151
152 protected:
Vitaly Bukac58a8282015-07-29 01:25:20 -0700153 explicit TypedValueBase(const TypedValueBase& other)
154 : PropValue(other), value_(other.value_) {}
155
Vitaly Buka79c05e92015-07-29 12:25:37 -0700156 TypedValueBase(const PropType& type, T value)
157 : PropValue(type), value_(value) {}
158
Vitaly Bukac58a8282015-07-29 01:25:20 -0700159 private:
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700160 T value_{}; // The value of the parameter in C++ data representation.
161};
162
Vitaly Bukac58a8282015-07-29 01:25:20 -0700163// A helper template base class for implementing value classes.
164template <typename Derived, typename T>
165class TypedValueWithClone : public TypedValueBase<T> {
166 public:
167 using Base = TypedValueWithClone<Derived, T>;
168
169 // Expose the custom constructor of the base class.
170 using TypedValueBase<T>::TypedValueBase;
171 using PropValue::GetPropType;
172
173 std::unique_ptr<PropValue> Clone() const override {
174 return std::unique_ptr<PropValue>{
175 new Derived{*static_cast<const Derived*>(this)}};
176 }
Vitaly Buka79c05e92015-07-29 12:25:37 -0700177
178 static std::unique_ptr<Derived> CreateFromJson(const base::Value& value,
179 const PropType& type,
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700180 ErrorPtr* error) {
Vitaly Buka79c05e92015-07-29 12:25:37 -0700181 T tmp_value;
182 if (!TypedValueFromJson(&value, &type, &tmp_value, error))
183 return nullptr;
184
185 // Only place where invalid value can exist.
186 std::unique_ptr<Derived> result{new Derived{type, tmp_value}};
187 if (!result->GetPropType()->ValidateConstraints(*result, error))
188 return nullptr;
189
190 return result;
191 }
Vitaly Bukac58a8282015-07-29 01:25:20 -0700192};
193
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700194// Value of type Integer.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700195class IntValue final : public TypedValueWithClone<IntValue, int> {
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700196 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700197 using Base::Base;
198 friend class TypedValueWithClone<IntValue, int>;
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700199 IntValue* GetInt() override { return this; }
200 IntValue const* GetInt() const override { return this; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700201};
202
203// Value of type Number.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700204class DoubleValue final : public TypedValueWithClone<DoubleValue, double> {
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700205 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700206 using Base::Base;
207 friend class TypedValueWithClone<DoubleValue, double>;
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700208 DoubleValue* GetDouble() override { return this; }
209 DoubleValue const* GetDouble() const override { return this; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700210};
211
212// Value of type String.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700213class StringValue final : public TypedValueWithClone<StringValue, std::string> {
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700214 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700215 using Base::Base;
216 friend class TypedValueWithClone<StringValue, std::string>;
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700217 StringValue* GetString() override { return this; }
218 StringValue const* GetString() const override { return this; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700219};
220
221// Value of type Boolean.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700222class BooleanValue final : public TypedValueWithClone<BooleanValue, bool> {
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700223 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700224 using Base::Base;
225 friend class TypedValueWithClone<BooleanValue, bool>;
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700226 BooleanValue* GetBoolean() override { return this; }
227 BooleanValue const* GetBoolean() const override { return this; }
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700228};
229
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700230// Value of type Object.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700231class ObjectValue final : public TypedValueWithClone<ObjectValue, ValueMap> {
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700232 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700233 using Base::Base;
234 friend class TypedValueWithClone<ObjectValue, ValueMap>;
Alex Vakulenko5a9e7182014-08-11 15:59:58 -0700235 ObjectValue* GetObject() override { return this; }
236 ObjectValue const* GetObject() const override { return this; }
Alex Vakulenko66ec2922014-06-17 15:30:22 -0700237};
Alex Vakulenko29e64442015-03-20 13:59:19 -0700238
239// Value of type Array.
Vitaly Bukac58a8282015-07-29 01:25:20 -0700240class ArrayValue final : public TypedValueWithClone<ArrayValue, ValueVector> {
Alex Vakulenko29e64442015-03-20 13:59:19 -0700241 public:
Vitaly Buka79c05e92015-07-29 12:25:37 -0700242 using Base::Base;
243 friend class TypedValueWithClone<ArrayValue, ValueVector>;
Alex Vakulenko29e64442015-03-20 13:59:19 -0700244 ArrayValue* GetArray() override { return this; }
245 ArrayValue const* GetArray() const override { return this; }
246};
247
Vitaly Bukab6f015a2015-07-09 14:59:23 -0700248} // namespace weave
Alex Vakulenkoe439a0f2014-05-21 12:26:47 -0700249
Vitaly Buka912b6982015-07-06 11:13:03 -0700250#endif // LIBWEAVE_SRC_COMMANDS_PROP_VALUES_H_