Vitaly Buka | 4615e0d | 2015-10-14 15:35:12 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Weave Authors. All rights reserved. |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
Stefan Sauer | 2d16dfa | 2015-09-25 17:08:35 +0200 | [diff] [blame] | 5 | #include "src/commands/command_dictionary.h" |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 6 | |
| 7 | #include <base/values.h> |
Vitaly Buka | 7b382ac | 2015-08-03 13:50:01 -0700 | [diff] [blame] | 8 | #include <weave/enum_to_string.h> |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 9 | |
Stefan Sauer | 2d16dfa | 2015-09-25 17:08:35 +0200 | [diff] [blame] | 10 | #include "src/commands/command_definition.h" |
| 11 | #include "src/commands/schema_constants.h" |
| 12 | #include "src/string_utils.h" |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 13 | |
Vitaly Buka | b6f015a | 2015-07-09 14:59:23 -0700 | [diff] [blame] | 14 | namespace weave { |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 15 | |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 16 | bool CommandDictionary::LoadCommands(const base::DictionaryValue& json, |
Alex Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 17 | const CommandDictionary* base_commands, |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 18 | ErrorPtr* error) { |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 19 | CommandMap new_defs; |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 20 | |
| 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 Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 26 | std::string package_name = package_iter.key(); |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 27 | const base::DictionaryValue* package_value = nullptr; |
| 28 | if (!package_iter.value().GetAsDictionary(&package_value)) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 29 | 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 Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 33 | 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 Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 38 | std::string command_name = command_iter.key(); |
| 39 | if (command_name.empty()) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 40 | 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 Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 44 | return false; |
| 45 | } |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 46 | const base::DictionaryValue* command_def_json = nullptr; |
| 47 | if (!command_iter.value().GetAsDictionary(&command_def_json)) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 48 | 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 Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 52 | return false; |
| 53 | } |
| 54 | // Construct the compound command name as "pkg_name.cmd_name". |
Vitaly Buka | 24d6fd5 | 2015-08-13 23:22:48 -0700 | [diff] [blame] | 55 | std::string full_command_name = Join(".", package_name, command_name); |
Alex Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 56 | |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 57 | const ObjectSchema* base_parameters_def = nullptr; |
Vitaly Buka | 4129dfa | 2015-04-29 12:16:58 -0700 | [diff] [blame] | 58 | const ObjectSchema* base_progress_def = nullptr; |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 59 | const ObjectSchema* base_results_def = nullptr; |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 60 | // By default make it available to all clients. |
| 61 | auto visibility = CommandDefinition::Visibility::GetAll(); |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 62 | UserRole minimal_role{UserRole::kUser}; |
Alex Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 63 | if (base_commands) { |
Anton Muhin | cfde869 | 2014-11-25 03:36:59 +0400 | [diff] [blame] | 64 | auto cmd = base_commands->FindCommand(full_command_name); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 65 | if (cmd) { |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 66 | base_parameters_def = cmd->GetParameters(); |
Vitaly Buka | 4129dfa | 2015-04-29 12:16:58 -0700 | [diff] [blame] | 67 | base_progress_def = cmd->GetProgress(); |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 68 | base_results_def = cmd->GetResults(); |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 69 | visibility = cmd->GetVisibility(); |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 70 | minimal_role = cmd->GetMinimalRole(); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 71 | } |
Alex Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 72 | |
| 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 Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 77 | if (!cmd) { |
Alex Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 78 | if (command_name.front() != '_') { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 79 | 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 Vakulenko | fd44869 | 2014-07-22 07:46:53 -0700 | [diff] [blame] | 84 | return false; |
| 85 | } |
| 86 | } |
| 87 | } |
| 88 | |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 89 | auto parameters_schema = BuildObjectSchema( |
Vitaly Buka | a647c85 | 2015-07-06 14:51:01 -0700 | [diff] [blame] | 90 | command_def_json, commands::attributes::kCommand_Parameters, |
| 91 | base_parameters_def, full_command_name, error); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 92 | if (!parameters_schema) |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 93 | return false; |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 94 | |
Vitaly Buka | 4129dfa | 2015-04-29 12:16:58 -0700 | [diff] [blame] | 95 | 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 Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 101 | auto results_schema = BuildObjectSchema( |
Vitaly Buka | a647c85 | 2015-07-06 14:51:01 -0700 | [diff] [blame] | 102 | command_def_json, commands::attributes::kCommand_Results, |
| 103 | base_results_def, full_command_name, error); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 104 | if (!results_schema) |
| 105 | return false; |
| 106 | |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 107 | std::string value; |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 108 | if (command_def_json->GetString(commands::attributes::kCommand_Visibility, |
| 109 | &value)) { |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 110 | if (!visibility.FromString(value, error)) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 111 | Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain, |
| 112 | errors::commands::kInvalidCommandVisibility, |
| 113 | "Error parsing command '%s'", |
| 114 | full_command_name.c_str()); |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 115 | return false; |
| 116 | } |
| 117 | } |
| 118 | |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 119 | if (command_def_json->GetString(commands::attributes::kCommand_Role, |
| 120 | &value)) { |
Vitaly Buka | 7197c1a | 2015-07-17 14:48:30 -0700 | [diff] [blame] | 121 | if (!StringToEnum(value, &minimal_role)) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 122 | 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 Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 129 | return false; |
| 130 | } |
| 131 | } |
| 132 | |
Vitaly Buka | 4ebd329 | 2015-09-23 18:04:17 -0700 | [diff] [blame] | 133 | std::unique_ptr<CommandDefinition> command_def{new CommandDefinition{ |
Vitaly Buka | 453c4dd | 2015-10-04 18:01:50 -0700 | [diff] [blame] | 134 | std::move(parameters_schema), std::move(progress_schema), |
Vitaly Buka | 4ebd329 | 2015-09-23 18:04:17 -0700 | [diff] [blame] | 135 | std::move(results_schema)}}; |
Alex Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 136 | command_def->SetVisibility(visibility); |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 137 | command_def->SetMinimalRole(minimal_role); |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 138 | new_defs.emplace(full_command_name, std::move(command_def)); |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 139 | |
| 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 Buka | 453c4dd | 2015-10-04 18:01:50 -0700 | [diff] [blame] | 151 | CHECK(iter == definitions_.end()) << "Definition for command '" |
| 152 | << pair.first |
| 153 | << "' overrides an earlier definition"; |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 154 | } |
| 155 | |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 156 | // Insert new definitions into the global map. |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 157 | for (auto& pair : new_defs) |
| 158 | definitions_.emplace(pair.first, std::move(pair.second)); |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 159 | return true; |
| 160 | } |
| 161 | |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 162 | std::unique_ptr<ObjectSchema> CommandDictionary::BuildObjectSchema( |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 163 | const base::DictionaryValue* command_def_json, |
| 164 | const char* property_name, |
| 165 | const ObjectSchema* base_def, |
| 166 | const std::string& command_name, |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 167 | ErrorPtr* error) { |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 168 | auto object_schema = ObjectSchema::Create(); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 169 | |
| 170 | const base::DictionaryValue* schema_def = nullptr; |
| 171 | if (!command_def_json->GetDictionaryWithoutPathExpansion(property_name, |
| 172 | &schema_def)) { |
Vitaly Buka | 4c32654 | 2015-05-31 21:48:25 -0700 | [diff] [blame] | 173 | if (base_def) |
| 174 | return base_def->Clone(); |
Vitaly Buka | 10cf5c7 | 2015-04-29 12:05:10 -0700 | [diff] [blame] | 175 | return object_schema; |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 176 | } |
| 177 | |
| 178 | if (!object_schema->FromJson(schema_def, base_def, error)) { |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 179 | Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain, |
| 180 | errors::commands::kInvalidObjectSchema, |
| 181 | "Invalid definition for command '%s'", |
| 182 | command_name.c_str()); |
Anton Muhin | 71fb9d5 | 2014-11-21 22:22:39 +0400 | [diff] [blame] | 183 | return {}; |
| 184 | } |
| 185 | |
| 186 | return object_schema; |
| 187 | } |
| 188 | |
Alex Vakulenko | 4510944 | 2014-07-29 11:07:10 -0700 | [diff] [blame] | 189 | std::unique_ptr<base::DictionaryValue> CommandDictionary::GetCommandsAsJson( |
Alex Vakulenko | 9ea5a32 | 2015-04-17 15:35:34 -0700 | [diff] [blame] | 190 | const std::function<bool(const CommandDefinition*)>& filter, |
| 191 | bool full_schema, |
Vitaly Buka | 0801a1f | 2015-08-14 10:03:46 -0700 | [diff] [blame] | 192 | ErrorPtr* error) const { |
Alex Vakulenko | 4510944 | 2014-07-29 11:07:10 -0700 | [diff] [blame] | 193 | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue); |
| 194 | for (const auto& pair : definitions_) { |
Alex Vakulenko | 9ea5a32 | 2015-04-17 15:35:34 -0700 | [diff] [blame] | 195 | // 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 Buka | b6b49e5 | 2015-05-01 10:53:06 -0700 | [diff] [blame] | 200 | std::unique_ptr<base::DictionaryValue> parameters = |
Vitaly Buka | 6942e1f | 2015-07-28 15:33:55 -0700 | [diff] [blame] | 201 | pair.second->GetParameters()->ToJson(full_schema, true); |
| 202 | CHECK(parameters); |
Vitaly Buka | b6b49e5 | 2015-05-01 10:53:06 -0700 | [diff] [blame] | 203 | // Progress and results are not part of public commandDefs. |
| 204 | |
Vitaly Buka | 24d6fd5 | 2015-08-13 23:22:48 -0700 | [diff] [blame] | 205 | auto parts = SplitAtFirst(pair.first, ".", true); |
| 206 | const std::string& package_name = parts.first; |
| 207 | const std::string& command_name = parts.second; |
| 208 | |
Alex Vakulenko | 4510944 | 2014-07-29 11:07:10 -0700 | [diff] [blame] | 209 | 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 Vakulenko | 5e86fee | 2015-04-17 08:47:45 -0700 | [diff] [blame] | 217 | command_def->Set(commands::attributes::kCommand_Parameters, |
Vitaly Buka | b6b49e5 | 2015-05-01 10:53:06 -0700 | [diff] [blame] | 218 | parameters.release()); |
Vitaly Buka | 6fed053 | 2015-05-14 16:57:23 -0700 | [diff] [blame] | 219 | command_def->SetString(commands::attributes::kCommand_Role, |
Vitaly Buka | 7197c1a | 2015-07-17 14:48:30 -0700 | [diff] [blame] | 220 | EnumToString(pair.second->GetMinimalRole())); |
Alex Vakulenko | 4510944 | 2014-07-29 11:07:10 -0700 | [diff] [blame] | 221 | package->SetWithoutPathExpansion(command_name, command_def); |
| 222 | } |
| 223 | return dict; |
| 224 | } |
| 225 | |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 226 | const CommandDefinition* CommandDictionary::FindCommand( |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 227 | const std::string& command_name) const { |
| 228 | auto pair = definitions_.find(command_name); |
Alex Vakulenko | 5ef7579 | 2015-03-19 15:50:44 -0700 | [diff] [blame] | 229 | return (pair != definitions_.end()) ? pair->second.get() : nullptr; |
Alex Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 230 | } |
| 231 | |
Alex Vakulenko | e03af6d | 2015-04-20 11:00:54 -0700 | [diff] [blame] | 232 | CommandDefinition* 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 Vakulenko | 7c36b67 | 2014-07-16 14:50:58 -0700 | [diff] [blame] | 238 | void CommandDictionary::Clear() { |
| 239 | definitions_.clear(); |
| 240 | } |
| 241 | |
Vitaly Buka | b6f015a | 2015-07-09 14:59:23 -0700 | [diff] [blame] | 242 | } // namespace weave |