blob: 11dc4bc70720a4d28d48996fba6b84b89ef9d9c4 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 2013-2017 Nest Labs, Inc.
* 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 "ChipDeviceController-ScriptDevicePairingDelegate.h"
#include "lib/support/TypeTraits.h"
#include <app/icd/client/DefaultICDClientStorage.h>
#include <controller/python/chip/native/PyChipError.h>
#include <setup_payload/ManualSetupPayloadGenerator.h>
#include <setup_payload/QRCodeSetupPayloadGenerator.h>
#include <string>
extern chip::app::DefaultICDClientStorage sICDClientStorage;
extern chip::Controller::CommissioningParameters sCommissioningParameters;
extern uint8_t sICDSymmetricKey[chip::Crypto::kAES_CCM128_Key_Length];
namespace chip {
namespace Controller {
namespace {
void OnWindowCompleteStatic(void * context, NodeId deviceId, CHIP_ERROR status, SetupPayload payload)
{
auto self = reinterpret_cast<ScriptDevicePairingDelegate *>(context);
self->OnOpenCommissioningWindow(deviceId, status, payload);
}
} // namespace
ScriptDevicePairingDelegate::ScriptDevicePairingDelegate() : mOpenWindowCallback(OnWindowCompleteStatic, this) {}
void ScriptDevicePairingDelegate::SetKeyExchangeCallback(DevicePairingDelegate_OnPairingCompleteFunct callback)
{
mOnPairingCompleteCallback = callback;
}
void ScriptDevicePairingDelegate::SetCommissioningCompleteCallback(DevicePairingDelegate_OnCommissioningCompleteFunct callback)
{
mOnCommissioningCompleteCallback = callback;
}
void ScriptDevicePairingDelegate::SetCommissioningWindowOpenCallback(DevicePairingDelegate_OnWindowOpenCompleteFunct callback)
{
mOnWindowOpenCompleteCallback = callback;
}
void ScriptDevicePairingDelegate::SetCommissioningSuccessCallback(DevicePairingDelegate_OnCommissioningSuccessFunct callback)
{
mOnCommissioningSuccessCallback = callback;
}
void ScriptDevicePairingDelegate::SetCommissioningFailureCallback(DevicePairingDelegate_OnCommissioningFailureFunct callback)
{
mOnCommissioningFailureCallback = callback;
}
void ScriptDevicePairingDelegate::SetFabricCheckCallback(DevicePairingDelegate_OnFabricCheckFunct callback)
{
mOnFabricCheckCallback = callback;
}
void ScriptDevicePairingDelegate::SetCommissioningStatusUpdateCallback(
DevicePairingDelegate_OnCommissioningStatusUpdateFunct callback)
{
mOnCommissioningStatusUpdateCallback = callback;
}
void ScriptDevicePairingDelegate::OnStatusUpdate(DevicePairingDelegate::Status status)
{
switch (status)
{
case DevicePairingDelegate::Status::SecurePairingSuccess:
ChipLogProgress(Zcl, "Secure Pairing Success");
break;
case DevicePairingDelegate::Status::SecurePairingFailed:
ChipLogError(Zcl, "Secure Pairing Failed");
if (mOnPairingCompleteCallback != nullptr && expectingPairingComplete)
{
// Incorrect state is the same error that chip-tool sends. We are also
// leveraging the on pairing complete callback to indicate that pairing
// has failed.
expectingPairingComplete = false;
mOnPairingCompleteCallback(ToPyChipError(CHIP_ERROR_INCORRECT_STATE));
}
break;
}
}
void ScriptDevicePairingDelegate::OnPairingComplete(CHIP_ERROR error)
{
if (mOnPairingCompleteCallback != nullptr && expectingPairingComplete)
{
expectingPairingComplete = false;
mOnPairingCompleteCallback(ToPyChipError(error));
}
}
void ScriptDevicePairingDelegate::OnCommissioningComplete(NodeId nodeId, CHIP_ERROR error)
{
if (mOnCommissioningCompleteCallback != nullptr)
{
mOnCommissioningCompleteCallback(nodeId, ToPyChipError(error));
}
}
void ScriptDevicePairingDelegate::OnCommissioningSuccess(PeerId peerId)
{
if (mOnCommissioningSuccessCallback != nullptr)
{
mOnCommissioningSuccessCallback(peerId);
}
}
void ScriptDevicePairingDelegate::OnCommissioningFailure(PeerId peerId, CHIP_ERROR error, CommissioningStage stageFailed,
Optional<Credentials::AttestationVerificationResult> additionalErrorInfo)
{
if (mOnCommissioningFailureCallback != nullptr)
{
mOnCommissioningFailureCallback(peerId, error, stageFailed, additionalErrorInfo);
}
}
void ScriptDevicePairingDelegate::OnCommissioningStatusUpdate(PeerId peerId, CommissioningStage stageCompleted, CHIP_ERROR error)
{
if (mOnCommissioningStatusUpdateCallback != nullptr)
{
mOnCommissioningStatusUpdateCallback(peerId, stageCompleted, error);
}
}
void ScriptDevicePairingDelegate::OnOpenCommissioningWindow(NodeId deviceId, CHIP_ERROR status, SetupPayload payload)
{
if (mOnWindowOpenCompleteCallback != nullptr)
{
std::string setupManualCode;
std::string setupQRCode;
ManualSetupPayloadGenerator(payload).payloadDecimalStringRepresentation(setupManualCode);
QRCodeSetupPayloadGenerator(payload).payloadBase38Representation(setupQRCode);
ChipLogProgress(Zcl, "SetupManualCode = %s", setupManualCode.c_str());
ChipLogProgress(Zcl, "SetupQRCode = %s", setupQRCode.c_str());
mOnWindowOpenCompleteCallback(deviceId, payload.setUpPINCode, setupManualCode.c_str(), setupQRCode.c_str(),
ToPyChipError(status));
}
if (mWindowOpener != nullptr)
{
Platform::Delete(mWindowOpener);
mWindowOpener = nullptr;
}
}
void ScriptDevicePairingDelegate::OnFabricCheck(NodeId matchingNodeId)
{
if (matchingNodeId == kUndefinedNodeId)
{
ChipLogProgress(Zcl, "No matching fabric found");
}
else
{
ChipLogProgress(Zcl, "Matching fabric found");
}
if (mOnFabricCheckCallback != nullptr)
{
mOnFabricCheckCallback(matchingNodeId);
}
}
Callback::Callback<Controller::OnOpenCommissioningWindow> *
ScriptDevicePairingDelegate::GetOpenWindowCallback(Controller::CommissioningWindowOpener * context)
{
mWindowOpener = context;
return &mOpenWindowCallback;
}
void ScriptDevicePairingDelegate::OnICDRegistrationComplete(ScopedNodeId nodeId, uint32_t icdCounter)
{
app::ICDClientInfo clientInfo;
clientInfo.peer_node = nodeId;
clientInfo.check_in_node = chip::ScopedNodeId(sCommissioningParameters.GetICDCheckInNodeId().Value(), nodeId.GetFabricIndex());
clientInfo.monitored_subject = sCommissioningParameters.GetICDMonitoredSubject().Value();
clientInfo.start_icd_counter = icdCounter;
clientInfo.client_type = sCommissioningParameters.GetICDClientType().Value();
CHIP_ERROR err = sICDClientStorage.SetKey(clientInfo, ByteSpan(sICDSymmetricKey));
if (err == CHIP_NO_ERROR)
{
err = sICDClientStorage.StoreEntry(clientInfo);
}
if (err != CHIP_NO_ERROR)
{
sICDClientStorage.RemoveKey(clientInfo);
ChipLogError(Controller, "Failed to persist symmetric key for " ChipLogFormatX64 ": %s",
ChipLogValueX64(nodeId.GetNodeId()), err.AsString());
return;
}
ChipLogProgress(Controller, "Saved ICD Symmetric key for " ChipLogFormatX64, ChipLogValueX64(nodeId.GetNodeId()));
ChipLogProgress(Controller,
"ICD Registration Complete for device " ChipLogFormatX64 " / Check-In NodeID: " ChipLogFormatX64
" / Monitored Subject: " ChipLogFormatX64 " / ICDCounter %u",
ChipLogValueX64(nodeId.GetNodeId()), ChipLogValueX64(sCommissioningParameters.GetICDCheckInNodeId().Value()),
ChipLogValueX64(clientInfo.monitored_subject), icdCounter);
}
void ScriptDevicePairingDelegate::OnICDStayActiveComplete(ScopedNodeId deviceId, uint32_t promisedActiveDuration)
{
ChipLogProgress(Controller, "ICD Stay Active Complete for device " ChipLogFormatX64 " / promisedActiveDuration: %u",
ChipLogValueX64(deviceId.GetNodeId()), promisedActiveDuration);
}
} // namespace Controller
} // namespace chip