Skip to content

Commit c0c674c

Browse files
[Fabric-Admin] Implement FabricSyncGetter APIs (project-chip#36108)
* [Fabric-Admin] Implement FabricSyncGetter APIs * Update examples/fabric-admin/device_manager/FabricSyncGetter.h Co-authored-by: Terence Hampson <thampson@google.com> --------- Co-authored-by: Terence Hampson <thampson@google.com>
1 parent d9c6de5 commit c0c674c

File tree

6 files changed

+210
-14
lines changed

6 files changed

+210
-14
lines changed

examples/fabric-admin/BUILD.gn

+2
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,8 @@ static_library("fabric-admin-utils") {
9090
"device_manager/DeviceSubscriptionManager.h",
9191
"device_manager/DeviceSynchronization.cpp",
9292
"device_manager/DeviceSynchronization.h",
93+
"device_manager/FabricSyncGetter.cpp",
94+
"device_manager/FabricSyncGetter.h",
9395
"device_manager/PairingManager.cpp",
9496
"device_manager/PairingManager.h",
9597
"device_manager/UniqueIdGetter.cpp",

examples/fabric-admin/commands/clusters/ReportCommand.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,6 @@ void ReportCommand::OnAttributeData(const app::ConcreteDataAttributePath & path,
4545
}
4646

4747
LogErrorOnFailure(RemoteDataModelLogger::LogAttributeAsJSON(path, data));
48-
49-
DeviceMgr().HandleAttributeData(path, *data);
5048
}
5149

5250
void ReportCommand::OnEventData(const app::EventHeader & eventHeader, TLV::TLVReader * data, const app::StatusIB * status)

examples/fabric-admin/device_manager/DeviceManager.cpp

+10-12
Original file line numberDiff line numberDiff line change
@@ -212,13 +212,18 @@ void DeviceManager::ReadSupportedDeviceCategories()
212212
return;
213213
}
214214

215-
StringBuilder<kMaxCommandSize> commandBuilder;
215+
ChipLogProgress(NotSpecified, "Read SupportedDeviceCategories from the remote bridge.");
216216

217-
commandBuilder.Add("commissionercontrol read supported-device-categories ");
218-
commandBuilder.AddFormat("%ld ", mRemoteBridgeNodeId);
219-
commandBuilder.AddFormat("%d", kAggregatorEndpointId);
217+
CHIP_ERROR error = mFabricSyncGetter.GetFabricSynchronizationData(
218+
[this](TLV::TLVReader & data) { this->HandleReadSupportedDeviceCategories(data); },
219+
PairingManager::Instance().CurrentCommissioner(), this->GetRemoteBridgeNodeId(), kAggregatorEndpointId);
220220

221-
PushCommand(commandBuilder.c_str());
221+
if (error != CHIP_NO_ERROR)
222+
{
223+
ChipLogError(NotSpecified,
224+
"Failed to read SupportedDeviceCategories from the remote bridge (NodeId: %lu). Error: %" CHIP_ERROR_FORMAT,
225+
mRemoteBridgeNodeId, error.Format());
226+
}
222227
}
223228

224229
void DeviceManager::HandleReadSupportedDeviceCategories(TLV::TLVReader & data)
@@ -421,13 +426,6 @@ void DeviceManager::HandleReverseOpenCommissioningWindow(TLV::TLVReader & data)
421426

