Skip to content

Commit b364ee6

Browse files
yufengwangcabzbarsky-apple
andauthoredDec 22, 2023
[Android] Return ConnectionFailureException which contains the connection state… (#31149)
* Return ConnectionFailureException which contains the connection state info in onConnectionFailure * Update src/app/CASESessionManager.h Co-authored-by: Boris Zbarsky <bzbarsky@apple.com> * Address review comments * Allow onFailure and onSetupFailure to be null * Update src/app/CASESessionManager.cpp Co-authored-by: Boris Zbarsky <bzbarsky@apple.com> * Address review comments --------- Co-authored-by: Boris Zbarsky <bzbarsky@apple.com>
1 parent 6484d4d commit b364ee6

9 files changed

+330
-10
lines changed
 

‎src/app/CASESessionManager.cpp

+66-2
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,56 @@ void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Cal
3636
uint8_t attemptCount, Callback::Callback<OnDeviceConnectionRetry> * onRetry
3737
#endif // CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
3838
)
39+
{
40+
FindOrEstablishSessionHelper(peerId, onConnection, onFailure, nullptr
41+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
42+
,
43+
attemptCount, onRetry
44+
#endif
45+
);
46+
}
47+
48+
void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Callback::Callback<OnDeviceConnected> * onConnection,
49+
Callback::Callback<OperationalSessionSetup::OnSetupFailure> * onSetupFailure
50+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
51+
,
52+
uint8_t attemptCount, Callback::Callback<OnDeviceConnectionRetry> * onRetry
53+
#endif
54+
)
55+
{
56+
FindOrEstablishSessionHelper(peerId, onConnection, nullptr, onSetupFailure
57+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
58+
,
59+
attemptCount, onRetry
60+
#endif
61+
);
62+
}
63+
64+
void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Callback::Callback<OnDeviceConnected> * onConnection,
65+
nullptr_t
66+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
67+
,
68+
uint8_t attemptCount, Callback::Callback<OnDeviceConnectionRetry> * onRetry
69+
#endif
70+
)
71+
{
72+
FindOrEstablishSessionHelper(peerId, onConnection, nullptr, nullptr
73+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
74+
,
75+
attemptCount, onRetry
76+
#endif
77+
);
78+
}
79+
80+
void CASESessionManager::FindOrEstablishSessionHelper(const ScopedNodeId & peerId,
81+
Callback::Callback<OnDeviceConnected> * onConnection,
82+
Callback::Callback<OnDeviceConnectionFailure> * onFailure,
83+
Callback::Callback<OperationalSessionSetup::OnSetupFailure> * onSetupFailure
84+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
85+
,
86+
uint8_t attemptCount, Callback::Callback<OnDeviceConnectionRetry> * onRetry
87+
#endif
88+
)
3989
{
4090
ChipLogDetail(CASESessionManager, "FindOrEstablishSession: PeerId = [%d:" ChipLogFormatX64 "]", peerId.GetFabricIndex(),
4191
ChipLogValueX64(peerId.GetNodeId()));
@@ -45,7 +95,6 @@ void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Cal
4595
if (session == nullptr)
4696
{
4797
ChipLogDetail(CASESessionManager, "FindOrEstablishSession: No existing OperationalSessionSetup instance found");
48-
4998
session = mConfig.sessionSetupPool->Allocate(mConfig.sessionInitParams, mConfig.clientPool, peerId, this);
5099

51100
if (session == nullptr)
@@ -54,6 +103,13 @@ void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Cal
54103
{
55104
onFailure->mCall(onFailure->mContext, peerId, CHIP_ERROR_NO_MEMORY);
56105
}
106+
107+
if (onSetupFailure != nullptr)
108+
{
109+
OperationalSessionSetup::ConnnectionFailureInfo failureInfo(peerId, CHIP_ERROR_NO_MEMORY,
110+
SessionEstablishmentStage::kUnknown);
111+
onSetupFailure->mCall(onSetupFailure->mContext, failureInfo);
112+
}
57113
return;
58114
}
59115
}
@@ -66,7 +122,15 @@ void CASESessionManager::FindOrEstablishSession(const ScopedNodeId & peerId, Cal
66122
}
67123
#endif // CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
68124

