2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 12:08:49 -07:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-06-23 09:47:54 -07:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
2016-12-12 15:59:12 -08:00
|
|
|
import * as chars from './chars';
|
2017-03-14 09:16:15 -07:00
|
|
|
import {CompileIdentifierMetadata, identifierModuleUrl, identifierName} from './compile_metadata';
|
2016-06-23 09:47:54 -07:00
|
|
|
|
2015-10-07 09:34:21 -07:00
|
|
|
export class ParseLocation {
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
|
|
|
public file: ParseSourceFile, public offset: number, public line: number,
|
|
|
|
public col: number) {}
|
2015-10-07 09:34:21 -07:00
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
toString(): string {
|
2017-03-02 09:37:01 -08:00
|
|
|
return this.offset != null ? `${this.file.url}@${this.line}:${this.col}` : this.file.url;
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
2016-12-12 15:59:12 -08:00
|
|
|
|
|
|
|
moveBy(delta: number): ParseLocation {
|
|
|
|
const source = this.file.content;
|
|
|
|
const len = source.length;
|
|
|
|
let offset = this.offset;
|
|
|
|
let line = this.line;
|
|
|
|
let col = this.col;
|
|
|
|
while (offset > 0 && delta < 0) {
|
|
|
|
offset--;
|
|
|
|
delta++;
|
|
|
|
const ch = source.charCodeAt(offset);
|
|
|
|
if (ch == chars.$LF) {
|
|
|
|
line--;
|
|
|
|
const priorLine = source.substr(0, offset - 1).lastIndexOf(String.fromCharCode(chars.$LF));
|
|
|
|
col = priorLine > 0 ? offset - priorLine : offset;
|
|
|
|
} else {
|
|
|
|
col--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (offset < len && delta > 0) {
|
|
|
|
const ch = source.charCodeAt(offset);
|
|
|
|
offset++;
|
|
|
|
delta--;
|
|
|
|
if (ch == chars.$LF) {
|
|
|
|
line++;
|
|
|
|
col = 0;
|
|
|
|
} else {
|
|
|
|
col++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new ParseLocation(this.file, offset, line, col);
|
|
|
|
}
|
2017-01-26 15:05:02 -08:00
|
|
|
|
|
|
|
// Return the source around the location
|
|
|
|
// Up to `maxChars` or `maxLines` on each side of the location
|
2017-03-24 09:59:58 -07:00
|
|
|
getContext(maxChars: number, maxLines: number): {before: string, after: string}|null {
|
2017-01-26 15:05:02 -08:00
|
|
|
const content = this.file.content;
|
|
|
|
let startOffset = this.offset;
|
|
|
|
|
2017-03-02 09:37:01 -08:00
|
|
|
if (startOffset != null) {
|
2017-01-26 15:05:02 -08:00
|
|
|
if (startOffset > content.length - 1) {
|
|
|
|
startOffset = content.length - 1;
|
|
|
|
}
|
|
|
|
let endOffset = startOffset;
|
|
|
|
let ctxChars = 0;
|
|
|
|
let ctxLines = 0;
|
|
|
|
|
|
|
|
while (ctxChars < maxChars && startOffset > 0) {
|
|
|
|
startOffset--;
|
|
|
|
ctxChars++;
|
|
|
|
if (content[startOffset] == '\n') {
|
|
|
|
if (++ctxLines == maxLines) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctxChars = 0;
|
|
|
|
ctxLines = 0;
|
|
|
|
while (ctxChars < maxChars && endOffset < content.length - 1) {
|
|
|
|
endOffset++;
|
|
|
|
ctxChars++;
|
|
|
|
if (content[endOffset] == '\n') {
|
|
|
|
if (++ctxLines == maxLines) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
before: content.substring(startOffset, this.offset),
|
|
|
|
after: content.substring(this.offset, endOffset + 1),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
2015-10-07 09:34:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export class ParseSourceFile {
|
|
|
|
constructor(public content: string, public url: string) {}
|
|
|
|
}
|
|
|
|
|
2016-02-16 16:46:51 -08:00
|
|
|
export class ParseSourceSpan {
|
2020-10-28 21:35:06 +00:00
|
|
|
/**
|
|
|
|
* Create an object that holds information about spans of tokens/nodes captured during
|
|
|
|
* lexing/parsing of text.
|
|
|
|
*
|
|
|
|
* @param start
|
|
|
|
* The location of the start of the span (having skipped leading trivia).
|
|
|
|
* Skipping leading trivia makes source-spans more "user friendly", since things like HTML
|
|
|
|
* elements will appear to begin at the start of the opening tag, rather than at the start of any
|
|
|
|
* leading trivia, which could include newlines.
|
|
|
|
*
|
|
|
|
* @param end
|
|
|
|
* The location of the end of the span.
|
|
|
|
*
|
|
|
|
* @param fullStart
|
|
|
|
* The start of the token without skipping the leading trivia.
|
|
|
|
* This is used by tooling that splits tokens further, such as extracting Angular interpolations
|
|
|
|
* from text tokens. Such tooling creates new source-spans relative to the original token's
|
|
|
|
* source-span. If leading trivia characters have been skipped then the new source-spans may be
|
|
|
|
* incorrectly offset.
|
|
|
|
*
|
|
|
|
* @param details
|
|
|
|
* Additional information (such as identifier names) that should be associated with the span.
|
|
|
|
*/
|
2016-07-28 02:27:07 -07:00
|
|
|
constructor(
|
2020-10-28 21:35:06 +00:00
|
|
|
public start: ParseLocation, public end: ParseLocation,
|
|
|
|
public fullStart: ParseLocation = start, public details: string|null = null) {}
|
2016-02-16 16:46:51 -08:00
|
|
|
|
|
|
|
toString(): string {
|
|
|
|
return this.start.file.content.substring(this.start.offset, this.end.offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-25 19:52:24 -07:00
|
|
|
export enum ParseErrorLevel {
|
|
|
|
WARNING,
|
2017-03-14 17:12:44 -07:00
|
|
|
ERROR,
|
2016-04-25 19:52:24 -07:00
|
|
|
}
|
|
|
|
|
2016-10-21 11:41:14 -07:00
|
|
|
export class ParseError {
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
|
|
|
public span: ParseSourceSpan, public msg: string,
|
2017-03-14 17:12:44 -07:00
|
|
|
public level: ParseErrorLevel = ParseErrorLevel.ERROR) {}
|
2015-10-07 09:34:21 -07:00
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
contextualMessage(): string {
|
2017-01-26 15:05:02 -08:00
|
|
|
const ctx = this.span.start.getContext(100, 3);
|
2017-11-09 14:06:02 -08:00
|
|
|
return ctx ? `${this.msg} ("${ctx.before}[${ParseErrorLevel[this.level]} ->]${ctx.after}")` :
|
|
|
|
this.msg;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
toString(): string {
|
2017-01-26 15:05:02 -08:00
|
|
|
const details = this.span.details ? `, ${this.span.details}` : '';
|
2017-11-09 14:06:02 -08:00
|
|
|
return `${this.contextualMessage()}: ${this.span.start}${details}`;
|
2015-10-07 09:34:21 -07:00
|
|
|
}
|
|
|
|
}
|
2017-03-14 09:16:15 -07:00
|
|
|
|
|
|
|
export function typeSourceSpan(kind: string, type: CompileIdentifierMetadata): ParseSourceSpan {
|
|
|
|
const moduleUrl = identifierModuleUrl(type);
|
|
|
|
const sourceFileName = moduleUrl != null ? `in ${kind} ${identifierName(type)} in ${moduleUrl}` :
|
|
|
|
`in ${kind} ${identifierName(type)}`;
|
|
|
|
const sourceFile = new ParseSourceFile('', sourceFileName);
|
|
|
|
return new ParseSourceSpan(
|
2017-03-24 09:59:58 -07:00
|
|
|
new ParseLocation(sourceFile, -1, -1, -1), new ParseLocation(sourceFile, -1, -1, -1));
|
2017-03-14 09:16:15 -07:00
|
|
|
}
|
2019-01-24 17:25:46 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates Source Span object for a given R3 Type for JIT mode.
|
|
|
|
*
|
|
|
|
* @param kind Component or Directive.
|
|
|
|
* @param typeName name of the Component or Directive.
|
|
|
|
* @param sourceUrl reference to Component or Directive source.
|
|
|
|
* @returns instance of ParseSourceSpan that represent a given Component or Directive.
|
|
|
|
*/
|
|
|
|
export function r3JitTypeSourceSpan(
|
|
|
|
kind: string, typeName: string, sourceUrl: string): ParseSourceSpan {
|
|
|
|
const sourceFileName = `in ${kind} ${typeName} in ${sourceUrl}`;
|
|
|
|
const sourceFile = new ParseSourceFile('', sourceFileName);
|
|
|
|
return new ParseSourceSpan(
|
|
|
|
new ParseLocation(sourceFile, -1, -1, -1), new ParseLocation(sourceFile, -1, -1, -1));
|
2019-07-16 12:18:32 -07:00
|
|
|
}
|