blob: 053e7aa1644412756022d99f2cd0e3243213060c [file] [log] [blame]
Vitaly Buka4615e0d2015-10-14 15:35:12 -07001// Copyright 2015 The Weave Authors. All rights reserved.
Alex Vakulenko7c36b672014-07-16 14:50:58 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Stefan Sauer2d16dfa2015-09-25 17:08:35 +02005#include "src/commands/command_dictionary.h"
Alex Vakulenko7c36b672014-07-16 14:50:58 -07006
7#include <base/values.h>
Vitaly Buka7b382ac2015-08-03 13:50:01 -07008#include <weave/enum_to_string.h>
Alex Vakulenko7c36b672014-07-16 14:50:58 -07009
Stefan Sauer2d16dfa2015-09-25 17:08:35 +020010#include "src/commands/command_definition.h"
11#include "src/commands/schema_constants.h"
12#include "src/string_utils.h"
Alex Vakulenko7c36b672014-07-16 14:50:58 -070013
Vitaly Bukab6f015a2015-07-09 14:59:23 -070014namespace weave {
Alex Vakulenko7c36b672014-07-16 14:50:58 -070015
Alex Vakulenko7c36b672014-07-16 14:50:58 -070016bool CommandDictionary::LoadCommands(const base::DictionaryValue& json,
Alex Vakulenkofd448692014-07-22 07:46:53 -070017 const CommandDictionary* base_commands,
Vitaly Buka0801a1f2015-08-14 10:03:46 -070018 ErrorPtr* error) {
Alex Vakulenko5ef75792015-03-19 15:50:44 -070019 CommandMap new_defs;
Alex Vakulenko7c36b672014-07-16 14:50:58 -070020
21 // |json| contains a list of nested objects with the following structure:
22 // {"<pkg_name>": {"<cmd_name>": {"parameters": {object_schema}}, ...}, ...}
23 // Iterate over packages
24 base::DictionaryValue::Iterator package_iter(json);
25 while (!package_iter.IsAtEnd()) {
Alex Vakulenkofd448692014-07-22 07:46:53 -070026 std::string package_name = package_iter.key();
Alex Vakulenko7c36b672014-07-16 14:50:58 -070027 const base::DictionaryValue* package_value = nullptr;
28 if (!package_iter.value().GetAsDictionary(&package_value)) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -070029 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
30 errors::commands::kTypeMismatch,
31 "Expecting an object for package '%s'",
32 package_name.c_str());
Alex Vakulenko7c36b672014-07-16 14:50:58 -070033 return false;
34 }
35 // Iterate over command definitions within the current package.
36 base::DictionaryValue::Iterator command_iter(*package_value);
37 while (!command_iter.IsAtEnd()) {
Alex Vakulenkofd448692014-07-22 07:46:53 -070038 std::string command_name = command_iter.key();
39 if (command_name.empty()) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -070040 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
41 errors::commands::kInvalidCommandName,
42 "Unnamed command encountered in package '%s'",
43 package_name.c_str());
Alex Vakulenkofd448692014-07-22 07:46:53 -070044 return false;
45 }
Anton Muhin71fb9d52014-11-21 22:22:39 +040046 const base::DictionaryValue* command_def_json = nullptr;
47 if (!command_iter.value().GetAsDictionary(&command_def_json)) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -070048 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
49 errors::commands::kTypeMismatch,
50 "Expecting an object for command '%s'",
51 command_name.c_str());
Alex Vakulenko7c36b672014-07-16 14:50:58 -070052 return false;
53 }
54 // Construct the compound command name as "pkg_name.cmd_name".
Vitaly Buka24d6fd52015-08-13 23:22:48 -070055 std::string full_command_name = Join(".", package_name, command_name);
Alex Vakulenkofd448692014-07-22 07:46:53 -070056
Anton Muhin71fb9d52014-11-21 22:22:39 +040057 const ObjectSchema* base_parameters_def = nullptr;
Vitaly Buka4129dfa2015-04-29 12:16:58 -070058 const ObjectSchema* base_progress_def = nullptr;
Anton Muhin71fb9d52014-11-21 22:22:39 +040059 const ObjectSchema* base_results_def = nullptr;
Alex Vakulenko5e86fee2015-04-17 08:47:45 -070060 // By default make it available to all clients.
61 auto visibility = CommandDefinition::Visibility::GetAll();
Vitaly Buka6fed0532015-05-14 16:57:23 -070062 UserRole minimal_role{UserRole::kUser};
Alex Vakulenkofd448692014-07-22 07:46:53 -070063 if (base_commands) {
Anton Muhincfde8692014-11-25 03:36:59 +040064 auto cmd = base_commands->FindCommand(full_command_name);
Anton Muhin71fb9d52014-11-21 22:22:39 +040065 if (cmd) {
Alex Vakulenko5ef75792015-03-19 15:50:44 -070066 base_parameters_def = cmd->GetParameters();
Vitaly Buka4129dfa2015-04-29 12:16:58 -070067 base_progress_def = cmd->GetProgress();
Alex Vakulenko5ef75792015-03-19 15:50:44 -070068 base_results_def = cmd->GetResults();
Alex Vakulenko5e86fee2015-04-17 08:47:45 -070069 visibility = cmd->GetVisibility();
Vitaly Buka6fed0532015-05-14 16:57:23 -070070 minimal_role = cmd->GetMinimalRole();
Anton Muhin71fb9d52014-11-21 22:22:39 +040071 }
Alex Vakulenkofd448692014-07-22 07:46:53 -070072
73 // If the base command dictionary was provided but the command was not
74 // found in it, this must be a custom (vendor) command. GCD spec states
75 // that all custom command names must begin with "_". Let's enforce
76 // this rule here.
Anton Muhin71fb9d52014-11-21 22:22:39 +040077 if (!cmd) {
Alex Vakulenkofd448692014-07-22 07:46:53 -070078 if (command_name.front() != '_') {
Vitaly Buka0801a1f2015-08-14 10:03:46 -070079 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
80 errors::commands::kInvalidCommandName,
81 "The name of custom command '%s' in package '%s'"
82 " must start with '_'",
83 command_name.c_str(), package_name.c_str());
Alex Vakulenkofd448692014-07-22 07:46:53 -070084 return false;
85 }
86 }
87 }
88
Anton Muhin71fb9d52014-11-21 22:22:39 +040089 auto parameters_schema = BuildObjectSchema(
Vitaly Bukaa647c852015-07-06 14:51:01 -070090 command_def_json, commands::attributes::kCommand_Parameters,
91 base_parameters_def, full_command_name, error);
Anton Muhin71fb9d52014-11-21 22:22:39 +040092 if (!parameters_schema)
Alex Vakulenko7c36b672014-07-16 14:50:58 -070093 return false;
Anton Muhin71fb9d52014-11-21 22:22:39 +040094
Vitaly Buka4129dfa2015-04-29 12:16:58 -070095 auto progress_schema = BuildObjectSchema(
96 command_def_json, commands::attributes::kCommand_Progress,
97 base_progress_def, full_command_name, error);
98 if (!progress_schema)
99 return false;
100
Anton Muhin71fb9d52014-11-21 22:22:39 +0400101 auto results_schema = BuildObjectSchema(
Vitaly Bukaa647c852015-07-06 14:51:01 -0700102 command_def_json, commands::attributes::kCommand_Results,
103 base_results_def, full_command_name, error);
Anton Muhin71fb9d52014-11-21 22:22:39 +0400104 if (!results_schema)
105 return false;
106
Alex Vakulenko5e86fee2015-04-17 08:47:45 -0700107 std::string value;
Vitaly Buka6fed0532015-05-14 16:57:23 -0700108 if (command_def_json->GetString(commands::attributes::kCommand_Visibility,
109 &value)) {
Alex Vakulenko5e86fee2015-04-17 08:47:45 -0700110 if (!visibility.FromString(value, error)) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700111 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
112 errors::commands::kInvalidCommandVisibility,
113 "Error parsing command '%s'",
114 full_command_name.c_str());
Alex Vakulenko5e86fee2015-04-17 08:47:45 -0700115 return false;
116 }
117 }
118
Vitaly Buka6fed0532015-05-14 16:57:23 -0700119 if (command_def_json->GetString(commands::attributes::kCommand_Role,
120 &value)) {
Vitaly Buka7197c1a2015-07-17 14:48:30 -0700121 if (!StringToEnum(value, &minimal_role)) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700122 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
123 errors::commands::kInvalidPropValue,
124 "Invalid role: '%s'", value.c_str());
125 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
126 errors::commands::kInvalidMinimalRole,
127 "Error parsing command '%s'",
128 full_command_name.c_str());
Vitaly Buka6fed0532015-05-14 16:57:23 -0700129 return false;
130 }
131 }
132
Vitaly Buka4ebd3292015-09-23 18:04:17 -0700133 std::unique_ptr<CommandDefinition> command_def{new CommandDefinition{
Vitaly Buka453c4dd2015-10-04 18:01:50 -0700134 std::move(parameters_schema), std::move(progress_schema),
Vitaly Buka4ebd3292015-09-23 18:04:17 -0700135 std::move(results_schema)}};
Alex Vakulenko5e86fee2015-04-17 08:47:45 -0700136 command_def->SetVisibility(visibility);
Vitaly Buka6fed0532015-05-14 16:57:23 -0700137 command_def->SetMinimalRole(minimal_role);
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700138 new_defs.emplace(full_command_name, std::move(command_def));
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700139
140 command_iter.Advance();
141 }
142 package_iter.Advance();
143 }
144
145 // Verify that newly loaded command definitions do not override existing
146 // definitions in another category. This is unlikely, but we don't want to let
147 // one vendor daemon to define the same commands already handled by another
148 // daemon on the same device.
149 for (const auto& pair : new_defs) {
150 auto iter = definitions_.find(pair.first);
Vitaly Buka453c4dd2015-10-04 18:01:50 -0700151 CHECK(iter == definitions_.end()) << "Definition for command '"
152 << pair.first
153 << "' overrides an earlier definition";
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700154 }
155
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700156 // Insert new definitions into the global map.
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700157 for (auto& pair : new_defs)
158 definitions_.emplace(pair.first, std::move(pair.second));
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700159 return true;
160}
161
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700162std::unique_ptr<ObjectSchema> CommandDictionary::BuildObjectSchema(
Anton Muhin71fb9d52014-11-21 22:22:39 +0400163 const base::DictionaryValue* command_def_json,
164 const char* property_name,
165 const ObjectSchema* base_def,
166 const std::string& command_name,
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700167 ErrorPtr* error) {
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700168 auto object_schema = ObjectSchema::Create();
Anton Muhin71fb9d52014-11-21 22:22:39 +0400169
170 const base::DictionaryValue* schema_def = nullptr;
171 if (!command_def_json->GetDictionaryWithoutPathExpansion(property_name,
172 &schema_def)) {
Vitaly Buka4c326542015-05-31 21:48:25 -0700173 if (base_def)
174 return base_def->Clone();
Vitaly Buka10cf5c72015-04-29 12:05:10 -0700175 return object_schema;
Anton Muhin71fb9d52014-11-21 22:22:39 +0400176 }
177
178 if (!object_schema->FromJson(schema_def, base_def, error)) {
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700179 Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain,
180 errors::commands::kInvalidObjectSchema,
181 "Invalid definition for command '%s'",
182 command_name.c_str());
Anton Muhin71fb9d52014-11-21 22:22:39 +0400183 return {};
184 }
185
186 return object_schema;
187}
188
Alex Vakulenko45109442014-07-29 11:07:10 -0700189std::unique_ptr<base::DictionaryValue> CommandDictionary::GetCommandsAsJson(
Alex Vakulenko9ea5a322015-04-17 15:35:34 -0700190 const std::function<bool(const CommandDefinition*)>& filter,
191 bool full_schema,
Vitaly Buka0801a1f2015-08-14 10:03:46 -0700192 ErrorPtr* error) const {
Alex Vakulenko45109442014-07-29 11:07:10 -0700193 std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue);
194 for (const auto& pair : definitions_) {
Alex Vakulenko9ea5a322015-04-17 15:35:34 -0700195 // Check if the command definition has the desired visibility.
196 // If not, then skip it.
197 if (!filter(pair.second.get()))
198 continue;
199
Vitaly Bukab6b49e52015-05-01 10:53:06 -0700200 std::unique_ptr<base::DictionaryValue> parameters =
Vitaly Buka6942e1f2015-07-28 15:33:55 -0700201 pair.second->GetParameters()->ToJson(full_schema, true);
202 CHECK(parameters);
Vitaly Bukab6b49e52015-05-01 10:53:06 -0700203 // Progress and results are not part of public commandDefs.
204
Vitaly Buka24d6fd52015-08-13 23:22:48 -0700205 auto parts = SplitAtFirst(pair.first, ".", true);
206 const std::string& package_name = parts.first;
207 const std::string& command_name = parts.second;
208
Alex Vakulenko45109442014-07-29 11:07:10 -0700209 base::DictionaryValue* package = nullptr;
210 if (!dict->GetDictionaryWithoutPathExpansion(package_name, &package)) {
211 // If this is the first time we encounter this package, create a JSON
212 // object for it.
213 package = new base::DictionaryValue;
214 dict->SetWithoutPathExpansion(package_name, package);
215 }
216 base::DictionaryValue* command_def = new base::DictionaryValue;
Alex Vakulenko5e86fee2015-04-17 08:47:45 -0700217 command_def->Set(commands::attributes::kCommand_Parameters,
Vitaly Bukab6b49e52015-05-01 10:53:06 -0700218 parameters.release());
Vitaly Buka6fed0532015-05-14 16:57:23 -0700219 command_def->SetString(commands::attributes::kCommand_Role,
Vitaly Buka7197c1a2015-07-17 14:48:30 -0700220 EnumToString(pair.second->GetMinimalRole()));
Alex Vakulenko45109442014-07-29 11:07:10 -0700221 package->SetWithoutPathExpansion(command_name, command_def);
222 }
223 return dict;
224}
225
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700226const CommandDefinition* CommandDictionary::FindCommand(
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700227 const std::string& command_name) const {
228 auto pair = definitions_.find(command_name);
Alex Vakulenko5ef75792015-03-19 15:50:44 -0700229 return (pair != definitions_.end()) ? pair->second.get() : nullptr;
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700230}
231
Alex Vakulenkoe03af6d2015-04-20 11:00:54 -0700232CommandDefinition* CommandDictionary::FindCommand(
233 const std::string& command_name) {
234 auto pair = definitions_.find(command_name);
235 return (pair != definitions_.end()) ? pair->second.get() : nullptr;
236}
237
Alex Vakulenko7c36b672014-07-16 14:50:58 -0700238void CommandDictionary::Clear() {
239 definitions_.clear();
240}
241
Vitaly Bukab6f015a2015-07-09 14:59:23 -0700242} // namespace weave