TypeScript/src/harness/fakes.ts

371 lines
14 KiB
TypeScript
Raw Normal View History

2018-04-24 19:48:55 +02:00
/**
* Fake implementations of various compiler dependencies.
*/
2017-11-23 03:59:59 +01:00
namespace fakes {
2018-02-06 07:27:55 +01:00
const processExitSentinel = new Error("System exit");
2018-04-19 20:30:03 +02:00
export interface SystemOptions {
2017-11-22 04:47:13 +01:00
executingFilePath?: string;
newLine?: "\r\n" | "\n";
2018-04-19 20:30:03 +02:00
env?: Record<string, string>;
}
2018-04-19 20:30:03 +02:00
/**
* A fake `ts.System` that leverages a virtual file system.
*/
export class System implements ts.System {
public readonly vfs: vfs.FileSystem;
2018-04-19 20:30:03 +02:00
public readonly args: string[] = [];
public readonly output: string[] = [];
public readonly newLine: string;
public readonly useCaseSensitiveFileNames: boolean;
public exitCode: number;
2017-11-22 04:47:13 +01:00
2018-04-19 20:30:03 +02:00
private readonly _executingFilePath: string | undefined;
private readonly _env: Record<string, string> | undefined;
2018-04-20 01:47:07 +02:00
constructor(vfs: vfs.FileSystem, { executingFilePath, newLine = "\r\n", env }: SystemOptions = {}) {
2018-02-06 07:27:55 +01:00
this.vfs = vfs.isReadonly ? vfs.shadow() : vfs;
this.useCaseSensitiveFileNames = !this.vfs.ignoreCase;
this.newLine = newLine;
this._executingFilePath = executingFilePath;
2018-04-19 20:30:03 +02:00
this._env = env;
}
public write(message: string) {
2018-04-19 20:30:03 +02:00
this.output.push(message);
}
public readFile(path: string) {
2018-04-19 20:30:03 +02:00
try {
const content = this.vfs.readFileSync(path, "utf8");
return content === undefined ? undefined : utils.removeByteOrderMark(content);
2018-04-19 20:30:03 +02:00
}
catch {
return undefined;
}
}
2017-11-22 04:47:13 +01:00
public writeFile(path: string, data: string, writeByteOrderMark?: boolean): void {
2018-04-19 20:30:03 +02:00
this.vfs.mkdirpSync(vpath.dirname(path));
2018-04-24 19:48:55 +02:00
this.vfs.writeFileSync(path, writeByteOrderMark ? utils.addUTF8ByteOrderMark(data) : data);
}
public fileExists(path: string) {
2018-04-19 20:30:03 +02:00
const stats = this._getStats(path);
return stats ? stats.isFile() : false;
}
public directoryExists(path: string) {
2018-04-19 20:30:03 +02:00
const stats = this._getStats(path);
return stats ? stats.isDirectory() : false;
}
public createDirectory(path: string): void {
this.vfs.mkdirpSync(path);
}
public getCurrentDirectory() {
return this.vfs.cwd();
}
public getDirectories(path: string) {
2018-04-19 20:30:03 +02:00
const result: string[] = [];
try {
for (const file of this.vfs.readdirSync(path)) {
if (this.vfs.statSync(vpath.combine(path, file)).isDirectory()) {
result.push(file);
}
}
}
catch { /*ignore*/ }
return result;
}
public readDirectory(path: string, extensions?: ReadonlyArray<string>, exclude?: ReadonlyArray<string>, include?: ReadonlyArray<string>, depth?: number): string[] {
2018-04-20 01:47:07 +02:00
return ts.matchFiles(path, extensions, exclude, include, this.useCaseSensitiveFileNames, this.getCurrentDirectory(), depth, path => this.getAccessibleFileSystemEntries(path));
}
public getAccessibleFileSystemEntries(path: string): ts.FileSystemEntries {
const files: string[] = [];
const directories: string[] = [];
try {
for (const file of this.vfs.readdirSync(path)) {
try {
const stats = this.vfs.statSync(vpath.combine(path, file));
if (stats.isFile()) {
files.push(file);
}
else if (stats.isDirectory()) {
directories.push(file);
2018-04-19 20:30:03 +02:00
}
}
2018-04-20 01:47:07 +02:00
catch { /*ignored*/ }
2018-04-19 20:30:03 +02:00
}
2018-04-20 01:47:07 +02:00
}
catch { /*ignored*/ }
return { files, directories };
}
public exit(exitCode?: number) {
this.exitCode = exitCode;
2018-02-06 07:27:55 +01:00
throw processExitSentinel;
2017-11-28 00:00:05 +01:00
}
public getFileSize(path: string) {
2018-04-19 20:30:03 +02:00
const stats = this._getStats(path);
return stats && stats.isFile() ? stats.size : 0;
}
public resolvePath(path: string) {
return vpath.resolve(this.vfs.cwd(), path);
}
public getExecutingFilePath() {
2018-04-19 20:30:03 +02:00
if (this._executingFilePath === undefined) return ts.notImplemented();
return this._executingFilePath;
}
public getModifiedTime(path: string) {
2018-04-19 20:30:03 +02:00
const stats = this._getStats(path);
return stats ? stats.mtime : undefined;
}
2018-05-22 04:40:09 +02:00
public setModifiedTime(path: string, time: Date) {
this.vfs.utimesSync(path, time, time);
}
public createHash(data: string): string {
2018-04-18 02:49:51 +02:00
return data;
}
public realpath(path: string) {
2018-01-22 08:25:40 +01:00
try {
return this.vfs.realpathSync(path);
}
catch {
return path;
}
}
2018-04-19 20:30:03 +02:00
public getEnvironmentVariable(name: string): string | undefined {
return this._env && this._env[name];
}
private _getStats(path: string) {
try {
return this.vfs.existsSync(path) ? this.vfs.statSync(path) : undefined;
2018-04-19 20:30:03 +02:00
}
catch {
return undefined;
}
}
}
/**
* A fake `ts.ParseConfigHost` that leverages a virtual file system.
*/
export class ParseConfigHost implements ts.ParseConfigHost {
public readonly sys: System;
constructor(sys: System | vfs.FileSystem) {
if (sys instanceof vfs.FileSystem) sys = new System(sys);
this.sys = sys;
}
public get vfs() {
return this.sys.vfs;
}
public get useCaseSensitiveFileNames() {
return this.sys.useCaseSensitiveFileNames;
}
public fileExists(fileName: string): boolean {
return this.sys.fileExists(fileName);
}
2018-04-19 20:30:03 +02:00
public directoryExists(directoryName: string): boolean {
return this.sys.directoryExists(directoryName);
}
2017-12-18 21:12:51 +01:00
2018-04-19 20:30:03 +02:00
public readFile(path: string): string | undefined {
return this.sys.readFile(path);
2017-12-18 21:12:51 +01:00
}
2018-04-19 20:30:03 +02:00
public readDirectory(path: string, extensions: string[], excludes: string[], includes: string[], depth: number): string[] {
return this.sys.readDirectory(path, extensions, excludes, includes, depth);
}
}
/**
* A fake `ts.CompilerHost` that leverages a virtual file system.
*/
export class CompilerHost implements ts.CompilerHost {
public readonly sys: System;
public readonly defaultLibLocation: string;
public readonly outputs: documents.TextDocument[] = [];
private readonly _outputsMap: collections.SortedMap<string, number>;
2018-04-19 20:30:03 +02:00
public readonly traces: string[] = [];
public readonly shouldAssertInvariants = !Harness.lightMode;
private _setParentNodes: boolean;
2018-04-24 19:48:55 +02:00
private _sourceFiles: collections.SortedMap<string, ts.SourceFile>;
2018-04-19 20:30:03 +02:00
private _parseConfigHost: ParseConfigHost;
private _newLine: string;
2018-04-20 01:47:07 +02:00
constructor(sys: System | vfs.FileSystem, options = ts.getDefaultCompilerOptions(), setParentNodes = false) {
if (sys instanceof vfs.FileSystem) sys = new System(sys);
2018-04-19 20:30:03 +02:00
this.sys = sys;
this.defaultLibLocation = sys.vfs.meta.get("defaultLibLocation") || "";
this._newLine = ts.getNewLineCharacter(options, () => this.sys.newLine);
2018-04-24 19:48:55 +02:00
this._sourceFiles = new collections.SortedMap<string, ts.SourceFile>({ comparer: sys.vfs.stringComparer, sort: "insertion" });
2018-04-19 20:30:03 +02:00
this._setParentNodes = setParentNodes;
this._outputsMap = new collections.SortedMap(this.vfs.stringComparer);
2018-04-19 20:30:03 +02:00
}
public get vfs() {
return this.sys.vfs;
}
public get parseConfigHost() {
return this._parseConfigHost || (this._parseConfigHost = new ParseConfigHost(this.sys));
}
public getCurrentDirectory(): string {
return this.sys.getCurrentDirectory();
}
public useCaseSensitiveFileNames(): boolean {
return this.sys.useCaseSensitiveFileNames;
}
public getNewLine(): string {
return this._newLine;
}
public getCanonicalFileName(fileName: string): string {
return this.sys.useCaseSensitiveFileNames ? fileName : fileName.toLowerCase();
}
2018-04-19 20:30:03 +02:00
public fileExists(fileName: string): boolean {
return this.sys.fileExists(fileName);
}
2018-04-19 20:30:03 +02:00
public directoryExists(directoryName: string): boolean {
return this.sys.directoryExists(directoryName);
}
2018-05-22 04:40:09 +02:00
public getModifiedTime(fileName: string) {
return this.sys.getModifiedTime(fileName);
}
public setModifiedTime(fileName: string, time: Date) {
return this.sys.setModifiedTime(fileName, time);
}
2018-04-19 20:30:03 +02:00
public getDirectories(path: string): string[] {
return this.sys.getDirectories(path);
}
2018-05-08 00:12:50 +02:00
public readDirectory(path: string, extensions?: ReadonlyArray<string>, exclude?: ReadonlyArray<string>, include?: ReadonlyArray<string>, depth?: number): string[] {
return this.sys.readDirectory(path, extensions, exclude, include, depth);
}
2018-04-19 20:30:03 +02:00
public readFile(path: string): string | undefined {
return this.sys.readFile(path);
}
public writeFile(fileName: string, content: string, writeByteOrderMark: boolean) {
2018-04-24 19:48:55 +02:00
if (writeByteOrderMark) content = utils.addUTF8ByteOrderMark(content);
2018-04-19 20:30:03 +02:00
this.sys.writeFile(fileName, content);
const document = new documents.TextDocument(fileName, content);
document.meta.set("fileName", fileName);
this.vfs.filemeta(fileName).set("document", document);
if (!this._outputsMap.has(document.file)) {
this._outputsMap.set(document.file, this.outputs.length);
2018-04-19 20:30:03 +02:00
this.outputs.push(document);
}
this.outputs[this._outputsMap.get(document.file)] = document;
2018-04-19 20:30:03 +02:00
}
public trace(s: string): void {
this.traces.push(s);
}
public realpath(path: string): string {
return this.sys.realpath(path);
}
public getDefaultLibLocation(): string {
return vpath.resolve(this.getCurrentDirectory(), this.defaultLibLocation);
}
public getDefaultLibFileName(options: ts.CompilerOptions): string {
return vpath.resolve(this.getDefaultLibLocation(), ts.getDefaultLibFileName(options));
2018-04-19 20:30:03 +02:00
}
public getSourceFile(fileName: string, languageVersion: number): ts.SourceFile | undefined {
const canonicalFileName = this.getCanonicalFileName(vpath.resolve(this.getCurrentDirectory(), fileName));
const existing = this._sourceFiles.get(canonicalFileName);
if (existing) return existing;
const content = this.readFile(canonicalFileName);
if (content === undefined) return undefined;
// A virtual file system may shadow another existing virtual file system. This
// allows us to reuse a common virtual file system structure across multiple
// tests. If a virtual file is a shadow, it is likely that the file will be
// reused across multiple tests. In that case, we cache the SourceFile we parse
// so that it can be reused across multiple tests to avoid the cost of
// repeatedly parsing the same file over and over (such as lib.d.ts).
const cacheKey = this.vfs.shadowRoot && `SourceFile[languageVersion=${languageVersion},setParentNodes=${this._setParentNodes}]`;
if (cacheKey) {
const meta = this.vfs.filemeta(canonicalFileName);
const sourceFileFromMetadata = meta.get(cacheKey) as ts.SourceFile | undefined;
if (sourceFileFromMetadata) {
this._sourceFiles.set(canonicalFileName, sourceFileFromMetadata);
return sourceFileFromMetadata;
}
}
const parsed = ts.createSourceFile(fileName, content, languageVersion, this._setParentNodes || this.shouldAssertInvariants);
if (this.shouldAssertInvariants) {
Utils.assertInvariants(parsed, /*parent*/ undefined);
}
this._sourceFiles.set(canonicalFileName, parsed);
if (cacheKey) {
// store the cached source file on the unshadowed file with the same version.
const stats = this.vfs.statSync(canonicalFileName);
let fs = this.vfs;
while (fs.shadowRoot) {
try {
const shadowRootStats = fs.shadowRoot.existsSync(canonicalFileName) && fs.shadowRoot.statSync(canonicalFileName);
2018-04-19 20:30:03 +02:00
if (shadowRootStats.dev !== stats.dev ||
shadowRootStats.ino !== stats.ino ||
shadowRootStats.mtimeMs !== stats.mtimeMs) {
break;
}
fs = fs.shadowRoot;
}
catch {
break;
}
}
if (fs !== this.vfs) {
fs.filemeta(canonicalFileName).set(cacheKey, parsed);
}
}
return parsed;
}
2017-11-28 02:01:15 +01:00
}
2018-02-06 07:27:55 +01:00
}
2017-11-28 02:01:15 +01:00