69-
session->Connect(onConnection, onFailure);
125+
if (onFailure != nullptr)
126+
{
127+
session->Connect(onConnection, onFailure);
128+
}
129+
130+
if (onSetupFailure != nullptr)
131+
{
132+
session->Connect(onConnection, onSetupFailure);
133+
}
70134
}
71135

72136
void CASESessionManager::ReleaseSessionsForFabric(FabricIndex fabricIndex)

‎src/app/CASESessionManager.h

+65
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,63 @@ class CASESessionManager : public OperationalSessionReleaseDelegate, public Sess
8787
#endif // CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
8888
);
8989

90+
/**
91+
* Find an existing session for the given node ID or trigger a new session request.
92+
*
93+
* The caller can optionally provide `onConnection` and `onSetupFailure`
94+
* callback objects. If provided, these will be used to inform the caller about successful or
95+
* failed connection establishment.
96+
*
97+
* If the connection is already established, the `onConnection` callback will be immediately called,
98+
* before `FindOrEstablishSession` returns.
99+
*
100+
* The `onSetupFailure` callback may be called before the `FindOrEstablishSession`
101+
* call returns, for error cases that are detected synchronously.
102+
*
103+
* The `attemptCount` parameter can be used to automatically retry multiple times if session setup is
104+
* not successful.
105+
*
106+
* @param peerId The node ID to find or establish a session with.
107+
* @param onConnection A callback to be called upon successful connection establishment.
108+
* @param onSetupFailure A callback to be called upon an extended device connection failure.
109+
* @param attemptCount The number of retry attempts if session setup fails (default is 1).
110+
* @param onRetry A callback to be called on a retry attempt (enabled by a config flag).
111+
*/
112+
void FindOrEstablishSession(const ScopedNodeId & peerId, Callback::Callback<OnDeviceConnected> * onConnection,
113+
Callback::Callback<OperationalSessionSetup::OnSetupFailure> * onSetupFailure
114+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
115+
,
116+
uint8_t attemptCount = 1, Callback::Callback<OnDeviceConnectionRetry> * onRetry = nullptr
117+
#endif // CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
118+
);
119+
120+
/**
121+
* Find an existing session for the given node ID or trigger a new session request.
122+
*
123+
* The caller can optionally provide `onConnection`
124+
* callback objects. If provided, these will be used to inform the caller about successful connection establishment.
125+
*
126+
* If the connection is already established, the `onConnection` callback will be immediately called,
127+
* before `FindOrEstablishSession` returns.
128+
*
129+
* The `attemptCount` parameter can be used to automatically retry multiple times if session setup is
130+
* not successful.
131+
*
132+
* This function allows passing 'nullptr' for the error handler to compile, which is useful in scenarios where error
133+
* handling is not needed.
134+
*
135+
* @param peerId The node ID to find or establish a session with.
136+
* @param onConnection A callback to be called upon successful connection establishment.
137+
* @param attemptCount The number of retry attempts if session setup fails (default is 1).
138+
* @param onRetry A callback to be called on a retry attempt (enabled by a config flag).
139+
*/
140+
void FindOrEstablishSession(const ScopedNodeId & peerId, Callback::Callback<OnDeviceConnected> * onConnection, nullptr_t
141+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
142+
,
143+
uint8_t attemptCount = 1, Callback::Callback<OnDeviceConnectionRetry> * onRetry = nullptr
144+
#endif // CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
145+
);
146+
90147
void ReleaseSessionsForFabric(FabricIndex fabricIndex);
91148

92149
void ReleaseAllSessions();
@@ -112,6 +169,14 @@ class CASESessionManager : public OperationalSessionReleaseDelegate, public Sess
112169

113170
Optional<SessionHandle> FindExistingSession(const ScopedNodeId & peerId) const;
114171

172+
void FindOrEstablishSessionHelper(const ScopedNodeId & peerId, Callback::Callback<OnDeviceConnected> * onConnection,
173+
Callback::Callback<OnDeviceConnectionFailure> * onFailure,
174+
Callback::Callback<OperationalSessionSetup::OnSetupFailure> * onSetupFailure,
175+
#if CHIP_DEVICE_CONFIG_ENABLE_AUTOMATIC_CASE_RETRIES
176+
uint8_t attemptCount, Callback::Callback<OnDeviceConnectionRetry> * onRetry
177+
#endif
178+
);
179+
115180
CASESessionManagerConfig mConfig;
116181
};
117182

