Get started with XMTP
The XMTP message API revolves around a network client that allows retrieving and sending messages to other network participants.
A client must be connected to a wallet on startup. If this is the very first time the client is created, the client will generate a key bundle that is used to encrypt and authenticate messages.
The key bundle persists encrypted in the network using a wallet signature. The public side of the key bundle is also regularly advertised on the network to allow parties to establish shared encryption keys.
Overview
This is an overview of the core concepts and lines of code needed to use XMTP successfully.
- JavaScript
- React
- Kotlin
- Swift
- Dart
- React Native
import { Client } from "@xmtp/xmtp-js";
import { Wallet } from "ethers";
// You'll want to replace this with a wallet from your application
const signer = Wallet.createRandom();
// Create the client with your wallet. This will connect to the XMTP development network by default
const xmtp = await Client.create(signer, { env: "dev" });
// Start a conversation with XMTP
const conversation = await xmtp.conversations.newConversation(
"0x3F11b27F323b62B159D2642964fa27C46C841897",
);
// Load all messages in the conversation
const messages = await conversation.messages();
// Send a message
await conversation.send("gm");
// Listen for new messages in the conversation
for await (const message of await conversation.streamMessages()) {
console.log(`[${message.senderAddress}]: ${message.content}`);
}
import {
Client,
useStreamMessages,
useClient,
useMessages,
useConversations,
useCanMessage,
useStartConversation,
} from "@xmtp/react-sdk";
const { client, initialize } = useClient();
const { conversations } = useConversations();
const { startConversation } = useStartConversation();
const { canMessage } = useCanMessage();
//Initialize
{
!isConnected && <button onClick={initXmtp}>Connect to XMTP</button>;
}
const initXmtp = async () => {
const options = {
persistConversations: false,
env: "dev",
};
await initialize({ keys, options, signer });
};
// Start a conversation with XMTP
const add = "0x3F11b27F323b62B159D2642964fa27C46C841897";
if (await canMessage(add)) {
const conversation = await startConversation(add, "hi");
}
//Stream messages
const [history, setHistory] = useState(null);
const { messages } = useMessages(conversation);
// Stream messages
const onMessage = useCallback((message) => {
setHistory((prevMessages) => {
const msgsnew = [...prevMessages, message];
return msgsnew;
});
}, []);
useStreamMessages(conversation, { onMessage });
// You'll want to replace this with a wallet from your application.
val account = PrivateKeyBuilder()
// Create the client with your wallet. This will connect to the XMTP `dev` network by default.
// The account is anything that conforms to the `XMTP.SigningKey` protocol.
val client = Client().create(account = account)
// Start a conversation with XMTP
val conversation =
client.conversations.newConversation("0x3F11b27F323b62B159D2642964fa27C46C841897")
// Load all messages in the conversation
val messages = conversation.messages()
// Send a message
conversation.send(text = "gm")
// Listen for new messages in the conversation
conversation.streamMessages().collect { print("${message.senderAddress}: ${message.body}") }
import XMTP
// You'll want to replace this with a wallet from your application.
let account = try PrivateKey.generate()
// Create the client with your wallet. This will connect to the XMTP `dev` network by default.
// The account is anything that conforms to the `XMTP.SigningKey` protocol.
let client = try await Client.create(account: account)
// Start a conversation with XMTP
let conversation = try await client.conversations.newConversation(
with: "0x3F11b27F323b62B159D2642964fa27C46C841897")
// Load all messages in the conversation
let messages = try await conversation.messages()
// Send a message
try await conversation.send(content: "gm")
// Listen for new messages in the conversation
for try await message in conversation.streamMessages() {
print("\(message.senderAddress): \(message.body)")
}
import 'package:xmtp/xmtp.dart' as xmtp;
import 'package:web3dart/credentials.dart';
import 'dart:math';
var signer = EthPrivateKey.createRandom(Random.secure());
var api = xmtp.Api.create();
var client = await xmtp.Client.createFromWallet(api, signer);
//List conversations
var conversations = await client.listConversations();
for (var convo in conversations) {
debugPrint('Saying GM to ${convo.peer}');
await client.sendMessage(convo, 'gm');
}
//Listen for new conversations
var listening = client.streamConversations().listen((convo) {
debugPrint('Got a new conversation with ${convo.peer}');
});
// When you want to stop listening:
await listening.cancel();
//Send messages
var convo = await client.newConversation("0x...");
await client.sendMessage(convo, 'gm');
//Listen for new messages
var listening = client.streamMessages(convo).listen((message) {
debugPrint('${message.sender}> ${message.content}');
});
// When you want to stop listening:
await listening.cancel();
import { Client, Conversation, DecodedMessage } from '@xmtp/react-native-sdk';
const [addressText, setAddressText] = useState("No Connected Address");
const [connected, setConnected] = useState(false);
const [client, setClient] = useState<Client | undefined>(undefined);
const [conversations, setConversations] = useState<Conversation[] | undefined>(undefined);
const [messages, setMessages] = useState<DecodedMessage[] | undefined>(undefined);
const [conversation, setConversation] = useState<Conversation | undefined>(undefined);
function connectRandomWallet() {
return async () => {
// NOTE: react-native-sdk testing
const client = await Client.createRandom('dev')
setClient(client);
const rnSDKAddress = await client.address;
// const address: string = await callIntoWebview("connectRandomWallet");
setAddressText('react-native-sdk npm address: ' + rnSDKAddress);
setConnected(true);
getConversations();
};
}
function getConversations() {
return async () => {
const conversations = await client?.conversations.list();
setConversations(conversations);
};
}
function getMessages(conversation: Conversation) {
return async () => {
const messages = await conversation?.messages();
setConversation(conversation);
setMessages(messages);
};
}
function sendMessage(message: string) {
return async () => {
await conversation?.send(message);
getMessages(conversation!!);
};
}
Install
To start with XMTP, install the XMTP client SDK:
- JavaScript
- React
- Kotlin
- Swift
- Dart
- React Native
npm install @xmtp/xmtp-js
npm i @xmtp/react-sdk
You can find the latest package version on Maven Central.
implementation 'org.xmtp:android:X.X.X'
Use Xcode to add to the project (File > Add Packages…) or add this to your Package.swift file
:
.package(url: "https://github.com/xmtp/xmtp-ios", branch: "main")
flutter pub add xmtp
npm i @xmtp/react-native-sdk
Need to send a test message?
Message this XMTP message bot to get an immediate automated reply:
gm.xmtp.eth
(0x937C0d4a6294cdfa575de17382c7076b579DC176
)
Troubleshooting
- If you run into issues with Buffer and polyfills, see these solutions