| 
									
										
										
										
											2016-06-23 09:47:54 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							|  |  |  |  * Copyright Google Inc. All Rights Reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Use of this source code is governed by an MIT-style license that can be | 
					
						
							|  |  |  |  * found in the LICENSE file at https://angular.io/license
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  | import {ERROR_ORIGINAL_ERROR, getDebugContext, getOriginalError} from './errors'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-16 14:35:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 22:47:11 +00:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * @whatItDoes Provides a hook for centralized exception handling. | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * @description | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * The default implementation of `ErrorHandler` prints error messages to the `console`. To | 
					
						
							|  |  |  |  * intercept error handling, write a custom exception handler that replaces this default as | 
					
						
							|  |  |  |  * appropriate for your app. | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * ### Example | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2016-08-30 18:07:40 -07:00
										 |  |  |  * class MyErrorHandler implements ErrorHandler { | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  *   handleError(error) { | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  *     // do something with the exception
 | 
					
						
							|  |  |  |  *   } | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-16 11:15:01 -07:00
										 |  |  |  * @NgModule({ | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |  *   providers: [{provide: ErrorHandler, useClass: MyErrorHandler}] | 
					
						
							| 
									
										
										
										
											2016-08-16 11:15:01 -07:00
										 |  |  |  * }) | 
					
						
							|  |  |  |  * class MyModule {} | 
					
						
							| 
									
										
										
										
											2015-04-13 21:00:52 -07:00
										 |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-25 15:00:05 -07:00
										 |  |  |  * @stable | 
					
						
							| 
									
										
										
										
											2015-03-31 22:47:11 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  | export class ErrorHandler { | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * @internal | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   _console: Console = console; | 
					
						
							| 
									
										
										
										
											2015-07-22 17:13:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 15:05:48 -08:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * @internal | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   rethrowError: boolean; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor(rethrowError: boolean = true) { this.rethrowError = rethrowError; } | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |   handleError(error: any): void { | 
					
						
							|  |  |  |     this._console.error(`EXCEPTION: ${this._extractMessage(error)}`); | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |     if (error instanceof Error) { | 
					
						
							|  |  |  |       const originalError = this._findOriginalError(error); | 
					
						
							|  |  |  |       const originalStack = this._findOriginalStack(error); | 
					
						
							|  |  |  |       const context = this._findContext(error); | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |       if (originalError) { | 
					
						
							|  |  |  |         this._console.error(`ORIGINAL EXCEPTION: ${this._extractMessage(originalError)}`); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-07-22 17:13:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |       if (originalStack) { | 
					
						
							|  |  |  |         this._console.error('ORIGINAL STACKTRACE:'); | 
					
						
							|  |  |  |         this._console.error(originalStack); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (context) { | 
					
						
							|  |  |  |         this._console.error('ERROR CONTEXT:'); | 
					
						
							|  |  |  |         this._console.error(context); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-07-22 17:13:42 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 15:05:48 -08:00
										 |  |  |     // We rethrow exceptions, so operations like 'bootstrap' will result in an error
 | 
					
						
							|  |  |  |     // when an error happens. If we do not rethrow, bootstrap will always succeed.
 | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |     if (this.rethrowError) throw error; | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-09 17:21:25 -07:00
										 |  |  |   /** @internal */ | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |   _extractMessage(error: any): string { | 
					
						
							|  |  |  |     return error instanceof Error ? error.message : error.toString(); | 
					
						
							| 
									
										
										
										
											2015-09-10 15:25:36 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-09 17:21:25 -07:00
										 |  |  |   /** @internal */ | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |   _findContext(error: any): any { | 
					
						
							|  |  |  |     if (error) { | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |       return getDebugContext(error) ? getDebugContext(error) : | 
					
						
							|  |  |  |                                       this._findContext(getOriginalError(error)); | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return null; | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-09 17:21:25 -07:00
										 |  |  |   /** @internal */ | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |   _findOriginalError(error: Error): any { | 
					
						
							|  |  |  |     let e = getOriginalError(error); | 
					
						
							|  |  |  |     while (e && getOriginalError(e)) { | 
					
						
							|  |  |  |       e = getOriginalError(e); | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return e; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-09 17:21:25 -07:00
										 |  |  |   /** @internal */ | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |   _findOriginalStack(error: Error): string { | 
					
						
							| 
									
										
										
										
											2016-09-23 15:05:43 -07:00
										 |  |  |     let e: any = error; | 
					
						
							|  |  |  |     let stack: string = e.stack; | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  |     while (e instanceof Error && getOriginalError(e)) { | 
					
						
							|  |  |  |       e = getOriginalError(e); | 
					
						
							| 
									
										
										
										
											2016-08-25 00:50:16 -07:00
										 |  |  |       if (e instanceof Error && e.stack) { | 
					
						
							|  |  |  |         stack = e.stack; | 
					
						
							| 
									
										
										
										
											2015-07-23 18:00:19 -07:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return stack; | 
					
						
							| 
									
										
										
										
											2015-02-16 14:35:27 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function wrappedError(message: string, originalError: any): Error { | 
					
						
							|  |  |  |   const msg = | 
					
						
							|  |  |  |       `${message} caused by: ${originalError instanceof Error ? originalError.message: originalError }`; | 
					
						
							|  |  |  |   const error = Error(msg); | 
					
						
							|  |  |  |   (error as any)[ERROR_ORIGINAL_ERROR] = originalError; | 
					
						
							|  |  |  |   return error; | 
					
						
							|  |  |  | } |