16
16
*/
17
17
18
18
#include " FabricAdmin.h"
19
+ #include < AppMain.h>
20
+ #include < bridge/include/FabricBridge.h>
21
+ #include < controller/CHIPDeviceControllerFactory.h>
19
22
20
23
using namespace ::chip;
21
24
@@ -28,16 +31,35 @@ constexpr uint32_t kCommissionPrepareTimeMs = 500;
28
31
} // namespace
29
32
30
33
FabricAdmin FabricAdmin::sInstance ;
34
+ app::DefaultICDClientStorage FabricAdmin::sICDClientStorage ;
35
+ app::CheckInHandler FabricAdmin::sCheckInHandler ;
31
36
32
37
FabricAdmin & FabricAdmin::Instance ()
33
38
{
34
39
if (!sInstance .mInitialized )
35
40
{
36
- sInstance .Init ();
41
+ VerifyOrDie ( sInstance .Init () == CHIP_NO_ERROR );
37
42
}
38
43
return sInstance ;
39
44
}
40
45
46
+ CHIP_ERROR FabricAdmin::Init ()
47
+ {
48
+ IcdManager::Instance ().SetDelegate (&sInstance );
49
+
50
+ ReturnLogErrorOnFailure (sICDClientStorage .Init (GetPersistentStorageDelegate (), GetSessionKeystore ()));
51
+
52
+ auto engine = chip::app::InteractionModelEngine::GetInstance ();
53
+ VerifyOrReturnError (engine != nullptr , CHIP_ERROR_INCORRECT_STATE);
54
+ ReturnLogErrorOnFailure (IcdManager::Instance ().Init (&sICDClientStorage , engine));
55
+ ReturnLogErrorOnFailure (sCheckInHandler .Init (Controller::DeviceControllerFactory::GetInstance ().GetSystemState ()->ExchangeMgr (),
56
+ &sICDClientStorage , &IcdManager::Instance (), engine));
57
+
58
+ mInitialized = true ;
59
+
60
+ return CHIP_NO_ERROR;
61
+ }
62
+
41
63
CHIP_ERROR FabricAdmin::OpenCommissioningWindow (Controller::CommissioningWindowVerifierParams params, FabricIndex fabricIndex)
42
64
{
43
65
ScopedNodeId scopedNodeId (params.GetNodeId (), fabricIndex);
@@ -116,6 +138,46 @@ CHIP_ERROR FabricAdmin::KeepActive(ScopedNodeId scopedNodeId, uint32_t stayActiv
116
138
return CHIP_NO_ERROR;
117
139
}
118
140
141
+ void FabricAdmin::OnCheckInCompleted (const app::ICDClientInfo & clientInfo)
142
+ {
143
+ // Accessing mPendingCheckIn should only be done while holding ChipStackLock
144
+ assertChipStackLockedByCurrentThread ();
145
+ ScopedNodeId scopedNodeId = clientInfo.peer_node ;
146
+ auto it = mPendingCheckIn .find (scopedNodeId);
147
+ VerifyOrReturn (it != mPendingCheckIn .end ());
148
+
149
+ KeepActiveDataForCheckIn checkInData = it->second ;
150
+ // Removed from pending map as check-in from this node has occured and we will handle the pending KeepActive
151
+ // request.
152
+ mPendingCheckIn .erase (scopedNodeId);
153
+
154
+ auto timeNow = System::SystemClock ().GetMonotonicTimestamp ();
155
+ if (timeNow > checkInData.mRequestExpiryTimestamp )
156
+ {
157
+ ChipLogError (NotSpecified,
158
+ " ICD check-in for device we have been waiting, came after KeepActive expiry. Request dropped for ID: "
159
+ " [%d:0x " ChipLogFormatX64 " ]" ,
160
+ scopedNodeId.GetFabricIndex (), ChipLogValueX64 (scopedNodeId.GetNodeId ()));
161
+ return ;
162
+ }
163
+
164
+ // TODO https://github.com/CHIP-Specifications/connectedhomeip-spec/issues/10448. Spec does
165
+ // not define what to do if we fail to send the StayActiveRequest. We are assuming that any
166
+ // further attempts to send a StayActiveRequest will result in a similar failure. Because
167
+ // there is no mechanism for us to communicate with the client that sent out the KeepActive
168
+ // command that there was a failure, we simply fail silently. After spec issue is
169
+ // addressed, we can implement what spec defines here.
170
+ auto onDone = [=](uint32_t promisedActiveDuration) {
171
+ bridge::FabricBridge::Instance ().ActiveChanged (scopedNodeId, promisedActiveDuration);
172
+ };
173
+ CHIP_ERROR err = StayActiveSender::SendStayActiveCommand (checkInData.mStayActiveDurationMs , clientInfo.peer_node ,
174
+ app::InteractionModelEngine::GetInstance (), onDone);
175
+ if (err != CHIP_NO_ERROR)
176
+ {
177
+ ChipLogError (NotSpecified, " Failed to send StayActive command %s" , err.AsString ());
178
+ }
179
+ }
180
+
119
181
void FabricAdmin::OnCommissioningComplete (NodeId deviceId, CHIP_ERROR err)
120
182
{
121
183
if (mNodeId != deviceId)
0 commit comments