‎src/controller/CHIPDeviceController.h

+26-2
Original file line numberDiff line numberDiff line change
@@ -220,11 +220,11 @@ class DLL_EXPORT DeviceController : public AbstractDnssdDiscoveryController
220220
* This function finds the device corresponding to deviceId, and establishes
221221
* a CASE session with it.
222222
*
223-
* Once the CASE session is successfully established the `onConnectedDevice`
223+
* Once the CASE session is successfully established the `onConnection`
224224
* callback is called. This can happen before GetConnectedDevice returns if
225225
* there is an existing CASE session.
226226
*
227-
* If a CASE sessions fails to be established, the `onError` callback will
227+
* If a CASE sessions fails to be established, the `onFailure` callback will
228228
* be called. This can also happen before GetConnectedDevice returns.
229229
*
230230
* An error return from this function means that neither callback has been
@@ -238,6 +238,30 @@ class DLL_EXPORT DeviceController : public AbstractDnssdDiscoveryController
238238
return CHIP_NO_ERROR;
239239
}
240240

241+
/**
242+
* This function finds the device corresponding to deviceId, and establishes
243+
* a CASE session with it.
244+
*
245+
* Once the CASE session is successfully established the `onConnection`
246+
* callback is called. This can happen before GetConnectedDevice returns if
247+
* there is an existing CASE session.
248+
*
249+
* If a CASE sessions fails to be established, the `onSetupFailure` callback will
250+
* be called. This can also happen before GetConnectedDevice returns.
251+
*
252+
* An error return from this function means that neither callback has been
253+
* called yet, and neither callback will be called in the future.
254+
*/
255+
CHIP_ERROR
256+
GetConnectedDevice(NodeId peerNodeId, Callback::Callback<OnDeviceConnected> * onConnection,
257+
chip::Callback::Callback<OperationalSessionSetup::OnSetupFailure> * onSetupFailure)
258+
{
259+
VerifyOrReturnError(mState == State::Initialized, CHIP_ERROR_INCORRECT_STATE);
260+
mSystemState->CASESessionMgr()->FindOrEstablishSession(ScopedNodeId(peerNodeId, GetFabricIndex()), onConnection,
261+
onSetupFailure);
262+
return CHIP_NO_ERROR;
263+
}
264+
241265
/**
242266
* @brief
243267
* Compute a PASE verifier and passcode ID for the desired setup pincode.

‎src/controller/java/AndroidCallbacks.cpp

+6-4
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
* limitations under the License.
1616
*/
1717
#include "AndroidCallbacks.h"
18+
#include <controller/java/AndroidConnectionFailureExceptions.h>
1819
#include <controller/java/AndroidControllerExceptions.h>
1920
#ifdef USE_JAVA_TLV_ENCODE_DECODE
2021
#include <controller/java/CHIPAttributeTLVValueDecoder.h>
@@ -114,7 +115,8 @@ void GetConnectedDeviceCallback::OnDeviceConnectedFn(void * context, Messaging::
114115
VerifyOrReturn(!env->ExceptionCheck(), env->ExceptionDescribe());
115116
}
116117

