UNPKG

@cosmology/ast

Version:
335 lines (306 loc) 11.8 kB
import * as t from '@babel/types'; import { getFieldOptionalityForAmino, getOneOfs } from '..'; import { identifier, objectMethod } from '../../../utils'; import { ProtoParseContext } from '../../context'; import { ProtoField, ProtoType } from '@cosmology/types'; import { arrayTypes, toAminoJSON, toAminoMessages } from './utils'; import { SymbolNames } from '../../types'; import { getAminoTypeName } from '../../amino'; const needsImplementation = (name: string, field: ProtoField) => { throw new Error(`need to implement toAminoJSON (${field.type} rules[${field.rule}] name[${name}])`); } export interface ToAminoJSONMethod { context: ProtoParseContext; field: ProtoField; isOneOf: boolean; isOptional: boolean; } export const toAminoJSONMethodFields = (context: ProtoParseContext, name: string, proto: ProtoType) => { const oneOfs = getOneOfs(proto); const fields = Object.keys(proto.fields ?? {}).reduce((m, fieldName) => { const field = { name: fieldName, ...proto.fields[fieldName] }; const isOneOf = oneOfs.includes(fieldName); const isOptional = getFieldOptionalityForAmino(context, field, isOneOf); const args: ToAminoJSONMethod = { context, field, isOneOf, isOptional }; // arrays if (field.rule === 'repeated') { switch (field.type) { case 'string': return [...m, toAminoJSON.array(args, arrayTypes.string(args))]; case 'bytes': return [...m, toAminoJSON.array(args, arrayTypes.bytes(args))]; case 'bool': return [...m, toAminoJSON.array(args, arrayTypes.bool())]; case 'double': return [...m, toAminoJSON.array(args, arrayTypes.double())]; case 'float': return [...m, toAminoJSON.array(args, arrayTypes.float())]; case 'int32': return [...m, toAminoJSON.array(args, arrayTypes.int32())]; case 'sint32': return [...m, toAminoJSON.array(args, arrayTypes.sint32())]; case 'uint32': return [...m, toAminoJSON.array(args, arrayTypes.uint32())]; case 'fixed32': return [...m, toAminoJSON.array(args, arrayTypes.fixed32())]; case 'sfixed32': return [...m, toAminoJSON.array(args, arrayTypes.sfixed32())]; case 'int64': return [...m, toAminoJSON.array(args, arrayTypes.int64(args))]; case 'sint64': return [...m, toAminoJSON.array(args, arrayTypes.sint64(args))]; case 'uint64': return [...m, toAminoJSON.array(args, arrayTypes.uint64(args))]; case 'fixed64': return [...m, toAminoJSON.array(args, arrayTypes.fixed64(args))]; case 'sfixed64': return [...m, toAminoJSON.array(args, arrayTypes.sfixed64(args))]; default: switch (field.parsedType.type) { case 'Enum': return [...m, toAminoJSON.array(args, arrayTypes.enum())]; case 'Type': return [...m, toAminoJSON.array(args, arrayTypes.type(args))]; } return needsImplementation(fieldName, field); } } if (field.keyType) { switch (field.keyType) { case 'string': case 'int32': case 'sint32': case 'uint32': case 'fixed32': case 'sfixed32': case 'int64': case 'sint64': case 'uint64': case 'fixed64': case 'sfixed64': return [...m, ...toAminoJSON.keyHash(args)]; default: return needsImplementation(fieldName, field); } } // casting Any types if (field.type === 'google.protobuf.Any') { switch (field.options?.['(cosmos_proto.accepts_interface)']) { case 'cosmos.crypto.PubKey': return [...m, toAminoJSON.pubkey(args)]; } } if (field.type === 'bytes') { // bytes [RawContractMessage] if (field.options?.['(gogoproto.casttype)'] === 'RawContractMessage') { return [...m, toAminoJSON.rawBytes(args)]; } // bytes [WASMByteCode] // TODO use a better option for this in proto source if (field.options?.['(gogoproto.customname)'] === 'WASMByteCode') { return [...m, toAminoJSON.wasmByteCode(args)]; } } // default types switch (field.type) { case 'string': return [...m, toAminoJSON.string(args)]; case 'double': return [...m, toAminoJSON.double(args)]; case 'float': return [...m, toAminoJSON.float(args)]; case 'bytes': return [...m, toAminoJSON.bytes(args)]; case 'bool': return [...m, toAminoJSON.bool(args)]; case 'int32': return [...m, toAminoJSON.int32(args)]; case 'sint32': return [...m, toAminoJSON.sint32(args)]; case 'uint32': return [...m, toAminoJSON.uint32(args)]; case 'fixed32': return [...m, toAminoJSON.fixed32(args)]; case 'sfixed32': return [...m, toAminoJSON.sfixed32(args)]; case 'int64': return [...m, toAminoJSON.int64(args)]; case 'sint64': return [...m, toAminoJSON.sint64(args)]; case 'uint64': return [...m, toAminoJSON.uint64(args)]; case 'fixed64': return [...m, toAminoJSON.fixed64(args)]; case 'sfixed64': return [...m, toAminoJSON.sfixed64(args)]; case 'google.protobuf.Duration': case 'Duration': return [...m, toAminoJSON.duration(args)]; case 'google.protobuf.Timestamp': case 'Timestamp': return [...m, toAminoJSON.timestamp(args)]; default: switch (field.parsedType.type) { case 'Enum': return [...m, toAminoJSON.enum(args)]; case 'Type': return [...m, toAminoJSON.type(args)]; } return needsImplementation(fieldName, field); } }, []); return fields; }; export const toAminoJSONMethod = (context: ProtoParseContext, name: string, proto: ProtoType) => { const fields = toAminoJSONMethodFields(context, name, proto); let varName = 'message'; if (!fields.length) { varName = '_'; } const AminoTypeName = SymbolNames.Amino(name); const body: t.Statement[] = []; // 1. some messages we parse specially if (proto.type === 'Type') { switch (proto.name) { case 'Duration': case 'google.protobuf.Duration': { body.push(toAminoMessages.duration(context, name, proto)); break; } case 'Timestamp': case 'google.protobuf.Timestamp': body.push(toAminoMessages.timestamp(context, name, proto)); break; case 'google.protobuf.Any': case 'Any': [].push.apply(body, toAminoMessages.anyType()) break; default: } } if (!body.length) { // 2. default to field-level parsing [].push.apply(body, [ t.variableDeclaration( 'const', [ t.variableDeclarator( identifier('obj', t.tsTypeAnnotation(t.tsAnyKeyword())), t.objectExpression([]) ) ] ), ...fields, // RETURN t.returnStatement(t.identifier('obj')) ]); } return objectMethod('method', t.identifier('toAmino'), [ identifier( varName, t.tsTypeAnnotation( t.tsTypeReference( t.identifier(name) ) ) ), ...(context.options.interfaces.enabled && context.options.interfaces.useUseInterfacesParams ? [ t.assignmentPattern( identifier( 'useInterfaces', t.tsTypeAnnotation(t.tsBooleanKeyword()) ), t.identifier( (context.pluginValue('interfaces.useByDefault') ?? true).toString() ) ) ] : []), ], t.blockStatement(body), false, false, false, t.tsTypeAnnotation( t.tsTypeReference( t.identifier(AminoTypeName) ) ) ); }; export const toAminoMsgMethod = (context: ProtoParseContext, name: string, proto: ProtoType) => { const varName = 'message'; const ReturnType = SymbolNames.AminoMsg(name); const TypeName = SymbolNames.Msg(name); const aminoType = getAminoTypeName(context, context.ref.proto, proto); if (!aminoType || aminoType.startsWith('/')) return; const body: t.Statement[] = []; // body body.push( t.returnStatement( t.objectExpression([ t.objectProperty( t.identifier('type'), t.stringLiteral(aminoType) ), t.objectProperty( t.identifier('value'), t.callExpression( t.memberExpression( t.identifier(TypeName), t.identifier('toAmino') ), [ t.identifier(varName), ...(context.options.interfaces.enabled && context.options.interfaces.useUseInterfacesParams ? [ t.identifier('useInterfaces') ] : []), ] ) ) ]) ) ); return objectMethod('method', t.identifier('toAminoMsg'), [ identifier( varName, t.tsTypeAnnotation( t.tsTypeReference( t.identifier(TypeName) ) ) ), ...(context.options.interfaces.enabled && context.options.interfaces.useUseInterfacesParams ? [ t.assignmentPattern( identifier( 'useInterfaces', t.tsTypeAnnotation(t.tsBooleanKeyword()) ), t.identifier( (context.pluginValue('interfaces.useByDefault') ?? true).toString() ) ) ] : []), ], t.blockStatement(body), false, false, false, t.tsTypeAnnotation( t.tsTypeReference( t.identifier(ReturnType) ) ) ); };