|  | // 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_manager.h" | 
|  |  | 
|  | #include <base/files/file_enumerator.h> | 
|  | #include <base/values.h> | 
|  | #include <chromeos/dbus/exported_object_manager.h> | 
|  | #include <chromeos/errors/error.h> | 
|  |  | 
|  | #include "buffet/commands/schema_constants.h" | 
|  | #include "buffet/utils.h" | 
|  |  | 
|  | using chromeos::dbus_utils::ExportedObjectManager; | 
|  |  | 
|  | namespace buffet { | 
|  |  | 
|  | CommandManager::CommandManager() | 
|  | : CommandManager(base::WeakPtr<ExportedObjectManager>{}) { | 
|  | } | 
|  |  | 
|  | CommandManager::CommandManager( | 
|  | const base::WeakPtr<ExportedObjectManager>& object_manager) | 
|  | : command_dispatcher_(object_manager) { | 
|  | command_queue_.AddOnCommandAddedCallback( | 
|  | base::Bind(&DBusCommandDispacher::OnCommandAdded, | 
|  | base::Unretained(&command_dispatcher_))); | 
|  | } | 
|  |  | 
|  | const CommandDictionary& CommandManager::GetCommandDictionary() const { | 
|  | return dictionary_; | 
|  | } | 
|  |  | 
|  | bool CommandManager::LoadBaseCommands(const base::DictionaryValue& json, | 
|  | chromeos::ErrorPtr* error) { | 
|  | return base_dictionary_.LoadCommands(json, "", nullptr, error); | 
|  | } | 
|  |  | 
|  | bool CommandManager::LoadBaseCommands(const base::FilePath& json_file_path, | 
|  | chromeos::ErrorPtr* error) { | 
|  | std::unique_ptr<const base::DictionaryValue> json = LoadJsonDict( | 
|  | json_file_path, error); | 
|  | if (!json) | 
|  | return false; | 
|  | return LoadBaseCommands(*json, error); | 
|  | } | 
|  |  | 
|  | bool CommandManager::LoadCommands(const base::DictionaryValue& json, | 
|  | const std::string& category, | 
|  | chromeos::ErrorPtr* error) { | 
|  | bool result = | 
|  | dictionary_.LoadCommands(json, category, &base_dictionary_, error); | 
|  | for (const auto& cb : on_command_changed_) | 
|  | cb.Run(); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool CommandManager::LoadCommands(const base::FilePath& json_file_path, | 
|  | chromeos::ErrorPtr* error) { | 
|  | std::unique_ptr<const base::DictionaryValue> json = LoadJsonDict( | 
|  | json_file_path, error); | 
|  | if (!json) | 
|  | return false; | 
|  | std::string category = json_file_path.BaseName().RemoveExtension().value(); | 
|  | return LoadCommands(*json, category, error); | 
|  | } | 
|  |  | 
|  | void CommandManager::Startup(const base::FilePath& definitions_path, | 
|  | const base::FilePath& test_definitions_path) { | 
|  | LOG(INFO) << "Initializing CommandManager."; | 
|  | // Load global standard GCD command dictionary. | 
|  | base::FilePath base_command_file{definitions_path.Append("gcd.json")}; | 
|  | LOG(INFO) << "Loading standard commands from " << base_command_file.value(); | 
|  | CHECK(LoadBaseCommands(base_command_file, nullptr)) | 
|  | << "Failed to load the standard command definitions."; | 
|  |  | 
|  | auto LoadPackages = [this](const base::FilePath& root, | 
|  | const base::FilePath::StringType& pattern) { | 
|  | base::FilePath device_command_dir{root.Append("commands")}; | 
|  | VLOG(2) << "Looking for commands in " << root.value(); | 
|  | base::FileEnumerator enumerator(device_command_dir, false, | 
|  | base::FileEnumerator::FILES, | 
|  | pattern); | 
|  | base::FilePath json_file_path = enumerator.Next(); | 
|  | while (!json_file_path.empty()) { | 
|  | LOG(INFO) << "Loading command schema from " << json_file_path.value(); | 
|  | CHECK(this->LoadCommands(json_file_path, nullptr)) | 
|  | << "Failed to load the command definition file."; | 
|  | json_file_path = enumerator.Next(); | 
|  | } | 
|  | }; | 
|  | LoadPackages(definitions_path, FILE_PATH_LITERAL("*.json")); | 
|  | LoadPackages(test_definitions_path, FILE_PATH_LITERAL("*test.json")); | 
|  | } | 
|  |  | 
|  | void CommandManager::AddCommand( | 
|  | std::unique_ptr<CommandInstance> command_instance) { | 
|  | command_queue_.Add(std::move(command_instance)); | 
|  | } | 
|  |  | 
|  | CommandInstance* CommandManager::FindCommand(const std::string& id) const { | 
|  | return command_queue_.Find(id); | 
|  | } | 
|  |  | 
|  | bool CommandManager::SetCommandVisibility( | 
|  | const std::vector<std::string>& command_names, | 
|  | CommandDefinition::Visibility visibility, | 
|  | chromeos::ErrorPtr* error) { | 
|  | if (command_names.empty()) | 
|  | return true; | 
|  |  | 
|  | std::vector<CommandDefinition*> definitions; | 
|  | definitions.reserve(command_names.size()); | 
|  |  | 
|  | // Find/validate command definitions first. | 
|  | for (const std::string& name : command_names) { | 
|  | CommandDefinition* def = dictionary_.FindCommand(name); | 
|  | if (!def) { | 
|  | chromeos::Error::AddToPrintf(error, FROM_HERE, errors::commands::kDomain, | 
|  | errors::commands::kInvalidCommandName, | 
|  | "Command '%s' is unknown", name.c_str()); | 
|  | return false; | 
|  | } | 
|  | definitions.push_back(def); | 
|  | } | 
|  |  | 
|  | // Now that we know that all the command names were valid, | 
|  | // update the respective commands' visibility. | 
|  | for (CommandDefinition* def : definitions) | 
|  | def->SetVisibility(visibility); | 
|  | for (const auto& cb : on_command_changed_) | 
|  | cb.Run(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | }  // namespace buffet |