buffet: Add command definition support for GCD command manager
Rudimentary framework of classes to represent GCD command manager
and list of registered device command definitions/schemas.
BUG=chromium:374861
TEST=USE=buffet P2_TEST_FILTER="buffet::*" FEATURES=test emerge-link platform
Change-Id: I3cd3d776879e8bd506aecd20df5cd89c65247d35
Reviewed-on: https://chromium-review.googlesource.com/208464
Tested-by: Alex Vakulenko <avakulenko@chromium.org>
Reviewed-by: Christopher Wiley <wiley@chromium.org>
Reviewed-by: Alex Vakulenko <avakulenko@chromium.org>
Commit-Queue: Alex Vakulenko <avakulenko@chromium.org>
diff --git a/buffet/commands/command_dictionary.cc b/buffet/commands/command_dictionary.cc
new file mode 100644
index 0000000..69f9e2b
--- /dev/null
+++ b/buffet/commands/command_dictionary.cc
@@ -0,0 +1,125 @@
+// Copyright 2014 The Chromium OS Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "buffet/commands/command_dictionary.h"
+
+#include <base/values.h>
+
+#include "buffet/commands/command_definition.h"
+#include "buffet/commands/schema_constants.h"
+#include "buffet/string_utils.h"
+
+namespace buffet {
+
+std::vector<std::string> CommandDictionary::GetCommandNamesByCategory(
+ const std::string& category) const {
+ std::vector<std::string> names;
+ for (const auto& pair : definitions_) {
+ if (pair.second->GetCategory() == category)
+ names.push_back(pair.first);
+ }
+ return names;
+}
+
+bool CommandDictionary::LoadCommands(const base::DictionaryValue& json,
+ const std::string& category,
+ ErrorPtr* error) {
+ std::map<std::string, std::shared_ptr<const CommandDefinition>> new_defs;
+
+ // |json| contains a list of nested objects with the following structure:
+ // {"<pkg_name>": {"<cmd_name>": {"parameters": {object_schema}}, ...}, ...}
+ // Iterate over packages
+ base::DictionaryValue::Iterator package_iter(json);
+ while (!package_iter.IsAtEnd()) {
+ std::string package = package_iter.key();
+ const base::DictionaryValue* package_value = nullptr;
+ if (!package_iter.value().GetAsDictionary(&package_value)) {
+ Error::AddToPrintf(error, commands::errors::kDomain,
+ commands::errors::kTypeMismatch,
+ "Expecting an object for package '%s'",
+ package.c_str());
+ return false;
+ }
+ // Iterate over command definitions within the current package.
+ base::DictionaryValue::Iterator command_iter(*package_value);
+ while (!command_iter.IsAtEnd()) {
+ std::string command = command_iter.key();
+ const base::DictionaryValue* command_value = nullptr;
+ if (!command_iter.value().GetAsDictionary(&command_value)) {
+ Error::AddToPrintf(error, commands::errors::kDomain,
+ commands::errors::kTypeMismatch,
+ "Expecting an object for command '%s'",
+ command.c_str());
+ return false;
+ }
+ // Construct the compound command name as "pkg_name.cmd_name".
+ std::string command_name = string_utils::Join('.', package, command);
+ // Get the "parameters" definition of the command and read it into
+ // an object schema.
+ const base::DictionaryValue* command_schema_def = nullptr;
+ if (!command_value->GetDictionaryWithoutPathExpansion(
+ commands::attributes::kCommand_Parameters, &command_schema_def)) {
+ Error::AddToPrintf(error, commands::errors::kDomain,
+ commands::errors::kPropertyMissing,
+ "Command definition '%s' is missing property '%s'",
+ command_name.c_str(),
+ commands::attributes::kCommand_Parameters);
+ return false;
+ }
+ auto command_schema = std::make_shared<ObjectSchema>();
+ if (!command_schema->FromJson(command_schema_def, nullptr, error)) {
+ Error::AddToPrintf(error, commands::errors::kDomain,
+ commands::errors::kInvalidObjectSchema,
+ "Invalid definition for command '%s'",
+ command_name.c_str());
+ return false;
+ }
+ auto command_def = std::make_shared<CommandDefinition>(category,
+ command_schema);
+ new_defs.insert(std::make_pair(command_name, command_def));
+
+ command_iter.Advance();
+ }
+ package_iter.Advance();
+ }
+
+ // Verify that newly loaded command definitions do not override existing
+ // definitions in another category. This is unlikely, but we don't want to let
+ // one vendor daemon to define the same commands already handled by another
+ // daemon on the same device.
+ for (const auto& pair : new_defs) {
+ auto iter = definitions_.find(pair.first);
+ if (iter != definitions_.end()) {
+ Error::AddToPrintf(error, commands::errors::kDomain,
+ commands::errors::kDuplicateCommandDef,
+ "Definition for command '%s' overrides an earlier "
+ "definition in category '%s'",
+ pair.first.c_str(),
+ iter->second->GetCategory().c_str());
+ return false;
+ }
+ }
+
+ // Now that we successfully loaded all the command definitions,
+ // remove previous definitions of commands from the same category.
+ std::vector<std::string> names = GetCommandNamesByCategory(category);
+ for (const std::string& name : names)
+ definitions_.erase(name);
+
+ // Insert new definitions into the global map.
+ definitions_.insert(new_defs.begin(), new_defs.end());
+ return true;
+}
+
+const CommandDefinition* CommandDictionary::FindCommand(
+ const std::string& command_name) const {
+ auto pair = definitions_.find(command_name);
+ return (pair != definitions_.end()) ? pair->second.get() : nullptr;
+}
+
+void CommandDictionary::Clear() {
+ definitions_.clear();
+}
+
+} // namespace buffet