| /* |
| * |
| * Copyright (c) 2021 Project CHIP Authors |
| * |
| * 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 <memory> |
| #include <type_traits> |
| |
| #include <app/CommandSender.h> |
| #include <app/DeviceProxy.h> |
| #include <lib/support/CodeUtils.h> |
| |
| #include <controller/python/chip/interaction_model/Delegate.h> |
| #include <cstdio> |
| #include <lib/support/logging/CHIPLogging.h> |
| |
| using namespace chip; |
| using namespace chip::app; |
| |
| using PyObject = void *; |
| |
| extern "C" { |
| chip::ChipError::StorageType pychip_CommandSender_SendCommand(void * appContext, DeviceProxy * device, chip::EndpointId endpointId, |
| chip::ClusterId clusterId, chip::CommandId commandId, |
| const uint8_t * payload, size_t length); |
| } |
| |
| namespace chip { |
| namespace python { |
| |
| using OnCommandSenderResponseCallback = void (*)(PyObject appContext, chip::EndpointId endpointId, chip::ClusterId clusterId, |
| chip::CommandId commandId, |
| std::underlying_type_t<Protocols::InteractionModel::Status> status, |
| chip::ClusterStatus clusterStatus, const uint8_t * payload, uint32_t length); |
| using OnCommandSenderErrorCallback = void (*)(PyObject appContext, |
| std::underlying_type_t<Protocols::InteractionModel::Status> status, |
| chip::ClusterStatus clusterStatus, uint32_t chiperror); |
| using OnCommandSenderDoneCallback = void (*)(PyObject appContext); |
| |
| OnCommandSenderResponseCallback gOnCommandSenderResponseCallback = nullptr; |
| OnCommandSenderErrorCallback gOnCommandSenderErrorCallback = nullptr; |
| OnCommandSenderDoneCallback gOnCommandSenderDoneCallback = nullptr; |
| |
| class CommandSenderCallback : public CommandSender::Callback |
| { |
| public: |
| CommandSenderCallback(PyObject appContext) : mAppContext(appContext) {} |
| |
| void OnResponse(CommandSender * apCommandSender, const ConcreteCommandPath & aPath, const app::StatusIB & aStatus, |
| TLV::TLVReader * aData) override |
| { |
| uint8_t buffer[CHIP_CONFIG_DEFAULT_UDP_MTU_SIZE]; |
| uint32_t size = 0; |
| // When the apData is nullptr, means we did not receive a valid attribute data from server, status will be some error |
| // status. |
| if (aData != nullptr) |
| { |
| // Python need to read from full TLV data the TLVReader may contain some unclean states. |
| TLV::TLVWriter writer; |
| writer.Init(buffer); |
| CHIP_ERROR err = writer.CopyContainer(TLV::AnonymousTag, *aData); |
| if (err != CHIP_NO_ERROR) |
| { |
| app::StatusIB status; |
| status.mStatus = Protocols::InteractionModel::Status::Failure; |
| this->OnError(apCommandSender, aStatus, err); |
| return; |
| } |
| size = writer.GetLengthWritten(); |
| } |
| |
| gOnCommandSenderResponseCallback( |
| mAppContext, aPath.mEndpointId, aPath.mClusterId, aPath.mCommandId, to_underlying(aStatus.mStatus), |
| aStatus.mClusterStatus.HasValue() ? aStatus.mClusterStatus.Value() : chip::python::kUndefinedClusterStatus, buffer, |
| size); |
| } |
| |
| void OnError(const CommandSender * apCommandSender, const app::StatusIB & aStatus, CHIP_ERROR aProtocolError) override |
| { |
| gOnCommandSenderErrorCallback(mAppContext, to_underlying(aStatus.mStatus), |
| aStatus.mClusterStatus.HasValue() ? aStatus.mClusterStatus.Value() |
| : chip::python::kUndefinedClusterStatus, |
| aProtocolError.AsInteger()); |
| } |
| |
| void OnDone(CommandSender * apCommandSender) override |
| { |
| gOnCommandSenderDoneCallback(mAppContext); |
| delete apCommandSender; |
| delete this; |
| }; |
| |
| private: |
| PyObject mAppContext = nullptr; |
| }; |
| |
| } // namespace python |
| } // namespace chip |
| |
| using namespace chip::python; |
| |
| extern "C" { |
| void pychip_CommandSender_InitCallbacks(OnCommandSenderResponseCallback onCommandSenderResponseCallback, |
| OnCommandSenderErrorCallback onCommandSenderErrorCallback, |
| OnCommandSenderDoneCallback onCommandSenderDoneCallback) |
| { |
| gOnCommandSenderResponseCallback = onCommandSenderResponseCallback; |
| gOnCommandSenderErrorCallback = onCommandSenderErrorCallback; |
| gOnCommandSenderDoneCallback = onCommandSenderDoneCallback; |
| } |
| |
| chip::ChipError::StorageType pychip_CommandSender_SendCommand(void * appContext, DeviceProxy * device, chip::EndpointId endpointId, |
| chip::ClusterId clusterId, chip::CommandId commandId, |
| const uint8_t * payload, size_t length) |
| { |
| CHIP_ERROR err = CHIP_NO_ERROR; |
| |
| std::unique_ptr<CommandSenderCallback> callback = std::make_unique<CommandSenderCallback>(appContext); |
| std::unique_ptr<CommandSender> sender = std::make_unique<CommandSender>(callback.get(), device->GetExchangeManager()); |
| |
| app::CommandPathParams cmdParams = { endpointId, /* group id */ 0, clusterId, commandId, |
| (app::CommandPathFlags::kEndpointIdValid) }; |
| |
| SuccessOrExit(err = sender->PrepareCommand(cmdParams)); |
| |
| { |
| auto writer = sender->GetCommandDataIBTLVWriter(); |
| TLV::TLVReader reader; |
| TLV::TLVType type; |
| VerifyOrExit(writer != nullptr, err = CHIP_ERROR_INCORRECT_STATE); |
| reader.Init(payload, length); |
| reader.Next(); |
| reader.EnterContainer(type); |
| while (reader.Next() == CHIP_NO_ERROR) |
| { |
| TLV::TLVReader tReader; |
| tReader.Init(reader); |
| writer->CopyElement(tReader); |
| } |
| } |
| |
| SuccessOrExit(err = sender->FinishCommand()); |
| SuccessOrExit(err = device->SendCommands(sender.get())); |
| |
| sender.release(); |
| callback.release(); |
| |
| exit: |
| return err.AsInteger(); |
| } |
| } |