2015-03-23 13:28:42 -07: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-19 20:10:52 -08:00
|
|
|
|
|
|
|
#import "RCTExceptionsManager.h"
|
|
|
|
|
2015-04-21 05:26:51 -07:00
|
|
|
#import "RCTDefines.h"
|
2015-04-09 10:39:28 -07:00
|
|
|
#import "RCTLog.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
#import "RCTRedBox.h"
|
2015-04-09 10:39:28 -07:00
|
|
|
#import "RCTRootView.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
@implementation RCTExceptionsManager
|
2015-03-11 13:53:30 -07:00
|
|
|
{
|
|
|
|
__weak id<RCTExceptionsManagerDelegate> _delegate;
|
2015-04-09 10:39:28 -07:00
|
|
|
NSUInteger _reloadRetries;
|
2015-03-11 13:53:30 -07:00
|
|
|
}
|
|
|
|
|
2015-08-19 11:27:43 -01:00
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
|
2015-04-08 05:42:43 -07:00
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-03-11 13:53:30 -07:00
|
|
|
- (instancetype)initWithDelegate:(id<RCTExceptionsManagerDelegate>)delegate
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
|
|
|
_delegate = delegate;
|
2015-04-09 10:39:28 -07:00
|
|
|
_maxReloadAttempts = 0;
|
2015-03-11 13:53:30 -07:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
return [self initWithDelegate:nil];
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2015-05-13 10:56:09 -07: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-08-19 11:27:43 -01:00
|
|
|
[_bridge.redBox showErrorMessage:message withStack:stack];
|
2015-05-13 10:56:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
RCT_EXPORT_METHOD(reportFatalException:(NSString *)message
|
2015-07-15 04:01:38 -07:00
|
|
|
stack:(NSArray *)stack
|
|
|
|
exceptionId:(__unused NSNumber *)exceptionId)
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-03-11 13:53:30 -07:00
|
|
|
if (_delegate) {
|
2015-05-13 10:56:09 -07:00
|
|
|
[_delegate handleFatalJSExceptionWithMessage:message stack:stack];
|
2015-04-09 10:39:28 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-19 11:27:43 -01:00
|
|
|
[_bridge.redBox showErrorMessage:message withStack:stack];
|
2015-04-21 09:48:29 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
if (!RCT_DEBUG) {
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
static NSUInteger reloadRetries = 0;
|
|
|
|
const NSUInteger maxMessageLength = 75;
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
if (reloadRetries < _maxReloadAttempts) {
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
reloadRetries++;
|
|
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTReloadNotification
|
|
|
|
object:nil];
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
} else {
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
if (message.length > maxMessageLength) {
|
|
|
|
message = [[message substringToIndex:maxMessageLength] stringByAppendingString:@"..."];
|
|
|
|
}
|
2015-04-19 12:55:46 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
NSMutableString *prettyStack = [NSMutableString stringWithString:@"\n"];
|
|
|
|
for (NSDictionary *frame in stack) {
|
|
|
|
[prettyStack appendFormat:@"%@@%@:%@\n", frame[@"methodName"], frame[@"lineNumber"], frame[@"column"]];
|
|
|
|
}
|
2015-04-09 10:39:28 -07:00
|
|
|
|
2015-05-06 17:23:57 -07:00
|
|
|
NSString *name = [@"Unhandled JS Exception: " stringByAppendingString:message];
|
|
|
|
[NSException raise:name format:@"Message: %@, stack: %@", message, prettyStack];
|
|
|
|
}
|
2015-03-11 13:53:30 -07:00
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-04-08 08:52:48 -07:00
|
|
|
RCT_EXPORT_METHOD(updateExceptionMessage:(NSString *)message
|
2015-07-15 04:01:38 -07:00
|
|
|
stack:(NSArray *)stack
|
|
|
|
exceptionId:(__unused NSNumber *)exceptionId)
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-05-06 17:23:57 -07:00
|
|
|
if (_delegate) {
|
2015-05-13 10:56:09 -07:00
|
|
|
[_delegate updateJSExceptionWithMessage:message stack:stack];
|
2015-05-06 17:23:57 -07:00
|
|
|
return;
|
|
|
|
}
|
2015-04-21 09:48:29 -07:00
|
|
|
|
2015-08-19 11:27:43 -01:00
|
|
|
[_bridge.redBox updateErrorMessage:message withStack:stack];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-05-13 10:56:09 -07:00
|
|
|
// Deprecated. Use reportFatalException directly instead.
|
|
|
|
RCT_EXPORT_METHOD(reportUnhandledException:(NSString *)message
|
|
|
|
stack:(NSArray *)stack)
|
|
|
|
{
|
2015-07-15 04:01:38 -07:00
|
|
|
[self reportFatalException:message stack:stack exceptionId:nil];
|
2015-05-13 10:56:09 -07:00
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
@end
|