The main use case for the generated source maps is to give
errors a meaningful context in terms of the original source
that the user wrote.
Related changes that are included in this commit:
* renamed virtual folders used for jit:
* ng://<module type>/module.ngfactory.js
* ng://<module type>/<comp type>.ngfactory.js
* ng://<module type>/<comp type>.html (for inline templates)
* error logging:
* all errors that happen in templates are logged
from the place of the nearest element.
* instead of logging error messages and stacks separately,
we log the actual error. This is needed so that browsers apply
source maps to the stack correctly.
* error type and error is logged as one log entry.
Note that long-stack-trace zone has a bug that
disables source maps for stack traces,
see https://github.com/angular/zone.js/issues/661.
BREAKING CHANGE:
- DebugNode.source no more returns the source location of a node.
Closes 14013
38 lines
1.3 KiB
TypeScript
38 lines
1.3 KiB
TypeScript
/**
|
|
* @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
|
|
*/
|
|
|
|
import {SourceMap} from '@angular/compiler/src/output/source_map';
|
|
const b64 = require('base64-js');
|
|
const SourceMapConsumer = require('source-map').SourceMapConsumer;
|
|
|
|
export interface SourceLocation {
|
|
line: number;
|
|
column: number;
|
|
source: string;
|
|
}
|
|
|
|
export function originalPositionFor(
|
|
sourceMap: SourceMap, genPosition: {line: number, column: number}): SourceLocation {
|
|
const smc = new SourceMapConsumer(sourceMap);
|
|
// Note: We don't return the original object as it also contains a `name` property
|
|
// which is always null and we don't want to include that in our assertions...
|
|
const {line, column, source} = smc.originalPositionFor(genPosition);
|
|
return {line, column, source};
|
|
}
|
|
|
|
export function extractSourceMap(source: string): SourceMap {
|
|
let idx = source.lastIndexOf('\n//#');
|
|
if (idx == -1) return null;
|
|
const smComment = source.slice(idx).trim();
|
|
const smB64 = smComment.split('sourceMappingURL=data:application/json;base64,')[1];
|
|
return smB64 ? JSON.parse(decodeB64String(smB64)) : null;
|
|
}
|
|
|
|
function decodeB64String(s: string): string {
|
|
return b64.toByteArray(s).reduce((s: string, c: number) => s + String.fromCharCode(c), '');
|
|
} |