@neo-one/smart-contract-compiler
Version:
NEO•ONE TypeScript smart contract compiler.
114 lines (112 loc) • 20 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
const tslib_1 = require("tslib");
const ts_utils_1 = require("@neo-one/ts-utils");
const typescript_1 = tslib_1.__importDefault(require("typescript"));
const Context_1 = require("./Context");
const utils_1 = require("./utils");
function createContext(sourceFiles, program, typeChecker, languageService, host) {
return new Context_1.Context(sourceFiles, program, typeChecker, languageService, host);
}
function updateContext(context, files) {
const { program, typeChecker, languageService } = createProgram(Object.keys(files), context.host, {
modifyHost: createModifyHostFiles(files),
withTestHarness: context.program.__withTestHarness,
});
const sourceFiles = utils_1.getAllSourceFiles(Object.keys(files).map((rootName) => ts_utils_1.tsUtils.file.getSourceFileOrThrow(program, rootName)), typeChecker);
return context.update(sourceFiles, program, typeChecker, languageService);
}
exports.updateContext = updateContext;
const CREATE_CONTEXT_OPTIONS_DEFAULT = {
withTestHarness: false,
};
const defaultModifyHost = () => {
};
const DEFAULT_MAKE_CONTEXT_OPTIONS = Object.assign({}, CREATE_CONTEXT_OPTIONS_DEFAULT, { modifyHost: defaultModifyHost });
exports.COMPILER_OPTIONS = {
target: typescript_1.default.ScriptTarget.ESNext,
module: typescript_1.default.ModuleKind.ESNext,
moduleResolution: typescript_1.default.ModuleResolutionKind.NodeJs,
noLib: true,
typeRoots: [],
pretty: true,
noEmit: true,
declaration: false,
allowSyntheticDefaultImports: true,
resolveJsonModule: false,
experimentalDecorators: true,
jsx: typescript_1.default.JsxEmit.React,
alwaysStrict: true,
strict: true,
skipLibCheck: true,
noUnusedLocals: true,
noImplicitReturns: true,
allowUnusedLabels: false,
noUnusedParameters: false,
allowUnreachableCode: false,
noFallthroughCasesInSwitch: true,
forceConsistentCasingInFileNames: true,
};
const makeContext = (rootNames, host, options = DEFAULT_MAKE_CONTEXT_OPTIONS) => {
const { program, typeChecker, languageService } = createProgram(rootNames, host, options);
const sourceFiles = utils_1.getAllSourceFiles(rootNames.map((rootName) => ts_utils_1.tsUtils.file.getSourceFileOrThrow(program, rootName)), typeChecker);
return createContext(sourceFiles, program, typeChecker, languageService, host);
};
const createModifyHostFiles = (files) => (host) => {
const originalFileExists = host.fileExists === undefined ? undefined : host.fileExists.bind(host);
host.fileExists = (file) => {
if (files[file] !== undefined) {
return true;
}
return originalFileExists === undefined ? false : originalFileExists(file);
};
const originalReadFile = host.readFile === undefined ? undefined : host.readFile.bind(host);
host.readFile = (file, ...args) => {
const foundFile = files[file];
if (foundFile !== undefined) {
return foundFile;
}
return originalReadFile === undefined ? undefined : originalReadFile(file, ...args);
};
};
const createProgram = (rootNames, host, { modifyHost = defaultModifyHost, withTestHarness = false } = DEFAULT_MAKE_CONTEXT_OPTIONS) => {
const servicesHost = host.createLanguageServiceHost(rootNames, exports.COMPILER_OPTIONS, withTestHarness);
modifyHost(servicesHost);
const languageService = typescript_1.default.createLanguageService(servicesHost);
const program = languageService.getProgram();
if (program === undefined) {
throw new Error('Something went wrong');
}
program.__withTestHarness = withTestHarness;
return {
program,
typeChecker: program.getTypeChecker(),
languageService,
};
};
exports.createContextForDir = async (dir, host, options = CREATE_CONTEXT_OPTIONS_DEFAULT) => {
const files = await host.getAllTypescriptFilesInDir(dir);
return makeContext(files, host, options);
};
exports.createContextForPath = (filePath, host, options = CREATE_CONTEXT_OPTIONS_DEFAULT) => makeContext([filePath], host, options);
exports.createContextForSnippet = (code, host, _a = CREATE_CONTEXT_OPTIONS_DEFAULT) => {
var { fileName: fileNameIn } = _a, rest = tslib_1.__rest(_a, ["fileName"]);
const fileName = host.createSnippetFile(fileNameIn);
const context = makeContext([fileName], host, Object.assign({}, rest, { modifyHost: createModifyHostFiles({ [fileName]: code }) }));
const sourceFile = ts_utils_1.tsUtils.file.getSourceFileOrThrow(context.program, fileName);
return {
context,
sourceFile,
};
};
exports.createContextForLanguageService = (filePath, languageService, host) => {
const program = languageService.getProgram();
if (program === undefined) {
throw new Error('Something went wrong');
}
const sourceFile = ts_utils_1.tsUtils.file.getSourceFileOrThrow(program, filePath);
const typeChecker = program.getTypeChecker();
const sourceFiles = utils_1.getAllSourceFiles([sourceFile], typeChecker);
return createContext(sourceFiles, program, typeChecker, languageService, host);
};
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["createContext.ts"],"names":[],"mappings":"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","file":"neo-one-smart-contract-compiler/src/createContext.js","sourcesContent":["import { tsUtils } from '@neo-one/ts-utils';\nimport ts from 'typescript';\nimport { Context } from './Context';\nimport { CompilerHost } from './types';\nimport { getAllSourceFiles } from './utils';\n\nfunction createContext(\n  sourceFiles: Set<ts.SourceFile>,\n  program: ts.Program,\n  typeChecker: ts.TypeChecker,\n  languageService: ts.LanguageService,\n  host: CompilerHost,\n): Context {\n  return new Context(sourceFiles, program, typeChecker, languageService, host);\n}\n\nexport function updateContext(context: Context, files: { readonly [fileName: string]: string | undefined }): Context {\n  const { program, typeChecker, languageService } = createProgram(Object.keys(files), context.host, {\n    modifyHost: createModifyHostFiles(files),\n    // tslint:disable-next-line no-any\n    withTestHarness: (context.program as any).__withTestHarness,\n  });\n\n  const sourceFiles = getAllSourceFiles(\n    Object.keys(files).map((rootName) => tsUtils.file.getSourceFileOrThrow(program, rootName)),\n    typeChecker,\n  );\n\n  return context.update(sourceFiles, program, typeChecker, languageService);\n}\n\nexport interface CreateContextOptions {\n  readonly withTestHarness?: boolean;\n}\n\nexport interface CreateContextSnippetOptions extends CreateContextOptions {\n  readonly fileName?: string;\n}\n\nconst CREATE_CONTEXT_OPTIONS_DEFAULT = {\n  withTestHarness: false,\n};\n\nconst defaultModifyHost = () => {\n  // do nothing\n};\n\ninterface MakeContextOptions extends CreateContextOptions {\n  readonly modifyHost?: (host: ts.LanguageServiceHost) => void;\n}\n\nconst DEFAULT_MAKE_CONTEXT_OPTIONS = {\n  ...CREATE_CONTEXT_OPTIONS_DEFAULT,\n  modifyHost: defaultModifyHost,\n};\n\nexport const COMPILER_OPTIONS: ts.CompilerOptions = {\n  target: ts.ScriptTarget.ESNext,\n  module: ts.ModuleKind.ESNext,\n  moduleResolution: ts.ModuleResolutionKind.NodeJs,\n\n  noLib: true,\n  typeRoots: [],\n\n  pretty: true,\n\n  noEmit: true,\n  declaration: false,\n\n  allowSyntheticDefaultImports: true,\n  resolveJsonModule: false,\n  experimentalDecorators: true,\n  jsx: ts.JsxEmit.React,\n\n  alwaysStrict: true,\n  strict: true,\n  skipLibCheck: true,\n  noUnusedLocals: true,\n  noImplicitReturns: true,\n  allowUnusedLabels: false,\n  noUnusedParameters: false,\n  allowUnreachableCode: false,\n  noFallthroughCasesInSwitch: true,\n  forceConsistentCasingInFileNames: true,\n};\n\nconst makeContext = (\n  rootNames: ReadonlyArray<string>,\n  host: CompilerHost,\n  options: MakeContextOptions = DEFAULT_MAKE_CONTEXT_OPTIONS,\n): Context => {\n  const { program, typeChecker, languageService } = createProgram(rootNames, host, options);\n\n  const sourceFiles = getAllSourceFiles(\n    rootNames.map((rootName) => tsUtils.file.getSourceFileOrThrow(program, rootName)),\n    typeChecker,\n  );\n\n  return createContext(sourceFiles, program, typeChecker, languageService, host);\n};\n\nconst createModifyHostFiles = (files: { readonly [fileName: string]: string | undefined }) => (\n  host: ts.LanguageServiceHost,\n) => {\n  const originalFileExists = host.fileExists === undefined ? undefined : host.fileExists.bind(host);\n  // tslint:disable-next-line no-object-mutation no-any\n  host.fileExists = (file) => {\n    if (files[file] !== undefined) {\n      return true;\n    }\n\n    return originalFileExists === undefined ? false : originalFileExists(file);\n  };\n\n  const originalReadFile = host.readFile === undefined ? undefined : host.readFile.bind(host);\n  // tslint:disable-next-line no-object-mutation no-any\n  host.readFile = (file, ...args: any[]) => {\n    const foundFile = files[file];\n    if (foundFile !== undefined) {\n      return foundFile;\n    }\n\n    return originalReadFile === undefined ? undefined : originalReadFile(file, ...args);\n  };\n};\n\nconst createProgram = (\n  rootNames: ReadonlyArray<string>,\n  host: CompilerHost,\n  { modifyHost = defaultModifyHost, withTestHarness = false }: MakeContextOptions = DEFAULT_MAKE_CONTEXT_OPTIONS,\n) => {\n  const servicesHost = host.createLanguageServiceHost(rootNames, COMPILER_OPTIONS, withTestHarness);\n  modifyHost(servicesHost);\n\n  const languageService = ts.createLanguageService(servicesHost);\n  const program = languageService.getProgram();\n  if (program === undefined) {\n    throw new Error('Something went wrong');\n  }\n\n  // tslint:disable-next-line no-any no-object-mutation\n  (program as any).__withTestHarness = withTestHarness;\n\n  return {\n    program,\n    typeChecker: program.getTypeChecker(),\n    languageService,\n  };\n};\n\nexport const createContextForDir = async (\n  dir: string,\n  host: CompilerHost,\n  options: CreateContextOptions = CREATE_CONTEXT_OPTIONS_DEFAULT,\n): Promise<Context> => {\n  const files = await host.getAllTypescriptFilesInDir(dir);\n\n  return makeContext(files, host, options);\n};\n\nexport const createContextForPath = (\n  filePath: string,\n  host: CompilerHost,\n  options: CreateContextOptions = CREATE_CONTEXT_OPTIONS_DEFAULT,\n): Context => makeContext([filePath], host, options);\n\nexport interface SnippetResult {\n  readonly context: Context;\n  readonly sourceFile: ts.SourceFile;\n}\n\nexport const createContextForSnippet = (\n  code: string,\n  host: CompilerHost,\n  { fileName: fileNameIn, ...rest }: CreateContextSnippetOptions = CREATE_CONTEXT_OPTIONS_DEFAULT,\n): SnippetResult => {\n  const fileName = host.createSnippetFile(fileNameIn);\n\n  const context = makeContext([fileName], host, {\n    ...rest,\n    modifyHost: createModifyHostFiles({ [fileName]: code }),\n  });\n  const sourceFile = tsUtils.file.getSourceFileOrThrow(context.program, fileName);\n\n  return {\n    context,\n    sourceFile,\n  };\n};\n\nexport const createContextForLanguageService = (\n  filePath: string,\n  languageService: ts.LanguageService,\n  host: CompilerHost,\n): Context => {\n  const program = languageService.getProgram();\n  if (program === undefined) {\n    throw new Error('Something went wrong');\n  }\n\n  const sourceFile = tsUtils.file.getSourceFileOrThrow(program, filePath);\n  const typeChecker = program.getTypeChecker();\n  const sourceFiles = getAllSourceFiles([sourceFile], typeChecker);\n\n  return createContext(sourceFiles, program, typeChecker, languageService, host);\n};\n"]}