Sign-in with Ronin
Overview
This guide shows you how to integrate "Sign-In with Ronin" into your dApp. "Sign-In with Ronin" is a standard extended from Sign-In with Ethereum (SIWE) to help users sign in to your dApp easily, providing a better user experience and improved security.
From the end user's perspective, the sign-in process is as follows:
- Connect Ronin Wallet: The user connects their Ronin Wallet to your dApp.
- Accept terms of service: The user accepts your terms of service by signing a message with their Ronin Wallet.
- Solve the challenge: The user solves a security challenge to prove they are human.
1. Connect Ronin Wallet | 2. Accept terms of service |
---|---|
Prerequisites
npm or yarn for package installation.
Steps
Step 1. Install dependencies
Firstly, you will need to get the @sky-mavis/tanto-connect
package to connect to Ronin Wallet, you can follow the steps from Install Tanto Connect
The dependency needed for signing in is the siwe
package. It helps you generate a formatted sign-in message more easily.
- Install the package using yarn or npm:
yarn add siwe@1.1.6 ethers@5.6.9
npm i siwe@1.1.6 ethers@5.6.9
- Import the package in your component:
import { SiweMessage } from "siwe";
Step 2. Generate a SIWE message
Using the SiweMessage
class, you can generate a formatted sign-in message. The message includes the following parameters:
domain
: The domain of your dApp.address
: The user's Ronin address.uri
: The URI of your dApp.version
: The version of the SIWE message.chainId
: The chain ID of the network. For the Ronin mainnet, use2020
. For the Saigon testnet, use2021
.nonce
: A unique nonce for the message.statement
: The terms of service that the user agrees to.expirationTime
: The expiration time of the message.
The following code snippet shows how to generate a SIWE message:
function SIWERequest() { const [connector, setConnector] = useState(null); const [signature, setSignature] = useState(); useEffect(() => { requestRoninWalletConnector().then((connector) => { setConnector(connector); }); }, []); async function onClickSignIn() { if (!connector) { return; } const accounts = await connector.requestAccounts(); if (!accounts) { return; } const currentAccount = accounts[0]; const currentDate = new Date(); currentDate.setDate(currentDate.getDate() + 1); const siweMessage = new SiweMessage({ domain: window.location.hostname, address: currentAccount, uri: window.location.origin, version: "1", chainId: 2020, nonce: "12345678", statement: "I accept the dApp's Terms of Service: https://example-dapp.com/terms-of-use", expirationTime: currentDate.toISOString(), }); const provider = await connector.getProvider(); const sig = await provider.request({ method: "personal_sign", params: [siweMessage.toMessage(), currentAccount], }); setSignature(sig); } if (signature) { return ( <div> <div>🎉 Congratulations! You are signed in.</div> <button onClick={() => setSignature(undefined)}>Reload</button> </div> ); } return <button onClick={onClickSignIn}>Sign in with Ronin</button>; }