Help & Support

Ethers SDK

Interact with the Lens Network using ethers.js. A template web app can be found here.

The Lens Network SDK provides support for ethers.js, a widely-used JavaScript library for interacting with EVM blockchains. This is achieved by building upon ZKsync-ethers v6, the official ZKsync SDK for ethers.js.

This section assumes you are familiar with the ethers concepts of Provider and Signer:

  • A Provider is a read-only connection to the blockchain, which allows querying the blockchain state, such as account, block or transaction details, querying event logs or evaluating read-only code using call.

  • A Signer wraps all operations that interact with an account. An account generally has a private key located somewhere, which can be used to sign a variety of types of payloads.

Getting Started

1

Install ZKsync-ethers

First, install ZKsync-ethers package:

npm install zksync-ethers ethers@6

ethers.js version 6 is a peer dependency of zksync-ethers package. Make sure to install it alongside ZKsync-ethers.

2

Install SDK

Then, install the @lens-network/sdk package:

npm install @lens-network/sdk@canary

3

Create Your Providers

Next, create the providers to interact with the Lens Network and the corresponding L1 (an Ethereum chain):

To interact with the Lens Network, use the SDK's getDefaultProvider function. This creates a specialized JsonRpcProvider instance that supports the lens and zks RPC method namespaces.

For interactions with Ethereum L1, create an ethers Provider connected to the corresponding network (Sepolia for testnets). The example below uses the getDefaultProvider function from the ethers package.

providers.ts
import { getDefaultProvider, Network } from "@lens-network/sdk/ethers";import { ethers } from "ethers";
// Lens Network (L2)export const lensProvider = getDefaultProvider(Network.Testnet);
// Ethereum L1export const ethProvider = ethers.getDefaultProvider("sepolia");

Create a Signer

Lens Network leverages ZKsync-ethers signers to facilitate interaction with an account. Below is a list of available signers:

Additionally, ZKsync-ether includes VoidSigner and L1VoidSigner classes. These are ZKsync's implementations of ethers' VoidSigner. They allow an address to be used in any API that accepts a Signer, even when no credentials are available for actual signing.

ZKsync Signers
import {  Wallet,  EIP712Signer,  Signer,  SmartAccount,  ECDSASmartAccount,  MultisigECDSASmartAccount,} from "zksync-ethers";

For a subset of these, the SDK provides extended versions that include Lens Network-specific features.

Lens Network Signers
import { Wallet, Signer } from "@lens-network/sdk/ethers";

Wallet

Create a Wallet instance using the L1 and/or L2 providers depending on which network you want to interact with.

import { Wallet } from "@lens-network/sdk/ethers";
import { lensProvider, ethProvider } from "./providers";
export const wallet = new Wallet(  process.env.PRIVATE_KEY as String,  lensProvider,  ethProvider);

You can also create an unconnected Wallet instance and connect it later:

import { Wallet } from "@lens-network/sdk/ethers";
import { lensProvider, ethProvider } from "./providers";
const unconnectedWallet = new Wallet(process.env.PRIVATE_KEY as String);
const lensWallet = unconnectedWallet.connect(lensProvider);
const ethWallet = unconnectedWallet.connectToL1(ethProvider);

Signer

Create a Signer instance that is connected to the BrowserProvider instance.

This class is to be used in a browser environment.

import { Signer } from "@lens-network/sdk/ethers";
import { browserProvider, lensProvider } from "./providers";
const network = await browserProvider.getNetwork();
const signer = Signer.from(  await browserProvider.getSigner(),  Number(network.chainId),  lensProvider);

L1Signer

Create an L1Signer instance to do ZKsync-related operations on L1. To do so you need to connect it to both the BrowserProvider and the LensProvider.

This class is to be used in a browser environment.

import { L1Signer } from "zksync-ethers";
import { browserProvider, lensProvider } from "./providers";
export const l1Signer = L1Signer.from(  await browserProvider.getSigner(),  lensProvider);

Actions

Ethers.js v6 enables interaction with the Lens Network through various actions, including sending transactions, signing messages, and managing wallet interactions.

Transactions

To send transactions, you can use the sendTransaction function for gas token transfers ($GRASS) or a Contract class instance to call contract functions.

Sending a Transaction:

import { signer } from "./signer";
const tx = await signer.sendTransaction({  to: "0xRecipientAddress",  value: parseEther("1.0"),});
await tx.wait();console.log(`Transaction Hash: ${tx.hash}`);

Executing Contract Function:

import { Contract } from "ethers";import { signer } from "./signer";import { contractAbi } from "./abi";
const contract = new Contract("0xContractAddress", contractAbi, signer);
const tx = await contract.functionName(arg1, arg2);await tx.wait();console.log(`Transaction Hash: ${tx.hash}`);

Signatures

To sign messages for cryptographic authentication, the signMessage or signTypedData functions can be used.

Sign Message

import { signer } from "./signer";
const message = "Hello, Lens Network!";const signature = await signer.signMessage(message);
console.log(`Signature: ${signature}`);

Sign Typed Data:

import { signer } from "./signer";
const domain = {  name: "Lens Testnet",  version: "1",  chainId: 37111,  verifyingContract: "0xYourContractAddress",};
const types = {  Permit: [    { name: "owner", type: "address" },    { name: "spender", type: "address" },    { name: "value", type: "uint256" },    { name: "nonce", type: "uint256" },    { name: "deadline", type: "uint256" },  ],};
const message = {  owner: "0xYourWalletAddress",  spender: "0xSpenderAddress",  value: BigInt(1000000000000000000),  nonce: BigInt(0),  deadline: BigInt(Math.floor(Date.now() / 1000) + 3600),};
const signature = await signer.signTypedData(domain, types, message);console.log(`Signature: ${signature}`);

Additional Options

Custom RPC Node

If you want to use a Lens Network RPC node other than the default one, you can create a custom provider like this:

providers.ts
import { Provider } from "@lens-network/sdk/ethers";
// Lens Network (L2)export const lensProvider = new Provider("https://custom-rpc-node.com");