UNPKG

@neo-one/smart-contract-compiler

Version:

NEO•ONE TypeScript smart contract compiler.

153 lines (151 loc) 26 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const tslib_1 = require("tslib"); const ts_utils_1 = require("@neo-one/ts-utils"); const lodash_1 = tslib_1.__importDefault(require("lodash")); const typescript_1 = tslib_1.__importDefault(require("typescript")); const DiagnosticCode_1 = require("../../../DiagnosticCode"); const DiagnosticMessage_1 = require("../../../DiagnosticMessage"); const constants_1 = require("../../constants"); const Helper_1 = require("../Helper"); const types_1 = require("../types"); class ArgumentsHelper extends Helper_1.Helper { emit(sb, node, optionsIn) { const options = sb.pushValueOptions(optionsIn); const args = typescript_1.default.isArrayLiteralExpression(node) ? ts_utils_1.tsUtils.expression.getElements(node) : ts_utils_1.tsUtils.argumented.getArgumentsArray(node); if (!typescript_1.default.isArrayLiteralExpression(node) && args.length > 0) { const signatureTypes = sb.context.analysis.extractSignaturesForCall(node, { error: true }); if (signatureTypes !== undefined) { args.forEach((arg, idx) => { const argType = sb.context.analysis.getType(arg); if (argType !== undefined) { const mismatch = signatureTypes.some(({ paramDecls, paramTypes }) => { const paramDecl = paramDecls[Math.min(idx, paramDecls.length - 1)]; let paramTypeIn = paramTypes.get(paramDecl); if (paramTypeIn !== undefined && ts_utils_1.tsUtils.parameter.isRestParameter(paramDecl)) { paramTypeIn = ts_utils_1.tsUtils.type_.getArrayType(paramTypeIn); } const paramType = paramTypeIn; const hasBuiltins = types_1.getHasBuiltins(sb.context, arg, argType); return (paramType === undefined || hasBuiltins.some((hasBuiltin) => !hasBuiltin(sb.context, paramDecl, paramType))); }); if (mismatch) { sb.context.reportError(arg, DiagnosticCode_1.DiagnosticCode.InvalidBuiltinUsage, DiagnosticMessage_1.DiagnosticMessage.InvalidBuiltinCallArgument); } } }); } } if (args.some((arg) => typescript_1.default.isSpreadElement(arg))) { sb.emitPushInt(node, 0); sb.emitOp(node, 'NEWARRAY'); args.forEach((arg) => { const handleArrayLike = () => { sb.emitOp(arg, 'TUCK'); sb.emitOp(arg, 'SWAP'); sb.emitOp(arg, 'APPEND'); }; const handleArray = (innerOptions) => { sb.emitHelper(arg, innerOptions, sb.helpers.unwrapArray); sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.arrReduce({ each: handleArrayLike, })); }; const handleMapLike = (innerOption) => { sb.emitOp(arg, 'ROT'); sb.emitOp(arg, 'ROT'); sb.emitPushInt(arg, 2); sb.emitOp(arg, 'PACK'); sb.emitHelper(arg, innerOption, sb.helpers.wrapArray); sb.emitOp(arg, 'OVER'); sb.emitOp(arg, 'SWAP'); sb.emitOp(arg, 'APPEND'); }; const handleMap = (innerOptions) => { sb.emitHelper(arg, innerOptions, sb.helpers.unwrapMap); sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.mapReduce({ each: handleMapLike, })); }; const handleSetLike = () => { sb.emitOp(arg, 'ROT'); sb.emitOp(arg, 'DROP'); sb.emitOp(arg, 'TUCK'); sb.emitOp(arg, 'SWAP'); sb.emitOp(arg, 'APPEND'); }; const handleSet = (innerOptions) => { sb.emitHelper(arg, innerOptions, sb.helpers.unwrapSet); sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.mapReduce({ each: handleSetLike, })); }; const handleArrayStorage = (innerOptions) => { sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.structuredStorageReduceVal({ type: constants_1.Types.ArrayStorage, each: handleArrayLike, })); }; const handleMapStorage = (innerOptions) => { sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.structuredStorageReduce({ type: constants_1.Types.MapStorage, each: handleMapLike, })); }; const handleSetStorage = (innerOptions) => { sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.structuredStorageReduce({ type: constants_1.Types.SetStorage, each: handleSetLike, })); }; const handleIterableIterator = (innerOptions) => { sb.emitOp(arg, 'SWAP'); sb.emitHelper(arg, innerOptions, sb.helpers.iterableIteratorReduce({ each: handleArrayLike, })); }; if (typescript_1.default.isSpreadElement(arg)) { const expr = ts_utils_1.tsUtils.expression.getExpression(arg); sb.visit(expr, options); sb.emitHelper(arg, options, sb.helpers.forIterableType({ type: sb.context.analysis.getType(expr), array: handleArray, map: handleMap, set: handleSet, arrayStorage: handleArrayStorage, mapStorage: handleMapStorage, setStorage: handleSetStorage, iterableIterator: handleIterableIterator, })); } else { sb.emitOp(node, 'DUP'); sb.visit(arg, options); sb.emitOp(node, 'APPEND'); } }); } else { const reversedElements = lodash_1.default.reverse([...args]); reversedElements.forEach((arg) => { sb.visit(arg, options); }); sb.emitPushInt(node, args.length); sb.emitOp(node, 'PACK'); } if (!optionsIn.pushValue) { sb.emitOp(node, 'DROP'); } } } exports.ArgumentsHelper = ArgumentsHelper; //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["ArgumentsHelper.ts"],"names":[],"mappings":"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","file":"neo-one-smart-contract-compiler/src/compile/helper/function/ArgumentsHelper.js","sourcesContent":["import { tsUtils } from '@neo-one/ts-utils';\nimport _ from 'lodash';\nimport ts from 'typescript';\nimport { DiagnosticCode } from '../../../DiagnosticCode';\nimport { DiagnosticMessage } from '../../../DiagnosticMessage';\nimport { Types } from '../../constants';\nimport { ScriptBuilder } from '../../sb';\nimport { VisitOptions } from '../../types';\nimport { Helper } from '../Helper';\nimport { getHasBuiltins } from '../types';\n\n// Input: []\n// Output: [argsArray]\nexport class ArgumentsHelper extends Helper<ts.CallExpression | ts.NewExpression | ts.ArrayLiteralExpression> {\n  public emit(\n    sb: ScriptBuilder,\n    node: ts.CallExpression | ts.NewExpression | ts.ArrayLiteralExpression,\n    optionsIn: VisitOptions,\n  ): void {\n    const options = sb.pushValueOptions(optionsIn);\n    const args = ts.isArrayLiteralExpression(node)\n      ? tsUtils.expression.getElements(node)\n      : tsUtils.argumented.getArgumentsArray(node);\n    if (!ts.isArrayLiteralExpression(node) && args.length > 0) {\n      const signatureTypes = sb.context.analysis.extractSignaturesForCall(node, { error: true });\n\n      if (signatureTypes !== undefined) {\n        args.forEach((arg, idx) => {\n          const argType = sb.context.analysis.getType(arg);\n          if (argType !== undefined) {\n            const mismatch = signatureTypes.some(({ paramDecls, paramTypes }) => {\n              const paramDecl = paramDecls[Math.min(idx, paramDecls.length - 1)];\n              let paramTypeIn = paramTypes.get(paramDecl);\n              if (paramTypeIn !== undefined && tsUtils.parameter.isRestParameter(paramDecl)) {\n                paramTypeIn = tsUtils.type_.getArrayType(paramTypeIn);\n              }\n              const paramType = paramTypeIn;\n              const hasBuiltins = getHasBuiltins(sb.context, arg, argType);\n\n              return (\n                paramType === undefined ||\n                hasBuiltins.some((hasBuiltin) => !hasBuiltin(sb.context, paramDecl, paramType))\n              );\n            });\n\n            if (mismatch) {\n              sb.context.reportError(\n                arg,\n                DiagnosticCode.InvalidBuiltinUsage,\n                DiagnosticMessage.InvalidBuiltinCallArgument,\n              );\n            }\n          }\n        });\n      }\n    }\n\n    if (args.some((arg) => ts.isSpreadElement(arg))) {\n      // [0]\n      sb.emitPushInt(node, 0);\n      // [arr]\n      sb.emitOp(node, 'NEWARRAY');\n      // [arr]\n      args.forEach((arg) => {\n        const handleArrayLike = () => {\n          // [arrOut, val, arrOut]\n          sb.emitOp(arg, 'TUCK');\n          // [val, arrOut, arrOut]\n          sb.emitOp(arg, 'SWAP');\n          // [arrOut]\n          sb.emitOp(arg, 'APPEND');\n        };\n\n        const handleArray = (innerOptions: VisitOptions) => {\n          // [arr, arrOut]\n          sb.emitHelper(arg, innerOptions, sb.helpers.unwrapArray);\n          // [arrOut, arr]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.arrReduce({\n              each: handleArrayLike,\n            }),\n          );\n        };\n\n        const handleMapLike = (innerOption: VisitOptions) => {\n          // [value, arrOut, key]\n          sb.emitOp(arg, 'ROT');\n          // [key, value, arrOut]\n          sb.emitOp(arg, 'ROT');\n          // [2, key, value, arrOut]\n          sb.emitPushInt(arg, 2);\n          // [arr, arrOut]\n          sb.emitOp(arg, 'PACK');\n          // [val, arrOut]\n          sb.emitHelper(arg, innerOption, sb.helpers.wrapArray);\n          // [arrOut, val, arrOut]\n          sb.emitOp(arg, 'OVER');\n          // [val, arrOut, arrOut]\n          sb.emitOp(arg, 'SWAP');\n          // [arrOut]\n          sb.emitOp(arg, 'APPEND');\n        };\n\n        const handleMap = (innerOptions: VisitOptions) => {\n          // [map, arrOut]\n          sb.emitHelper(arg, innerOptions, sb.helpers.unwrapMap);\n          // [arrOut, map]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.mapReduce({\n              each: handleMapLike,\n            }),\n          );\n        };\n\n        const handleSetLike = () => {\n          // [value, arrOut, key]\n          sb.emitOp(arg, 'ROT');\n          // [arrOut, key]\n          sb.emitOp(arg, 'DROP');\n          // [arrOut, key, arrOut]\n          sb.emitOp(arg, 'TUCK');\n          // [key, arrOut, arrOut]\n          sb.emitOp(arg, 'SWAP');\n          // [arrOut]\n          sb.emitOp(arg, 'APPEND');\n        };\n\n        const handleSet = (innerOptions: VisitOptions) => {\n          // [map, arrOut]\n          sb.emitHelper(arg, innerOptions, sb.helpers.unwrapSet);\n          // [arrOut, map]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.mapReduce({\n              each: handleSetLike,\n            }),\n          );\n        };\n\n        const handleArrayStorage = (innerOptions: VisitOptions) => {\n          // [arrOut, val]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.structuredStorageReduceVal({\n              type: Types.ArrayStorage,\n              each: handleArrayLike,\n            }),\n          );\n        };\n\n        const handleMapStorage = (innerOptions: VisitOptions) => {\n          // [arrOut, val]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.structuredStorageReduce({\n              type: Types.MapStorage,\n              each: handleMapLike,\n            }),\n          );\n        };\n\n        const handleSetStorage = (innerOptions: VisitOptions) => {\n          // [arrOut, val]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.structuredStorageReduce({\n              type: Types.SetStorage,\n              each: handleSetLike,\n            }),\n          );\n        };\n\n        const handleIterableIterator = (innerOptions: VisitOptions) => {\n          // [arrOut, val]\n          sb.emitOp(arg, 'SWAP');\n          sb.emitHelper(\n            arg,\n            innerOptions,\n            sb.helpers.iterableIteratorReduce({\n              each: handleArrayLike,\n            }),\n          );\n        };\n\n        if (ts.isSpreadElement(arg)) {\n          const expr = tsUtils.expression.getExpression(arg);\n          // [iterable, arr]\n          sb.visit(expr, options);\n          // [arr]\n          sb.emitHelper(\n            arg,\n            options,\n            sb.helpers.forIterableType({\n              type: sb.context.analysis.getType(expr),\n              array: handleArray,\n              map: handleMap,\n              set: handleSet,\n              arrayStorage: handleArrayStorage,\n              mapStorage: handleMapStorage,\n              setStorage: handleSetStorage,\n              iterableIterator: handleIterableIterator,\n            }),\n          );\n        } else {\n          // [arr, arr]\n          sb.emitOp(node, 'DUP');\n          // [val, arr, arr]\n          sb.visit(arg, options);\n          // [arr]\n          sb.emitOp(node, 'APPEND');\n        }\n      });\n    } else {\n      const reversedElements = _.reverse([...args]);\n      reversedElements.forEach((arg) => {\n        sb.visit(arg, options);\n      });\n      // [length, ...vals]\n      sb.emitPushInt(node, args.length);\n      // [arr]\n      sb.emitOp(node, 'PACK');\n    }\n\n    if (!optionsIn.pushValue) {\n      sb.emitOp(node, 'DROP');\n    }\n  }\n}\n"]}