/* * * Copyright (c) 2023 Project CHIP Authors * All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "LightingAppCommandDelegate.h" #include #include #include #include #include #include #include #include #include using namespace chip; using namespace chip::app; using namespace chip::app::Clusters; using namespace chip::DeviceLayer; LightingAppCommandHandler * LightingAppCommandHandler::FromJSON(const char * json) { Json::Reader reader; Json::Value value; if (!reader.parse(json, value)) { ChipLogError(NotSpecified, "AllClusters App: Error parsing JSON with error %s:", reader.getFormattedErrorMessages().c_str()); return nullptr; } if (value.empty() || !value.isObject()) { ChipLogError(NotSpecified, "AllClusters App: Invalid JSON command received"); return nullptr; } if (!value.isMember("Name") || !value["Name"].isString()) { ChipLogError(NotSpecified, "AllClusters App: Invalid JSON command received: command name is missing"); return nullptr; } return Platform::New(std::move(value)); } void LightingAppCommandHandler::HandleCommand(intptr_t context) { auto * self = reinterpret_cast(context); std::string name = self->mJsonValue["Name"].asString(); VerifyOrExit(!self->mJsonValue.empty(), ChipLogError(NotSpecified, "Invalid JSON event command received")); if (name == "SoftwareFault") { self->OnSoftwareFaultEventHandler(Clusters::SoftwareDiagnostics::Events::SoftwareFault::Id); } else if (name == "HardwareFaultChange") { self->OnGeneralFaultEventHandler(Clusters::GeneralDiagnostics::Events::HardwareFaultChange::Id); } else if (name == "RadioFaultChange") { self->OnGeneralFaultEventHandler(Clusters::GeneralDiagnostics::Events::RadioFaultChange::Id); } else if (name == "NetworkFaultChange") { self->OnGeneralFaultEventHandler(Clusters::GeneralDiagnostics::Events::NetworkFaultChange::Id); } else if (name == "SwitchLatched") { uint8_t newPosition = static_cast(self->mJsonValue["NewPosition"].asUInt()); self->OnSwitchLatchedHandler(newPosition); } else if (name == "InitialPress") { uint8_t newPosition = static_cast(self->mJsonValue["NewPosition"].asUInt()); self->OnSwitchInitialPressedHandler(newPosition); } else if (name == "LongPress") { uint8_t newPosition = static_cast(self->mJsonValue["NewPosition"].asUInt()); self->OnSwitchLongPressedHandler(newPosition); } else if (name == "ShortRelease") { uint8_t previousPosition = static_cast(self->mJsonValue["PreviousPosition"].asUInt()); self->OnSwitchShortReleasedHandler(previousPosition); } else if (name == "LongRelease") { uint8_t previousPosition = static_cast(self->mJsonValue["PreviousPosition"].asUInt()); self->OnSwitchLongReleasedHandler(previousPosition); } else if (name == "MultiPressOngoing") { uint8_t newPosition = static_cast(self->mJsonValue["NewPosition"].asUInt()); uint8_t count = static_cast(self->mJsonValue["CurrentNumberOfPressesCounted"].asUInt()); self->OnSwitchMultiPressOngoingHandler(newPosition, count); } else if (name == "MultiPressComplete") { uint8_t previousPosition = static_cast(self->mJsonValue["PreviousPosition"].asUInt()); uint8_t count = static_cast(self->mJsonValue["TotalNumberOfPressesCounted"].asUInt()); self->OnSwitchMultiPressCompleteHandler(previousPosition, count); } else if (name == "PowerOnReboot") { self->OnRebootSignalHandler(BootReasonType::kPowerOnReboot); } else if (name == "BrownOutReset") { self->OnRebootSignalHandler(BootReasonType::kBrownOutReset); } else if (name == "SoftwareWatchdogReset") { self->OnRebootSignalHandler(BootReasonType::kSoftwareWatchdogReset); } else if (name == "HardwareWatchdogReset") { self->OnRebootSignalHandler(BootReasonType::kHardwareWatchdogReset); } else if (name == "SoftwareUpdateCompleted") { self->OnRebootSignalHandler(BootReasonType::kSoftwareUpdateCompleted); } else if (name == "SoftwareReset") { self->OnRebootSignalHandler(BootReasonType::kSoftwareReset); } else { ChipLogError(NotSpecified, "Unhandled command: Should never happens"); } exit: Platform::Delete(self); } bool LightingAppCommandHandler::IsClusterPresentOnAnyEndpoint(ClusterId clusterId) { EnabledEndpointsWithServerCluster enabledEndpoints(clusterId); return (enabledEndpoints.begin() != enabledEndpoints.end()); } void LightingAppCommandHandler::OnRebootSignalHandler(BootReasonType bootReason) { if (ConfigurationMgr().StoreBootReason(static_cast(bootReason)) == CHIP_NO_ERROR) { Server::GetInstance().GenerateShutDownEvent(); PlatformMgr().ScheduleWork([](intptr_t) { PlatformMgr().StopEventLoopTask(); }); } else { ChipLogError(NotSpecified, "Failed to store boot reason:%d", static_cast(bootReason)); } } void LightingAppCommandHandler::OnGeneralFaultEventHandler(uint32_t eventId) { if (!IsClusterPresentOnAnyEndpoint(Clusters::GeneralDiagnostics::Id)) return; if (eventId == Clusters::GeneralDiagnostics::Events::HardwareFaultChange::Id) { GeneralFaults previous; GeneralFaults current; using GeneralDiagnostics::HardwareFaultEnum; // On Linux Simulation, set following hardware faults statically. ReturnOnFailure(previous.add(to_underlying(HardwareFaultEnum::kRadio))); ReturnOnFailure(previous.add(to_underlying(HardwareFaultEnum::kPowerSource))); ReturnOnFailure(current.add(to_underlying(HardwareFaultEnum::kRadio))); ReturnOnFailure(current.add(to_underlying(HardwareFaultEnum::kSensor))); ReturnOnFailure(current.add(to_underlying(HardwareFaultEnum::kPowerSource))); ReturnOnFailure(current.add(to_underlying(HardwareFaultEnum::kUserInterfaceFault))); Clusters::GeneralDiagnosticsServer::Instance().OnHardwareFaultsDetect(previous, current); } else if (eventId == Clusters::GeneralDiagnostics::Events::RadioFaultChange::Id) { GeneralFaults previous; GeneralFaults current; // On Linux Simulation, set following radio faults statically. ReturnOnFailure(previous.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kWiFiFault))); ReturnOnFailure(previous.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kThreadFault))); ReturnOnFailure(current.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kWiFiFault))); ReturnOnFailure(current.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kCellularFault))); ReturnOnFailure(current.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kThreadFault))); ReturnOnFailure(current.add(to_underlying(GeneralDiagnostics::RadioFaultEnum::kNFCFault))); Clusters::GeneralDiagnosticsServer::Instance().OnRadioFaultsDetect(previous, current); } else if (eventId == Clusters::GeneralDiagnostics::Events::NetworkFaultChange::Id) { GeneralFaults previous; GeneralFaults current; // On Linux Simulation, set following radio faults statically. ReturnOnFailure(previous.add(to_underlying(Clusters::GeneralDiagnostics::NetworkFaultEnum::kHardwareFailure))); ReturnOnFailure(previous.add(to_underlying(Clusters::GeneralDiagnostics::NetworkFaultEnum::kNetworkJammed))); ReturnOnFailure(current.add(to_underlying(Clusters::GeneralDiagnostics::NetworkFaultEnum::kHardwareFailure))); ReturnOnFailure(current.add(to_underlying(Clusters::GeneralDiagnostics::NetworkFaultEnum::kNetworkJammed))); ReturnOnFailure(current.add(to_underlying(Clusters::GeneralDiagnostics::NetworkFaultEnum::kConnectionFailed))); Clusters::GeneralDiagnosticsServer::Instance().OnNetworkFaultsDetect(previous, current); } else { ChipLogError(NotSpecified, "Unknow event ID:%d", eventId); } } void LightingAppCommandHandler::OnSoftwareFaultEventHandler(uint32_t eventId) { VerifyOrReturn(eventId == Clusters::SoftwareDiagnostics::Events::SoftwareFault::Id, ChipLogError(NotSpecified, "Unknown software fault event received")); if (!IsClusterPresentOnAnyEndpoint(Clusters::SoftwareDiagnostics::Id)) return; Clusters::SoftwareDiagnostics::Events::SoftwareFault::Type softwareFault; char threadName[kMaxThreadNameLength + 1]; softwareFault.id = static_cast(getpid()); Platform::CopyString(threadName, std::to_string(softwareFault.id).c_str()); softwareFault.name.SetValue(CharSpan::fromCharString(threadName)); std::time_t result = std::time(nullptr); // Using size of 50 as it is double the expected 25 characters "Www Mmm dd hh:mm:ss yyyy\n". char timeChar[50]; if (std::strftime(timeChar, sizeof(timeChar), "%c", std::localtime(&result))) { softwareFault.faultRecording.SetValue(ByteSpan(Uint8::from_const_char(timeChar), strlen(timeChar))); } Clusters::SoftwareDiagnosticsServer::Instance().OnSoftwareFaultDetect(softwareFault); } void LightingAppCommandHandler::OnSwitchLatchedHandler(uint8_t newPosition) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, newPosition); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to set CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The latching switch is moved to a new position:%d", newPosition); Clusters::SwitchServer::Instance().OnSwitchLatch(endpoint, newPosition); } void LightingAppCommandHandler::OnSwitchInitialPressedHandler(uint8_t newPosition) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, newPosition); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to set CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The new position when the momentary switch starts to be pressed:%d", newPosition); Clusters::SwitchServer::Instance().OnInitialPress(endpoint, newPosition); } void LightingAppCommandHandler::OnSwitchLongPressedHandler(uint8_t newPosition) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, newPosition); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to set CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The new position when the momentary switch has been pressed for a long time:%d", newPosition); Clusters::SwitchServer::Instance().OnLongPress(endpoint, newPosition); } void LightingAppCommandHandler::OnSwitchShortReleasedHandler(uint8_t previousPosition) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, 0); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to reset CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The the previous value of the CurrentPosition when the momentary switch has been released:%d", previousPosition); Clusters::SwitchServer::Instance().OnShortRelease(endpoint, previousPosition); } void LightingAppCommandHandler::OnSwitchLongReleasedHandler(uint8_t previousPosition) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, 0); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to reset CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The the previous value of the CurrentPosition when the momentary switch has been released after having been " "pressed for a long time:%d", previousPosition); Clusters::SwitchServer::Instance().OnLongRelease(endpoint, previousPosition); } void LightingAppCommandHandler::OnSwitchMultiPressOngoingHandler(uint8_t newPosition, uint8_t count) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, newPosition); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to set CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The new position when the momentary switch has been pressed in a multi-press sequence:%d", newPosition); ChipLogDetail(NotSpecified, "%d times the momentary switch has been pressed", count); Clusters::SwitchServer::Instance().OnMultiPressOngoing(endpoint, newPosition, count); } void LightingAppCommandHandler::OnSwitchMultiPressCompleteHandler(uint8_t previousPosition, uint8_t count) { EndpointId endpoint = 0; Protocols::InteractionModel::Status status = Switch::Attributes::CurrentPosition::Set(endpoint, 0); VerifyOrReturn(Protocols::InteractionModel::Status::Success == status, ChipLogError(NotSpecified, "Failed to reset CurrentPosition attribute")); ChipLogDetail(NotSpecified, "The previous position when the momentary switch has been pressed in a multi-press sequence:%d", previousPosition); ChipLogDetail(NotSpecified, "%d times the momentary switch has been pressed", count); Clusters::SwitchServer::Instance().OnMultiPressComplete(endpoint, previousPosition, count); } void LightingAppCommandDelegate::OnEventCommandReceived(const char * json) { auto handler = LightingAppCommandHandler::FromJSON(json); if (nullptr == handler) { ChipLogError(NotSpecified, "AllClusters App: Unable to instantiate a command handler"); return; } chip::DeviceLayer::PlatformMgr().ScheduleWork(LightingAppCommandHandler::HandleCommand, reinterpret_cast(handler)); }