2015-03-23 20:28:42 +00:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-02-20 04:10:52 +00:00
|
|
|
|
|
|
|
#import "RCTExceptionsManager.h"
|
|
|
|
|
2015-04-21 12:26:51 +00:00
|
|
|
#import "RCTDefines.h"
|
2015-04-09 17:39:28 +00:00
|
|
|
#import "RCTLog.h"
|
2015-02-20 04:10:52 +00:00
|
|
|
#import "RCTRedBox.h"
|
2015-04-09 17:39:28 +00:00
|
|
|
#import "RCTRootView.h"
|
2015-02-20 04:10:52 +00:00
|
|
|
|
|
|
|
@implementation RCTExceptionsManager
|
2015-03-11 20:53:30 +00:00
|
|
|
{
|
|
|
|
__weak id<RCTExceptionsManagerDelegate> _delegate;
|
2015-04-09 17:39:28 +00:00
|
|
|
NSUInteger _reloadRetries;
|
2015-03-11 20:53:30 +00:00
|
|
|
}
|
|
|
|
|
2015-04-08 12:42:43 +00:00
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-03-11 20:53:30 +00:00
|
|
|
- (instancetype)initWithDelegate:(id<RCTExceptionsManagerDelegate>)delegate
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
|
|
|
_delegate = delegate;
|
2015-04-09 17:39:28 +00:00
|
|
|
_maxReloadAttempts = 0;
|
2015-03-11 20:53:30 +00:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
return [self initWithDelegate:nil];
|
|
|
|
}
|
2015-02-20 04:10:52 +00:00
|
|
|
|
2015-05-13 17:56:09 +00:00
|
|
|
RCT_EXPORT_METHOD(reportSoftException:(NSString *)message
|
|
|
|
stack:(NSArray *)stack)
|
|
|
|
{
|
|
|
|
// TODO(#7070533): report a soft error to the server
|
|
|
|
if (_delegate) {
|
|
|
|
[_delegate handleSoftJSExceptionWithMessage:message stack:stack];
|
|
|
|
return;
|
|
|
|
}
|
2015-06-11 17:39:52 +00:00
|
|
|
[[RCTRedBox sharedInstance] showErrorMessage:message withStack:stack];
|
2015-05-13 17:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
RCT_EXPORT_METHOD(reportFatalException:(NSString *)message
|
2015-07-15 11:01:38 +00:00
|
|
|
stack:(NSArray *)stack
|
|
|
|
exceptionId:(__unused NSNumber *)exceptionId)
|
2015-02-20 04:10:52 +00:00
|
|
|
{
|
2015-03-11 20:53:30 +00:00
|
|
|
if (_delegate) {
|
2015-05-13 17:56:09 +00:00
|
|
|
[_delegate handleFatalJSExceptionWithMessage:message stack:stack];
|
2015-04-09 17:39:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-04-21 16:48:29 +00:00
|
|
|
[[RCTRedBox sharedInstance] showErrorMessage:message withStack:stack];
|
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
if (!RCT_DEBUG) {
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
static NSUInteger reloadRetries = 0;
|
|
|
|
const NSUInteger maxMessageLength = 75;
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
if (reloadRetries < _maxReloadAttempts) {
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
reloadRetries++;
|
|
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTReloadNotification
|
|
|
|
object:nil];
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
} else {
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
if (message.length > maxMessageLength) {
|
|
|
|
message = [[message substringToIndex:maxMessageLength] stringByAppendingString:@"..."];
|
|
|
|
}
|
2015-04-19 19:55:46 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
NSMutableString *prettyStack = [NSMutableString stringWithString:@"\n"];
|
|
|
|
for (NSDictionary *frame in stack) {
|
|
|
|
[prettyStack appendFormat:@"%@@%@:%@\n", frame[@"methodName"], frame[@"lineNumber"], frame[@"column"]];
|
|
|
|
}
|
2015-04-09 17:39:28 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
NSString *name = [@"Unhandled JS Exception: " stringByAppendingString:message];
|
|
|
|
[NSException raise:name format:@"Message: %@, stack: %@", message, prettyStack];
|
|
|
|
}
|
2015-03-11 20:53:30 +00:00
|
|
|
}
|
2015-02-20 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
2015-04-08 15:52:48 +00:00
|
|
|
RCT_EXPORT_METHOD(updateExceptionMessage:(NSString *)message
|
2015-07-15 11:01:38 +00:00
|
|
|
stack:(NSArray *)stack
|
|
|
|
exceptionId:(__unused NSNumber *)exceptionId)
|
2015-02-20 04:10:52 +00:00
|
|
|
{
|
2015-05-07 00:23:57 +00:00
|
|
|
if (_delegate) {
|
2015-05-13 17:56:09 +00:00
|
|
|
[_delegate updateJSExceptionWithMessage:message stack:stack];
|
2015-05-07 00:23:57 +00:00
|
|
|
return;
|
|
|
|
}
|
2015-04-21 16:48:29 +00:00
|
|
|
|
2015-05-07 00:23:57 +00:00
|
|
|
[[RCTRedBox sharedInstance] updateErrorMessage:message withStack:stack];
|
2015-02-20 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
2015-05-13 17:56:09 +00:00
|
|
|
// Deprecated. Use reportFatalException directly instead.
|
|
|
|
RCT_EXPORT_METHOD(reportUnhandledException:(NSString *)message
|
|
|
|
stack:(NSArray *)stack)
|
|
|
|
{
|
2015-07-15 11:01:38 +00:00
|
|
|
[self reportFatalException:message stack:stack exceptionId:nil];
|
2015-05-13 17:56:09 +00:00
|
|
|
}
|
2015-02-20 04:10:52 +00:00
|
|
|
@end
|