117-
void GetConnectedDeviceCallback::OnDeviceConnectionFailureFn(void * context, const ScopedNodeId & peerId, CHIP_ERROR error)
118+
void GetConnectedDeviceCallback::OnDeviceConnectionFailureFn(void * context,
119+
const OperationalSessionSetup::ConnnectionFailureInfo & failureInfo)
118120
{
119121
JNIEnv * env = JniReferences::GetInstance().GetEnvForCurrentThread();
120122
VerifyOrReturn(env != nullptr, ChipLogError(Controller, "Could not get JNIEnv for current thread"));
@@ -135,15 +137,15 @@ void GetConnectedDeviceCallback::OnDeviceConnectionFailureFn(void * context, con
135137
VerifyOrReturn(failureMethod != nullptr, ChipLogError(Controller, "Could not find onConnectionFailure method"));
136138

137139
jthrowable exception;
138-
CHIP_ERROR err = AndroidControllerExceptions::GetInstance().CreateAndroidControllerException(env, ErrorStr(error),
139-
error.AsInteger(), exception);
140+
CHIP_ERROR err = AndroidConnectionFailureExceptions::GetInstance().CreateAndroidConnectionFailureException(
141+
env, failureInfo.error.Format(), failureInfo.error.AsInteger(), failureInfo.sessionStage, exception);
140142
VerifyOrReturn(
141143
err == CHIP_NO_ERROR,
142144
ChipLogError(Controller,
143145
"Unable to create AndroidControllerException on GetConnectedDeviceCallback::OnDeviceConnectionFailureFn: %s",
144146
ErrorStr(err)));
145147
DeviceLayer::StackUnlock unlock;
146-
env->CallVoidMethod(javaCallback, failureMethod, peerId.GetNodeId(), exception);
148+
env->CallVoidMethod(javaCallback, failureMethod, failureInfo.peerId.GetNodeId(), exception);
147149
VerifyOrReturn(!env->ExceptionCheck(), env->ExceptionDescribe());
148150
}
149151

‎src/controller/java/AndroidCallbacks.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -39,10 +39,10 @@ struct GetConnectedDeviceCallback
3939
~GetConnectedDeviceCallback();
4040

4141
static void OnDeviceConnectedFn(void * context, Messaging::ExchangeManager & exchangeMgr, const SessionHandle & sessionHandle);
42-
static void OnDeviceConnectionFailureFn(void * context, const ScopedNodeId & peerId, CHIP_ERROR error);
42+
static void OnDeviceConnectionFailureFn(void * context, const OperationalSessionSetup::ConnnectionFailureInfo & failureInfo);
4343

4444
Callback::Callback<OnDeviceConnected> mOnSuccess;
45-
Callback::Callback<OnDeviceConnectionFailure> mOnFailure;
45+
Callback::Callback<OperationalSessionSetup::OnSetupFailure> mOnFailure;
4646
JniGlobalReference mWrapperCallbackRef;
4747
JniGlobalReference mJavaCallbackRef;
4848
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
/*
2+
*
3+
* Copyright (c) 2023 Project CHIP Authors
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+
#include "AndroidConnectionFailureExceptions.h"
19+
20+
#include <lib/core/CHIPError.h>
21+
#include <lib/support/CHIPJNIError.h>
22+
#include <lib/support/JniReferences.h>
23+
#include <lib/support/JniTypeWrappers.h>
24+
25+
namespace chip {
26+
27+
CHIP_ERROR AndroidConnectionFailureExceptions::CreateAndroidConnectionFailureException(JNIEnv * env, const char * message,
28+
uint32_t errorCode,
29+
SessionEstablishmentStage state,
30+
jthrowable & outEx)
31+
{
32+
jclass controllerExceptionCls;
33+
CHIP_ERROR err = JniReferences::GetInstance().GetLocalClassRef(env, "chip/devicecontroller/ConnectionFailureException",
34+
controllerExceptionCls);
35+
VerifyOrReturnError(err == CHIP_NO_ERROR, CHIP_JNI_ERROR_TYPE_NOT_FOUND);
36+
37+
jmethodID exceptionConstructor = env->GetMethodID(controllerExceptionCls, "<init>", "(JILjava/lang/String;)V");
38+
outEx = static_cast<jthrowable>(env->NewObject(controllerExceptionCls, exceptionConstructor, static_cast<jlong>(errorCode),
39+
static_cast<jint>(state), env->NewStringUTF(message)));
40+
VerifyOrReturnError(outEx != nullptr, CHIP_JNI_ERROR_TYPE_NOT_FOUND);
41+
return CHIP_NO_ERROR;
42+
}
43+
44+
} // namespace chip
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
/*
2+
*
3+
* Copyright (c) 2023 Project CHIP Authors
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+
#pragma once
19+
20+
#include <app/OperationalSessionSetup.h>
21+
#include <jni.h>
22+
#include <lib/core/CHIPError.h>
23+
24+
namespace chip {
25+
class AndroidConnectionFailureExceptions
26+
{
27+
public:
28+
AndroidConnectionFailureExceptions(const AndroidConnectionFailureExceptions &) = delete;
29+
AndroidConnectionFailureExceptions(const AndroidConnectionFailureExceptions &&) = delete;
30+
AndroidConnectionFailureExceptions & operator=(const AndroidConnectionFailureExceptions &) = delete;
31+
32+
static AndroidConnectionFailureExceptions & GetInstance()
33+
{
34+
static AndroidConnectionFailureExceptions androidConnectionFailureExceptions;
35+
return androidConnectionFailureExceptions;
36+
}
37+
38+
/**
39+
* Creates a Java ConnectionFailureException object in outEx.
40+
*/
41+
CHIP_ERROR CreateAndroidConnectionFailureException(JNIEnv * env, const char * message, uint32_t errorCode,
42+
SessionEstablishmentStage state, jthrowable & outEx);
43+
44+
private:
45+
AndroidConnectionFailureExceptions() {}
46+
};
47+
} // namespace chip

