mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
4c998fd05d
Summary: ## Motivation The concept behind JSCallInvoker doesn't necessarily have to apply only to the JS thread. On Android, we need to re-use this abstraction to allow execution of async method calls on the NativeModules thread. Reviewed By: PeteTheHeat Differential Revision: D17377313 fbshipit-source-id: 3d9075cbfce0b908d800a366947cfd16a3013d1c
136 lines
4.2 KiB
Plaintext
136 lines
4.2 KiB
Plaintext
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#import "RCTExceptionsManager.h"
|
|
|
|
#import <FBReactNativeSpec/FBReactNativeSpec.h>
|
|
#import <React/RCTConvert.h>
|
|
#import <React/RCTDefines.h>
|
|
#import <React/RCTLog.h>
|
|
#import <React/RCTRedBox.h>
|
|
#import <React/RCTRootView.h>
|
|
|
|
#import "CoreModulesPlugins.h"
|
|
|
|
@interface RCTExceptionsManager() <NativeExceptionsManagerSpec>
|
|
|
|
@end
|
|
|
|
@implementation RCTExceptionsManager
|
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (instancetype)initWithDelegate:(id<RCTExceptionsManagerDelegate>)delegate
|
|
{
|
|
if ((self = [self init])) {
|
|
_delegate = delegate;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(reportSoftException:(NSString *)message
|
|
stack:(NSArray<NSDictionary *> *)stack
|
|
exceptionId:(double)exceptionId)
|
|
{
|
|
[_bridge.redBox showErrorMessage:message withStack:stack errorCookie:((int)exceptionId)];
|
|
|
|
if (_delegate) {
|
|
[_delegate handleSoftJSExceptionWithMessage:message stack:stack exceptionId:[NSNumber numberWithDouble:exceptionId]];
|
|
}
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(reportFatalException:(NSString *)message
|
|
stack:(NSArray<NSDictionary *> *)stack
|
|
exceptionId:(double) exceptionId)
|
|
{
|
|
[_bridge.redBox showErrorMessage:message withStack:stack errorCookie:((int)exceptionId)];
|
|
|
|
if (_delegate) {
|
|
[_delegate handleFatalJSExceptionWithMessage:message stack:stack exceptionId:[NSNumber numberWithDouble:exceptionId]];
|
|
}
|
|
|
|
static NSUInteger reloadRetries = 0;
|
|
if (!RCT_DEBUG && reloadRetries < _maxReloadAttempts) {
|
|
reloadRetries++;
|
|
[_bridge reload];
|
|
} else {
|
|
NSString *description = [@"Unhandled JS Exception: " stringByAppendingString:message];
|
|
NSDictionary *errorInfo = @{ NSLocalizedDescriptionKey: description, RCTJSStackTraceKey: stack };
|
|
RCTFatal([NSError errorWithDomain:RCTErrorDomain code:0 userInfo:errorInfo]);
|
|
}
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(updateExceptionMessage:(NSString *)message
|
|
stack:(NSArray<NSDictionary *> *)stack
|
|
exceptionId:(double)exceptionId)
|
|
{
|
|
[_bridge.redBox updateErrorMessage:message withStack:stack errorCookie:((int)exceptionId)];
|
|
|
|
if (_delegate && [_delegate respondsToSelector:@selector(updateJSExceptionWithMessage:stack:exceptionId:)]) {
|
|
[_delegate updateJSExceptionWithMessage:message stack:stack exceptionId:[NSNumber numberWithDouble:exceptionId]];
|
|
}
|
|
}
|
|
|
|
// Deprecated. Use reportFatalException directly instead.
|
|
RCT_EXPORT_METHOD(reportUnhandledException:(NSString *)message
|
|
stack:(NSArray<NSDictionary *> *)stack)
|
|
{
|
|
[self reportFatalException:message stack:stack exceptionId:-1];
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(dismissRedbox)
|
|
{
|
|
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(reportException:(JS::NativeExceptionsManager::ExceptionData &)data)
|
|
{
|
|
NSString *message = data.message();
|
|
double exceptionId = data.id_();
|
|
|
|
// Reserialize data.stack() into an array of untyped dictionaries.
|
|
// TODO: (moti) T53588496 Replace `(NSArray<NSDictionary *> *)stack` in
|
|
// reportFatalException etc with a typed interface.
|
|
NSMutableArray<NSDictionary *> *stackArray = [NSMutableArray<NSDictionary *> new];
|
|
for (auto frame: data.stack()) {
|
|
NSMutableDictionary * frameDict = [NSMutableDictionary new];
|
|
if (frame.column().hasValue()) {
|
|
frameDict[@"column"] = @(frame.column().value());
|
|
}
|
|
frameDict[@"file"] = frame.file();
|
|
if (frame.lineNumber().hasValue()) {
|
|
frameDict[@"lineNumber"] = @(frame.lineNumber().value());
|
|
}
|
|
frameDict[@"methodName"] = frame.methodName();
|
|
if (frame.collapse().hasValue()) {
|
|
frameDict[@"collapse"] = @(frame.collapse().value());
|
|
}
|
|
[stackArray addObject:frameDict];
|
|
}
|
|
|
|
if (data.isFatal()) {
|
|
[self reportFatalException:message stack:stackArray exceptionId:exceptionId];
|
|
} else {
|
|
[self reportSoftException:message stack:stackArray exceptionId:exceptionId];
|
|
}
|
|
}
|
|
|
|
- (std::shared_ptr<facebook::react::TurboModule>)getTurboModuleWithJsInvoker:
|
|
(std::shared_ptr<facebook::react::CallInvoker>)jsInvoker
|
|
{
|
|
return std::make_shared<facebook::react::NativeExceptionsManagerSpecJSI>(self, jsInvoker);
|
|
}
|
|
|
|
@end
|
|
|
|
Class RCTExceptionsManagerCls(void)
|
|
{
|
|
return RCTExceptionsManager.class;
|
|
}
|