mirror of
https://github.com/cosmos/cosmjs.git
synced 2025-03-11 14:09:15 +00:00
Deduplicate makeSignBytes
This commit is contained in:
parent
44182e96b7
commit
2d14efa4cd
@ -1,5 +1,5 @@
|
||||
/* eslint-disable @typescript-eslint/camelcase */
|
||||
import { marshalTx, sortJson, unmarshalTx } from "@cosmwasm/sdk";
|
||||
import { makeSignBytes, marshalTx, types, unmarshalTx } from "@cosmwasm/sdk";
|
||||
import {
|
||||
Address,
|
||||
ChainId,
|
||||
@ -14,7 +14,6 @@ import {
|
||||
TxCodec,
|
||||
UnsignedTransaction,
|
||||
} from "@iov/bcp";
|
||||
import { Encoding } from "@iov/encoding";
|
||||
|
||||
import { CosmosBech32Prefix, isValidAddress, pubkeyToAddress } from "./address";
|
||||
import { Caip5 } from "./caip5";
|
||||
@ -22,8 +21,6 @@ import { parseTx } from "./decode";
|
||||
import { buildSignedTx, buildUnsignedTx } from "./encode";
|
||||
import { nonceToAccountNumber, nonceToSequence, TokenInfos } from "./types";
|
||||
|
||||
const { toUtf8 } = Encoding;
|
||||
|
||||
export class CosmWasmCodec implements TxCodec {
|
||||
private readonly prefix: CosmosBech32Prefix;
|
||||
private readonly tokens: TokenInfos;
|
||||
@ -34,18 +31,19 @@ export class CosmWasmCodec implements TxCodec {
|
||||
}
|
||||
|
||||
public bytesToSign(unsigned: UnsignedTransaction, nonce: Nonce): SigningJob {
|
||||
const memo = (unsigned as any).memo;
|
||||
const built = buildUnsignedTx(unsigned, this.tokens);
|
||||
|
||||
const signMsg = sortJson({
|
||||
account_number: nonceToAccountNumber(nonce).toString(),
|
||||
chain_id: Caip5.decode(unsigned.chainId),
|
||||
fee: (built.value as any).fee,
|
||||
memo: memo,
|
||||
msgs: (built.value as any).msg,
|
||||
sequence: nonceToSequence(nonce).toString(),
|
||||
});
|
||||
const signBytes = toUtf8(JSON.stringify(signMsg));
|
||||
const nonceInfo: types.NonceInfo = {
|
||||
account_number: nonceToAccountNumber(nonce),
|
||||
sequence: nonceToSequence(nonce),
|
||||
};
|
||||
const signBytes = makeSignBytes(
|
||||
built.value.msg[0],
|
||||
built.value.fee,
|
||||
Caip5.decode(unsigned.chainId),
|
||||
built.value.memo || "",
|
||||
nonceInfo,
|
||||
);
|
||||
|
||||
return {
|
||||
bytes: signBytes as SignableBytes,
|
||||
|
@ -23,13 +23,9 @@ const maxAcct = 1 << 23;
|
||||
// tslint:disable-next-line:no-bitwise
|
||||
const maxSeq = 1 << 20;
|
||||
|
||||
// NonceInfo is the data we need from account to create a nonce
|
||||
// Use this so no confusion about order of arguments
|
||||
export type NonceInfo = Pick<types.BaseAccount, "account_number" | "sequence">;
|
||||
|
||||
// this (lossily) encodes the two pieces of info (uint64) needed to sign into
|
||||
// one (53-bit) number. Cross your fingers.
|
||||
export function accountToNonce({ account_number: account, sequence }: NonceInfo): Nonce {
|
||||
export function accountToNonce({ account_number: account, sequence }: types.NonceInfo): Nonce {
|
||||
// we allow 23 bits (8 million) for accounts, and 20 bits (1 million) for tx/account
|
||||
// let's fix this soon
|
||||
if (account > maxAcct) {
|
||||
|
3
packages/bcp/types/types.d.ts
vendored
3
packages/bcp/types/types.d.ts
vendored
@ -15,7 +15,6 @@ export interface TokenInfo {
|
||||
readonly fractionalDigits: number;
|
||||
}
|
||||
export declare type TokenInfos = ReadonlyArray<TokenInfo>;
|
||||
export declare type NonceInfo = Pick<types.BaseAccount, "account_number" | "sequence">;
|
||||
export declare function accountToNonce({ account_number: account, sequence }: NonceInfo): Nonce;
|
||||
export declare function accountToNonce({ account_number: account, sequence }: types.NonceInfo): Nonce;
|
||||
export declare function nonceToAccountNumber(nonce: Nonce): number;
|
||||
export declare function nonceToSequence(nonce: Nonce): number;
|
||||
|
@ -1,11 +1,11 @@
|
||||
import { Secp256k1 } from "@iov/crypto";
|
||||
import { Encoding } from "@iov/encoding";
|
||||
|
||||
import { StdSignature, StdTx } from "./types";
|
||||
import { Msg, NonceInfo, StdFee, StdSignature, StdTx } from "./types";
|
||||
|
||||
const { toBase64 } = Encoding;
|
||||
const { toBase64, toUtf8 } = Encoding;
|
||||
|
||||
export function sortJson(json: any): any {
|
||||
function sortJson(json: any): any {
|
||||
if (typeof json !== "object" || json === null) {
|
||||
return json;
|
||||
}
|
||||
@ -28,6 +28,26 @@ export function marshalTx(tx: StdTx): Uint8Array {
|
||||
return Encoding.toUtf8(json);
|
||||
}
|
||||
|
||||
export function makeSignBytes(
|
||||
msg: Msg,
|
||||
fee: StdFee,
|
||||
chainId: string,
|
||||
memo: string,
|
||||
account: NonceInfo,
|
||||
): Uint8Array {
|
||||
const signMsg = sortJson({
|
||||
// eslint-disable-next-line @typescript-eslint/camelcase
|
||||
account_number: account.account_number.toString(),
|
||||
// eslint-disable-next-line @typescript-eslint/camelcase
|
||||
chain_id: chainId,
|
||||
fee: fee,
|
||||
memo: memo,
|
||||
msgs: msg,
|
||||
sequence: account.sequence.toString(),
|
||||
});
|
||||
return toUtf8(JSON.stringify(signMsg));
|
||||
}
|
||||
|
||||
export function encodeSecp256k1Signature(pubkey: Uint8Array, signature: Uint8Array): StdSignature {
|
||||
return {
|
||||
// eslint-disable-next-line @typescript-eslint/camelcase
|
||||
|
@ -1,6 +1,6 @@
|
||||
import * as types from "./types";
|
||||
|
||||
export { unmarshalTx } from "./decoding";
|
||||
export { encodeSecp256k1Signature, marshalTx, sortJson } from "./encoding";
|
||||
export { encodeSecp256k1Signature, makeSignBytes, marshalTx } from "./encoding";
|
||||
export { RestClient, TxsResponse } from "./restclient";
|
||||
export { types };
|
||||
|
@ -3,13 +3,13 @@ import { ChainId, PrehashType, SignableBytes } from "@iov/bcp";
|
||||
import { Encoding } from "@iov/encoding";
|
||||
import { HdPaths, Secp256k1HdWallet } from "@iov/keycontrol";
|
||||
|
||||
import { encodeSecp256k1Signature, marshalTx, sortJson } from "./encoding";
|
||||
import { encodeSecp256k1Signature, makeSignBytes, marshalTx } from "./encoding";
|
||||
import { RestClient } from "./restclient";
|
||||
import contract from "./testdata/contract.json";
|
||||
import data from "./testdata/cosmoshub.json";
|
||||
import { MsgStoreCode, StdTx } from "./types";
|
||||
import { MsgStoreCode, StdFee, StdTx } from "./types";
|
||||
|
||||
const { fromBase64, toUtf8 } = Encoding;
|
||||
const { fromBase64 } = Encoding;
|
||||
|
||||
const httpUrl = "http://localhost:1317";
|
||||
const defaultNetworkId = "testing";
|
||||
@ -75,41 +75,25 @@ describe("RestClient", () => {
|
||||
builder: "v0.0.1",
|
||||
},
|
||||
};
|
||||
|
||||
const unsigned: StdTx = {
|
||||
msg: [theMsg],
|
||||
memo: memo,
|
||||
signatures: [],
|
||||
fee: {
|
||||
amount: [
|
||||
{
|
||||
amount: "5000",
|
||||
denom: "ucosm",
|
||||
},
|
||||
],
|
||||
gas: "89000000",
|
||||
},
|
||||
const fee: StdFee = {
|
||||
amount: [
|
||||
{
|
||||
amount: "5000",
|
||||
denom: "ucosm",
|
||||
},
|
||||
],
|
||||
gas: "89000000",
|
||||
};
|
||||
|
||||
const client = new RestClient(httpUrl);
|
||||
const account = (await client.authAccounts(faucetAddress)).result.value;
|
||||
|
||||
const signMsg = sortJson({
|
||||
account_number: account.account_number.toString(),
|
||||
chain_id: defaultNetworkId,
|
||||
fee: unsigned.fee,
|
||||
memo: memo,
|
||||
msgs: unsigned.msg,
|
||||
sequence: account.sequence.toString(),
|
||||
});
|
||||
|
||||
const signBytes = toUtf8(JSON.stringify(signMsg)) as SignableBytes;
|
||||
const signBytes = makeSignBytes(theMsg, fee, defaultNetworkId, memo, account) as SignableBytes;
|
||||
const rawSignature = await wallet.createTransactionSignature(signer, signBytes, PrehashType.Sha256);
|
||||
const signature = encodeSecp256k1Signature(signer.pubkey.data, rawSignature);
|
||||
|
||||
const tx: StdTx = {
|
||||
msg: unsigned.msg,
|
||||
fee: unsigned.fee,
|
||||
msg: [theMsg],
|
||||
fee: fee,
|
||||
memo: memo,
|
||||
signatures: [signature],
|
||||
};
|
||||
|
@ -97,3 +97,6 @@ export interface BaseAccount {
|
||||
readonly account_number: number;
|
||||
readonly sequence: number;
|
||||
}
|
||||
|
||||
/** The data we need from BaseAccount to create a nonce */
|
||||
export type NonceInfo = Pick<BaseAccount, "account_number" | "sequence">;
|
||||
|
10
packages/sdk/types/encoding.d.ts
vendored
10
packages/sdk/types/encoding.d.ts
vendored
@ -1,4 +1,10 @@
|
||||
import { StdSignature, StdTx } from "./types";
|
||||
export declare function sortJson(json: any): any;
|
||||
import { Msg, NonceInfo, StdFee, StdSignature, StdTx } from "./types";
|
||||
export declare function marshalTx(tx: StdTx): Uint8Array;
|
||||
export declare function makeSignBytes(
|
||||
msg: Msg,
|
||||
fee: StdFee,
|
||||
chainId: string,
|
||||
memo: string,
|
||||
account: NonceInfo,
|
||||
): Uint8Array;
|
||||
export declare function encodeSecp256k1Signature(pubkey: Uint8Array, signature: Uint8Array): StdSignature;
|
||||
|
2
packages/sdk/types/index.d.ts
vendored
2
packages/sdk/types/index.d.ts
vendored
@ -1,5 +1,5 @@
|
||||
import * as types from "./types";
|
||||
export { unmarshalTx } from "./decoding";
|
||||
export { encodeSecp256k1Signature, marshalTx, sortJson } from "./encoding";
|
||||
export { encodeSecp256k1Signature, makeSignBytes, marshalTx } from "./encoding";
|
||||
export { RestClient, TxsResponse } from "./restclient";
|
||||
export { types };
|
||||
|
2
packages/sdk/types/types.d.ts
vendored
2
packages/sdk/types/types.d.ts
vendored
@ -69,4 +69,6 @@ export interface BaseAccount {
|
||||
readonly account_number: number;
|
||||
readonly sequence: number;
|
||||
}
|
||||
/** The data we need from BaseAccount to create a nonce */
|
||||
export declare type NonceInfo = Pick<BaseAccount, "account_number" | "sequence">;
|
||||
export {};
|
||||
|
Loading…
x
Reference in New Issue
Block a user