@iov/multichain
@iov/multichain exposes high-level functionality to work with multiple
blockchains. It uses the keymanagement functionality of UserProfile
, and the
generic blockchain connection of BlockchainConnection
, and pulls them together
into one MultiChainSigner
, which can query state and sign transactions on
multiple blockchains. The examples below show a basic usage of
MultiChainSigner
. You may also want to experiment with
@iov/cli
as a developer tool to familiarize yourself with this functionality.
Full Api Docs
Full API Docs from the latest release are hosted at: https://iov-one.github.io/iov-core-docs/
If you want to generate documentation for a development branch, please run
yarn docs
in the current directory and go to
docs/index.html.
Examples
Here are some example use cases. They all build on each other and assume all
imports from above. I also use await
syntax here, which works inside of async
functions and experimentally in the
@iov/cli
REPL. (All imports are done for you in the REPL as well, so you can skip the
import statements. They are provided for guidance when integrating into your own
codebase).
Before starting, either run from source in ../iov-cli
via
yarn build && ./bin/iov-cli
or install from npm via
npm install -g @iov/cli; iov-cli
. Inside the cli the remaining code should
work verbatim.
Key Management
Create a random mnemonic:
import { Bip39, Random } from "@iov/crypto";
// 16 bytes -> 12 word phrase
const entropy16 = Random.getBytes(16);
const mnemonic12 = Bip39.encode(entropy16).toString();
console.log(mnemonic12);
// 32 bytes -> 24 word phrase
const entropy32 = Random.getBytes(32);
const mnemonic24 = Bip39.encode(entropy32).toString();
console.log(mnemonic24);
Create a new profile with two wallets:
import { Ed25519HdWallet, UserProfile } from "@iov/keycontrol";
const profile = new UserProfile();
const wallet1 = profile.addWallet(Ed25519HdWallet.fromMnemonic(mnemonic12));
const wallet2 = profile.addWallet(Ed25519HdWallet.fromMnemonic(mnemonic24));
Inspect the profile:
// look at profile (value reads current state)
console.log(profile.wallets.value);
// listen to the profile (stream of updates, good for reactive UI)
const sub = profile.wallets.updates.subscribe({
next: (wallets) => console.log(wallets),
});
profile.setWalletLabel(wallet1.id, "12 words");
profile.setWalletLabel(wallet2.id, "24 words");
Create identies on the two wallets:
import { ChainId } from "@iov/bcp";
import { HdPaths } from "@iov/keycontrol";
import { fromHex, toHex } from "@iov/encoding";
const chainId = "iov-exchangenet" as ChainId;
// this creates two different public key identities, generated from the
// first mnemonic using two different SLIP-0010 paths
const id1a = await profile.createIdentity(wallet1.id, chainId, HdPaths.iov(0));
const id1b = await profile.createIdentity(wallet1.id, chainId, HdPaths.iov(1));
console.log(id1a);
console.log(id1a.pubkey.algo, toHex(id1a.pubkey.data));
console.log(id1b.pubkey.algo, toHex(id1b.pubkey.data));
// this creates a different key from the second mnemonic,
// this uses the same HD path as id1a, but different seed.
const id2 = await profile.createIdentity(wallet2.id, chainId, HdPaths.iov(0));
console.log(id2.pubkey.algo, toHex(id2.pubkey.data));
// we can also add labels to the individual identies
profile.setIdentityLabel(id1a, "main account");
console.log(profile.getIdentities(wallet1.id));
Save and reload keyring:
const levelup = require("levelup");
// this is for local leveldb in node
const leveldown = require("leveldown");
// use this for indexdb storage in browser
// const browsedown = require('browsedown');
const db = levelup(leveldown("./my_secret_keys"));
// const db = levelup(browsedown('keystore'));
const passphrase = "is seven words enough for the checker?";
await profile.storeIn(db, passphrase);
// this throws an error:
// await UserProfile.loadFrom(db, "garbage");
const loaded = await UserProfile.loadFrom(db, passphrase);
// and we have the same data
console.log(loaded.wallets.value);
const ids = profile.getIdentities(loaded.wallets.value[0].id);
console.log(ids);
console.log(toHex(ids[0].pubkey.data));
console.log(toHex(id1a.pubkey.data));
Interacting with BCP Blockchain
The main use of private keypairs is not just to generate and organize them, but
to actually sign transactions (or encrypt/decrypt messages... not there yet). To
demonstrate this part, we need a working blockchain. If you are ambitious, you
can check out bcp-demo, and build the
bov
and tendermint
binaries, construct your genesis file and run the client
against your one-node "dev net"...
But, if you just want to see how the client works, let's run against IOV's testnet and use the faucet to get some tokens. As of December 17, 2019, the current testnet is located at http://rpc-private-a-x-exchangenet.iov.one:16657/.
To connect, you need to know the address of the rpc server (above). It is also
helpful to know the chainId
of the chain. You can find that quite easily by
looking at the
genesis file under
.result.genesis.chain_id
. In our case this is iov-exchangenet
.
Executing the commands
Discover the address for your identity. This is chain-dependent, so we need to
use the chain-dependent TxCodec
to generate it. In our case, bnsCodec:
import { bnsCodec } from "@iov/bns";
const addr = bnsCodec.identityToAddress(id1a);
console.log(addr);
If you are running your own "dev-net" give that address plenty of tokens in the
genesis file, by running bov init IOV $ADDR
.
Now, connect to the network:
import { createBnsConnector, MultiChainSigner } from "@iov/multichain";
const signer = new MultiChainSigner(profile);
await signer.addChain(
createBnsConnector("ws://rpc-private-a-x-exchangenet.iov.one:16657"),
);
console.log(signer.chainIds()[0]); // is this what you got yourself?
List the tokens on the network:
const connection = signer.connection(chainId);
const tokens = await connection.getAllTokens();
console.log(tokens);
Query the testnet for some existing genesis accounts:
// this is pulled from the genesis account
import { Address } from "@iov/bcp";
const alice = "tiov1c9eprq0gxdmwl9u25j568zj7ylqgc7ajyu8wxr" as Address;
const acct = await connection.getAccount({ address: alice });
console.log(acct);
If you are running the testnet faucet, just ask for some free money.
import { TokenTicker } from "@iov/bcp";
import { IovFaucet } from "@iov/faucets";
const faucet = new IovFaucet("http://faucet.x-exchangenet.iov.one:8080/");
await faucet.credit(addr, "ALT" as TokenTicker);
Then query your account:
const mine = await connection.getAccount({ address: addr });
console.log(mine); // should show non-empty array for balance
console.log(mine.balance[0]);
const addr2 = bnsCodec.identityToAddress(id2);
console.log(addr2);
let yours = await connection.getAccount({ address: addr2 });
console.log(yours); // should be undefined
Send a transaction to second id:
import { SendTransaction, TokenTicker } from "@iov/bcp";
const sendTx: SendTransaction = {
kind: "bcp/send",
chainId: chainId,
sender: addr, // this account must have money
recipient: addr2,
memo: "My first transaction",
amount: {
// 10.11 ALT (9 sig figs in tx codec)
quantity: "10110000000",
fractionalDigits: 9,
tokenTicker: "ALT" as TokenTicker,
},
};
// we must have the private key for the transaction creator (id1a)
await signer.signAndPost(id1a, sendTx);
// and we have a balance on the recipient now
yours = await connection.getAccount({ address: addr2 });
console.log(yours); // should show non-empty array for balance
console.log(yours.balance[0]);
Now, query the transaction history:
const history = await connection.searchTx({ sentFromOrTo: addr2 });
console.log(history);
const first = history[0].transaction as SendTransaction;
console.log(first.amount);
// address of recipient
console.log(toHex(first.recipient));
// public key of sender
console.log(toHex(first.signer.data));
// address of sender
const sender = bnsCodec.identityToAddress(first);
console.log(sender);
Reactive Clients
If you query, the data can get stale, and you may be tempted to start polling the blockchain. Don't! Instead we offer event streams for anyone wishing to generate a reactive application. You can simply log these values, or feed them into a reducer to capture their value.
// these are helpers for consuming streams
// lastValue will always store the last value,
// asArray will append to an array with list of all tx that were streamed
import { asArray, lastValue } from "@iov/stream";
const liveHeight = lastValue(
client.watchBlockHeaders().map((header) => header.height),
);
// if you wait a few seconds, you should see the block-height increase
console.log(liveHeight.value());
console.log(liveHeight.value());
console.log(liveHeight.value());
// you can also watch an account balance
const liveBalance = lastValue(client.watchAccount({ address: addr }));
console.log(liveBalance.value());
// or a list of all transactions that touch this account
const liveTx = asArray(client.liveTx({ address: addr }));
console.log(liveTx.value());
Now, go ahead, send some tokens to this account in another window, and read the value of the account and tx again, watch them grow.
License
This package is part of the IOV-Core repository, licensed under the Apache License 2.0 (see NOTICE and LICENSE).