422427
void DeviceManager::HandleAttributeData(const app::ConcreteDataAttributePath & path, TLV::TLVReader & data)
423428
{
424-
if (path.mClusterId == CommissionerControl::Id &&
425-
path.mAttributeId == CommissionerControl::Attributes::SupportedDeviceCategories::Id)
426-
{
427-
HandleReadSupportedDeviceCategories(data);
428-
return;
429-
}
430-
431429
if (path.mClusterId == Descriptor::Id && path.mAttributeId == Descriptor::Attributes::PartsList::Id)
432430
{
433431
HandleAttributePartsListUpdate(data);

examples/fabric-admin/device_manager/DeviceManager.h

+2
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
#include <app-common/zap-generated/cluster-objects.h>
2222
#include <device_manager/BridgeSubscription.h>
23+
#include <device_manager/FabricSyncGetter.h>
2324
#include <device_manager/PairingManager.h>
2425
#include <platform/CHIPDeviceLayer.h>
2526

@@ -212,6 +213,7 @@ class DeviceManager : public PairingDelegate
212213
uint64_t mRequestId = 0;
213214

214215
BridgeSubscription mBridgeSubscriber;
216+
FabricSyncGetter mFabricSyncGetter;
215217
};
216218

217219
/**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,121 @@
1+
/*
2+
* Copyright (c) 2024 Project CHIP Authors
3+
* All rights reserved.
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*
17+
*/
18+
19+
#include "FabricSyncGetter.h"
20+
21+
using namespace ::chip;
22+
using namespace ::chip::app;
23+
using chip::app::ReadClient;
24+
25+
namespace {
26+
27+
void OnDeviceConnectedWrapper(void * context, Messaging::ExchangeManager & exchangeMgr, const SessionHandle & sessionHandle)
28+
{
29+
reinterpret_cast<FabricSyncGetter *>(context)->OnDeviceConnected(exchangeMgr, sessionHandle);
30+
}
31+
32+
void OnDeviceConnectionFailureWrapper(void * context, const ScopedNodeId & peerId, CHIP_ERROR error)
33+
{
34+
reinterpret_cast<FabricSyncGetter *>(context)->OnDeviceConnectionFailure(peerId, error);
35+
}
36+
37+
} // namespace
38+
39+
FabricSyncGetter::FabricSyncGetter() :
40+
mOnDeviceConnectedCallback(OnDeviceConnectedWrapper, this),
41+
mOnDeviceConnectionFailureCallback(OnDeviceConnectionFailureWrapper, this)
42+
{}
43+
44+
CHIP_ERROR FabricSyncGetter::GetFabricSynchronizationData(OnDoneCallback onDoneCallback, Controller::DeviceController & controller,
45+
NodeId nodeId, EndpointId endpointId)
46+
{
47+
assertChipStackLockedByCurrentThread();
48+
49+
mEndpointId = endpointId;
50+
mOnDoneCallback = onDoneCallback;
51+
52+
CHIP_ERROR err = controller.GetConnectedDevice(nodeId, &mOnDeviceConnectedCallback, &mOnDeviceConnectionFailureCallback);
53+
if (err != CHIP_NO_ERROR)
54+
{
55+
ChipLogError(NotSpecified, "Failed to connect to remote fabric bridge %" CHIP_ERROR_FORMAT, err.Format());
56+
}
57+
return err;
58+
}
59+
60+
void FabricSyncGetter::OnAttributeData(const ConcreteDataAttributePath & path, TLV::TLVReader * data, const StatusIB & status)
61+
{
62+
VerifyOrDie(path.mClusterId == Clusters::CommissionerControl::Id);
63+
64+
if (!status.IsSuccess())
65+
{
66+
ChipLogError(NotSpecified, "Response Failure: %" CHIP_ERROR_FORMAT, status.ToChipError().Format());
67+
return;
68+
}
69+
70+
switch (path.mAttributeId)
71+
{
72+
case Clusters::CommissionerControl::Attributes::SupportedDeviceCategories::Id: {
73+
mOnDoneCallback(*data);
74+
break;
75+
}
76+
default:
77+
break;
78+
}
79+
}
80+
81+
void FabricSyncGetter::OnError(CHIP_ERROR error)
82+
{
83+
ChipLogProgress(NotSpecified, "Error Getting SupportedDeviceCategories: %" CHIP_ERROR_FORMAT, error.Format());
84+
}
85+
86+
void FabricSyncGetter::OnDone(ReadClient * apReadClient)
87+
{
88+
ChipLogProgress(NotSpecified, "Reading SupportedDeviceCategories is done.");
89+
}
90+
91+
void FabricSyncGetter::OnDeviceConnected(Messaging::ExchangeManager & exchangeMgr, const SessionHandle & sessionHandle)
92+
{
93+
mClient = std::make_unique<ReadClient>(app::InteractionModelEngine::GetInstance(), &exchangeMgr, *this /* callback */,
94+
ReadClient::InteractionType::Read);
95+
VerifyOrDie(mClient);
96+
97+
AttributePathParams readPaths[1];
98+
readPaths[0] = AttributePathParams(mEndpointId, Clusters::CommissionerControl::Id,
99+
Clusters::CommissionerControl::Attributes::SupportedDeviceCategories::Id);
100+
101+
ReadPrepareParams readParams(sessionHandle);
102+
103+
readParams.mpAttributePathParamsList = readPaths;
104+
readParams.mAttributePathParamsListSize = 1;
105+
106+
CHIP_ERROR err = mClient->SendRequest(readParams);
107+
108+
if (err != CHIP_NO_ERROR)
109+
{
110+
ChipLogError(NotSpecified, "Failed to read SupportedDeviceCategories from the bridged device.");
111+
OnDone(nullptr);
112+
return;
113+
}
114+
}
115+
116+
void FabricSyncGetter::OnDeviceConnectionFailure(const ScopedNodeId & peerId, CHIP_ERROR error)
117+
{
118+
ChipLogError(NotSpecified, "FabricSyncGetter failed to connect to " ChipLogFormatX64, ChipLogValueX64(peerId.GetNodeId()));
119+
120+
OnDone(nullptr);
121+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright (c) 2024 Project CHIP Authors
3+
* All rights reserved.
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*
17+
*/
18+
19+
#pragma once
20+
21+
#include <app/ReadClient.h>
22+
#include <controller/CHIPDeviceController.h>
23+
24+
#include <memory>
25+
#include <optional>
26+
27+
/**
28+
* @brief Class used to get FabricSynchronization from SupportedDeviceCategories attribute of Commissioner Control Cluster.
29+
*
30+
* Functionality:
31+
* - Establishes a CASE session to communicate with the remote bridge.
32+
* - Retrieves the attribute data from the endpoint which host Aggregator.
33+
* - Provides callbacks for success, error, and completion when retrieving data.
34+
*/
35+
class FabricSyncGetter : public chip::app::ReadClient::Callback
36+
{
37+
public:
38+
using OnDoneCallback = std::function<void(chip::TLV::TLVReader & data)>;
39+
40+
FabricSyncGetter();
41+
42+
/**
43+
* @brief Initiates the process of retrieving fabric synchronization data from the target device.
44+
*
45+
* @param onDoneCallback A callback function to be invoked when the data retrieval is complete.
46+
* @param controller The device controller used to establish a session with the target device.
47+
* @param nodeId The Node ID of the target device.
48+
* @param endpointId The Endpoint ID from which to retrieve the fabric synchronization data.
49+
* @return CHIP_ERROR Returns an error if the process fails, CHIP_NO_ERROR on success.
50+
*/
51+
CHIP_ERROR GetFabricSynchronizationData(OnDoneCallback onDoneCallback, chip::Controller::DeviceController & controller,
52+
chip::NodeId nodeId, chip::EndpointId endpointId);
53+
54+
///////////////////////////////////////////////////////////////
55+
// ReadClient::Callback implementation
56+
///////////////////////////////////////////////////////////////
57+
void OnAttributeData(const chip::app::ConcreteDataAttributePath & path, chip::TLV::TLVReader * data,
58+
const chip::app::StatusIB & status) override;
59+
void OnError(CHIP_ERROR error) override;
60+
void OnDone(chip::app::ReadClient * apReadClient) override;
61+
62+
///////////////////////////////////////////////////////////////
63+
// callbacks for CASE session establishment
64+
///////////////////////////////////////////////////////////////
65+
void OnDeviceConnected(chip::Messaging::ExchangeManager & exchangeMgr, const chip::SessionHandle & sessionHandle);
66+
void OnDeviceConnectionFailure(const chip::ScopedNodeId & peerId, CHIP_ERROR error);
67+
68+
private:
69+
std::unique_ptr<chip::app::ReadClient> mClient;
70+
71+
OnDoneCallback mOnDoneCallback;
72+
chip::EndpointId mEndpointId;
73+
chip::Callback::Callback<chip::OnDeviceConnected> mOnDeviceConnectedCallback;
74+
chip::Callback::Callback<chip::OnDeviceConnectionFailure> mOnDeviceConnectionFailureCallback;
75+
};

0 commit comments

Comments
 (0)