| 
									
										
										
										
											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-03-14 09:16:15 -07:00
										 |  |  | import {ERROR_ORIGINAL_ERROR, getDebugContext, getErrorLogger, getOriginalError} from './errors'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 13:19:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-15 13:56:24 -07:00
										 |  |  |   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 { | 
					
						
							| 
									
										
										
										
											2017-03-14 09:16:15 -07:00
										 |  |  |     const originalError = this._findOriginalError(error); | 
					
						
							|  |  |  |     const context = this._findContext(error); | 
					
						
							|  |  |  |     // Note: Browser consoles show the place from where console.error was called.
 | 
					
						
							|  |  |  |     // We can use this to give users additional information about the error.
 | 
					
						
							|  |  |  |     const errorLogger = getErrorLogger(error); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     errorLogger(this._console, `ERROR`, error); | 
					
						
							|  |  |  |     if (originalError) { | 
					
						
							|  |  |  |       errorLogger(this._console, `ORIGINAL ERROR`, originalError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (context) { | 
					
						
							|  |  |  |       errorLogger(this._console, 'ERROR CONTEXT', context); | 
					
						
							| 
									
										
										
										
											2015-07-22 17:13:42 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-15 13:56:24 -07: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
										 |  |  |   _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-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; | 
					
						
							|  |  |  | } |