mirror of
https://github.com/nbd-wtf/nostr-tools.git
synced 2025-12-09 08:38:50 +00:00
209 lines
4.4 KiB
Markdown
209 lines
4.4 KiB
Markdown
# nostr-tools
|
|
|
|
Tools for developing [Nostr](https://github.com/fiatjaf/nostr) clients.
|
|
|
|
## Usage
|
|
|
|
### Generating a private key and a public key
|
|
|
|
```js
|
|
import { generatePrivateKey, getPublicKey } from 'nostr-tools'
|
|
|
|
let sk = generatePrivateKey() # `sk` is a hex string
|
|
let pk = getPublicKey(sk) # `pk` is a hex string
|
|
```
|
|
|
|
### Creating, signing and verifying events
|
|
|
|
```js
|
|
import {
|
|
validateEvent,
|
|
verifySignature,
|
|
signEvent,
|
|
getEventHash,
|
|
getPublicKey
|
|
} from 'nostr-tools'
|
|
|
|
let event = {
|
|
kind: 1,
|
|
created_at: Math.floor(Date.now() / 1000),
|
|
tags: [],
|
|
content: 'hello'
|
|
}
|
|
|
|
event.id = getEventHash(event.id)
|
|
event.pubkey = getPublicKey(privateKey)
|
|
event.sig = await signEvent(event, privateKey)
|
|
|
|
let ok = validateEvent(event)
|
|
let veryOk = await verifySignature(event)
|
|
```
|
|
|
|
### Interacting with a relay
|
|
|
|
```js
|
|
import {
|
|
relayInit,
|
|
generatePrivateKey,
|
|
getPublicKey,
|
|
getEventHash,
|
|
signEvent
|
|
} from 'nostr-tools'
|
|
|
|
const relay = relayInit('wss://relay.example.com')
|
|
relay.connect()
|
|
|
|
relay.on('connect', () => {
|
|
console.log(`connected to ${relay.url}`)
|
|
})
|
|
relay.on('error', () => {
|
|
console.log(`failed to connect to ${relay.url}`)
|
|
})
|
|
|
|
// let's query for an event that exists
|
|
let sub = relay.sub([
|
|
{
|
|
ids: ['d7dd5eb3ab747e16f8d0212d53032ea2a7cadef53837e5a6c66d42849fcb9027']
|
|
}
|
|
])
|
|
sub.on('event', event => {
|
|
console.log('we got the event we wanted:', event)
|
|
})
|
|
sub.on('eose', () => {
|
|
sub.unsub()
|
|
})
|
|
|
|
// let's publish a new event while simultaneously monitoring the relay for it
|
|
let sk = generatePrivateKey()
|
|
let pk = getPublicKey(sk)
|
|
|
|
let sub = relay.sub([
|
|
{
|
|
kinds: [1],
|
|
authors: [pk]
|
|
}
|
|
])
|
|
|
|
sub.on('event', event => {
|
|
console.log('got event:', event)
|
|
})
|
|
|
|
let event = {
|
|
kind: 1,
|
|
pubkey: pk,
|
|
created_at: Math.floor(Date.now() / 1000),
|
|
tags: [],
|
|
content: 'hello world'
|
|
}
|
|
event.id = getEventHash(event)
|
|
event.sig = await signEvent(event, sk)
|
|
|
|
let pub = relay.publish(event)
|
|
pub.on('ok', () => {
|
|
console.log(`{relay.url} has accepted our event`)
|
|
})
|
|
pub.on('seen', () => {
|
|
console.log(`we saw the event on {relay.url}`)
|
|
})
|
|
pub.on('failed', reason => {
|
|
console.log(`failed to publish to {relay.url}: ${reason}`)
|
|
})
|
|
|
|
await relay.close()
|
|
```
|
|
|
|
### Querying profile data from a NIP-05 address
|
|
|
|
```js
|
|
import {nip05} from 'nostr-tools'
|
|
|
|
let profile = await nip05.queryProfile('jb55.com')
|
|
console.log(profile.pubkey)
|
|
// prints: 32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245
|
|
console.log(profile.relays)
|
|
// prints: [wss://relay.damus.io]
|
|
|
|
// on nodejs, install node-fetch@2 and call this first:
|
|
nip05.useFetchImplementation(require('node-fetch'))
|
|
```
|
|
|
|
### Encoding and decoding NIP-19 codes
|
|
|
|
```js
|
|
import {nip19, generatePrivateKey, getPublicKey} from 'nostr-tools'
|
|
|
|
let sk = generatePrivateKey()
|
|
let nsec = nip19.nsecEncode(sk)
|
|
let {type, data} = nip19.decode(nsec)
|
|
assert(type === 'nsec')
|
|
assert(data === sk)
|
|
|
|
let pk = getPublicKey(generatePrivateKey())
|
|
let npub = nip19.npubEncode(pk)
|
|
let {type, data} = nip19.decode(npub)
|
|
assert(type === 'npub')
|
|
assert(data === pk)
|
|
|
|
let pk = getPublicKey(generatePrivateKey())
|
|
let relays = [
|
|
'wss://relay.nostr.example.mydomain.example.com',
|
|
'wss://nostr.banana.com'
|
|
]
|
|
let nprofile = nip19.nprofileEncode({pubkey: pk, relays})
|
|
let {type, data} = nip19.decode(nprofile)
|
|
assert(type === 'nprofile')
|
|
assert(data.pubkey === pk)
|
|
assert(data.relays.length === 2)
|
|
```
|
|
|
|
### Encrypting and decrypting direct messages
|
|
|
|
```js
|
|
import {nip04, getPublicKey, generatePrivateKey} from 'nostr-tools'
|
|
|
|
// sender
|
|
let sk1 = generatePrivateKey()
|
|
let pk1 = getPublicKey(sk1)
|
|
|
|
// receiver
|
|
let sk2 = generatePrivateKey()
|
|
let pk2 = getPublicKey(sk2)
|
|
|
|
// on the sender side
|
|
let message = 'hello'
|
|
let ciphertext = nip04.encrypt(sk1, pk2, 'hello')
|
|
|
|
let event = {
|
|
kind: 4,
|
|
pubkey: pk1,
|
|
tags: [['p', pk2]],
|
|
content: ciphertext,
|
|
...otherProperties
|
|
}
|
|
|
|
sendEvent(event)
|
|
|
|
// on the receiver side
|
|
|
|
sub.on('event', (event) => {
|
|
let sender = event.tags.find(([k, v]) => k === 'p' && && v && v !== '')[1]
|
|
pk1 === sender
|
|
let plaintext = nip04.decrypt(sk2, pk1, event.content)
|
|
})
|
|
```
|
|
|
|
Please consult the tests or [the source code](https://github.com/fiatjaf/nostr-tools) for more information that isn't available here.
|
|
|
|
### Using from the browser (if you don't want to use a bundler)
|
|
|
|
```html
|
|
<script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js"></script>
|
|
<script>
|
|
window.NostrTools.generatePrivateKey('...') // and so on
|
|
</script>
|
|
```
|
|
|
|
## License
|
|
|
|
Public domain.
|