| // Copyright 2015 The Weave 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 "examples/daemon/common/daemon.h" |
| |
| #include <weave/device.h> |
| #include <weave/provider/task_runner.h> |
| |
| #include <base/bind.h> |
| #include <base/memory/weak_ptr.h> |
| |
| // SampleHandler is a command handler example. |
| // It implements the following commands: |
| // - _hello: handle a command with an argument and set its results. |
| // - _ping: update device state. |
| // - _countdown: handle long running command and report progress. |
| class SampleHandler { |
| public: |
| SampleHandler(weave::provider::TaskRunner* task_runner) |
| : task_runner_{task_runner} {} |
| void Register(weave::Device* device) { |
| device_ = device; |
| |
| device->AddCommandDefinitionsFromJson(R"({ |
| "_sample": { |
| "_hello": { |
| "minimalRole": "user", |
| "parameters": { |
| "_name": {"type": "string"} |
| } |
| }, |
| "_ping": { |
| "minimalRole": "user" |
| }, |
| "_countdown": { |
| "minimalRole": "user", |
| "parameters": { |
| "_seconds": {"type": "integer", "minimum": 1, "maximum": 25} |
| } |
| } |
| } |
| })"); |
| |
| device->AddStateDefinitionsFromJson(R"({ |
| "_sample": {"_ping_count": {"type": "integer"}} |
| })"); |
| |
| device->SetStatePropertiesFromJson(R"({ |
| "_sample": {"_ping_count": 0} |
| })", |
| nullptr); |
| |
| device->AddCommandHandler("_sample._hello", |
| base::Bind(&SampleHandler::OnHelloCommand, |
| weak_ptr_factory_.GetWeakPtr())); |
| device->AddCommandHandler("_sample._ping", |
| base::Bind(&SampleHandler::OnPingCommand, |
| weak_ptr_factory_.GetWeakPtr())); |
| device->AddCommandHandler("_sample._countdown", |
| base::Bind(&SampleHandler::OnCountdownCommand, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| private: |
| void OnHelloCommand(const std::weak_ptr<weave::Command>& command) { |
| auto cmd = command.lock(); |
| if (!cmd) |
| return; |
| LOG(INFO) << "received command: " << cmd->GetName(); |
| |
| const auto& params = cmd->GetParameters(); |
| std::string name; |
| if (!params.GetString("_name", &name)) { |
| weave::ErrorPtr error; |
| weave::Error::AddTo(&error, FROM_HERE, "example", |
| "invalid_parameter_value", "Name is missing"); |
| cmd->Abort(error.get(), nullptr); |
| return; |
| } |
| |
| base::DictionaryValue result; |
| result.SetString("_reply", "Hello " + name); |
| cmd->Complete(result, nullptr); |
| LOG(INFO) << cmd->GetName() << " command finished: " << result; |
| } |
| |
| void OnPingCommand(const std::weak_ptr<weave::Command>& command) { |
| auto cmd = command.lock(); |
| if (!cmd) |
| return; |
| LOG(INFO) << "received command: " << cmd->GetName(); |
| |
| base::DictionaryValue state; |
| state.SetInteger("_sample._ping_count", ++ping_count_); |
| device_->SetStateProperties(state, nullptr); |
| LOG(INFO) << "New state: " << device_->GetState(); |
| |
| base::DictionaryValue result; |
| cmd->Complete(result, nullptr); |
| |
| LOG(INFO) << cmd->GetName() << " command finished: " << result; |
| } |
| |
| void OnCountdownCommand(const std::weak_ptr<weave::Command>& command) { |
| auto cmd = command.lock(); |
| if (!cmd) |
| return; |
| LOG(INFO) << "received command: " << cmd->GetName(); |
| |
| const auto& params = cmd->GetParameters(); |
| int seconds; |
| if (!params.GetInteger("_seconds", &seconds)) |
| seconds = 10; |
| |
| LOG(INFO) << "starting countdown"; |
| DoTick(cmd, seconds); |
| } |
| |
| void DoTick(const std::weak_ptr<weave::Command>& command, int seconds) { |
| auto cmd = command.lock(); |
| if (!cmd) |
| return; |
| |
| if (seconds > 0) { |
| const auto& params = cmd->GetParameters(); |
| std::string todo; |
| params.GetString("_todo", &todo); |
| LOG(INFO) << "countdown tick: " << seconds << " seconds left"; |
| |
| base::DictionaryValue progress; |
| progress.SetInteger("_seconds_left", seconds); |
| cmd->SetProgress(progress, nullptr); |
| task_runner_->PostDelayedTask( |
| FROM_HERE, |
| base::Bind(&SampleHandler::DoTick, weak_ptr_factory_.GetWeakPtr(), |
| command, --seconds), |
| base::TimeDelta::FromSeconds(1)); |
| return; |
| } |
| |
| base::DictionaryValue result; |
| cmd->Complete(result, nullptr); |
| LOG(INFO) << "countdown finished"; |
| LOG(INFO) << cmd->GetName() << " command finished: " << result; |
| } |
| |
| weave::Device* device_{nullptr}; |
| weave::provider::TaskRunner* task_runner_{nullptr}; |
| |
| int ping_count_{0}; |
| base::WeakPtrFactory<SampleHandler> weak_ptr_factory_{this}; |
| }; |
| |
| int main(int argc, char** argv) { |
| Daemon::Options opts; |
| if (!opts.Parse(argc, argv)) { |
| Daemon::Options::ShowUsage(argv[0]); |
| return 1; |
| } |
| Daemon daemon{opts}; |
| SampleHandler handler{daemon.GetTaskRunner()}; |
| handler.Register(daemon.GetDevice()); |
| daemon.Run(); |
| return 0; |
| } |