XMTP JS client library
Disclaimer: Pre-Stable Alpha
The XMTP protocol is in the early stages of development. This pre-stable alpha library is being provided for evaluation, feedback, and community contribution. It has not undergone a formal security audit and is not intended for production applications. Significant breaking revisions should be expected for all pre-stable alpha software.
Installation
This library is not yet public on npm. It can be installed from this repo to your project directory using npm install xmtp/xmtp-js
, or from npm using a private access token:
export NPM_TOKEN=$YOUR_NPM_TOKEN
echo "//registry.npmjs.org/:_authToken=${NPM_TOKEN}" > .npmrc
npm install @xmtp/xmtp-js
Additional configuration is required in React environments due to the removal of polyfills from Webpack 5.
Create React App
Use react-scripts
prior to version 5.0.0
. For example:
npx create-react-app --scripts-version 4.0.2
Or downgrade after creating your app.
Next.js
In next.config.js
:
webpack: (config, { isServer }) => {
if (!isServer) {
config.resolve.fallback.fs = false
}
return config
}
Usage
The 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, or optionally in local storage. The public side of the key bundle is also regularly advertised on the network to allow parties to establish shared encryption keys. All this happens transparently, without requiring any additional code.
import { Client } from '@xmtp/xmtp-js'
import { Wallet } from 'ethers'
const wallet = Wallet.createRandom()
const xmtp = await Client.create(wallet)
const conversation = await xmtp.conversations.newConversation(
'0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045'
)
const messages = await conversation.messages()
await conversation.send('gm')
for await (const message of conversation.streamMessages()) {
console.log(`[${message.senderAddress}]: ${message.text}`)
}
Creating a Client
A Client is created with Client.create(wallet: ethers.Signer): Promise<Client>
that requires passing in a connected Wallet. The Client will request a wallet signature in 2 cases:
- To sign the newly generated key bundle. This happens only the very first time when key bundle is not found in storage.
- To sign a random salt used to encrypt the key bundle in storage. This happens every time the Client is started (including the very first time).
The Client will connect to the XMTP playnet by default. ClientOptions can be used to override this and other parameters of the network connection.
import { Client } from '@xmtp/xmtp-js'
const xmtp = await Client.create(wallet)
Configuring the Client
The client's network connection and key storage method can be configured with these optional parameters of Client.create
:
Parameter | Default | Description |
---|
env | 'testnet' | Connect to the specified network environment (currently only 'testnet' ). |
waitForPeersTimeoutMs | 10000 | Wait this long for an initial peer connection. |
keyStoreType | networkTopicStoreV1 | Persist the wallet's key bundle to the network, or optionally to localStorage . |
codecs | [TextCodec] | Add codecs to support additional content types. |
maxContentSize | 100M | Maximum message content size in bytes. |
Conversations
Most of the time, when interacting with the network, you'll want to do it through conversations
. Conversations are between two wallets.
import { Client } from '@xmtp/xmtp-js'
const xmtp = await Client.create(wallet)
const conversations = xmtp.conversations
List existing conversations
You can get a list of all conversations that have had 1 or more messages exchanged in the last 30 days.
const allConversations = await xmtp.conversations.list()
for (const conversation of allConversations) {
console.log(`Saying GM to ${conversation.peerAddress}`)
await conversation.send('gm')
}
Listen for new conversations
You can also listen for new conversations being started in real-time. This will allow applications to display incoming messages from new contacts.
Warning: this stream will continue infinitely. To end the stream you can either break from the loop, or call await stream.return()
const stream = xmtp.conversations.stream()
for await (const conversation of stream) {
console.log(`New conversation started with ${conversation.peerAddress}`)
await conversation.send('Hi there!')
break
}
Start a new conversation
You can create a new conversation with any Ethereum address on the XMTP network.
const newConversation = await xmtp.conversations.newConversation(
'0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045'
)
Sending messages
To be able to send a message, the recipient must have already started their Client at least once and consequently advertised their key bundle on the network. Messages are addressed using wallet addresses. The message payload can be a plain string, but other types of content can be supported through the use of SendOptions (see Different types of content for more details)
const conversation = await xmtp.conversations.newConversation(
'0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045'
)
await conversation.send('Hello world')
List messages in a conversation
You can receive the complete message history in a conversation by calling conversation.messages()
for (const conversation of await xmtp.conversations.list()) {
const opts = {
startTime: new Date(new Date().setDate(new Date().getDate() - 1)),
endTime: new Date(),
}
const messagesInConversation = await conversation.messages(opts)
}
Pre-Stable Alpha Limitation: After 7 days, messages are deleted from the network and cannot be retrieved. We also forcibly wipe the playnet on Mondays.
Listen for new messages in a conversation
You can listen for any new messages (incoming or outgoing) in a conversation by calling conversation.streamMessages()
.
A successfully received message (that makes it through the decoding and decryption without throwing) can be trusted to be authentic, i.e. that it was sent by the owner of the message.senderAddress
wallet and that it wasn't modified in transit. The message.sent
timestamp can be trusted to have been set by the sender.
The Stream returned by the stream
methods is an asynchronous iterator and as such usable by a for-await-of loop. Note however that it is by its nature infinite, so any looping construct used with it will not terminate, unless the termination is explicitly initiated (by breaking the loop or by an external call to Stream.return()
)
const conversation = await xmtp.conversations.newConversation(
'0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045'
)
for await (const message of conversation.streamMessages()) {
if (message.senderAddress === xmtp.address) {
continue
}
console.log(`New message from ${message.senderAddress}: ${message.text}`)
}
Different types of content
All the send functions support SendOptions
as an optional parameter. Option contentType
allows specifying different types of content than the default simple string, which is identified with content type identifier ContentTypeText
. Support for other types of content can be added by registering additional ContentCodecs
with the Client
. Every codec is associated with a content type identifier, ContentTypeId
, which is used to signal to the Client which codec should be used to process the content that is being sent or received. See XIP-5 for more details on codecs and content types. New Codecs and content types are defined through XRCs.
If there is a concern that the recipient may not be able to handle particular content type, the sender can use contentFallback
option to provide a string that describes the content being sent. If the recipient fails to decode the original content, the fallback will replace it and can be used to inform the recipient what the original content was.
xmtp.registerCodec:(new NumberCodec())
conversation.send(3.14, {
contentType: ContentTypeNumber,
contentFallback: 'sending you a pie'
})
Additional codecs can be configured through the ClientOptions
parameter of Client.create
. The codecs
option is a list of codec instances that should be added to the default set of codecs (currently only the TextCodec
). If a codec is added for a content type that is already in the default set, it will replace the original codec.
import { CompositeCodec } from '@xmtp/xmtp-js'
const xmtp = Client.create(wallet, { codecs: [new CompositeCodec()] })
Compression
Message content can be optionally compressed using the compression
option. The value of the option is the name of the compression algorithm to use. Currently supported are gzip
and deflate
. Compression is applied to the bytes produced by the content codec.
Content will be decompressed transparently on the receiving end. Note that Client
enforces maximum content size. The default limit can be overridden through the ClientOptions
. Consequently a message that would expand beyond that limit on the receiving end will fail to decode.
conversation.send('#'.repeat(1000), {
compression: 'deflate',
})
Under the hood
Using xmtp.conversations
hides the details of this, but for the curious this is how sending a message on XMTP works. The first message and first response between two parties is sent to three separate Waku content topics:
- Sender's introduction topic
- Recipient's introduction topic
- Conversation topic shared by the sender and the recipient
This is used to establish a shared secret and negotiate a topic to communicate on. Any following messages are sent to the conversation topic only.
The introduction topics allow the participants to reconstruct the list of conversations that they participate(d) in.
The conversation topics carry the contents of the conversations.
Developing
Auto-releasing and commit conventions
A new version of this package will be automatically published whenever there is a merge to the main
branch. Specifically, new GitHub releases and tags will be created, and a new NPM package version will be published. The release version increment type is derived from the commits that were bundled in the merge to main
, using semantic-release commit message conventions.
The table below shows example commits and the resulting release type:
Commit message | Release type |
---|
fix(pencil): stop graphite breaking when too much pressure applied | Patch Fix Release |
feat(pencil): add 'graphiteWidth' option | Minor Feature Release |
perf(pencil): remove graphiteWidth option
BREAKING CHANGE: The graphiteWidth option has been removed.
The default graphite width of 10mm is always used for performance reasons. | Major Breaking Release (Note that the BREAKING CHANGE: token must be in the footer of the commit) |
This is currently configured to use the Angular Commit Message Conventions.
Prerequisites
Node
Please make sure you have a Node version compatible with that specified in the root .nvmrc
file. We recommend using nvm
to manage local node versions - find install instructions appropriate for your system here.
Buf
You will need to install Buf in your environment in order to npm build
this package from source.
brew install bufbuild/buf/buf