mirror of
https://github.com/microsoft/TypeScript.git
synced 2026-02-15 03:23:08 -06:00
337 lines
12 KiB
TypeScript
337 lines
12 KiB
TypeScript
/// <reference path="..\..\src\compiler\sys.ts" />
|
|
/// <reference path="..\..\src\harness\harness.ts" />
|
|
/// <reference path="..\..\src\harness\harnessLanguageService.ts" />
|
|
/// <reference path="..\..\src\harness\runnerbase.ts" />
|
|
/// <reference path="..\..\src\harness\typeWriter.ts" />
|
|
|
|
interface FileInformation {
|
|
contents: string;
|
|
codepage: number;
|
|
}
|
|
|
|
interface FindFileResult {
|
|
}
|
|
|
|
interface IOLogFile {
|
|
path: string;
|
|
codepage: number;
|
|
result?: FileInformation;
|
|
}
|
|
|
|
interface IOLog {
|
|
timestamp: string;
|
|
arguments: string[];
|
|
executingPath: string;
|
|
currentDirectory: string;
|
|
useCustomLibraryFile?: boolean;
|
|
filesRead: IOLogFile[];
|
|
filesWritten: {
|
|
path: string;
|
|
contents: string;
|
|
bom: boolean;
|
|
}[];
|
|
filesDeleted: string[];
|
|
filesAppended: {
|
|
path: string;
|
|
contents: string;
|
|
}[];
|
|
fileExists: {
|
|
path: string;
|
|
result?: boolean;
|
|
}[];
|
|
filesFound: {
|
|
path: string;
|
|
pattern: string;
|
|
result?: FindFileResult;
|
|
}[];
|
|
dirs: {
|
|
path: string;
|
|
re: string;
|
|
re_m: boolean;
|
|
re_g: boolean;
|
|
re_i: boolean;
|
|
opts: { recursive?: boolean; };
|
|
result?: string[];
|
|
}[];
|
|
dirExists: {
|
|
path: string;
|
|
result?: boolean;
|
|
}[];
|
|
dirsCreated: string[];
|
|
pathsResolved: {
|
|
path: string;
|
|
result?: string;
|
|
}[];
|
|
directoriesRead: {
|
|
path: string,
|
|
extensions: string[],
|
|
exclude: string[],
|
|
include: string[],
|
|
result: string[]
|
|
}[];
|
|
}
|
|
|
|
interface PlaybackControl {
|
|
startReplayFromFile(logFileName: string): void;
|
|
startReplayFromString(logContents: string): void;
|
|
startReplayFromData(log: IOLog): void;
|
|
endReplay(): void;
|
|
startRecord(logFileName: string): void;
|
|
endRecord(): void;
|
|
}
|
|
|
|
namespace Playback {
|
|
let recordLog: IOLog = undefined;
|
|
let replayLog: IOLog = undefined;
|
|
let recordLogFileNameBase = "";
|
|
|
|
interface Memoized<T> {
|
|
(s: string): T;
|
|
reset(): void;
|
|
}
|
|
|
|
function memoize<T>(func: (s: string) => T): Memoized<T> {
|
|
const lookup = new ts.StringMap<T>();
|
|
const run: Memoized<T> = <Memoized<T>>((s: string) =>
|
|
ts.getOrUpdateAndAllowUndefined(lookup, s, func));
|
|
run.reset = () => {
|
|
lookup.clear();
|
|
};
|
|
|
|
return run;
|
|
}
|
|
|
|
export interface PlaybackIO extends Harness.IO, PlaybackControl { }
|
|
|
|
export interface PlaybackSystem extends ts.System, PlaybackControl { }
|
|
|
|
function createEmptyLog(): IOLog {
|
|
return {
|
|
timestamp: (new Date()).toString(),
|
|
arguments: [],
|
|
currentDirectory: "",
|
|
filesRead: [],
|
|
directoriesRead: [],
|
|
filesWritten: [],
|
|
filesDeleted: [],
|
|
filesAppended: [],
|
|
fileExists: [],
|
|
filesFound: [],
|
|
dirs: [],
|
|
dirExists: [],
|
|
dirsCreated: [],
|
|
pathsResolved: [],
|
|
executingPath: ""
|
|
};
|
|
}
|
|
|
|
function initWrapper(wrapper: PlaybackSystem, underlying: ts.System): void;
|
|
function initWrapper(wrapper: PlaybackIO, underlying: Harness.IO): void;
|
|
function initWrapper(wrapper: PlaybackSystem | PlaybackIO, underlying: ts.System | Harness.IO): void {
|
|
ts.forEach(Object.keys(underlying), prop => {
|
|
(<any>wrapper)[prop] = (<any>underlying)[prop];
|
|
});
|
|
|
|
wrapper.startReplayFromString = logString => {
|
|
wrapper.startReplayFromData(JSON.parse(logString));
|
|
};
|
|
wrapper.startReplayFromData = log => {
|
|
replayLog = log;
|
|
// Remove non-found files from the log (shouldn't really need them, but we still record them for diagnostic purposes)
|
|
replayLog.filesRead = replayLog.filesRead.filter(f => f.result.contents !== undefined);
|
|
};
|
|
|
|
wrapper.endReplay = () => {
|
|
replayLog = undefined;
|
|
};
|
|
|
|
wrapper.startRecord = (fileNameBase) => {
|
|
recordLogFileNameBase = fileNameBase;
|
|
recordLog = createEmptyLog();
|
|
|
|
if (typeof underlying.args !== "function") {
|
|
recordLog.arguments = underlying.args;
|
|
}
|
|
};
|
|
|
|
wrapper.startReplayFromFile = logFn => {
|
|
wrapper.startReplayFromString(underlying.readFile(logFn));
|
|
};
|
|
wrapper.endRecord = () => {
|
|
if (recordLog !== undefined) {
|
|
let i = 0;
|
|
const fn = () => recordLogFileNameBase + i + ".json";
|
|
while (underlying.fileExists(fn())) i++;
|
|
underlying.writeFile(fn(), JSON.stringify(recordLog));
|
|
recordLog = undefined;
|
|
}
|
|
};
|
|
|
|
wrapper.fileExists = recordReplay(wrapper.fileExists, underlying)(
|
|
path => callAndRecord(underlying.fileExists(path), recordLog.fileExists, { path }),
|
|
memoize(path => {
|
|
// If we read from the file, it must exist
|
|
if (findFileByPath(wrapper, replayLog.filesRead, path, /*throwFileNotFoundError*/ false)) {
|
|
return true;
|
|
}
|
|
else {
|
|
return findResultByFields(replayLog.fileExists, { path }, /*defaultValue*/ false);
|
|
}
|
|
})
|
|
);
|
|
|
|
wrapper.getExecutingFilePath = () => {
|
|
if (replayLog !== undefined) {
|
|
return replayLog.executingPath;
|
|
}
|
|
else if (recordLog !== undefined) {
|
|
return recordLog.executingPath = underlying.getExecutingFilePath();
|
|
}
|
|
else {
|
|
return underlying.getExecutingFilePath();
|
|
}
|
|
};
|
|
|
|
wrapper.getCurrentDirectory = () => {
|
|
if (replayLog !== undefined) {
|
|
return replayLog.currentDirectory || "";
|
|
}
|
|
else if (recordLog !== undefined) {
|
|
return recordLog.currentDirectory = underlying.getCurrentDirectory();
|
|
}
|
|
else {
|
|
return underlying.getCurrentDirectory();
|
|
}
|
|
};
|
|
|
|
wrapper.resolvePath = recordReplay(wrapper.resolvePath, underlying)(
|
|
path => callAndRecord(underlying.resolvePath(path), recordLog.pathsResolved, { path }),
|
|
memoize(path => findResultByFields(replayLog.pathsResolved, { path }, !ts.isRootedDiskPath(ts.normalizeSlashes(path)) && replayLog.currentDirectory ? replayLog.currentDirectory + "/" + path : ts.normalizeSlashes(path))));
|
|
|
|
wrapper.readFile = recordReplay(wrapper.readFile, underlying)(
|
|
(path: string) => {
|
|
const result = underlying.readFile(path);
|
|
const logEntry = { path, codepage: 0, result: { contents: result, codepage: 0 } };
|
|
recordLog.filesRead.push(logEntry);
|
|
return result;
|
|
},
|
|
memoize(path => findFileByPath(wrapper, replayLog.filesRead, path, /*throwFileNotFoundError*/ true).contents));
|
|
|
|
wrapper.readDirectory = recordReplay(wrapper.readDirectory, underlying)(
|
|
(path, extensions, exclude, include) => {
|
|
const result = (<ts.System>underlying).readDirectory(path, extensions, exclude, include);
|
|
const logEntry = { path, extensions, exclude, include, result };
|
|
recordLog.directoriesRead.push(logEntry);
|
|
return result;
|
|
},
|
|
(path, extensions, exclude) => {
|
|
// Because extensions is an array of all allowed extension, we will want to merge each of the replayLog.directoriesRead into one
|
|
// if each of the directoriesRead has matched path with the given path (directory with same path but different extension will considered
|
|
// different entry).
|
|
// TODO (yuisu): We can certainly remove these once we recapture the RWC using new API
|
|
const normalizedPath = ts.normalizePath(path).toLowerCase();
|
|
const result: string[] = [];
|
|
for (const directory of replayLog.directoriesRead) {
|
|
if (ts.normalizeSlashes(directory.path).toLowerCase() === normalizedPath) {
|
|
result.push(...directory.result);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
});
|
|
|
|
wrapper.writeFile = recordReplay(wrapper.writeFile, underlying)(
|
|
(path: string, contents: string) => callAndRecord(underlying.writeFile(path, contents), recordLog.filesWritten, { path, contents, bom: false }),
|
|
(path: string, contents: string) => noOpReplay("writeFile"));
|
|
|
|
wrapper.exit = (exitCode) => {
|
|
if (recordLog !== undefined) {
|
|
wrapper.endRecord();
|
|
}
|
|
underlying.exit(exitCode);
|
|
};
|
|
}
|
|
|
|
function recordReplay<T extends Function>(original: T, underlying: any) {
|
|
function createWrapper(record: T, replay: T): T {
|
|
return <any>(function () {
|
|
if (replayLog !== undefined) {
|
|
return replay.apply(undefined, arguments);
|
|
}
|
|
else if (recordLog !== undefined) {
|
|
return record.apply(undefined, arguments);
|
|
}
|
|
else {
|
|
return original.apply(underlying, arguments);
|
|
}
|
|
});
|
|
}
|
|
return createWrapper;
|
|
}
|
|
|
|
function callAndRecord<T, U>(underlyingResult: T, logArray: U[], logEntry: U): T {
|
|
if (underlyingResult !== undefined) {
|
|
(<any>logEntry).result = underlyingResult;
|
|
}
|
|
logArray.push(logEntry);
|
|
return underlyingResult;
|
|
}
|
|
|
|
function findResultByFields<T>(logArray: { result?: T }[], expectedFields: {}, defaultValue?: T): T {
|
|
const predicate = (entry: { result?: T }) => {
|
|
return Object.getOwnPropertyNames(expectedFields).every((name) => (<any>entry)[name] === (<any>expectedFields)[name]);
|
|
};
|
|
const results = logArray.filter(entry => predicate(entry));
|
|
if (results.length === 0) {
|
|
if (defaultValue !== undefined) {
|
|
return defaultValue;
|
|
}
|
|
else {
|
|
throw new Error("No matching result in log array for: " + JSON.stringify(expectedFields));
|
|
}
|
|
}
|
|
return results[0].result;
|
|
}
|
|
|
|
function findFileByPath(wrapper: { resolvePath(s: string): string }, logArray: IOLogFile[],
|
|
expectedPath: string, throwFileNotFoundError: boolean): FileInformation {
|
|
const normalizedName = ts.normalizePath(expectedPath).toLowerCase();
|
|
// Try to find the result through normal fileName
|
|
for (const log of logArray) {
|
|
if (ts.normalizeSlashes(log.path).toLowerCase() === normalizedName) {
|
|
return log.result;
|
|
}
|
|
}
|
|
|
|
// If we got here, we didn't find a match
|
|
if (throwFileNotFoundError) {
|
|
throw new Error("No matching result in log array for path: " + expectedPath);
|
|
}
|
|
else {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
function noOpReplay(name: string) {
|
|
// console.log("Swallowed write operation during replay: " + name);
|
|
}
|
|
|
|
export function wrapIO(underlying: Harness.IO): PlaybackIO {
|
|
const wrapper: PlaybackIO = <any>{};
|
|
initWrapper(wrapper, underlying);
|
|
|
|
wrapper.directoryName = (path): string => { throw new Error("NotSupported"); };
|
|
wrapper.createDirectory = (path): void => { throw new Error("NotSupported"); };
|
|
wrapper.directoryExists = (path): boolean => { throw new Error("NotSupported"); };
|
|
wrapper.deleteFile = (path): void => { throw new Error("NotSupported"); };
|
|
wrapper.listFiles = (path, filter, options): string[] => { throw new Error("NotSupported"); };
|
|
|
|
return wrapper;
|
|
}
|
|
|
|
export function wrapSystem(underlying: ts.System): PlaybackSystem {
|
|
const wrapper: PlaybackSystem = <any>{};
|
|
initWrapper(wrapper, underlying);
|
|
return wrapper;
|
|
}
|
|
} |