@neo-one/smart-contract-compiler
Version:
NEO•ONE TypeScript smart contract compiler.
139 lines (137 loc) • 25.4 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
const ts_utils_1 = require("@neo-one/ts-utils");
const utils_1 = require("@neo-one/utils");
const constants_1 = require("../../../constants");
const utils_2 = require("../../../utils");
const constants_2 = require("../../constants");
const BuiltinMemberCall_1 = require("../BuiltinMemberCall");
class SmartContractForBase extends BuiltinMemberCall_1.BuiltinMemberCall {
emitCall(sb, func, node, optionsIn) {
const options = sb.pushValueOptions(optionsIn);
const returnType = sb.context.analysis.getType(node);
if (returnType === undefined) {
return;
}
const properties = ts_utils_1.tsUtils.type_.getProperties(returnType);
const props = properties.map((prop) => {
const propType = sb.context.analysis.getTypeOfSymbol(prop, node);
if (propType === undefined) {
return undefined;
}
const propName = ts_utils_1.tsUtils.symbol.getName(prop);
if (constants_1.IGNORED_PROPERTIES.has(propName)) {
return undefined;
}
if (constants_1.BUILTIN_PROPERTIES.has(propName)) {
return undefined;
}
const propNode = ts_utils_1.tsUtils.symbol.getValueDeclarationOrThrow(prop);
if (ts_utils_1.tsUtils.modifier.isStatic(propNode) ||
ts_utils_1.tsUtils.modifier.isProtected(propNode) ||
ts_utils_1.tsUtils.modifier.isPrivate(propNode)) {
return undefined;
}
const result = sb.context.analysis.extractSignatureForType(propNode, propType, { error: true });
if (result === undefined) {
return {
paramDecls: [],
paramTypes: new Map(),
returnType: propType,
prop: propNode,
propName,
accessor: true,
isReadonly: ts_utils_1.tsUtils.modifier.isReadonly(propNode),
};
}
return Object.assign({}, result, { prop: propNode, propName, accessor: false, isReadonly: false });
});
const handleParams = (prop, paramDecls, paramTypes, innerOptions) => {
sb.emitHelper(prop, innerOptions, sb.helpers.parameters({
params: paramDecls,
asArgsArr: true,
map: (param, innerInnerOptions, isRestElement) => {
let type = paramTypes.get(param);
if (type !== undefined && isRestElement) {
type = ts_utils_1.tsUtils.type_.getArrayType(type);
}
sb.emitHelper(param, innerInnerOptions, sb.helpers.unwrapValRecursive({ type }));
},
}));
};
const addressName = sb.scope.addUnique();
this.emitInitial(sb, func, node, addressName, options);
sb.emitHelper(node, options, sb.helpers.createObject);
props
.filter(utils_1.utils.notNull)
.forEach(({ prop, propName, paramDecls, paramTypes, returnType: propReturnType, accessor, isReadonly }) => {
sb.emitOp(prop, 'DUP');
sb.emitPushString(prop, propName);
if (accessor && !isReadonly) {
sb.emitHelper(prop, options, sb.helpers.createFunctionArray({
body: (innerOptionsIn) => {
const innerOptions = sb.pushValueOptions(innerOptionsIn);
sb.emitPushInt(prop, 0);
sb.emitOp(prop, 'PICKITEM');
sb.emitHelper(prop, innerOptions, sb.helpers.unwrapValRecursive({ type: propReturnType }));
sb.emitPushInt(prop, 1);
sb.emitOp(prop, 'PACK');
sb.emitPushString(prop, utils_2.getSetterName(propName));
const callBuffer = Buffer.from([0, 2]);
this.emitInvoke(sb, func, node, prop, addressName, callBuffer, sb.noPushValueOptions(innerOptions));
sb.emitHelper(prop, innerOptions, sb.helpers.wrapUndefined);
sb.emitHelper(prop, innerOptions, sb.helpers.return);
},
}));
sb.emitHelper(prop, options, sb.helpers.createFunctionObject({
property: constants_2.InternalObjectProperty.Call,
}));
}
sb.emitHelper(prop, options, sb.helpers.createFunctionArray({
body: (innerOptionsIn) => {
const innerOptions = sb.pushValueOptions(innerOptionsIn);
if (accessor) {
sb.emitOp(prop, 'DROP');
sb.emitPushInt(prop, 0);
sb.emitOp(prop, 'NEWARRAY');
}
else {
handleParams(prop, paramDecls, paramTypes, innerOptions);
}
sb.emitPushString(prop, propName);
const isVoidReturn = propReturnType !== undefined && ts_utils_1.tsUtils.type_.isVoid(propReturnType);
const callBuffer = Buffer.from([isVoidReturn ? 0 : 1, 2]);
this.emitInvoke(sb, func, node, prop, addressName, callBuffer, innerOptions);
if (isVoidReturn) {
sb.emitHelper(prop, innerOptions, sb.helpers.wrapUndefined);
}
else {
sb.emitHelper(prop, innerOptions, sb.helpers.wrapValRecursive({
type: propReturnType,
}));
}
sb.emitHelper(prop, innerOptions, sb.helpers.return);
},
}));
sb.emitHelper(prop, options, sb.helpers.createFunctionObject({
property: constants_2.InternalObjectProperty.Call,
}));
if (accessor) {
sb.emitHelper(prop, options, sb.helpers.setAccessorPropertyObjectProperty({ hasGet: true, hasSet: !isReadonly }));
}
else {
sb.emitHelper(prop, options, sb.helpers.setDataPropertyObjectProperty);
}
});
this.emitAdditionalProperties(sb, func, node, options);
if (!optionsIn.pushValue) {
sb.emitOp(node, 'DROP');
}
}
emitInitial(_sb, _func, _node, _addressName, _options) {
}
emitAdditionalProperties(_sb, _func, _node, _options) {
}
}
exports.SmartContractForBase = SmartContractForBase;
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["SmartContractForBase.ts"],"names":[],"mappings":"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","file":"neo-one-smart-contract-compiler/src/compile/builtins/contract/SmartContractForBase.js","sourcesContent":["import { tsUtils } from '@neo-one/ts-utils';\nimport { utils } from '@neo-one/utils';\nimport ts from 'typescript';\nimport { BUILTIN_PROPERTIES, IGNORED_PROPERTIES } from '../../../constants';\nimport { getSetterName } from '../../../utils';\nimport { InternalObjectProperty } from '../../constants';\nimport { ScriptBuilder } from '../../sb';\nimport { Name } from '../../scope';\nimport { VisitOptions } from '../../types';\nimport { BuiltinMemberCall } from '../BuiltinMemberCall';\nimport { MemberLikeExpression } from '../types';\n\nexport abstract class SmartContractForBase extends BuiltinMemberCall {\n  public emitCall(\n    sb: ScriptBuilder,\n    func: MemberLikeExpression,\n    node: ts.CallExpression,\n    optionsIn: VisitOptions,\n  ): void {\n    const options = sb.pushValueOptions(optionsIn);\n    const returnType = sb.context.analysis.getType(node);\n    if (returnType === undefined) {\n      /* istanbul ignore next */\n      return;\n    }\n\n    const properties = tsUtils.type_.getProperties(returnType);\n    const props = properties.map((prop) => {\n      const propType = sb.context.analysis.getTypeOfSymbol(prop, node);\n      if (propType === undefined) {\n        /* istanbul ignore next */\n        return undefined;\n      }\n\n      const propName = tsUtils.symbol.getName(prop);\n      if (IGNORED_PROPERTIES.has(propName)) {\n        return undefined;\n      }\n      if (BUILTIN_PROPERTIES.has(propName)) {\n        return undefined;\n      }\n      const propNode = tsUtils.symbol.getValueDeclarationOrThrow(prop);\n      if (\n        tsUtils.modifier.isStatic(propNode) ||\n        tsUtils.modifier.isProtected(propNode) ||\n        tsUtils.modifier.isPrivate(propNode)\n      ) {\n        return undefined;\n      }\n\n      const result = sb.context.analysis.extractSignatureForType(propNode, propType, { error: true });\n      if (result === undefined) {\n        // Must be a property, not a method\n        return {\n          paramDecls: [],\n          paramTypes: new Map<ts.ParameterDeclaration, ts.Type | undefined>(),\n          returnType: propType,\n          prop: propNode,\n          propName,\n          accessor: true,\n          isReadonly: tsUtils.modifier.isReadonly(propNode),\n        };\n      }\n\n      return { ...result, prop: propNode, propName, accessor: false, isReadonly: false };\n    });\n\n    const handleParams = (\n      prop: ts.Declaration,\n      paramDecls: ReadonlyArray<ts.ParameterDeclaration>,\n      paramTypes: Map<ts.ParameterDeclaration, ts.Type | undefined>,\n      innerOptions: VisitOptions,\n    ) => {\n      // [params]\n      sb.emitHelper(\n        prop,\n        innerOptions,\n        sb.helpers.parameters({\n          params: paramDecls,\n          asArgsArr: true,\n          map: (param, innerInnerOptions, isRestElement) => {\n            let type = paramTypes.get(param);\n            if (type !== undefined && isRestElement) {\n              type = tsUtils.type_.getArrayType(type);\n            }\n            // [value]\n            sb.emitHelper(param, innerInnerOptions, sb.helpers.unwrapValRecursive({ type }));\n          },\n        }),\n      );\n    };\n\n    const addressName = sb.scope.addUnique();\n    this.emitInitial(sb, func, node, addressName, options);\n\n    // [objectVal]\n    sb.emitHelper(node, options, sb.helpers.createObject);\n    // [objectVal]\n    props\n      .filter(utils.notNull)\n      .forEach(({ prop, propName, paramDecls, paramTypes, returnType: propReturnType, accessor, isReadonly }) => {\n        // [objectVal, objectVal]\n        sb.emitOp(prop, 'DUP');\n        // [string, objectVal, objectVal]\n        sb.emitPushString(prop, propName);\n        if (accessor && !isReadonly) {\n          sb.emitHelper(\n            prop,\n            options,\n            sb.helpers.createFunctionArray({\n              body: (innerOptionsIn) => {\n                const innerOptions = sb.pushValueOptions(innerOptionsIn);\n                // [0, argsarr]\n                sb.emitPushInt(prop, 0);\n                // [val]\n                sb.emitOp(prop, 'PICKITEM');\n                // [value]\n                sb.emitHelper(prop, innerOptions, sb.helpers.unwrapValRecursive({ type: propReturnType }));\n                // [1, value]\n                sb.emitPushInt(prop, 1);\n                // [params]\n                sb.emitOp(prop, 'PACK');\n                // [string, params]\n                sb.emitPushString(prop, getSetterName(propName));\n\n                const callBuffer = Buffer.from([0, 2]);\n                this.emitInvoke(sb, func, node, prop, addressName, callBuffer, sb.noPushValueOptions(innerOptions));\n                // [val]\n                sb.emitHelper(prop, innerOptions, sb.helpers.wrapUndefined);\n                // []\n                sb.emitHelper(prop, innerOptions, sb.helpers.return);\n              },\n            }),\n          );\n          sb.emitHelper(\n            prop,\n            options,\n            sb.helpers.createFunctionObject({\n              property: InternalObjectProperty.Call,\n            }),\n          );\n        }\n        // [farr, string, objectVal, objectVal]\n        sb.emitHelper(\n          prop,\n          options,\n          sb.helpers.createFunctionArray({\n            body: (innerOptionsIn) => {\n              const innerOptions = sb.pushValueOptions(innerOptionsIn);\n              if (accessor) {\n                // []\n                sb.emitOp(prop, 'DROP');\n                // [number]\n                sb.emitPushInt(prop, 0);\n                // [params]\n                sb.emitOp(prop, 'NEWARRAY');\n              } else {\n                // [params]\n                handleParams(prop, paramDecls, paramTypes, innerOptions);\n              }\n              // [string, params]\n              sb.emitPushString(prop, propName);\n\n              const isVoidReturn = propReturnType !== undefined && tsUtils.type_.isVoid(propReturnType);\n              const callBuffer = Buffer.from([isVoidReturn ? 0 : 1, 2]);\n              this.emitInvoke(sb, func, node, prop, addressName, callBuffer, innerOptions);\n\n              if (isVoidReturn) {\n                sb.emitHelper(prop, innerOptions, sb.helpers.wrapUndefined);\n              } else {\n                // [val]\n                sb.emitHelper(\n                  prop,\n                  innerOptions,\n                  sb.helpers.wrapValRecursive({\n                    type: propReturnType,\n                  }),\n                );\n              }\n\n              // []\n              sb.emitHelper(prop, innerOptions, sb.helpers.return);\n            },\n          }),\n        );\n        // [fobj, string, objectVal, objectVal]\n        sb.emitHelper(\n          prop,\n          options,\n          sb.helpers.createFunctionObject({\n            property: InternalObjectProperty.Call,\n          }),\n        );\n        if (accessor) {\n          // [objectVal]\n          sb.emitHelper(\n            prop,\n            options,\n            sb.helpers.setAccessorPropertyObjectProperty({ hasGet: true, hasSet: !isReadonly }),\n          );\n        } else {\n          // [objectVal]\n          sb.emitHelper(prop, options, sb.helpers.setDataPropertyObjectProperty);\n        }\n      });\n\n    // [objectVal]\n    this.emitAdditionalProperties(sb, func, node, options);\n\n    if (!optionsIn.pushValue) {\n      sb.emitOp(node, 'DROP');\n    }\n  }\n\n  protected emitInitial(\n    _sb: ScriptBuilder,\n    _func: MemberLikeExpression,\n    _node: ts.CallExpression,\n    _addressName: Name,\n    _options: VisitOptions,\n  ): void {\n    // do nothing\n  }\n\n  protected emitAdditionalProperties(\n    _sb: ScriptBuilder,\n    _func: MemberLikeExpression,\n    _node: ts.CallExpression,\n    _options: VisitOptions,\n  ): void {\n    // do nothing\n  }\n\n  protected abstract emitInvoke(\n    sb: ScriptBuilder,\n    func: MemberLikeExpression,\n    node: ts.CallExpression,\n    prop: ts.Declaration,\n    addressName: Name,\n    callBuffer: Buffer,\n    optionsIn: VisitOptions,\n  ): void;\n}\n"]}