Skip to content

Commit ae1841a

Browse files
committed
Convert to JSException only NSException from sync methods (#50193)
Summary: Pull Request resolved: #50193 This fix makes sure that we convert to JSException only NSException thrwn by sync methods. Currently, nothing in the stack will be capable of understanding that js error if it is triggered by an exception raised by an asyc method. See reactwg/react-native-new-architecture#276 for further details We need to cherry pick this in 0.78 and 0.79 ## Changelog: [iOS][Fixed] - Make sure the TM infra does not crash on NSException when triggered by async method Reviewed By: fabriziocucci Differential Revision: D71619229 fbshipit-source-id: b87aef5dd2720a2641c8da0904da651866370dc6
1 parent aa75e94 commit ae1841a

File tree

1 file changed

+24
-13
lines changed
  • packages/react-native/ReactCommon/react/nativemodule/core/platform/ios/ReactCommon

1 file changed

+24
-13
lines changed

packages/react-native/ReactCommon/react/nativemodule/core/platform/ios/ReactCommon/RCTTurboModule.mm

+24-13
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ static int32_t getUniqueId()
5757

5858
static jsi::String convertNSStringToJSIString(jsi::Runtime &runtime, NSString *value)
5959
{
60-
return jsi::String::createFromUtf8(runtime, [value UTF8String] ?: "");
60+
return jsi::String::createFromUtf8(runtime, [value UTF8String] ? [value UTF8String] : "");
6161
}
6262

6363
static jsi::Object convertNSDictionaryToJSIObject(jsi::Runtime &runtime, NSDictionary *value)
@@ -203,7 +203,11 @@ id convertJSIValueToObjCObject(jsi::Runtime &runtime, const jsi::Value &value, s
203203
/**
204204
* Creates JSError with current JS runtime and NSException stack trace.
205205
*/
206-
static jsi::JSError convertNSExceptionToJSError(jsi::Runtime &runtime, NSException *exception)
206+
static jsi::JSError convertNSExceptionToJSError(
207+
jsi::Runtime &runtime,
208+
NSException *exception,
209+
const std::string &moduleName,
210+
const std::string &methodName)
207211
{
208212
std::string reason = [exception.reason UTF8String];
209213

@@ -214,7 +218,8 @@ id convertJSIValueToObjCObject(jsi::Runtime &runtime, const jsi::Value &value, s
214218
cause.setProperty(
215219
runtime, "stackReturnAddresses", convertNSArrayToJSIArray(runtime, exception.callStackReturnAddresses));
216220

217-
jsi::Value error = createJSRuntimeError(runtime, "Exception in HostFunction: " + reason);
221+
std::string message = moduleName + "." + methodName + " raised an exception: " + reason;
222+
jsi::Value error = createJSRuntimeError(runtime, message);
218223
error.asObject(runtime).setProperty(runtime, "cause", std::move(cause));
219224
return {runtime, std::move(error)};
220225
}
@@ -346,28 +351,34 @@ id convertJSIValueToObjCObject(jsi::Runtime &runtime, const jsi::Value &value, s
346351
}
347352

348353
if (isSync) {
349-
TurboModulePerfLogger::syncMethodCallExecutionStart(moduleName, methodNameStr.c_str());
354+
TurboModulePerfLogger::syncMethodCallExecutionStart(moduleName, methodName);
350355
} else {
351-
TurboModulePerfLogger::asyncMethodCallExecutionStart(moduleName, methodNameStr.c_str(), asyncCallCounter);
356+
TurboModulePerfLogger::asyncMethodCallExecutionStart(moduleName, methodName, asyncCallCounter);
352357
}
353358

354359
@try {
355360
[inv invokeWithTarget:strongModule];
356361
} @catch (NSException *exception) {
357-
throw convertNSExceptionToJSError(runtime, exception);
362+
if (isSync) {
363+
// We can only convert NSException to JSError in sync method calls.
364+
// See https://github.com/reactwg/react-native-new-architecture/discussions/276#discussioncomment-12567155
365+
throw convertNSExceptionToJSError(runtime, exception, std::string{moduleName}, methodNameStr);
366+
} else {
367+
@throw exception;
368+
}
358369
} @finally {
359370
[retainedObjectsForInvocation removeAllObjects];
360371
}
361372

362373
if (!isSync) {
363-
TurboModulePerfLogger::asyncMethodCallExecutionEnd(moduleName, methodNameStr.c_str(), asyncCallCounter);
374+
TurboModulePerfLogger::asyncMethodCallExecutionEnd(moduleName, methodName, asyncCallCounter);
364375
return;
365376
}
366377

367378
void *rawResult;
368379
[inv getReturnValue:&rawResult];
369380
result = (__bridge id)rawResult;
370-
TurboModulePerfLogger::syncMethodCallExecutionEnd(moduleName, methodNameStr.c_str());
381+
TurboModulePerfLogger::syncMethodCallExecutionEnd(moduleName, methodName);
371382
};
372383

373384
if (isSync) {
@@ -409,23 +420,23 @@ TraceSection s(
409420
}
410421

411422
if (shouldVoidMethodsExecuteSync_) {
412-
TurboModulePerfLogger::syncMethodCallExecutionStart(moduleName, methodNameStr.c_str());
423+
TurboModulePerfLogger::syncMethodCallExecutionStart(moduleName, methodName);
413424
} else {
414-
TurboModulePerfLogger::asyncMethodCallExecutionStart(moduleName, methodNameStr.c_str(), asyncCallCounter);
425+
TurboModulePerfLogger::asyncMethodCallExecutionStart(moduleName, methodName, asyncCallCounter);
415426
}
416427

417428
@try {
418429
[inv invokeWithTarget:strongModule];
419430
} @catch (NSException *exception) {
420-
throw convertNSExceptionToJSError(runtime, exception);
431+
throw convertNSExceptionToJSError(runtime, exception, std::string{moduleName}, methodNameStr);
421432
} @finally {
422433
[retainedObjectsForInvocation removeAllObjects];
423434
}
424435

425436
if (shouldVoidMethodsExecuteSync_) {
426-
TurboModulePerfLogger::syncMethodCallExecutionEnd(moduleName, methodNameStr.c_str());
437+
TurboModulePerfLogger::syncMethodCallExecutionEnd(moduleName, methodName);
427438
} else {
428-
TurboModulePerfLogger::asyncMethodCallExecutionEnd(moduleName, methodNameStr.c_str(), asyncCallCounter);
439+
TurboModulePerfLogger::asyncMethodCallExecutionEnd(moduleName, methodName, asyncCallCounter);
429440
}
430441

431442
return;

0 commit comments

Comments
 (0)