@neo-one/smart-contract-compiler
Version:
NEO•ONE TypeScript smart contract compiler.
89 lines (87 loc) • 15.9 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
const tslib_1 = require("tslib");
const ts_utils_1 = require("@neo-one/ts-utils");
const utils_1 = require("@neo-one/utils");
const typescript_1 = tslib_1.__importDefault(require("typescript"));
const createContext_1 = require("./createContext");
const errors_1 = require("./errors");
exports.scanContext = (context) => {
const smartContract = ts_utils_1.tsUtils.symbol.getDeclarations(context.builtins.getValueSymbol('SmartContract'))[0];
if (!typescript_1.default.isClassDeclaration(smartContract)) {
throw new Error('Something went wrong!');
}
const { contracts, dependencies } = ts_utils_1.tsUtils.class_
.getExtendors(context.program, context.languageService, smartContract)
.reduce((acc, derived) => {
if (!ts_utils_1.tsUtils.modifier.isAbstract(derived) &&
!ts_utils_1.tsUtils.file.isDeclarationFile(ts_utils_1.tsUtils.node.getSourceFile(derived))) {
const filePath = ts_utils_1.tsUtils.file.getFilePath(ts_utils_1.tsUtils.node.getSourceFile(derived));
const name = ts_utils_1.tsUtils.node.getNameOrThrow(derived);
const existing = acc.contracts[filePath];
if (existing !== undefined) {
throw new errors_1.MultipleContractsInFileError(filePath);
}
const references = [
...new Set(ts_utils_1.tsUtils.reference
.findReferencesAsNodes(context.program, context.languageService, derived)
.map((reference) => ts_utils_1.tsUtils.file.getFilePath(ts_utils_1.tsUtils.node.getSourceFile(reference)))),
];
const dependency = { filePath, name };
const dependenciesOut = references.reduce((innerAcc, reference) => {
let filePathDependencies = innerAcc[reference];
if (filePathDependencies === undefined) {
filePathDependencies = [];
}
return Object.assign({}, innerAcc, { [reference]: [...filePathDependencies, dependency] });
}, acc.dependencies);
return {
contracts: Object.assign({}, acc.contracts, { [filePath]: {
filePath,
name,
dependencies: [],
} }),
dependencies: dependenciesOut,
};
}
return acc;
}, { contracts: {}, dependencies: {} });
const unsortedContracts = Object.values(contracts).map((contract) => {
const filePathDependencies = dependencies[contract.filePath];
return Object.assign({}, contract, { dependencies: filePathDependencies === undefined ? [] : filePathDependencies });
});
return topographicalSort(unsortedContracts);
};
const topographicalSort = (contracts) => {
const contractToDependencies = contracts.reduce((acc, contract) => (Object.assign({}, acc, { [contract.filePath]: new Set(contract.dependencies.map((dep) => dep.filePath)) })), {});
const mutableOut = [];
const satisfied = contracts.filter((contract) => contract.dependencies.length === 0);
let remaining = contracts.filter((contract) => contract.dependencies.length !== 0);
while (satisfied.length > 0) {
const node = satisfied.shift();
if (node === undefined) {
break;
}
mutableOut.push(node);
remaining = remaining
.map((contract) => {
const deps = contractToDependencies[contract.filePath];
deps.delete(node.filePath);
if (deps.size === 0) {
satisfied.push(contract);
return undefined;
}
return contract;
})
.filter(utils_1.utils.notNull);
}
if (mutableOut.length !== contracts.length) {
throw new errors_1.CircularLinkedDependencyError(contracts.map((contract) => contract.name));
}
return mutableOut;
};
exports.scan = async (dir, host) => {
const context = await createContext_1.createContextForDir(dir, host);
return exports.scanContext(context);
};
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["scan.ts"],"names":[],"mappings":"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","file":"neo-one-smart-contract-compiler/src/scan.js","sourcesContent":["import { tsUtils } from '@neo-one/ts-utils';\nimport { utils } from '@neo-one/utils';\nimport ts from 'typescript';\nimport { Context } from './Context';\nimport { createContextForDir } from './createContext';\nimport { CircularLinkedDependencyError, MultipleContractsInFileError } from './errors';\nimport { CompilerHost } from './types';\n\nexport interface ContractDependency {\n  readonly filePath: string;\n  readonly name: string;\n}\nexport interface Contract {\n  readonly filePath: string;\n  readonly name: string;\n  readonly dependencies: ReadonlyArray<ContractDependency>;\n}\nexport type Contracts = ReadonlyArray<Contract>;\n\ninterface FilePathToContract {\n  readonly [filePath: string]: Contract;\n}\ninterface FilePathToDependencies {\n  readonly [filePath: string]: ReadonlyArray<ContractDependency>;\n}\n\nexport const scanContext = (context: Context): Contracts => {\n  const smartContract = tsUtils.symbol.getDeclarations(context.builtins.getValueSymbol('SmartContract'))[0];\n  if (!ts.isClassDeclaration(smartContract)) {\n    throw new Error('Something went wrong!');\n  }\n\n  const { contracts, dependencies } = tsUtils.class_\n    .getExtendors(context.program, context.languageService, smartContract)\n    .reduce<{ contracts: FilePathToContract; dependencies: FilePathToDependencies }>(\n      (acc, derived) => {\n        if (\n          !tsUtils.modifier.isAbstract(derived) &&\n          !tsUtils.file.isDeclarationFile(tsUtils.node.getSourceFile(derived))\n        ) {\n          const filePath = tsUtils.file.getFilePath(tsUtils.node.getSourceFile(derived));\n          const name = tsUtils.node.getNameOrThrow(derived);\n          const existing = acc.contracts[filePath] as Contract | undefined;\n          if (existing !== undefined) {\n            throw new MultipleContractsInFileError(filePath);\n          }\n\n          const references = [\n            ...new Set(\n              tsUtils.reference\n                .findReferencesAsNodes(context.program, context.languageService, derived)\n                .map((reference) => tsUtils.file.getFilePath(tsUtils.node.getSourceFile(reference))),\n            ),\n          ];\n\n          const dependency = { filePath, name };\n          const dependenciesOut = references.reduce((innerAcc, reference) => {\n            let filePathDependencies = innerAcc[reference] as ReadonlyArray<ContractDependency> | undefined;\n            if (filePathDependencies === undefined) {\n              filePathDependencies = [];\n            }\n\n            return {\n              ...innerAcc,\n              [reference]: [...filePathDependencies, dependency],\n            };\n          }, acc.dependencies);\n\n          return {\n            contracts: {\n              ...acc.contracts,\n              [filePath]: {\n                filePath,\n                name,\n                dependencies: [],\n              },\n            },\n            dependencies: dependenciesOut,\n          };\n        }\n\n        return acc;\n      },\n      { contracts: {}, dependencies: {} },\n    );\n\n  const unsortedContracts = Object.values(contracts).map((contract) => {\n    const filePathDependencies = dependencies[contract.filePath] as ReadonlyArray<ContractDependency> | undefined;\n\n    return {\n      ...contract,\n      dependencies: filePathDependencies === undefined ? [] : filePathDependencies,\n    };\n  });\n\n  return topographicalSort(unsortedContracts);\n};\n\nconst topographicalSort = (contracts: Contracts): Contracts => {\n  const contractToDependencies = contracts.reduce<{ [filePath: string]: Set<string> }>(\n    (acc, contract) => ({\n      ...acc,\n      [contract.filePath]: new Set(contract.dependencies.map((dep) => dep.filePath)),\n    }),\n    {},\n  );\n  const mutableOut: Contract[] = [];\n  const satisfied = contracts.filter((contract) => contract.dependencies.length === 0);\n  let remaining = contracts.filter((contract) => contract.dependencies.length !== 0);\n  // tslint:disable-next-line no-loop-statement\n  while (satisfied.length > 0) {\n    // tslint:disable-next-line no-array-mutation\n    const node = satisfied.shift();\n    if (node === undefined) {\n      /* istanbul ignore next */\n      break;\n    }\n\n    mutableOut.push(node);\n    remaining = remaining\n      .map((contract) => {\n        const deps = contractToDependencies[contract.filePath];\n        deps.delete(node.filePath);\n        if (deps.size === 0) {\n          // tslint:disable-next-line no-array-mutation\n          satisfied.push(contract);\n\n          return undefined;\n        }\n\n        return contract;\n      })\n      .filter(utils.notNull);\n  }\n\n  if (mutableOut.length !== contracts.length) {\n    throw new CircularLinkedDependencyError(contracts.map((contract) => contract.name));\n  }\n\n  return mutableOut;\n};\n\nexport const scan = async (dir: string, host: CompilerHost): Promise<Contracts> => {\n  const context = await createContextForDir(dir, host);\n\n  return scanContext(context);\n};\n"]}