blob: 1fad323e125c2a914c3afbb3d62c5c5d7db03397 [file] [log] [blame]
/*
* Copyright (c) 2024 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.
*
*/
#pragma once
#include <app/CommandSender.h>
#include <controller/CHIPDeviceController.h>
/**
* @class CommissionerControl
* @brief This class handles sending CHIP commands related to commissioning, including sending
* commissioning approval requests and commissioning nodes.
*
* The class acts as a command sender and implements the `chip::app::CommandSender::Callback` interface
* to handle responses, errors, and completion events for the commands it sends. It relies on external
* CCTRL delegate and server mechanisms to manage the overall protocol and state transitions, including
* processing the CommissioningRequestResult and invoking CommissionNode.
*/
class CommissionerControl : public chip::app::CommandSender::Callback
{
public:
CommissionerControl() :
mOnDeviceConnectedCallback(OnDeviceConnectedFn, this), mOnDeviceConnectionFailureCallback(OnDeviceConnectionFailureFn, this)
{}
/**
* @brief Initializes the CommissionerControl with a DeviceCommissioner, NodeId, and EndpointId.
*
* @param commissioner The DeviceCommissioner to use for the commissioning process.
* @param nodeId The node ID of the remote fabric bridge.
* @param endpointId The endpoint on which to send CommissionerControl commands.
*/
void Init(chip::Controller::DeviceCommissioner & commissioner, chip::NodeId nodeId, chip::EndpointId endpointId);
/**
* @brief Sends a RequestCommissioningApproval command to the device.
*
* @param requestId The unique request ID.
* @param vendorId The vendor ID of the device.
* @param productId The product ID of the device.
* @param label Optional label for the device.
* @return CHIP_ERROR CHIP_NO_ERROR on success, or an appropriate error code on failure.
*/
CHIP_ERROR RequestCommissioningApproval(uint64_t requestId, uint16_t vendorId, uint16_t productId,
chip::Optional<chip::CharSpan> label);
/**
* @brief Sends a CommissionNode command to the device.
*
* @param requestId The unique request ID.
* @param responseTimeoutSeconds Timeout for the response in seconds.
* @return CHIP_ERROR CHIP_NO_ERROR on success, or an appropriate error code on failure.
*/
CHIP_ERROR CommissionNode(uint64_t requestId, uint16_t responseTimeoutSeconds);
/////////// CommandSender Callback Interface /////////
virtual void OnResponse(chip::app::CommandSender * client, const chip::app::ConcreteCommandPath & path,
const chip::app::StatusIB & status, chip::TLV::TLVReader * data) override;
virtual void OnError(const chip::app::CommandSender * client, CHIP_ERROR error) override;
virtual void OnDone(chip::app::CommandSender * client) override;
private:
static constexpr uint16_t kMaxDeviceLabelLength = 64;
enum class CommandType : uint8_t
{
kUndefined = 0,
kRequestCommissioningApproval = 1,
kCommissionNode = 2,
};
template <class T>
CHIP_ERROR SendCommand(chip::DeviceProxy * device, chip::EndpointId endpointId, chip::ClusterId clusterId,
chip::CommandId commandId, const T & value)
{
chip::app::CommandPathParams commandPath = { endpointId, clusterId, commandId,
(chip::app::CommandPathFlags::kEndpointIdValid) };
mCommandSender = std::make_unique<chip::app::CommandSender>(this, device->GetExchangeManager(), false, false,
device->GetSecureSession().Value()->AllowsLargePayload());
VerifyOrReturnError(mCommandSender != nullptr, CHIP_ERROR_NO_MEMORY);
chip::app::CommandSender::AddRequestDataParameters addRequestDataParams(chip::NullOptional);
ReturnErrorOnFailure(mCommandSender->AddRequestData(commandPath, value, addRequestDataParams));
ReturnErrorOnFailure(mCommandSender->SendCommandRequest(device->GetSecureSession().Value()));
return CHIP_NO_ERROR;
}
CHIP_ERROR SendCommandForType(CommandType commandType, chip::DeviceProxy * device);
static void OnDeviceConnectedFn(void * context, chip::Messaging::ExchangeManager & exchangeMgr,
const chip::SessionHandle & sessionHandle);
static void OnDeviceConnectionFailureFn(void * context, const chip::ScopedNodeId & peerId, CHIP_ERROR error);
// Private data members
chip::Controller::DeviceCommissioner * mCommissioner = nullptr;
std::unique_ptr<chip::app::CommandSender> mCommandSender;
chip::NodeId mDestinationId = chip::kUndefinedNodeId;
chip::EndpointId mEndpointId = chip::kRootEndpointId;
CommandType mCommandType = CommandType::kUndefined;
char mLabelBuffer[kMaxDeviceLabelLength];
chip::Callback::Callback<chip::OnDeviceConnected> mOnDeviceConnectedCallback;
chip::Callback::Callback<chip::OnDeviceConnectionFailure> mOnDeviceConnectionFailureCallback;
chip::app::Clusters::CommissionerControl::Commands::RequestCommissioningApproval::Type mRequestCommissioningApproval;
chip::app::Clusters::CommissionerControl::Commands::CommissionNode::Type mCommissionNode;
};