‎src/controller/java/BUILD.gn

+3
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@ shared_library("jni") {
4545
"AndroidClusterExceptions.h",
4646
"AndroidCommissioningWindowOpener.cpp",
4747
"AndroidCommissioningWindowOpener.h",
48+
"AndroidConnectionFailureExceptions.cpp",
49+
"AndroidConnectionFailureExceptions.h",
4850
"AndroidControllerExceptions.cpp",
4951
"AndroidControllerExceptions.h",
5052
"AndroidCurrentFabricRemover.cpp",
@@ -439,6 +441,7 @@ android_library("java") {
439441
"src/chip/devicecontroller/ChipCommandType.java",
440442
"src/chip/devicecontroller/ChipDeviceController.java",
441443
"src/chip/devicecontroller/ChipDeviceControllerException.java",
444+
"src/chip/devicecontroller/ConnectionFailureException.java",
442445
"src/chip/devicecontroller/ControllerParams.java",
443446
"src/chip/devicecontroller/DeviceAttestationDelegate.java",
444447
"src/chip/devicecontroller/DiscoveredDevice.java",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
/*
2+
* Copyright (c) 2023 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+
package chip.devicecontroller;
19+
20+
@SuppressWarnings("serial")
21+
public class ConnectionFailureException extends ChipDeviceControllerException {
22+
public enum ConnectionState {
23+
UNKNOWN, // Unknown state
24+
NOT_IN_KEY_EXCHANGE, // Not currently in key exchange process
25+
SENT_SIGMA1, // Sigma1 message sent
26+
RECEIVED_SIGMA1, // Sigma1 message received
27+
SENT_SIGMA2, // Sigma2 message sent
28+
RECEIVED_SIGMA2, // Sigma2 message received
29+
SENT_SIGMA3, // Sigma3 message sent
30+
RECEIVED_SIGMA3 // Sigma3 message received
31+
}
32+
33+
private ConnectionState connectionState;
34+
35+
public ConnectionFailureException() {
36+
super();
37+
}
38+
39+
public ConnectionFailureException(long errorCode, int connectionState, String message) {
40+
super(errorCode, message);
41+
this.connectionState = mapIntToConnectionState(connectionState);
42+
}
43+
44+
public ConnectionState getConnectionState() {
45+
return connectionState;
46+
}
47+
48+
// Helper method to map an int to ConnectionState enum
49+
private ConnectionState mapIntToConnectionState(int value) {
50+
switch (value) {
51+
case 0:
52+
return ConnectionState.UNKNOWN;
53+
case 1:
54+
return ConnectionState.NOT_IN_KEY_EXCHANGE;
55+
case 2:
56+
return ConnectionState.SENT_SIGMA1;
57+
case 3:
58+
return ConnectionState.RECEIVED_SIGMA1;
59+
case 4:
60+
return ConnectionState.SENT_SIGMA2;
61+
case 5:
62+
return ConnectionState.RECEIVED_SIGMA2;
63+
case 6:
64+
return ConnectionState.SENT_SIGMA3;
65+
case 7:
66+
return ConnectionState.RECEIVED_SIGMA3;
67+
default:
68+
throw new IllegalArgumentException("Invalid connection state value: " + value);
69+
}
70+
}
71+
}

0 commit comments

Comments
 (0)
Please sign in to comment.