182 lines
4.8 KiB
JavaScript
182 lines
4.8 KiB
JavaScript
import type SourceMap from "./source-map";
|
|
import trimEnd from "lodash/trimEnd";
|
|
|
|
const SPACES_RE = /^[ \t]+$/;
|
|
|
|
/**
|
|
* The Buffer class exists to manage the queue of tokens being pushed onto the output string
|
|
* in such a way that the final string buffer is treated as write-only until the final .get()
|
|
* call. This allows V8 to optimize the output efficiently by not requiring it to store the
|
|
* string in contiguous memory.
|
|
*/
|
|
|
|
export default class Buffer {
|
|
constructor(map: ?SourceMap) {
|
|
this._map = map;
|
|
}
|
|
|
|
_map: SourceMap = null;
|
|
_buf: Array = [];
|
|
_last: string = "";
|
|
_queue: Array = [];
|
|
|
|
_position: Object = {
|
|
line: 1,
|
|
column: 0,
|
|
};
|
|
_sourcePosition: Object = {
|
|
line: null,
|
|
column: null,
|
|
filename: null,
|
|
};
|
|
|
|
/**
|
|
* Get the final string output from the buffer, along with the sourcemap if one exists.
|
|
*/
|
|
|
|
get(): Object {
|
|
this._flush();
|
|
|
|
return {
|
|
code: trimEnd(this._buf.join("")),
|
|
map: this._map ? this._map.get() : null,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Add a string to the buffer that cannot be reverted.
|
|
*/
|
|
|
|
append(str: string): void {
|
|
this._flush();
|
|
const { line, column, filename } = this._sourcePosition;
|
|
this._append(str, line, column, filename);
|
|
}
|
|
|
|
/**
|
|
* Add a string to the buffer than can be reverted.
|
|
*/
|
|
|
|
queue(str: string): void {
|
|
// Drop trailing spaces when a newline is inserted.
|
|
if (str === "\n") while (this._queue.length > 0 && SPACES_RE.test(this._queue[0][0])) this._queue.shift();
|
|
|
|
const { line, column, filename } = this._sourcePosition;
|
|
this._queue.unshift([str, line, column, filename]);
|
|
}
|
|
|
|
_flush(): void {
|
|
let item;
|
|
while (item = this._queue.pop()) this._append(...item);
|
|
}
|
|
|
|
_append(str: string, line: number, column: number, filename: ?string): void {
|
|
// If there the line is ending, adding a new mapping marker is redundant
|
|
if (this._map && str[0] !== "\n") {
|
|
this._map.mark(this._position.line, this._position.column, line, column, filename);
|
|
}
|
|
|
|
this._buf.push(str);
|
|
this._last = str[str.length - 1];
|
|
|
|
for (let i = 0; i < str.length; i++) {
|
|
if (str[i] === "\n") {
|
|
this._position.line++;
|
|
this._position.column = 0;
|
|
} else {
|
|
this._position.column++;
|
|
}
|
|
}
|
|
}
|
|
|
|
removeTrailingNewline(): void {
|
|
if (this._queue.length > 0 && this._queue[0][0] === "\n") this._queue.shift();
|
|
}
|
|
|
|
removeLastSemicolon(): void {
|
|
if (this._queue.length > 0 && this._queue[0][0] === ";") this._queue.shift();
|
|
}
|
|
|
|
endsWith(suffix: string): boolean {
|
|
// Fast path to avoid iterating over this._queue.
|
|
if (suffix.length === 1) {
|
|
let last;
|
|
if (this._queue.length > 0) {
|
|
const str = this._queue[0][0];
|
|
last = str[str.length - 1];
|
|
} else {
|
|
last = this._last;
|
|
}
|
|
|
|
return last === suffix;
|
|
}
|
|
|
|
const end = this._last + this._queue.reduce((acc, item) => item[0] + acc, "");
|
|
if (suffix.length <= end.length) {
|
|
return end.slice(-suffix.length) === suffix;
|
|
}
|
|
|
|
// We assume that everything being matched is at most a single token plus some whitespace,
|
|
// which everything currently is, but otherwise we'd have to expand _last or check _buf.
|
|
return false;
|
|
}
|
|
|
|
hasContent(): boolean {
|
|
return this._queue.length > 0 || !!this._last;
|
|
}
|
|
|
|
/**
|
|
* Sets a given position as the current source location so generated code after this call
|
|
* will be given this position in the sourcemap.
|
|
*/
|
|
|
|
source(prop: string, loc: Location): void {
|
|
if (prop && !loc) return;
|
|
|
|
let pos = loc ? loc[prop] : null;
|
|
|
|
this._sourcePosition.line = pos ? pos.line : null;
|
|
this._sourcePosition.column = pos ? pos.column : null;
|
|
this._sourcePosition.filename = loc && loc.filename || null;
|
|
}
|
|
|
|
/**
|
|
* Call a callback with a specific source location and restore on completion.
|
|
*/
|
|
|
|
withSource(prop: string, loc: Location, cb: () => void): void {
|
|
if (!this._map) return cb();
|
|
|
|
// Use the call stack to manage a stack of "source location" data.
|
|
let originalLine = this._sourcePosition.line;
|
|
let originalColumn = this._sourcePosition.column;
|
|
let originalFilename = this._sourcePosition.filename;
|
|
|
|
this.source(prop, loc);
|
|
|
|
cb();
|
|
|
|
this._sourcePosition.line = originalLine;
|
|
this._sourcePosition.column = originalColumn;
|
|
this._sourcePosition.filename = originalFilename;
|
|
}
|
|
|
|
getCurrentColumn(): number {
|
|
const extra = this._queue.reduce((acc, item) => item[0] + acc, "");
|
|
const lastIndex = extra.lastIndexOf("\n");
|
|
|
|
return lastIndex === -1 ? this._position.column + extra.length : (extra.length - 1 - lastIndex);
|
|
}
|
|
|
|
getCurrentLine(): number {
|
|
const extra = this._queue.reduce((acc, item) => item[0] + acc, "");
|
|
|
|
let count = 0;
|
|
for (let i = 0; i < extra.length; i++) {
|
|
if (extra[i] === "\n") count++;
|
|
}
|
|
|
|
return this._position.line + count;
|
|
}
|
|
}
|