2020-06-12 03:38:55 -04:00
|
|
|
_section: Getting Started @<getting-started>
|
2019-08-21 01:53:47 -04:00
|
|
|
|
|
|
|
|
2020-06-12 03:38:55 -04:00
|
|
|
_subsection: Installing @<installing>
|
2019-08-21 01:53:47 -04:00
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
Ethers' various Classes and Functions are available to import
|
2020-02-02 07:58:29 -05:00
|
|
|
manually from sub-packages under the [@ethersproject](link-ethers-npm)
|
2020-06-12 03:38:55 -04:00
|
|
|
organization but for most projects, the umbrella package is the
|
|
|
|
easiest way to get started.
|
2019-08-21 01:53:47 -04:00
|
|
|
|
2020-04-19 02:18:20 -04:00
|
|
|
_code: @lang<shell>
|
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
/home/ricmoo> npm install --save ethers
|
2019-08-21 01:53:47 -04:00
|
|
|
|
|
|
|
|
2020-06-12 03:38:55 -04:00
|
|
|
_subsection: Importing @<importing>
|
2019-08-21 01:53:47 -04:00
|
|
|
|
|
|
|
_heading: Node.js
|
|
|
|
|
2020-05-08 03:24:40 -04:00
|
|
|
_code: node.js require @lang<script>
|
2020-04-19 02:18:20 -04:00
|
|
|
|
|
|
|
const { ethers } = require("ethers");
|
|
|
|
|
2020-05-08 03:24:40 -04:00
|
|
|
_code: ES6 or TypeScript @lang<script>
|
|
|
|
|
2020-04-19 02:18:20 -04:00
|
|
|
import { ethers } from "ethers";
|
|
|
|
|
2019-08-21 01:53:47 -04:00
|
|
|
|
|
|
|
_heading: Web Browser
|
|
|
|
|
|
|
|
It is generally better practice (for security reasons) to copy the
|
2020-02-02 07:58:29 -05:00
|
|
|
[ethers library](link-ethers-js) to your own webserver and serve it
|
|
|
|
yourself.
|
2019-08-21 01:53:47 -04:00
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
For quick demos or prototyping though, you can load it in your
|
2019-08-21 01:53:47 -04:00
|
|
|
Web Applications from our CDN.
|
|
|
|
|
2020-05-08 03:24:40 -04:00
|
|
|
|
|
|
|
_code: ES6 in the Browser @lang<html>
|
2020-04-19 02:18:20 -04:00
|
|
|
|
2020-07-04 22:46:47 -04:00
|
|
|
<script type="module">
|
2021-06-04 01:17:56 -04:00
|
|
|
import { ethers } from "https://cdn.ethers.io/lib/ethers-5.2.esm.min.js";
|
2020-07-04 22:46:47 -04:00
|
|
|
// Your code here...
|
|
|
|
</script>
|
2020-05-08 03:24:40 -04:00
|
|
|
|
|
|
|
|
|
|
|
_code: ES3 (UMD) in the Browser @lang<html>
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
<script src="https://cdn.ethers.io/lib/ethers-5.2.umd.min.js"
|
2020-10-06 17:18:09 -04:00
|
|
|
type="application/javascript"></script>
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
_subsection: Common Terminology @<getting-started--glossary>
|
|
|
|
|
|
|
|
This section needs work...
|
|
|
|
|
|
|
|
_table: Common Terms
|
|
|
|
|
|
|
|
$Provider: A Provider (in ethers) is a class which provides an abstraction
|
|
|
|
for a connection to the Ethereum Network. It provides read-only
|
|
|
|
access to the Blockchain and its status.
|
|
|
|
$Signer: A Signer is a class which (usually) in some way directly or
|
|
|
|
indirectly has access to a private key, which can sign messages
|
|
|
|
and transactions to authorize the network to charge your account
|
|
|
|
ether to perform operations.
|
|
|
|
$Contract: A Contract is an abstraction which represents a connection to a
|
2020-07-07 14:41:23 +02:00
|
|
|
specific contract on the Ethereum Network, so that applications
|
2020-10-03 13:30:15 -03:00
|
|
|
can use it like a normal JavaScript object.
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
| **Provider** | $Provider |
|
|
|
|
| **Signer** | $Signer |
|
|
|
|
| **Contract** | $Contract |
|
|
|
|
|
|
|
|
|
2021-10-04 10:52:34 -04:00
|
|
|
_subsection: Connecting to Ethereum: MetaMask @<getting-started--connecting>
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
The quickest and easiest way to experiment and begin developing on
|
|
|
|
Ethereum is to use [[link-metamask]], which is a browser extension
|
|
|
|
that provides:
|
|
|
|
|
|
|
|
- A connection to the Ethereum network (a [[Provider]])
|
2020-07-03 01:44:17 -04:00
|
|
|
- Holds your private key and can sign things (a [[Signer]])
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2021-10-04 10:52:34 -04:00
|
|
|
_code: Connecting to MetaMask @lang<script>
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// A Web3Provider wraps a standard Web3 provider, which is
|
2021-10-04 10:52:34 -04:00
|
|
|
// what MetaMask injects as window.ethereum into each page
|
2020-06-12 03:38:55 -04:00
|
|
|
const provider = new ethers.providers.Web3Provider(window.ethereum)
|
|
|
|
|
2022-02-03 15:28:04 -05:00
|
|
|
// MetaMask requires requesting permission to connect users accounts
|
|
|
|
await provider.send("eth_requestAccounts", []);
|
|
|
|
|
2021-10-04 10:52:34 -04:00
|
|
|
// The MetaMask plugin also allows signing transactions to
|
2020-06-12 03:38:55 -04:00
|
|
|
// send ether and pay to change state within the blockchain.
|
2020-07-07 14:41:23 +02:00
|
|
|
// For this, you need the account signer...
|
2020-06-12 03:38:55 -04:00
|
|
|
const signer = provider.getSigner()
|
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
_subsection: Connecting to Ethereum: RPC @<getting-started--connecting-rpc>
|
|
|
|
|
|
|
|
The [JSON-RPC API](link-jsonrpc) is another popular method for interacting
|
|
|
|
with Ethereum and is available in all major Ethereum node implementations
|
|
|
|
(e.g. [[link-geth]] and [[link-parity]]) as well as many
|
|
|
|
third-party web services (e.g. [[link-infura]]). It typically provides:
|
|
|
|
|
|
|
|
- A connection to the Ethereum network (a [[Provider]])
|
2021-04-02 03:55:06 -04:00
|
|
|
- Holds your private key and can sign things (a [[Signer]])
|
2020-07-07 14:41:23 +02:00
|
|
|
|
|
|
|
_code: Connecting to an RPC client @lang<script>
|
|
|
|
|
|
|
|
// If you don't specify a //url//, Ethers connects to the default
|
|
|
|
// (i.e. ``http:/\/localhost:8545``)
|
|
|
|
const provider = new ethers.providers.JsonRpcProvider();
|
|
|
|
|
|
|
|
// The provider also allows signing transactions to
|
|
|
|
// send ether and pay to change state within the blockchain.
|
|
|
|
// For this, we need the account signer...
|
|
|
|
const signer = provider.getSigner()
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
_heading: Querying the Blockchain @<getting-started--querying>
|
|
|
|
|
|
|
|
Once you have a [[Provider]], you have a read-only connection to the
|
2020-07-07 14:41:23 +02:00
|
|
|
blockchain, which you can use to query the current state, fetch historic
|
2020-06-12 03:38:55 -04:00
|
|
|
logs, look up deployed code and so on.
|
|
|
|
|
|
|
|
_code: Basic Queries @lang<javascript>
|
|
|
|
|
|
|
|
// Look up the current block number
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
2021-06-10 17:38:38 -04:00
|
|
|
await provider.getBlockNumber()
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
// Get the balance of an account (by address or ENS name, if supported by network)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
2020-06-12 03:38:55 -04:00
|
|
|
balance = await provider.getBalance("ethers.eth")
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
// Often you need to format the output to something more user-friendly,
|
|
|
|
// such as in ether (instead of wei)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
2020-06-12 03:38:55 -04:00
|
|
|
ethers.utils.formatEther(balance)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
// If a user enters a string in an input field, you may need
|
2020-06-12 03:38:55 -04:00
|
|
|
// to convert it from ether (as a string) to wei (as a BigNumber)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
2020-06-12 03:38:55 -04:00
|
|
|
ethers.utils.parseEther("1.0")
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
_heading: Writing to the Blockchain @<getting-started--sending>
|
|
|
|
|
|
|
|
_code: Sending Ether @lang<script>
|
|
|
|
|
|
|
|
// Send 1 ether to an ens name.
|
|
|
|
const tx = signer.sendTransaction({
|
|
|
|
to: "ricmoo.firefly.eth",
|
|
|
|
value: ethers.utils.parseEther("1.0")
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
_subsection: Contracts @<getting-started--contracts>
|
|
|
|
|
|
|
|
A Contract is an abstraction of program code which lives on the
|
|
|
|
Ethereum blockchain.
|
|
|
|
|
|
|
|
The [[Contract]] object makes it easier to use an on-chain
|
2020-07-07 14:41:23 +02:00
|
|
|
Contract as a normal JavaScript object, with the methods
|
2020-06-12 03:38:55 -04:00
|
|
|
mapped to encoding and decoding data for you.
|
|
|
|
|
2020-09-08 01:11:29 -04:00
|
|
|
If you are familiar with Databases, this is similar to an //Object Relational Mapper// (ORM).
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
In order to communicate with the Contract on-chain, this class
|
|
|
|
needs to know what methods are available and how to encode and
|
2021-02-04 17:21:11 -05:00
|
|
|
decode the data, which is what the //Application Binary Interface// (ABI)
|
2020-06-12 03:38:55 -04:00
|
|
|
provides.
|
|
|
|
|
2020-07-14 14:05:55 +02:00
|
|
|
This class is a //meta-class//, which means its methods are constructed
|
|
|
|
at runtime, and when you pass in the ABI to the constructor it uses it
|
2020-06-12 03:38:55 -04:00
|
|
|
to determine which methods to add.
|
|
|
|
|
2020-07-14 14:05:55 +02:00
|
|
|
While an on-chain Contract may have many methods available, you can safely ignore
|
2020-06-12 03:38:55 -04:00
|
|
|
any methods you don't need or use, providing a smaller subset of the ABI to
|
|
|
|
the contract.
|
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
An ABI often comes from the Solidity or Vyper compiler, but you can use the
|
|
|
|
Human-Readable ABI in code, which the following examples use.
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
_code: Connecting to the DAI Contract @lang<javascript>
|
|
|
|
|
2020-07-07 14:41:23 +02:00
|
|
|
// You can also use an ENS name for the contract address
|
2020-06-12 03:38:55 -04:00
|
|
|
const daiAddress = "dai.tokens.ethers.eth";
|
|
|
|
|
|
|
|
// The ERC-20 Contract ABI, which is a common contract interface
|
|
|
|
// for tokens (this is the Human-Readable ABI format)
|
|
|
|
const daiAbi = [
|
2020-07-07 14:41:23 +02:00
|
|
|
// Some details about the token
|
2020-06-12 03:38:55 -04:00
|
|
|
"function name() view returns (string)",
|
|
|
|
"function symbol() view returns (string)",
|
|
|
|
|
|
|
|
// Get the account balance
|
|
|
|
"function balanceOf(address) view returns (uint)",
|
|
|
|
|
|
|
|
// Send some of your tokens to someone else
|
|
|
|
"function transfer(address to, uint amount)",
|
|
|
|
|
|
|
|
// An event triggered whenever anyone transfers to someone else
|
|
|
|
"event Transfer(address indexed from, address indexed to, uint amount)"
|
|
|
|
];
|
|
|
|
|
|
|
|
// The Contract object
|
|
|
|
const daiContract = new ethers.Contract(daiAddress, daiAbi, provider);
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: _page.daiAbi = daiAbi;
|
|
|
|
//_hide: _page.daiContract = daiContract;
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
_heading: Read-Only Methods @<getting-started--reading>
|
|
|
|
|
|
|
|
_code: Querying the DAI Contract @lang<javascript>
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: const daiContract = _page.daiContract;
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Get the ERC-20 token name
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
|
|
|
await daiContract.name()
|
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2020-10-03 13:30:15 -03:00
|
|
|
// Get the ERC-20 token symbol (for tickers and UIs)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
|
|
|
await daiContract.symbol()
|
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Get the balance of an address
|
|
|
|
balance = await daiContract.balanceOf("ricmoo.firefly.eth")
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: balance
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Format the DAI for displaying to the user
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
2020-06-12 03:38:55 -04:00
|
|
|
ethers.utils.formatUnits(balance, 18)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
_heading: State Changing Methods @<getting-started--writing>
|
|
|
|
|
|
|
|
_code: Sending DAI @lang<script>
|
|
|
|
|
|
|
|
// The DAI Contract is currently connected to the Provider,
|
2020-07-14 14:05:55 +02:00
|
|
|
// which is read-only. You need to connect to a Signer, so
|
|
|
|
// that you can pay to send state-changing transactions.
|
2020-06-12 03:38:55 -04:00
|
|
|
const daiWithSigner = contract.connect(signer);
|
|
|
|
|
|
|
|
// Each DAI has 18 decimal places
|
|
|
|
const dai = ethers.utils.parseUnits("1.0", 18);
|
|
|
|
|
|
|
|
// Send 1 DAI to "ricmoo.firefly.eth"
|
|
|
|
tx = daiWithSigner.transfer("ricmoo.firefly.eth", dai);
|
|
|
|
|
|
|
|
|
|
|
|
_heading: Listening to Events @<getting-started--events>
|
|
|
|
|
|
|
|
_code: Listening to Events @lang<javascript>
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: const daiContract = _page.daiContract;
|
|
|
|
//_hide: const formatEther = ethers.utils.formatEther;
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Receive an event when ANY transfer occurs
|
|
|
|
daiContract.on("Transfer", (from, to, amount, event) => {
|
|
|
|
console.log(`${ from } sent ${ formatEther(amount) } to ${ to}`);
|
|
|
|
// The event object contains the verbatim log data, the
|
|
|
|
// EventFragment and functions to fetch the block,
|
|
|
|
// transaction and receipt and event functions
|
|
|
|
});
|
|
|
|
|
|
|
|
// A filter for when a specific address receives tokens
|
|
|
|
myAddress = "0x8ba1f109551bD432803012645Ac136ddd64DBA72";
|
|
|
|
filter = daiContract.filters.Transfer(null, myAddress)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: filter
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Receive an event when that filter occurs
|
|
|
|
daiContract.on(filter, (from, to, amount, event) => {
|
|
|
|
// The to will always be "address"
|
|
|
|
console.log(`I got ${ formatEther(amount) } from ${ from }.`);
|
|
|
|
});
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: daiContract.removeAllListeners(); /* Don't want to block the docs from compiling... */
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
_heading: Query Historic Events @<getting-started--history>
|
|
|
|
|
|
|
|
_code: Filtering Historic Events @lang<javascript>
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: const signer = new ethers.VoidSigner("0x8ba1f109551bD432803012645Ac136ddd64DBA72");
|
|
|
|
//_hide: const daiContract = _page.daiContract;
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// Get the address of the Signer
|
|
|
|
myAddress = await signer.getAddress()
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: myAddress
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2021-02-04 17:21:11 -05:00
|
|
|
// Filter for all token transfers from me
|
2020-06-12 03:38:55 -04:00
|
|
|
filterFrom = daiContract.filters.Transfer(myAddress, null);
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: filterFrom
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2021-02-04 17:21:11 -05:00
|
|
|
// Filter for all token transfers to me
|
2020-06-12 03:38:55 -04:00
|
|
|
filterTo = daiContract.filters.Transfer(null, myAddress);
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: filterTo
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2022-03-04 16:17:26 -05:00
|
|
|
// List all transfers sent from me in a specific block range
|
2021-06-04 01:17:56 -04:00
|
|
|
//_result:
|
|
|
|
await daiContract.queryFilter(filterFrom, 9843470, 9843480)
|
|
|
|
//_log:
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// The following have had the results omitted due to the
|
|
|
|
// number of entries; but they provide some useful examples
|
|
|
|
//
|
|
|
|
|
2020-07-14 14:05:55 +02:00
|
|
|
// List all transfers sent in the last 10,000 blocks
|
2021-06-04 01:17:56 -04:00
|
|
|
await daiContract.queryFilter(filterFrom, -10000)
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// List all transfers ever sent to me
|
2021-06-04 01:17:56 -04:00
|
|
|
await daiContract.queryFilter(filterTo)
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
_subsection: Signing Messages @<getting-started--signing>
|
|
|
|
|
|
|
|
_code: Signing Messages @lang<javascript>
|
|
|
|
|
2021-06-04 01:17:56 -04:00
|
|
|
//_hide: const signer = ethers.Wallet.createRandom();
|
2020-06-12 03:38:55 -04:00
|
|
|
|
2020-07-14 14:05:55 +02:00
|
|
|
// To sign a simple string, which are used for
|
|
|
|
// logging into a service, such as CryptoKitties,
|
2020-06-12 03:38:55 -04:00
|
|
|
// pass the string in.
|
|
|
|
signature = await signer.signMessage("Hello World");
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: signature
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// A common case is also signing a hash, which is 32
|
|
|
|
// bytes. It is important to note, that to sign binary
|
|
|
|
// data it MUST be an Array (or TypedArray)
|
|
|
|
//
|
|
|
|
|
2020-07-14 14:05:55 +02:00
|
|
|
// This string is 66 characters long
|
2020-06-12 03:38:55 -04:00
|
|
|
message = "0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"
|
|
|
|
|
|
|
|
// This array representation is 32 bytes long
|
|
|
|
messageBytes = ethers.utils.arrayify(message);
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: messageBytes
|
2020-06-12 03:38:55 -04:00
|
|
|
|
|
|
|
// To sign a hash, you most often want to sign the bytes
|
|
|
|
signature = await signer.signMessage(messageBytes)
|
2021-06-04 01:17:56 -04:00
|
|
|
//_log: signature
|