Skip to content

Commit

Permalink
fix: Added support for lowercase addresses
Browse files Browse the repository at this point in the history
Convert addresses to checksums first
  • Loading branch information
Alex Risch authored and Alex Risch committed Feb 7, 2024
1 parent ddae69a commit bb53546
Show file tree
Hide file tree
Showing 8 changed files with 151 additions and 10 deletions.
6 changes: 6 additions & 0 deletions .github/workflows/release.yml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,12 @@ on:
branches:
- main
- beta
workflow_dispatch:
inputs:
branch:
description: 'Branch name'
required: true
default: 'main'
jobs:
release:
name: Release
Expand Down
56 changes: 56 additions & 0 deletions example/src/tests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -841,3 +841,59 @@ test('calls preEnableIdentityCallback when supplied', async () => {

return isCallbackCalled
})

test('correctly handles lowercase addresses', async () => {
const bob = await Client.createRandom({ env: 'local' })
await delayToPropogate()
const alice = await Client.createRandom({ env: 'local' })
await delayToPropogate()
if (bob.address === alice.address) {
throw new Error('bob and alice should be different')
}

const bobConversation = await bob.conversations.newConversation(
alice.address.toLocaleLowerCase()
)
await delayToPropogate()
if (!bobConversation) {
throw new Error('bobConversation should exist')
}
const aliceConversation = (await alice.conversations.list())[0]
if (!aliceConversation) {
throw new Error('aliceConversation should exist')
}

await bob.contacts.deny([aliceConversation.peerAddress.toLocaleLowerCase()])
await delayToPropogate()
const deniedState = await bob.contacts.isDenied(aliceConversation.peerAddress)
const allowedState = await bob.contacts.isAllowed(
aliceConversation.peerAddress
)
if (!deniedState) {
throw new Error(`contacts denied by bo should be denied not ${deniedState}`)
}

if (allowedState) {
throw new Error(
`contacts denied by bo should be denied not ${allowedState}`
)
}
const deniedLowercaseState = await bob.contacts.isDenied(
aliceConversation.peerAddress.toLocaleLowerCase()
)
const allowedLowercaseState = await bob.contacts.isAllowed(
aliceConversation.peerAddress.toLocaleLowerCase()
)
if (!deniedLowercaseState) {
throw new Error(
`contacts denied by bo should be denied not ${deniedLowercaseState}`
)
}

if (allowedLowercaseState) {
throw new Error(
`contacts denied by bo should be denied not ${allowedLowercaseState}`
)
}
return true
})
7 changes: 5 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,8 @@
{
"name": "beta",
"prerelease": true
}
},
"+([0-9])?(.{+([0-9]),x}).x"
]
},
"publishConfig": {
Expand All @@ -45,9 +46,11 @@
"homepage": "https://github.com/xmtp/xmtp-react-native#readme",
"dependencies": {
"@msgpack/msgpack": "^3.0.0-beta2",
"@noble/hashes": "^1.3.3",
"@xmtp/proto": "^3.25.0",
"buffer": "^6.0.3",
"ethers": "^5.7.2"
"ethers": "^5.7.2",
"text-encoding": "^0.7.0"
},
"devDependencies": {
"@babel/plugin-proposal-export-namespace-from": "^7.18.9",
Expand Down
11 changes: 8 additions & 3 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import {
import { Conversation } from './lib/Conversation'
import { DecodedMessage } from './lib/DecodedMessage'
import type { Query } from './lib/Query'
import { getAddress } from './utils/address'

export { ReactionCodec } from './lib/NativeCodecs/ReactionCodec'
export { ReplyCodec } from './lib/NativeCodecs/ReplyCodec'
Expand Down Expand Up @@ -105,15 +106,19 @@ export async function canMessage(
clientAddress: string,
peerAddress: string
): Promise<boolean> {
return await XMTPModule.canMessage(clientAddress, peerAddress)
return await XMTPModule.canMessage(clientAddress, getAddress(peerAddress))
}

export async function staticCanMessage(
peerAddress: string,
environment: 'local' | 'dev' | 'production',
appVersion?: string | undefined
): Promise<boolean> {
return await XMTPModule.staticCanMessage(peerAddress, environment, appVersion)
return await XMTPModule.staticCanMessage(
getAddress(peerAddress),
environment,
appVersion
)
}

export async function encryptAttachment(
Expand Down Expand Up @@ -212,7 +217,7 @@ export async function createConversation<ContentTypes>(
JSON.parse(
await XMTPModule.createConversation(
client.address,
peerAddress,
getAddress(peerAddress),
JSON.stringify(context || {})
)
)
Expand Down
17 changes: 13 additions & 4 deletions src/lib/Contacts.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { Client } from './Client'
import { ConsentListEntry } from './ConsentListEntry'
import * as XMTPModule from '../index'
import { getAddress } from '../utils/address'

export default class Contacts {
client: Client<any>
Expand All @@ -10,19 +11,27 @@ export default class Contacts {
}

async isAllowed(address: string): Promise<boolean> {
return await XMTPModule.isAllowed(this.client.address, address)
return await XMTPModule.isAllowed(this.client.address, getAddress(address))
}

async isDenied(address: string): Promise<boolean> {
return await XMTPModule.isDenied(this.client.address, address)
return await XMTPModule.isDenied(this.client.address, getAddress(address))
}

async deny(addresses: string[]): Promise<void> {
return await XMTPModule.denyContacts(this.client.address, addresses)
const checkSummedAddresses = addresses.map((address) => getAddress(address))
return await XMTPModule.denyContacts(
this.client.address,
checkSummedAddresses
)
}

async allow(addresses: string[]): Promise<void> {
return await XMTPModule.allowContacts(this.client.address, addresses)
const checkSummedAddresses = addresses.map((address) => getAddress(address))
return await XMTPModule.allowContacts(
this.client.address,
checkSummedAddresses
)
}

async refreshConsentList(): Promise<ConsentListEntry[]> {
Expand Down
4 changes: 3 additions & 1 deletion src/lib/Conversations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import { Conversation } from './Conversation'
import { DecodedMessage } from './DecodedMessage'
import { ConversationContext } from '../XMTP.types'
import * as XMTPModule from '../index'
import { getAddress } from '../utils/address'

export default class Conversations<ContentTypes> {
client: Client<ContentTypes>
Expand Down Expand Up @@ -50,9 +51,10 @@ export default class Conversations<ContentTypes> {
peerAddress: string,
context?: ConversationContext
): Promise<Conversation<ContentTypes>> {
const checksumAddress = getAddress(peerAddress)
return await XMTPModule.createConversation(
this.client,
peerAddress,
checksumAddress,
context
)
}
Expand Down
50 changes: 50 additions & 0 deletions src/utils/address.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import { keccak_256 } from '@noble/hashes/sha3'
import { TextEncoder } from 'text-encoding'

const addressRegex = /^0x[a-fA-F0-9]{40}$/
const encoder = new TextEncoder()

export function stringToBytes(value: string): Uint8Array {
const bytes = encoder.encode(value)
return bytes
}

export function keccak256(value: Uint8Array): Uint8Array {
const bytes = keccak_256(value)
return bytes
}

export function isAddress(address: string): boolean {
return addressRegex.test(address)
}

export function checksumAddress(address_: string, chainId?: number): string {
const hexAddress = chainId
? `${chainId}${address_.toLowerCase()}`
: address_.substring(2).toLowerCase()
const hash = keccak256(stringToBytes(hexAddress))

const address = (
chainId ? hexAddress.substring(`${chainId}0x`.length) : hexAddress
).split('')
for (let i = 0; i < 40; i += 2) {
if (hash[i >> 1] >> 4 >= 8 && address[i]) {
address[i] = address[i].toUpperCase()
}
if ((hash[i >> 1] & 0x0f) >= 8 && address[i + 1]) {
address[i + 1] = address[i + 1].toUpperCase()
}
}

return `0x${address.join('')}`
}

const addressCache = new Map<string, string>()

export function getAddress(address: string, chainId?: number): string {
if (addressCache.has(address)) return addressCache.get(address) as string
if (!isAddress(address)) throw new Error('Invalid address' + address)
const checksumedAddress = checksumAddress(address, chainId)
addressCache.set(address, checksumedAddress)
return checksumedAddress
}
10 changes: 10 additions & 0 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -1849,6 +1849,11 @@
resolved "https://registry.yarnpkg.com/@nicolo-ribaudo/chokidar-2/-/chokidar-2-2.1.8-no-fsevents.3.tgz#323d72dd25103d0c4fbdce89dadf574a787b1f9b"
integrity sha512-s88O1aVtXftvp5bCPB7WnmXc5IwOZZ7YPuwNPt+GtOOXpPvad1LfbmjYv+qII7zP6RU2QGnqve27dnLycEnyEQ==

"@noble/hashes@^1.3.3":
version "1.3.3"
resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.3.3.tgz#39908da56a4adc270147bb07968bf3b16cfe1699"
integrity sha512-V7/fPHgl+jsVPXqqeOzT8egNj2iBIVt+ECeMMG8TdcnTikP3oaBtUVqpT/gYCR68aEBJSF+XbYUxStjbFMqIIA==

"@nodelib/[email protected]":
version "2.1.5"
resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5"
Expand Down Expand Up @@ -8345,6 +8350,11 @@ test-exclude@^6.0.0:
glob "^7.1.4"
minimatch "^3.0.4"

text-encoding@^0.7.0:
version "0.7.0"
resolved "https://registry.yarnpkg.com/text-encoding/-/text-encoding-0.7.0.tgz#f895e836e45990624086601798ea98e8f36ee643"
integrity sha512-oJQ3f1hrOnbRLOcwKz0Liq2IcrvDeZRHXhd9RgLrsT+DjWY/nty1Hi7v3dtkaEYbPYe0mUoOfzRrMwfXXwgPUA==

text-extensions@^2.0.0:
version "2.4.0"
resolved "https://registry.yarnpkg.com/text-extensions/-/text-extensions-2.4.0.tgz#a1cfcc50cf34da41bfd047cc744f804d1680ea34"
Expand Down

0 comments on commit bb53546

Please sign in to comment.