mirror of
https://github.com/nbd-wtf/nostr-tools.git
synced 2025-12-08 16:28:49 +00:00
Compare commits
145 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d773012658 | ||
|
|
b8f91c37fa | ||
|
|
2da3528362 | ||
|
|
315e9a472c | ||
|
|
a2b1bf0338 | ||
|
|
861a77e2b3 | ||
|
|
9132b722f3 | ||
|
|
ae2f97655b | ||
|
|
5b78a829c7 | ||
|
|
de26ee98c5 | ||
|
|
1437bbdb0f | ||
|
|
57354b9fb4 | ||
|
|
924075b803 | ||
|
|
666a02027e | ||
|
|
eff9ea9579 | ||
|
|
ca174e6cd8 | ||
|
|
4ba9c8886b | ||
|
|
7dbd86eb5c | ||
|
|
3e839db6f2 | ||
|
|
cb370fbf4f | ||
|
|
c015b6e794 | ||
|
|
52079f6e75 | ||
|
|
ef28b2eb73 | ||
|
|
2a422774fb | ||
|
|
b80f8a0bcc | ||
|
|
dd603e47d8 | ||
|
|
ba26b92973 | ||
|
|
aec8ff5946 | ||
|
|
e498c9144d | ||
|
|
42d47abba1 | ||
|
|
303c35120c | ||
|
|
4a738c93d0 | ||
|
|
2a11c9ec91 | ||
|
|
cbe3a9d683 | ||
|
|
2944a932b8 | ||
|
|
6b39de04d7 | ||
|
|
9a612e59a2 | ||
|
|
266dbdf766 | ||
|
|
19ae9837a7 | ||
|
|
4188f2c596 | ||
|
|
97bded8f5b | ||
|
|
174d36a440 | ||
|
|
0177b130c3 | ||
|
|
05eb62da5b | ||
|
|
3c4019a154 | ||
|
|
e7e8db1dbd | ||
|
|
44a679e642 | ||
|
|
c1172caf1d | ||
|
|
86f37d6003 | ||
|
|
3daade322c | ||
|
|
fcf10541c8 | ||
|
|
548abb5d4a | ||
|
|
1e5bfe856b | ||
|
|
3266b4d4c2 | ||
|
|
a0b950ab12 | ||
|
|
be741159d7 | ||
|
|
9c50b2c655 | ||
|
|
bbb09420fe | ||
|
|
2e85f7a5fe | ||
|
|
b22e2465cc | ||
|
|
43ce7f9377 | ||
|
|
5a55c670fb | ||
|
|
bf0c4d4988 | ||
|
|
50fe7c2a8b | ||
|
|
29270c8c9d | ||
|
|
cb29d62033 | ||
|
|
0d237405d9 | ||
|
|
659ad36b62 | ||
|
|
d062ab8afd | ||
|
|
94f841f347 | ||
|
|
c1d03cf00b | ||
|
|
29ecdfc5ec | ||
|
|
d3fc4734b4 | ||
|
|
66d0b8a4e1 | ||
|
|
e2ec7a4b55 | ||
|
|
a72e47135a | ||
|
|
de7bbfc6a2 | ||
|
|
f2d421fa4f | ||
|
|
cae06fc4fe | ||
|
|
5c538efa38 | ||
|
|
013daae91b | ||
|
|
75660e7ff1 | ||
|
|
4c2d2b5ce6 | ||
|
|
aba266b8e6 | ||
|
|
d7dcc75ebe | ||
|
|
b18510b460 | ||
|
|
b04e0d16c0 | ||
|
|
633696bf46 | ||
|
|
bf975c9a87 | ||
|
|
7aa4f09769 | ||
|
|
f646fcd889 | ||
|
|
1d89038375 | ||
|
|
0b5b35714c | ||
|
|
e398617fdc | ||
|
|
1b236faa7b | ||
|
|
7064e0b828 | ||
|
|
4f6976f6f8 | ||
|
|
a61cde77ea | ||
|
|
23d95acb26 | ||
|
|
13ac04b8f8 | ||
|
|
45b25c5bf5 | ||
|
|
ee76d69b4b | ||
|
|
21433049b8 | ||
|
|
e8ff68f0b3 | ||
|
|
1b77d6e080 | ||
|
|
76d3a91600 | ||
|
|
6f334f31a7 | ||
|
|
9c009ac543 | ||
|
|
a87099fa5c | ||
|
|
475a22a95f | ||
|
|
54e352d8e2 | ||
|
|
235a1c50cb | ||
|
|
dfc2107569 | ||
|
|
986b9d0cce | ||
|
|
753ff323ea | ||
|
|
f8c3e20f3d | ||
|
|
87a91c2daf | ||
|
|
4f1dc9ef1c | ||
|
|
faa1a9d556 | ||
|
|
97d838f254 | ||
|
|
260400b24d | ||
|
|
6e5ab34a54 | ||
|
|
9562c408b3 | ||
|
|
4f4de458e9 | ||
|
|
88454de628 | ||
|
|
9f5984d78d | ||
|
|
80df21d47f | ||
|
|
296e99d2a4 | ||
|
|
1cd9847ad5 | ||
|
|
fa31fdca78 | ||
|
|
5876acd67a | ||
|
|
44efd49bc0 | ||
|
|
f4f9bece6e | ||
|
|
e217f751da | ||
|
|
d0ae8b36a2 | ||
|
|
fd945757be | ||
|
|
c12ddd3c53 | ||
|
|
1e9f828e3e | ||
|
|
0a5eaac088 | ||
|
|
e858698cb9 | ||
|
|
b349ee577d | ||
|
|
849a2ac3f3 | ||
|
|
c18b94677c | ||
|
|
f306cec716 | ||
|
|
88247e56c1 |
19
.devcontainer/Dockerfile
Executable file
19
.devcontainer/Dockerfile
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
FROM node:20
|
||||||
|
|
||||||
|
RUN npm install typescript eslint prettier -g
|
||||||
|
|
||||||
|
# Install bun
|
||||||
|
RUN curl -fsSL https://bun.sh/install | bash
|
||||||
|
|
||||||
|
# Install just
|
||||||
|
WORKDIR /usr/bin
|
||||||
|
RUN wget https://github.com/casey/just/releases/download/1.26.0/just-1.26.0-x86_64-unknown-linux-musl.tar.gz
|
||||||
|
RUN tar -xzf just-1.26.0-x86_64-unknown-linux-musl.tar.gz
|
||||||
|
RUN chmod +x ./just
|
||||||
|
RUN rm just-1.26.0-x86_64-unknown-linux-musl.tar.gz
|
||||||
|
|
||||||
|
WORKDIR /nostr-tools
|
||||||
|
ENV LANG C.UTF-8
|
||||||
|
|
||||||
|
# The run the start script
|
||||||
|
CMD [ "/bin/bash" ]
|
||||||
19
.devcontainer/devcontainer.json
Executable file
19
.devcontainer/devcontainer.json
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"name": "Nostr Tools",
|
||||||
|
"dockerComposeFile": [
|
||||||
|
"docker-compose.yml"
|
||||||
|
],
|
||||||
|
"service": "nostr-tools-dev",
|
||||||
|
"workspaceFolder": "/nostr-tools",
|
||||||
|
"customizations": {
|
||||||
|
"vscode": {
|
||||||
|
"extensions": [
|
||||||
|
"ms-vscode.vscode-typescript-next",
|
||||||
|
"eamodio.gitlens",
|
||||||
|
"dbaeumer.vscode-eslint",
|
||||||
|
"manishsencha.readme-preview",
|
||||||
|
"wix.vscode-import-cost"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
13
.devcontainer/docker-compose.yml
Executable file
13
.devcontainer/docker-compose.yml
Executable file
@@ -0,0 +1,13 @@
|
|||||||
|
version: '3.9'
|
||||||
|
|
||||||
|
services:
|
||||||
|
nostr-tools-dev:
|
||||||
|
image: nostr-tools-dev
|
||||||
|
container_name: nostr-tools-dev
|
||||||
|
build:
|
||||||
|
context: ../.
|
||||||
|
dockerfile: ./.devcontainer/Dockerfile
|
||||||
|
working_dir: /nostr-tools
|
||||||
|
volumes:
|
||||||
|
- ..:/nostr-tools:cached
|
||||||
|
tty: true
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
"extends": ["prettier"],
|
"extends": ["prettier"],
|
||||||
|
|
||||||
"parser": "@typescript-eslint/parser",
|
"parser": "@typescript-eslint/parser",
|
||||||
"plugins": ["@typescript-eslint", "babel"],
|
"plugins": ["@typescript-eslint"],
|
||||||
|
|
||||||
"parserOptions": {
|
"parserOptions": {
|
||||||
"ecmaVersion": 9,
|
"ecmaVersion": 9,
|
||||||
@@ -116,7 +116,8 @@
|
|||||||
"no-unexpected-multiline": 2,
|
"no-unexpected-multiline": 2,
|
||||||
"no-unneeded-ternary": [2, { "defaultAssignment": false }],
|
"no-unneeded-ternary": [2, { "defaultAssignment": false }],
|
||||||
"no-unreachable": 2,
|
"no-unreachable": 2,
|
||||||
"no-unused-vars": [2, { "vars": "local", "args": "none", "varsIgnorePattern": "^_" }],
|
"no-unused-vars": "off",
|
||||||
|
"@typescript-eslint/no-unused-vars": [2, { "vars": "local", "args": "none", "varsIgnorePattern": "^_" }],
|
||||||
"no-useless-call": 2,
|
"no-useless-call": 2,
|
||||||
"no-useless-constructor": 2,
|
"no-useless-constructor": 2,
|
||||||
"no-with": 2,
|
"no-with": 2,
|
||||||
|
|||||||
231
README.md
231
README.md
@@ -1,19 +1,27 @@
|
|||||||
#  nostr-tools
|
#  [](https://jsr.io/@nostr/tools) nostr-tools
|
||||||
|
|
||||||
Tools for developing [Nostr](https://github.com/fiatjaf/nostr) clients.
|
Tools for developing [Nostr](https://github.com/fiatjaf/nostr) clients.
|
||||||
|
|
||||||
Only depends on _@scure_ and _@noble_ packages.
|
Only depends on _@scure_ and _@noble_ packages.
|
||||||
|
|
||||||
This package is only providing lower-level functionality. If you want an easy-to-use fully-fledged solution that abstracts the hard parts of Nostr and makes decisions on your behalf, take a look at [NDK](https://github.com/nostr-dev-kit/ndk) and [@snort/system](https://www.npmjs.com/package/@snort/system).
|
This package is only providing lower-level functionality. If you want higher-level features, take a look at [@nostr/gadgets](https://jsr.io/@nostr/gadgets) which is based on this library and expands upon it and has other goodies (it's only available on jsr).
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
npm install nostr-tools # or yarn add nostr-tools
|
# npm
|
||||||
|
npm install --save nostr-tools
|
||||||
|
|
||||||
|
# jsr
|
||||||
|
npx jsr add @nostr/tools
|
||||||
```
|
```
|
||||||
|
|
||||||
If using TypeScript, this package requires TypeScript >= 5.0.
|
If using TypeScript, this package requires TypeScript >= 5.0.
|
||||||
|
|
||||||
|
## Documentation
|
||||||
|
|
||||||
|
https://jsr.io/@nostr/tools/doc
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
### Generating a private key and a public key
|
### Generating a private key and a public key
|
||||||
@@ -28,9 +36,9 @@ let pk = getPublicKey(sk) // `pk` is a hex string
|
|||||||
To get the secret key in hex format, use
|
To get the secret key in hex format, use
|
||||||
|
|
||||||
```js
|
```js
|
||||||
import { bytestohex, hexToBytes } from '@noble/hashes/utils' // already an installed dependency
|
import { bytesToHex, hexToBytes } from '@noble/hashes/utils' // already an installed dependency
|
||||||
|
|
||||||
let skHex = bytestohex(sk)
|
let skHex = bytesToHex(sk)
|
||||||
let backToBytes = hexToBytes(skHex)
|
let backToBytes = hexToBytes(skHex)
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -49,43 +57,57 @@ let event = finalizeEvent({
|
|||||||
let isGood = verifyEvent(event)
|
let isGood = verifyEvent(event)
|
||||||
```
|
```
|
||||||
|
|
||||||
### Interacting with a relay
|
### Interacting with one or multiple relays
|
||||||
|
|
||||||
|
Doesn't matter what you do, you always should be using a `SimplePool`:
|
||||||
|
|
||||||
```js
|
```js
|
||||||
import { finalizeEvent, generateSecretKey, getPublicKey } from 'nostr-tools/pure'
|
import { finalizeEvent, generateSecretKey, getPublicKey } from 'nostr-tools/pure'
|
||||||
import { Relay } from 'nostr-tools/relay'
|
import { SimplePool } from 'nostr-tools/pool'
|
||||||
|
|
||||||
const relay = await Relay.connect('wss://relay.example.com')
|
const pool = new SimplePool()
|
||||||
console.log(`connected to ${relay.url}`)
|
|
||||||
|
|
||||||
// let's query for an event that exists
|
const relays = ['wss://relay.example.com', 'wss://relay.example2.com']
|
||||||
const sub = relay.subscribe([
|
|
||||||
|
// let's query for one event that exists
|
||||||
|
const event = pool.get(
|
||||||
|
relays,
|
||||||
{
|
{
|
||||||
ids: ['d7dd5eb3ab747e16f8d0212d53032ea2a7cadef53837e5a6c66d42849fcb9027'],
|
ids: ['d7dd5eb3ab747e16f8d0212d53032ea2a7cadef53837e5a6c66d42849fcb9027'],
|
||||||
},
|
},
|
||||||
], {
|
)
|
||||||
onevent(event) {
|
if (event) {
|
||||||
console.log('we got the event we wanted:', event)
|
console.log('it exists indeed on this relay:', event)
|
||||||
|
}
|
||||||
|
|
||||||
|
// let's query for more than one event that exists
|
||||||
|
const events = pool.querySync(
|
||||||
|
relays,
|
||||||
|
{
|
||||||
|
kinds: [1],
|
||||||
|
limit: 10
|
||||||
},
|
},
|
||||||
oneose() {
|
)
|
||||||
sub.close()
|
if (events) {
|
||||||
}
|
console.log('it exists indeed on this relay:', events)
|
||||||
})
|
}
|
||||||
|
|
||||||
// let's publish a new event while simultaneously monitoring the relay for it
|
// let's publish a new event while simultaneously monitoring the relay for it
|
||||||
let sk = generateSecretKey()
|
let sk = generateSecretKey()
|
||||||
let pk = getPublicKey(sk)
|
let pk = getPublicKey(sk)
|
||||||
|
|
||||||
relay.sub([
|
pool.subscribe(
|
||||||
|
['wss://a.com', 'wss://b.com', 'wss://c.com'],
|
||||||
{
|
{
|
||||||
kinds: [1],
|
kinds: [1],
|
||||||
authors: [pk],
|
authors: [pk],
|
||||||
},
|
},
|
||||||
], {
|
{
|
||||||
onevent(event) {
|
onevent(event) {
|
||||||
console.log('got event:', event)
|
console.log('got event:', event)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
)
|
||||||
|
|
||||||
let eventTemplate = {
|
let eventTemplate = {
|
||||||
kind: 1,
|
kind: 1,
|
||||||
@@ -96,7 +118,7 @@ let eventTemplate = {
|
|||||||
|
|
||||||
// this assigns the pubkey, calculates the event id and signs the event in a single step
|
// this assigns the pubkey, calculates the event id and signs the event in a single step
|
||||||
const signedEvent = finalizeEvent(eventTemplate, sk)
|
const signedEvent = finalizeEvent(eventTemplate, sk)
|
||||||
await relay.publish(signedEvent)
|
await Promise.any(pool.publish(['wss://a.com', 'wss://b.com'], signedEvent))
|
||||||
|
|
||||||
relay.close()
|
relay.close()
|
||||||
```
|
```
|
||||||
@@ -104,63 +126,123 @@ relay.close()
|
|||||||
To use this on Node.js you first must install `ws` and call something like this:
|
To use this on Node.js you first must install `ws` and call something like this:
|
||||||
|
|
||||||
```js
|
```js
|
||||||
import { useWebSocketImplementation } from 'nostr-tools/relay'
|
import { useWebSocketImplementation } from 'nostr-tools/pool'
|
||||||
useWebSocketImplementation(require('ws'))
|
// or import { useWebSocketImplementation } from 'nostr-tools/relay' if you're using the Relay directly
|
||||||
|
|
||||||
|
import WebSocket from 'ws'
|
||||||
|
useWebSocketImplementation(WebSocket)
|
||||||
```
|
```
|
||||||
|
|
||||||
### Interacting with multiple relays
|
### Parsing references (mentions) from a content based on NIP-27
|
||||||
|
|
||||||
```js
|
```js
|
||||||
import { SimplePool } from 'nostr-tools/pool'
|
import * as nip27 from '@nostr/tools/nip27'
|
||||||
|
|
||||||
const pool = new SimplePool()
|
for (let block of nip27.parse(evt.content)) {
|
||||||
|
switch (block.type) {
|
||||||
let relays = ['wss://relay.example.com', 'wss://relay.example2.com']
|
case 'text':
|
||||||
|
console.log(block.text)
|
||||||
let h = pool.subscribeMany(
|
break
|
||||||
[...relays, 'wss://relay.example3.com'],
|
case 'reference': {
|
||||||
[
|
if ('id' in block.pointer) {
|
||||||
{
|
console.log("it's a nevent1 uri", block.pointer)
|
||||||
authors: ['32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245'],
|
} else if ('identifier' in block.pointer) {
|
||||||
},
|
console.log("it's a naddr1 uri", block.pointer)
|
||||||
],
|
} else {
|
||||||
{
|
console.log("it's an npub1 or nprofile1 uri", block.pointer)
|
||||||
onevent(event) {
|
}
|
||||||
// this will only be called once the first time the event is received
|
break
|
||||||
// ...
|
|
||||||
},
|
|
||||||
oneose() {
|
|
||||||
h.close()
|
|
||||||
}
|
}
|
||||||
|
case 'url': {
|
||||||
|
console.log("it's a normal url:", block.url)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
case 'image':
|
||||||
|
case 'video':
|
||||||
|
case 'audio':
|
||||||
|
console.log("it's a media url:", block.url)
|
||||||
|
case 'relay':
|
||||||
|
console.log("it's a websocket url, probably a relay address:", block.url)
|
||||||
|
default:
|
||||||
|
break
|
||||||
}
|
}
|
||||||
)
|
}
|
||||||
|
|
||||||
await Promise.any(pool.publish(relays, newEvent))
|
|
||||||
console.log('published to at least one relay!')
|
|
||||||
|
|
||||||
let events = await pool.querySync(relays, [{ kinds: [0, 1] }])
|
|
||||||
let event = await pool.get(relays, {
|
|
||||||
ids: ['44e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245'],
|
|
||||||
})
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Parsing references (mentions) from a content using NIP-10 and NIP-27
|
### Connecting to a bunker using NIP-46
|
||||||
|
|
||||||
```js
|
```js
|
||||||
import { parseReferences } from 'nostr-tools/references'
|
import { generateSecretKey, getPublicKey } from '@nostr/tools/pure'
|
||||||
|
import { BunkerSigner, parseBunkerInput } from '@nostr/tools/nip46'
|
||||||
|
import { SimplePool } from '@nostr/tools/pool'
|
||||||
|
|
||||||
let references = parseReferences(event)
|
// the client needs a local secret key (which is generally persisted) for communicating with the bunker
|
||||||
let simpleAugmentedContent = event.content
|
const localSecretKey = generateSecretKey()
|
||||||
for (let i = 0; i < references.length; i++) {
|
|
||||||
let { text, profile, event, address } = references[i]
|
// parse a bunker URI
|
||||||
let augmentedReference = profile
|
const bunkerPointer = await parseBunkerInput('bunker://abcd...?relay=wss://relay.example.com')
|
||||||
? `<strong>@${profilesCache[profile.pubkey].name}</strong>`
|
if (!bunkerPointer) {
|
||||||
: event
|
throw new Error('Invalid bunker input')
|
||||||
? `<em>${eventsCache[event.id].content.slice(0, 5)}</em>`
|
}
|
||||||
: address
|
|
||||||
? `<a href="${text}">[link]</a>`
|
// create the bunker instance
|
||||||
: text
|
const pool = new SimplePool()
|
||||||
simpleAugmentedContent.replaceAll(text, augmentedReference)
|
const bunker = new BunkerSigner(localSecretKey, bunkerPointer, { pool })
|
||||||
|
await bunker.connect()
|
||||||
|
|
||||||
|
// and use it
|
||||||
|
const pubkey = await bunker.getPublicKey()
|
||||||
|
const event = await bunker.signEvent({
|
||||||
|
kind: 1,
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello from bunker!'
|
||||||
|
})
|
||||||
|
|
||||||
|
// cleanup
|
||||||
|
await signer.close()
|
||||||
|
pool.close([])
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parsing thread from any note based on NIP-10
|
||||||
|
|
||||||
|
```js
|
||||||
|
import * as nip10 from '@nostr/tools/nip10'
|
||||||
|
|
||||||
|
// event is a nostr event with tags
|
||||||
|
const refs = nip10.parse(event)
|
||||||
|
|
||||||
|
// get the root event of the thread
|
||||||
|
if (refs.root) {
|
||||||
|
console.log('root event:', refs.root.id)
|
||||||
|
console.log('root event relay hints:', refs.root.relays)
|
||||||
|
console.log('root event author:', refs.root.author)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get the immediate parent being replied to
|
||||||
|
if (refs.reply) {
|
||||||
|
console.log('reply to:', refs.reply.id)
|
||||||
|
console.log('reply relay hints:', refs.reply.relays)
|
||||||
|
console.log('reply author:', refs.reply.author)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get any mentioned events
|
||||||
|
for (let mention of refs.mentions) {
|
||||||
|
console.log('mentioned event:', mention.id)
|
||||||
|
console.log('mention relay hints:', mention.relays)
|
||||||
|
console.log('mention author:', mention.author)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get any quoted events
|
||||||
|
for (let quote of refs.quotes) {
|
||||||
|
console.log('quoted event:', quote.id)
|
||||||
|
console.log('quote relay hints:', quote.relays)
|
||||||
|
}
|
||||||
|
|
||||||
|
// get any referenced profiles
|
||||||
|
for (let profile of refs.profiles) {
|
||||||
|
console.log('referenced profile:', profile.pubkey)
|
||||||
|
console.log('profile relay hints:', profile.relays)
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -183,6 +265,17 @@ import { useFetchImplementation } from 'nostr-tools/nip05'
|
|||||||
useFetchImplementation(require('node-fetch'))
|
useFetchImplementation(require('node-fetch'))
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Including NIP-07 types
|
||||||
|
```js
|
||||||
|
import type { WindowNostr } from 'nostr-tools/nip07'
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface Window {
|
||||||
|
nostr?: WindowNostr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
### Encoding and decoding NIP-19 codes
|
### Encoding and decoding NIP-19 codes
|
||||||
|
|
||||||
```js
|
```js
|
||||||
|
|||||||
151
abstract-pool.ts
151
abstract-pool.ts
@@ -1,28 +1,44 @@
|
|||||||
import { AbstractRelay as AbstractRelay, SubscriptionParams, Subscription } from './abstract-relay.ts'
|
/* global WebSocket */
|
||||||
|
|
||||||
|
import {
|
||||||
|
AbstractRelay as AbstractRelay,
|
||||||
|
SubscriptionParams,
|
||||||
|
Subscription,
|
||||||
|
type AbstractRelayConstructorOptions,
|
||||||
|
} from './abstract-relay.ts'
|
||||||
import { normalizeURL } from './utils.ts'
|
import { normalizeURL } from './utils.ts'
|
||||||
|
|
||||||
import type { Event, Nostr } from './core.ts'
|
import type { Event, EventTemplate, Nostr, VerifiedEvent } from './core.ts'
|
||||||
import { type Filter } from './filter.ts'
|
import { type Filter } from './filter.ts'
|
||||||
import { alwaysTrue } from './helpers.ts'
|
import { alwaysTrue } from './helpers.ts'
|
||||||
|
|
||||||
export type SubCloser = { close: () => void }
|
export type SubCloser = { close: () => void }
|
||||||
|
|
||||||
export type SubscribeManyParams = Omit<SubscriptionParams, 'onclose' | 'id'> & {
|
export type AbstractPoolConstructorOptions = AbstractRelayConstructorOptions & {}
|
||||||
|
|
||||||
|
export type SubscribeManyParams = Omit<SubscriptionParams, 'onclose'> & {
|
||||||
maxWait?: number
|
maxWait?: number
|
||||||
onclose?: (reasons: string[]) => void
|
onclose?: (reasons: string[]) => void
|
||||||
|
onauth?: (event: EventTemplate) => Promise<VerifiedEvent>
|
||||||
|
// Deprecated: use onauth instead
|
||||||
|
doauth?: (event: EventTemplate) => Promise<VerifiedEvent>
|
||||||
id?: string
|
id?: string
|
||||||
|
label?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export class AbstractSimplePool {
|
export class AbstractSimplePool {
|
||||||
private relays = new Map<string, AbstractRelay>()
|
protected relays: Map<string, AbstractRelay> = new Map()
|
||||||
public seenOn: Map<string, Set<AbstractRelay>> = new Map()
|
public seenOn: Map<string, Set<AbstractRelay>> = new Map()
|
||||||
public trackRelays: boolean = false
|
public trackRelays: boolean = false
|
||||||
|
|
||||||
public verifyEvent: Nostr['verifyEvent']
|
public verifyEvent: Nostr['verifyEvent']
|
||||||
public trustedRelayURLs: Set<string> = new Set()
|
public trustedRelayURLs: Set<string> = new Set()
|
||||||
|
|
||||||
constructor(opts: { verifyEvent: Nostr['verifyEvent'] }) {
|
private _WebSocket?: typeof WebSocket
|
||||||
|
|
||||||
|
constructor(opts: AbstractPoolConstructorOptions) {
|
||||||
this.verifyEvent = opts.verifyEvent
|
this.verifyEvent = opts.verifyEvent
|
||||||
|
this._WebSocket = opts.websocketImplementation
|
||||||
}
|
}
|
||||||
|
|
||||||
async ensureRelay(url: string, params?: { connectionTimeout?: number }): Promise<AbstractRelay> {
|
async ensureRelay(url: string, params?: { connectionTimeout?: number }): Promise<AbstractRelay> {
|
||||||
@@ -32,6 +48,7 @@ export class AbstractSimplePool {
|
|||||||
if (!relay) {
|
if (!relay) {
|
||||||
relay = new AbstractRelay(url, {
|
relay = new AbstractRelay(url, {
|
||||||
verifyEvent: this.trustedRelayURLs.has(url) ? alwaysTrue : this.verifyEvent,
|
verifyEvent: this.trustedRelayURLs.has(url) ? alwaysTrue : this.verifyEvent,
|
||||||
|
websocketImplementation: this._WebSocket,
|
||||||
})
|
})
|
||||||
if (params?.connectionTimeout) relay.connectionTimeout = params.connectionTimeout
|
if (params?.connectionTimeout) relay.connectionTimeout = params.connectionTimeout
|
||||||
this.relays.set(url, relay)
|
this.relays.set(url, relay)
|
||||||
@@ -47,7 +64,27 @@ export class AbstractSimplePool {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
subscribe(relays: string[], filter: Filter, params: SubscribeManyParams): SubCloser {
|
||||||
|
params.onauth = params.onauth || params.doauth
|
||||||
|
|
||||||
|
return this.subscribeMap(
|
||||||
|
relays.map(url => ({ url, filter })),
|
||||||
|
params,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
subscribeMany(relays: string[], filters: Filter[], params: SubscribeManyParams): SubCloser {
|
subscribeMany(relays: string[], filters: Filter[], params: SubscribeManyParams): SubCloser {
|
||||||
|
params.onauth = params.onauth || params.doauth
|
||||||
|
|
||||||
|
return this.subscribeMap(
|
||||||
|
relays.flatMap(url => filters.map(filter => ({ url, filter }))),
|
||||||
|
params,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
subscribeMap(requests: { url: string; filter: Filter }[], params: SubscribeManyParams): SubCloser {
|
||||||
|
params.onauth = params.onauth || params.doauth
|
||||||
|
|
||||||
if (this.trackRelays) {
|
if (this.trackRelays) {
|
||||||
params.receivedEvent = (relay: AbstractRelay, id: string) => {
|
params.receivedEvent = (relay: AbstractRelay, id: string) => {
|
||||||
let set = this.seenOn.get(id)
|
let set = this.seenOn.get(id)
|
||||||
@@ -65,8 +102,9 @@ export class AbstractSimplePool {
|
|||||||
// batch all EOSEs into a single
|
// batch all EOSEs into a single
|
||||||
const eosesReceived: boolean[] = []
|
const eosesReceived: boolean[] = []
|
||||||
let handleEose = (i: number) => {
|
let handleEose = (i: number) => {
|
||||||
|
if (eosesReceived[i]) return // do not act twice for the same relay
|
||||||
eosesReceived[i] = true
|
eosesReceived[i] = true
|
||||||
if (eosesReceived.filter(a => a).length === relays.length) {
|
if (eosesReceived.filter(a => a).length === requests.length) {
|
||||||
params.oneose?.()
|
params.oneose?.()
|
||||||
handleEose = () => {}
|
handleEose = () => {}
|
||||||
}
|
}
|
||||||
@@ -74,9 +112,10 @@ export class AbstractSimplePool {
|
|||||||
// batch all closes into a single
|
// batch all closes into a single
|
||||||
const closesReceived: string[] = []
|
const closesReceived: string[] = []
|
||||||
let handleClose = (i: number, reason: string) => {
|
let handleClose = (i: number, reason: string) => {
|
||||||
|
if (closesReceived[i]) return // do not act twice for the same relay
|
||||||
handleEose(i)
|
handleEose(i)
|
||||||
closesReceived[i] = reason
|
closesReceived[i] = reason
|
||||||
if (closesReceived.filter(a => a).length === relays.length) {
|
if (closesReceived.filter(a => a).length === requests.length) {
|
||||||
params.onclose?.(closesReceived)
|
params.onclose?.(closesReceived)
|
||||||
handleClose = () => {}
|
handleClose = () => {}
|
||||||
}
|
}
|
||||||
@@ -93,12 +132,8 @@ export class AbstractSimplePool {
|
|||||||
|
|
||||||
// open a subscription in all given relays
|
// open a subscription in all given relays
|
||||||
const allOpened = Promise.all(
|
const allOpened = Promise.all(
|
||||||
relays.map(normalizeURL).map(async (url, i, arr) => {
|
requests.map(async ({ url, filter }, i) => {
|
||||||
if (arr.indexOf(url) !== i) {
|
url = normalizeURL(url)
|
||||||
// duplicate
|
|
||||||
handleClose(i, 'duplicate url')
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let relay: AbstractRelay
|
let relay: AbstractRelay
|
||||||
try {
|
try {
|
||||||
@@ -110,10 +145,31 @@ export class AbstractSimplePool {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
let subscription = relay.subscribe(filters, {
|
let subscription = relay.subscribe([filter], {
|
||||||
...params,
|
...params,
|
||||||
oneose: () => handleEose(i),
|
oneose: () => handleEose(i),
|
||||||
onclose: reason => handleClose(i, reason),
|
onclose: reason => {
|
||||||
|
if (reason.startsWith('auth-required: ') && params.onauth) {
|
||||||
|
relay
|
||||||
|
.auth(params.onauth)
|
||||||
|
.then(() => {
|
||||||
|
relay.subscribe([filter], {
|
||||||
|
...params,
|
||||||
|
oneose: () => handleEose(i),
|
||||||
|
onclose: reason => {
|
||||||
|
handleClose(i, reason) // the second time we won't try to auth anymore
|
||||||
|
},
|
||||||
|
alreadyHaveEvent: localAlreadyHaveEventHandler,
|
||||||
|
eoseTimeout: params.maxWait,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
handleClose(i, `auth was required and attempted, but failed with: ${err}`)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
handleClose(i, reason)
|
||||||
|
}
|
||||||
|
},
|
||||||
alreadyHaveEvent: localAlreadyHaveEventHandler,
|
alreadyHaveEvent: localAlreadyHaveEventHandler,
|
||||||
eoseTimeout: params.maxWait,
|
eoseTimeout: params.maxWait,
|
||||||
})
|
})
|
||||||
@@ -132,11 +188,29 @@ export class AbstractSimplePool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
subscribeEose(
|
||||||
|
relays: string[],
|
||||||
|
filter: Filter,
|
||||||
|
params: Pick<SubscribeManyParams, 'label' | 'id' | 'onevent' | 'onclose' | 'maxWait' | 'onauth' | 'doauth'>,
|
||||||
|
): SubCloser {
|
||||||
|
params.onauth = params.onauth || params.doauth
|
||||||
|
|
||||||
|
const subcloser = this.subscribe(relays, filter, {
|
||||||
|
...params,
|
||||||
|
oneose() {
|
||||||
|
subcloser.close()
|
||||||
|
},
|
||||||
|
})
|
||||||
|
return subcloser
|
||||||
|
}
|
||||||
|
|
||||||
subscribeManyEose(
|
subscribeManyEose(
|
||||||
relays: string[],
|
relays: string[],
|
||||||
filters: Filter[],
|
filters: Filter[],
|
||||||
params: Pick<SubscribeManyParams, 'id' | 'onevent' | 'onclose' | 'maxWait'>,
|
params: Pick<SubscribeManyParams, 'label' | 'id' | 'onevent' | 'onclose' | 'maxWait' | 'onauth' | 'doauth'>,
|
||||||
): SubCloser {
|
): SubCloser {
|
||||||
|
params.onauth = params.onauth || params.doauth
|
||||||
|
|
||||||
const subcloser = this.subscribeMany(relays, filters, {
|
const subcloser = this.subscribeMany(relays, filters, {
|
||||||
...params,
|
...params,
|
||||||
oneose() {
|
oneose() {
|
||||||
@@ -149,11 +223,11 @@ export class AbstractSimplePool {
|
|||||||
async querySync(
|
async querySync(
|
||||||
relays: string[],
|
relays: string[],
|
||||||
filter: Filter,
|
filter: Filter,
|
||||||
params?: Pick<SubscribeManyParams, 'id' | 'maxWait'>,
|
params?: Pick<SubscribeManyParams, 'label' | 'id' | 'maxWait'>,
|
||||||
): Promise<Event[]> {
|
): Promise<Event[]> {
|
||||||
return new Promise(async resolve => {
|
return new Promise(async resolve => {
|
||||||
const events: Event[] = []
|
const events: Event[] = []
|
||||||
this.subscribeManyEose(relays, [filter], {
|
this.subscribeEose(relays, filter, {
|
||||||
...params,
|
...params,
|
||||||
onevent(event: Event) {
|
onevent(event: Event) {
|
||||||
events.push(event)
|
events.push(event)
|
||||||
@@ -168,7 +242,7 @@ export class AbstractSimplePool {
|
|||||||
async get(
|
async get(
|
||||||
relays: string[],
|
relays: string[],
|
||||||
filter: Filter,
|
filter: Filter,
|
||||||
params?: Pick<SubscribeManyParams, 'id' | 'maxWait'>,
|
params?: Pick<SubscribeManyParams, 'label' | 'id' | 'maxWait'>,
|
||||||
): Promise<Event | null> {
|
): Promise<Event | null> {
|
||||||
filter.limit = 1
|
filter.limit = 1
|
||||||
const events = await this.querySync(relays, filter, params)
|
const events = await this.querySync(relays, filter, params)
|
||||||
@@ -176,7 +250,11 @@ export class AbstractSimplePool {
|
|||||||
return events[0] || null
|
return events[0] || null
|
||||||
}
|
}
|
||||||
|
|
||||||
publish(relays: string[], event: Event): Promise<string>[] {
|
publish(
|
||||||
|
relays: string[],
|
||||||
|
event: Event,
|
||||||
|
options?: { onauth?: (evt: EventTemplate) => Promise<VerifiedEvent> },
|
||||||
|
): Promise<string>[] {
|
||||||
return relays.map(normalizeURL).map(async (url, i, arr) => {
|
return relays.map(normalizeURL).map(async (url, i, arr) => {
|
||||||
if (arr.indexOf(url) !== i) {
|
if (arr.indexOf(url) !== i) {
|
||||||
// duplicate
|
// duplicate
|
||||||
@@ -184,7 +262,38 @@ export class AbstractSimplePool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let r = await this.ensureRelay(url)
|
let r = await this.ensureRelay(url)
|
||||||
return r.publish(event)
|
return r
|
||||||
|
.publish(event)
|
||||||
|
.catch(async err => {
|
||||||
|
if (err instanceof Error && err.message.startsWith('auth-required: ') && options?.onauth) {
|
||||||
|
await r.auth(options.onauth)
|
||||||
|
return r.publish(event) // retry
|
||||||
|
}
|
||||||
|
throw err
|
||||||
|
})
|
||||||
|
.then(reason => {
|
||||||
|
if (this.trackRelays) {
|
||||||
|
let set = this.seenOn.get(event.id)
|
||||||
|
if (!set) {
|
||||||
|
set = new Set()
|
||||||
|
this.seenOn.set(event.id, set)
|
||||||
|
}
|
||||||
|
set.add(r)
|
||||||
|
}
|
||||||
|
return reason
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
listConnectionStatus(): Map<string, boolean> {
|
||||||
|
const map = new Map<string, boolean>()
|
||||||
|
this.relays.forEach((relay, url) => map.set(url, relay.connected))
|
||||||
|
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy(): void {
|
||||||
|
this.relays.forEach(conn => conn.close())
|
||||||
|
this.relays = new Map()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,22 @@
|
|||||||
/* global WebSocket */
|
/* global WebSocket */
|
||||||
|
|
||||||
import type { Event, EventTemplate, VerifiedEvent, Nostr } from './core.ts'
|
import type { Event, EventTemplate, VerifiedEvent, Nostr, NostrEvent } from './core.ts'
|
||||||
import { matchFilters, type Filter } from './filter.ts'
|
import { matchFilters, type Filter } from './filter.ts'
|
||||||
import { getHex64, getSubscriptionId } from './fakejson.ts'
|
import { getHex64, getSubscriptionId } from './fakejson.ts'
|
||||||
import { Queue, normalizeURL } from './utils.ts'
|
import { Queue, normalizeURL } from './utils.ts'
|
||||||
import { makeAuthEvent } from './nip42.ts'
|
import { makeAuthEvent } from './nip42.ts'
|
||||||
import { yieldThread } from './helpers.ts'
|
import { yieldThread } from './helpers.ts'
|
||||||
|
|
||||||
var _WebSocket: typeof WebSocket
|
export type AbstractRelayConstructorOptions = {
|
||||||
|
verifyEvent: Nostr['verifyEvent']
|
||||||
|
websocketImplementation?: typeof WebSocket
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
export class SendingOnClosedConnection extends Error {
|
||||||
_WebSocket = WebSocket
|
constructor(message: string, relay: string) {
|
||||||
} catch {}
|
super(`Tried to send message '${message} on a closed connection to ${relay}.`)
|
||||||
|
this.name = 'SendingOnClosedConnection'
|
||||||
export function useWebSocketImplementation(websocketImplementation: any) {
|
}
|
||||||
_WebSocket = websocketImplementation
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class AbstractRelay {
|
export class AbstractRelay {
|
||||||
@@ -29,6 +31,7 @@ export class AbstractRelay {
|
|||||||
|
|
||||||
public baseEoseTimeout: number = 4400
|
public baseEoseTimeout: number = 4400
|
||||||
public connectionTimeout: number = 4400
|
public connectionTimeout: number = 4400
|
||||||
|
public publishTimeout: number = 4400
|
||||||
public openSubs: Map<string, Subscription> = new Map()
|
public openSubs: Map<string, Subscription> = new Map()
|
||||||
private connectionTimeoutHandle: ReturnType<typeof setTimeout> | undefined
|
private connectionTimeoutHandle: ReturnType<typeof setTimeout> | undefined
|
||||||
|
|
||||||
@@ -39,15 +42,19 @@ export class AbstractRelay {
|
|||||||
private incomingMessageQueue = new Queue<string>()
|
private incomingMessageQueue = new Queue<string>()
|
||||||
private queueRunning = false
|
private queueRunning = false
|
||||||
private challenge: string | undefined
|
private challenge: string | undefined
|
||||||
|
private authPromise: Promise<string> | undefined
|
||||||
private serial: number = 0
|
private serial: number = 0
|
||||||
private verifyEvent: Nostr['verifyEvent']
|
private verifyEvent: Nostr['verifyEvent']
|
||||||
|
|
||||||
constructor(url: string, opts: { verifyEvent: Nostr['verifyEvent'] }) {
|
private _WebSocket: typeof WebSocket
|
||||||
|
|
||||||
|
constructor(url: string, opts: AbstractRelayConstructorOptions) {
|
||||||
this.url = normalizeURL(url)
|
this.url = normalizeURL(url)
|
||||||
this.verifyEvent = opts.verifyEvent
|
this.verifyEvent = opts.verifyEvent
|
||||||
|
this._WebSocket = opts.websocketImplementation || WebSocket
|
||||||
}
|
}
|
||||||
|
|
||||||
static async connect(url: string, opts: { verifyEvent: Nostr['verifyEvent'] }): Promise<AbstractRelay> {
|
static async connect(url: string, opts: AbstractRelayConstructorOptions): Promise<AbstractRelay> {
|
||||||
const relay = new AbstractRelay(url, opts)
|
const relay = new AbstractRelay(url, opts)
|
||||||
await relay.connect()
|
await relay.connect()
|
||||||
return relay
|
return relay
|
||||||
@@ -78,6 +85,7 @@ export class AbstractRelay {
|
|||||||
if (this.connectionPromise) return this.connectionPromise
|
if (this.connectionPromise) return this.connectionPromise
|
||||||
|
|
||||||
this.challenge = undefined
|
this.challenge = undefined
|
||||||
|
this.authPromise = undefined
|
||||||
this.connectionPromise = new Promise((resolve, reject) => {
|
this.connectionPromise = new Promise((resolve, reject) => {
|
||||||
this.connectionTimeoutHandle = setTimeout(() => {
|
this.connectionTimeoutHandle = setTimeout(() => {
|
||||||
reject('connection timed out')
|
reject('connection timed out')
|
||||||
@@ -87,8 +95,9 @@ export class AbstractRelay {
|
|||||||
}, this.connectionTimeout)
|
}, this.connectionTimeout)
|
||||||
|
|
||||||
try {
|
try {
|
||||||
this.ws = new _WebSocket(this.url)
|
this.ws = new this._WebSocket(this.url)
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
clearTimeout(this.connectionTimeoutHandle)
|
||||||
reject(err)
|
reject(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -100,7 +109,8 @@ export class AbstractRelay {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.ws.onerror = ev => {
|
this.ws.onerror = ev => {
|
||||||
reject((ev as any).message)
|
clearTimeout(this.connectionTimeoutHandle)
|
||||||
|
reject((ev as any).message || 'websocket error')
|
||||||
if (this._connected) {
|
if (this._connected) {
|
||||||
this._connected = false
|
this._connected = false
|
||||||
this.connectionPromise = undefined
|
this.connectionPromise = undefined
|
||||||
@@ -109,7 +119,9 @@ export class AbstractRelay {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.ws.onclose = async () => {
|
this.ws.onclose = ev => {
|
||||||
|
clearTimeout(this.connectionTimeoutHandle)
|
||||||
|
reject((ev as any).message || 'websocket closed')
|
||||||
if (this._connected) {
|
if (this._connected) {
|
||||||
this._connected = false
|
this._connected = false
|
||||||
this.connectionPromise = undefined
|
this.connectionPromise = undefined
|
||||||
@@ -173,7 +185,7 @@ export class AbstractRelay {
|
|||||||
switch (data[0]) {
|
switch (data[0]) {
|
||||||
case 'EVENT': {
|
case 'EVENT': {
|
||||||
const so = this.openSubs.get(data[1] as string) as Subscription
|
const so = this.openSubs.get(data[1] as string) as Subscription
|
||||||
const event = data[2] as Event
|
const event = data[2] as NostrEvent
|
||||||
if (this.verifyEvent(event) && matchFilters(so.filters, event)) {
|
if (this.verifyEvent(event) && matchFilters(so.filters, event)) {
|
||||||
so.onevent(event)
|
so.onevent(event)
|
||||||
}
|
}
|
||||||
@@ -200,9 +212,12 @@ export class AbstractRelay {
|
|||||||
const ok: boolean = data[2]
|
const ok: boolean = data[2]
|
||||||
const reason: string = data[3]
|
const reason: string = data[3]
|
||||||
const ep = this.openEventPublishes.get(id) as EventPublishResolver
|
const ep = this.openEventPublishes.get(id) as EventPublishResolver
|
||||||
if (ok) ep.resolve(reason)
|
if (ep) {
|
||||||
else ep.reject(new Error(reason))
|
clearTimeout(ep.timeout)
|
||||||
this.openEventPublishes.delete(id)
|
if (ok) ep.resolve(reason)
|
||||||
|
else ep.reject(new Error(reason))
|
||||||
|
this.openEventPublishes.delete(id)
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
case 'CLOSED': {
|
case 'CLOSED': {
|
||||||
@@ -228,7 +243,7 @@ export class AbstractRelay {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async send(message: string) {
|
public async send(message: string) {
|
||||||
if (!this.connectionPromise) throw new Error('sending on closed connection')
|
if (!this.connectionPromise) throw new SendingOnClosedConnection(message, this.url)
|
||||||
|
|
||||||
this.connectionPromise.then(() => {
|
this.connectionPromise.then(() => {
|
||||||
this.ws?.send(message)
|
this.ws?.send(message)
|
||||||
@@ -236,18 +251,35 @@ export class AbstractRelay {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async auth(signAuthEvent: (evt: EventTemplate) => Promise<VerifiedEvent>): Promise<string> {
|
public async auth(signAuthEvent: (evt: EventTemplate) => Promise<VerifiedEvent>): Promise<string> {
|
||||||
if (!this.challenge) throw new Error("can't perform auth, no challenge was received")
|
const challenge = this.challenge
|
||||||
const evt = await signAuthEvent(makeAuthEvent(this.url, this.challenge))
|
if (!challenge) throw new Error("can't perform auth, no challenge was received")
|
||||||
const ret = new Promise<string>((resolve, reject) => {
|
if (this.authPromise) return this.authPromise
|
||||||
this.openEventPublishes.set(evt.id, { resolve, reject })
|
|
||||||
|
this.authPromise = new Promise<string>(async (resolve, reject) => {
|
||||||
|
const evt = await signAuthEvent(makeAuthEvent(this.url, challenge))
|
||||||
|
const timeout = setTimeout(() => {
|
||||||
|
const ep = this.openEventPublishes.get(evt.id) as EventPublishResolver
|
||||||
|
if (ep) {
|
||||||
|
ep.reject(new Error('auth timed out'))
|
||||||
|
this.openEventPublishes.delete(evt.id)
|
||||||
|
}
|
||||||
|
}, this.publishTimeout)
|
||||||
|
this.openEventPublishes.set(evt.id, { resolve, reject, timeout })
|
||||||
|
this.send('["AUTH",' + JSON.stringify(evt) + ']')
|
||||||
})
|
})
|
||||||
this.send('["AUTH",' + JSON.stringify(evt) + ']')
|
return this.authPromise
|
||||||
return ret
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async publish(event: Event): Promise<string> {
|
public async publish(event: Event): Promise<string> {
|
||||||
const ret = new Promise<string>((resolve, reject) => {
|
const ret = new Promise<string>((resolve, reject) => {
|
||||||
this.openEventPublishes.set(event.id, { resolve, reject })
|
const timeout = setTimeout(() => {
|
||||||
|
const ep = this.openEventPublishes.get(event.id) as EventPublishResolver
|
||||||
|
if (ep) {
|
||||||
|
ep.reject(new Error('publish timed out'))
|
||||||
|
this.openEventPublishes.delete(event.id)
|
||||||
|
}
|
||||||
|
}, this.publishTimeout)
|
||||||
|
this.openEventPublishes.set(event.id, { resolve, reject, timeout })
|
||||||
})
|
})
|
||||||
this.send('["EVENT",' + JSON.stringify(event) + ']')
|
this.send('["EVENT",' + JSON.stringify(event) + ']')
|
||||||
return ret
|
return ret
|
||||||
@@ -259,19 +291,25 @@ export class AbstractRelay {
|
|||||||
const ret = new Promise<number>((resolve, reject) => {
|
const ret = new Promise<number>((resolve, reject) => {
|
||||||
this.openCountRequests.set(id, { resolve, reject })
|
this.openCountRequests.set(id, { resolve, reject })
|
||||||
})
|
})
|
||||||
this.send('["COUNT","' + id + '",' + JSON.stringify(filters) + ']')
|
this.send('["COUNT","' + id + '",' + JSON.stringify(filters).substring(1))
|
||||||
return ret
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
public subscribe(filters: Filter[], params: Partial<SubscriptionParams>): Subscription {
|
public subscribe(
|
||||||
|
filters: Filter[],
|
||||||
|
params: Partial<SubscriptionParams> & { label?: string; id?: string },
|
||||||
|
): Subscription {
|
||||||
const subscription = this.prepareSubscription(filters, params)
|
const subscription = this.prepareSubscription(filters, params)
|
||||||
subscription.fire()
|
subscription.fire()
|
||||||
return subscription
|
return subscription
|
||||||
}
|
}
|
||||||
|
|
||||||
public prepareSubscription(filters: Filter[], params: Partial<SubscriptionParams> & { id?: string }): Subscription {
|
public prepareSubscription(
|
||||||
|
filters: Filter[],
|
||||||
|
params: Partial<SubscriptionParams> & { label?: string; id?: string },
|
||||||
|
): Subscription {
|
||||||
this.serial++
|
this.serial++
|
||||||
const id = params.id || 'sub:' + this.serial
|
const id = params.id || (params.label ? params.label + ':' : 'sub:') + this.serial
|
||||||
const subscription = new Subscription(this, id, filters, params)
|
const subscription = new Subscription(this, id, filters, params)
|
||||||
this.openSubs.set(id, subscription)
|
this.openSubs.set(id, subscription)
|
||||||
return subscription
|
return subscription
|
||||||
@@ -348,7 +386,15 @@ export class Subscription {
|
|||||||
if (!this.closed && this.relay.connected) {
|
if (!this.closed && this.relay.connected) {
|
||||||
// if the connection was closed by the user calling .close() we will send a CLOSE message
|
// if the connection was closed by the user calling .close() we will send a CLOSE message
|
||||||
// otherwise this._open will be already set to false so we will skip this
|
// otherwise this._open will be already set to false so we will skip this
|
||||||
this.relay.send('["CLOSE",' + JSON.stringify(this.id) + ']')
|
try {
|
||||||
|
this.relay.send('["CLOSE",' + JSON.stringify(this.id) + ']')
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof SendingOnClosedConnection) {
|
||||||
|
/* doesn't matter, it's ok */
|
||||||
|
} else {
|
||||||
|
throw err
|
||||||
|
}
|
||||||
|
}
|
||||||
this.closed = true
|
this.closed = true
|
||||||
}
|
}
|
||||||
this.relay.openSubs.delete(this.id)
|
this.relay.openSubs.delete(this.id)
|
||||||
@@ -373,4 +419,5 @@ export type CountResolver = {
|
|||||||
export type EventPublishResolver = {
|
export type EventPublishResolver = {
|
||||||
resolve: (reason: string) => void
|
resolve: (reason: string) => void
|
||||||
reject: (err: Error) => void
|
reject: (err: Error) => void
|
||||||
|
timeout: ReturnType<typeof setTimeout>
|
||||||
}
|
}
|
||||||
|
|||||||
305
core.test.ts
305
core.test.ts
@@ -1,293 +1,18 @@
|
|||||||
import { describe, test, expect } from 'bun:test'
|
import { test, expect } from 'bun:test'
|
||||||
|
import { sortEvents } from './core.ts'
|
||||||
|
|
||||||
import {
|
test('sortEvents', () => {
|
||||||
finalizeEvent,
|
const events = [
|
||||||
serializeEvent,
|
{ id: 'abc123', pubkey: 'key1', created_at: 1610000000, kind: 1, tags: [], content: 'Hello', sig: 'sig1' },
|
||||||
getEventHash,
|
{ id: 'abc124', pubkey: 'key2', created_at: 1620000000, kind: 1, tags: [], content: 'World', sig: 'sig2' },
|
||||||
validateEvent,
|
{ id: 'abc125', pubkey: 'key3', created_at: 1620000000, kind: 1, tags: [], content: '!', sig: 'sig3' },
|
||||||
verifyEvent,
|
]
|
||||||
verifiedSymbol,
|
|
||||||
getPublicKey,
|
|
||||||
generateSecretKey,
|
|
||||||
} from './pure.ts'
|
|
||||||
import { ShortTextNote } from './kinds.ts'
|
|
||||||
import { bytesToHex, hexToBytes } from '@noble/hashes/utils'
|
|
||||||
|
|
||||||
test('private key generation', () => {
|
const sortedEvents = sortEvents(events)
|
||||||
expect(bytesToHex(generateSecretKey())).toMatch(/[a-f0-9]{64}/)
|
|
||||||
})
|
expect(sortedEvents).toEqual([
|
||||||
|
{ id: 'abc124', pubkey: 'key2', created_at: 1620000000, kind: 1, tags: [], content: 'World', sig: 'sig2' },
|
||||||
test('public key generation', () => {
|
{ id: 'abc125', pubkey: 'key3', created_at: 1620000000, kind: 1, tags: [], content: '!', sig: 'sig3' },
|
||||||
expect(getPublicKey(generateSecretKey())).toMatch(/[a-f0-9]{64}/)
|
{ id: 'abc123', pubkey: 'key1', created_at: 1610000000, kind: 1, tags: [], content: 'Hello', sig: 'sig1' },
|
||||||
})
|
])
|
||||||
|
|
||||||
test('public key from private key deterministic', () => {
|
|
||||||
let sk = generateSecretKey()
|
|
||||||
let pk = getPublicKey(sk)
|
|
||||||
|
|
||||||
for (let i = 0; i < 5; i++) {
|
|
||||||
expect(getPublicKey(sk)).toEqual(pk)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('finalizeEvent', () => {
|
|
||||||
test('should create a signed event from a template', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const template = {
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
}
|
|
||||||
|
|
||||||
const event = finalizeEvent(template, privateKey)
|
|
||||||
|
|
||||||
expect(event.kind).toEqual(template.kind)
|
|
||||||
expect(event.tags).toEqual(template.tags)
|
|
||||||
expect(event.content).toEqual(template.content)
|
|
||||||
expect(event.created_at).toEqual(template.created_at)
|
|
||||||
expect(event.pubkey).toEqual(publicKey)
|
|
||||||
expect(typeof event.id).toEqual('string')
|
|
||||||
expect(typeof event.sig).toEqual('string')
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('serializeEvent', () => {
|
|
||||||
test('should serialize a valid event object', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const unsignedEvent = {
|
|
||||||
pubkey: publicKey,
|
|
||||||
created_at: 1617932115,
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
}
|
|
||||||
|
|
||||||
const serializedEvent = serializeEvent(unsignedEvent)
|
|
||||||
|
|
||||||
expect(serializedEvent).toEqual(
|
|
||||||
JSON.stringify([
|
|
||||||
0,
|
|
||||||
publicKey,
|
|
||||||
unsignedEvent.created_at,
|
|
||||||
unsignedEvent.kind,
|
|
||||||
unsignedEvent.tags,
|
|
||||||
unsignedEvent.content,
|
|
||||||
]),
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should throw an error for an invalid event object', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const invalidEvent = {
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
created_at: 1617932115,
|
|
||||||
pubkey: publicKey, // missing content
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(() => {
|
|
||||||
// @ts-expect-error
|
|
||||||
serializeEvent(invalidEvent)
|
|
||||||
}).toThrow("can't serialize event with wrong or missing properties")
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('getEventHash', () => {
|
|
||||||
test('should return the correct event hash', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const unsignedEvent = {
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
pubkey: publicKey,
|
|
||||||
}
|
|
||||||
|
|
||||||
const eventHash = getEventHash(unsignedEvent)
|
|
||||||
|
|
||||||
expect(typeof eventHash).toEqual('string')
|
|
||||||
expect(eventHash.length).toEqual(64)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('validateEvent', () => {
|
|
||||||
test('should return true for a valid event object', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const unsignedEvent = {
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
pubkey: publicKey,
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValid = validateEvent(unsignedEvent)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for a non object event', () => {
|
|
||||||
const nonObjectEvent = ''
|
|
||||||
const isValid = validateEvent(nonObjectEvent)
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an event object with missing properties', () => {
|
|
||||||
const invalidEvent = {
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
created_at: 1617932115, // missing content and pubkey
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValid = validateEvent(invalidEvent)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an empty object', () => {
|
|
||||||
const emptyObj = {}
|
|
||||||
|
|
||||||
const isValid = validateEvent(emptyObj)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an object with invalid properties', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const invalidEvent = {
|
|
||||||
kind: 1,
|
|
||||||
tags: [],
|
|
||||||
created_at: '1617932115', // should be a number
|
|
||||||
pubkey: publicKey,
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValid = validateEvent(invalidEvent)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an object with an invalid public key', () => {
|
|
||||||
const invalidEvent = {
|
|
||||||
kind: 1,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
pubkey: 'invalid_pubkey',
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValid = validateEvent(invalidEvent)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an object with invalid tags', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const publicKey = getPublicKey(privateKey)
|
|
||||||
|
|
||||||
const invalidEvent = {
|
|
||||||
kind: 1,
|
|
||||||
tags: {}, // should be an array
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
pubkey: publicKey,
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValid = validateEvent(invalidEvent)
|
|
||||||
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('verifyEvent', () => {
|
|
||||||
test('should return true for a valid event signature', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const event = finalizeEvent(
|
|
||||||
{
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
},
|
|
||||||
privateKey,
|
|
||||||
)
|
|
||||||
|
|
||||||
const isValid = verifyEvent(event)
|
|
||||||
expect(isValid).toEqual(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an invalid event signature', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
|
||||||
{
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
},
|
|
||||||
privateKey,
|
|
||||||
)
|
|
||||||
|
|
||||||
// tamper with the signature
|
|
||||||
event.sig = event.sig.replace(/^.{3}/g, '666')
|
|
||||||
|
|
||||||
const isValid = verifyEvent(event)
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false when verifying an event with a different private key', () => {
|
|
||||||
const privateKey1 = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
|
|
||||||
const privateKey2 = hexToBytes('5b4a34f4e4b23c63ad55a35e3f84a3b53d96dbf266edf521a8358f71d19cbf67')
|
|
||||||
const publicKey2 = getPublicKey(privateKey2)
|
|
||||||
|
|
||||||
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
|
||||||
{
|
|
||||||
kind: ShortTextNote,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
},
|
|
||||||
privateKey1,
|
|
||||||
)
|
|
||||||
|
|
||||||
// verify with different private key
|
|
||||||
const isValid = verifyEvent({
|
|
||||||
...event,
|
|
||||||
pubkey: publicKey2,
|
|
||||||
})
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false for an invalid event id', () => {
|
|
||||||
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
|
||||||
|
|
||||||
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
|
||||||
{
|
|
||||||
kind: 1,
|
|
||||||
tags: [],
|
|
||||||
content: 'Hello, world!',
|
|
||||||
created_at: 1617932115,
|
|
||||||
},
|
|
||||||
privateKey,
|
|
||||||
)
|
|
||||||
|
|
||||||
// tamper with the id
|
|
||||||
event.id = event.id.replace(/^.{3}/g, '666')
|
|
||||||
|
|
||||||
const isValid = verifyEvent(event)
|
|
||||||
expect(isValid).toEqual(false)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|||||||
16
core.ts
16
core.ts
@@ -43,9 +43,23 @@ export function validateEvent<T>(event: T): event is T & UnsignedEvent {
|
|||||||
let tag = event.tags[i]
|
let tag = event.tags[i]
|
||||||
if (!Array.isArray(tag)) return false
|
if (!Array.isArray(tag)) return false
|
||||||
for (let j = 0; j < tag.length; j++) {
|
for (let j = 0; j < tag.length; j++) {
|
||||||
if (typeof tag[j] === 'object') return false
|
if (typeof tag[j] !== 'string') return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort events in reverse-chronological order by the `created_at` timestamp,
|
||||||
|
* and then by the event `id` (lexicographically) in case of ties.
|
||||||
|
* This mutates the array.
|
||||||
|
*/
|
||||||
|
export function sortEvents(events: Event[]): Event[] {
|
||||||
|
return events.sort((a: NostrEvent, b: NostrEvent): number => {
|
||||||
|
if (a.created_at !== b.created_at) {
|
||||||
|
return b.created_at - a.created_at
|
||||||
|
}
|
||||||
|
return a.id.localeCompare(b.id)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ describe('Filter', () => {
|
|||||||
until: 200,
|
until: 200,
|
||||||
'#tag': ['value'],
|
'#tag': ['value'],
|
||||||
}
|
}
|
||||||
|
|
||||||
const event = buildEvent({
|
const event = buildEvent({
|
||||||
id: '123',
|
id: '123',
|
||||||
kind: 1,
|
kind: 1,
|
||||||
@@ -21,39 +20,21 @@ describe('Filter', () => {
|
|||||||
created_at: 150,
|
created_at: 150,
|
||||||
tags: [['tag', 'value']],
|
tags: [['tag', 'value']],
|
||||||
})
|
})
|
||||||
|
|
||||||
const result = matchFilter(filter, event)
|
const result = matchFilter(filter, event)
|
||||||
|
|
||||||
expect(result).toEqual(true)
|
expect(result).toEqual(true)
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should return false when the event id is not in the filter', () => {
|
test('should return false when the event id is not in the filter', () => {
|
||||||
const filter = { ids: ['123', '456'] }
|
const filter = { ids: ['123', '456'] }
|
||||||
|
|
||||||
const event = buildEvent({ id: '789' })
|
const event = buildEvent({ id: '789' })
|
||||||
|
|
||||||
const result = matchFilter(filter, event)
|
const result = matchFilter(filter, event)
|
||||||
|
|
||||||
expect(result).toEqual(false)
|
expect(result).toEqual(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should return true when the event id starts with a prefix', () => {
|
|
||||||
const filter = { ids: ['22', '00'] }
|
|
||||||
|
|
||||||
const event = buildEvent({ id: '001' })
|
|
||||||
|
|
||||||
const result = matchFilter(filter, event)
|
|
||||||
|
|
||||||
expect(result).toEqual(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return false when the event kind is not in the filter', () => {
|
test('should return false when the event kind is not in the filter', () => {
|
||||||
const filter = { kinds: [1, 2, 3] }
|
const filter = { kinds: [1, 2, 3] }
|
||||||
|
|
||||||
const event = buildEvent({ kind: 4 })
|
const event = buildEvent({ kind: 4 })
|
||||||
|
|
||||||
const result = matchFilter(filter, event)
|
const result = matchFilter(filter, event)
|
||||||
|
|
||||||
expect(result).toEqual(false)
|
expect(result).toEqual(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -154,25 +135,8 @@ describe('Filter', () => {
|
|||||||
{ ids: ['456'], kinds: [2], authors: ['def'] },
|
{ ids: ['456'], kinds: [2], authors: ['def'] },
|
||||||
{ ids: ['789'], kinds: [3], authors: ['ghi'] },
|
{ ids: ['789'], kinds: [3], authors: ['ghi'] },
|
||||||
]
|
]
|
||||||
|
|
||||||
const event = buildEvent({ id: '789', kind: 3, pubkey: 'ghi' })
|
const event = buildEvent({ id: '789', kind: 3, pubkey: 'ghi' })
|
||||||
|
|
||||||
const result = matchFilters(filters, event)
|
const result = matchFilters(filters, event)
|
||||||
|
|
||||||
expect(result).toEqual(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('should return true when at least one prefix matches the event', () => {
|
|
||||||
const filters = [
|
|
||||||
{ ids: ['1'], kinds: [1], authors: ['a'] },
|
|
||||||
{ ids: ['4'], kinds: [2], authors: ['d'] },
|
|
||||||
{ ids: ['9'], kinds: [3], authors: ['g'] },
|
|
||||||
]
|
|
||||||
|
|
||||||
const event = buildEvent({ id: '987', kind: 3, pubkey: 'ghi' })
|
|
||||||
|
|
||||||
const result = matchFilters(filters, event)
|
|
||||||
|
|
||||||
expect(result).toEqual(true)
|
expect(result).toEqual(true)
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -201,11 +165,8 @@ describe('Filter', () => {
|
|||||||
{ ids: ['456'], kinds: [2], authors: ['def'] },
|
{ ids: ['456'], kinds: [2], authors: ['def'] },
|
||||||
{ ids: ['789'], kinds: [3], authors: ['ghi'] },
|
{ ids: ['789'], kinds: [3], authors: ['ghi'] },
|
||||||
]
|
]
|
||||||
|
|
||||||
const event = buildEvent({ id: '100', kind: 4, pubkey: 'jkl' })
|
const event = buildEvent({ id: '100', kind: 4, pubkey: 'jkl' })
|
||||||
|
|
||||||
const result = matchFilters(filters, event)
|
const result = matchFilters(filters, event)
|
||||||
|
|
||||||
expect(result).toEqual(false)
|
expect(result).toEqual(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -221,9 +182,7 @@ describe('Filter', () => {
|
|||||||
pubkey: 'def',
|
pubkey: 'def',
|
||||||
created_at: 200,
|
created_at: 200,
|
||||||
})
|
})
|
||||||
|
|
||||||
const result = matchFilters(filters, event)
|
const result = matchFilters(filters, event)
|
||||||
|
|
||||||
expect(result).toEqual(false)
|
expect(result).toEqual(false)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@@ -256,6 +215,16 @@ describe('Filter', () => {
|
|||||||
expect(getFilterLimit({ kinds: [0, 3], authors: ['alex', 'fiatjaf'] })).toEqual(4)
|
expect(getFilterLimit({ kinds: [0, 3], authors: ['alex', 'fiatjaf'] })).toEqual(4)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('should handle parameterized replaceable events', () => {
|
||||||
|
expect(getFilterLimit({ kinds: [30078], authors: ['alex'] })).toEqual(Infinity)
|
||||||
|
expect(getFilterLimit({ kinds: [30078], authors: ['alex'], '#d': ['ditto'] })).toEqual(1)
|
||||||
|
expect(getFilterLimit({ kinds: [30078], authors: ['alex'], '#d': ['ditto', 'soapbox'] })).toEqual(2)
|
||||||
|
expect(getFilterLimit({ kinds: [30078], authors: ['alex', 'fiatjaf'], '#d': ['ditto', 'soapbox'] })).toEqual(4)
|
||||||
|
expect(
|
||||||
|
getFilterLimit({ kinds: [30000, 30078], authors: ['alex', 'fiatjaf'], '#d': ['ditto', 'soapbox'] }),
|
||||||
|
).toEqual(8)
|
||||||
|
})
|
||||||
|
|
||||||
test('should return Infinity for authors with regular kinds', () => {
|
test('should return Infinity for authors with regular kinds', () => {
|
||||||
expect(getFilterLimit({ kinds: [1], authors: ['alex'] })).toEqual(Infinity)
|
expect(getFilterLimit({ kinds: [1], authors: ['alex'] })).toEqual(Infinity)
|
||||||
})
|
})
|
||||||
@@ -263,5 +232,9 @@ describe('Filter', () => {
|
|||||||
test('should return Infinity for empty filters', () => {
|
test('should return Infinity for empty filters', () => {
|
||||||
expect(getFilterLimit({})).toEqual(Infinity)
|
expect(getFilterLimit({})).toEqual(Infinity)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('empty tags return 0', () => {
|
||||||
|
expect(getFilterLimit({ '#p': [] })).toEqual(0)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
37
filter.ts
37
filter.ts
@@ -1,5 +1,5 @@
|
|||||||
import { Event } from './core.ts'
|
import { Event } from './core.ts'
|
||||||
import { isReplaceableKind } from './kinds.ts'
|
import { isAddressableKind, isReplaceableKind } from './kinds.ts'
|
||||||
|
|
||||||
export type Filter = {
|
export type Filter = {
|
||||||
ids?: string[]
|
ids?: string[]
|
||||||
@@ -14,15 +14,13 @@ export type Filter = {
|
|||||||
|
|
||||||
export function matchFilter(filter: Filter, event: Event): boolean {
|
export function matchFilter(filter: Filter, event: Event): boolean {
|
||||||
if (filter.ids && filter.ids.indexOf(event.id) === -1) {
|
if (filter.ids && filter.ids.indexOf(event.id) === -1) {
|
||||||
if (!filter.ids.some(prefix => event.id.startsWith(prefix))) {
|
return false
|
||||||
return false
|
}
|
||||||
}
|
if (filter.kinds && filter.kinds.indexOf(event.kind) === -1) {
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
if (filter.kinds && filter.kinds.indexOf(event.kind) === -1) return false
|
|
||||||
if (filter.authors && filter.authors.indexOf(event.pubkey) === -1) {
|
if (filter.authors && filter.authors.indexOf(event.pubkey) === -1) {
|
||||||
if (!filter.authors.some(prefix => event.pubkey.startsWith(prefix))) {
|
return false
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let f in filter) {
|
for (let f in filter) {
|
||||||
@@ -41,7 +39,9 @@ export function matchFilter(filter: Filter, event: Event): boolean {
|
|||||||
|
|
||||||
export function matchFilters(filters: Filter[], event: Event): boolean {
|
export function matchFilters(filters: Filter[], event: Event): boolean {
|
||||||
for (let i = 0; i < filters.length; i++) {
|
for (let i = 0; i < filters.length; i++) {
|
||||||
if (matchFilter(filters[i], event)) return true
|
if (matchFilter(filters[i], event)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
@@ -72,17 +72,34 @@ export function mergeFilters(...filters: Filter[]): Filter {
|
|||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Calculate the intrinsic limit of a filter. This function may return `Infinity`. */
|
/**
|
||||||
|
* Calculate the intrinsic limit of a filter.
|
||||||
|
* This function returns a positive integer, or `Infinity` if there is no intrinsic limit.
|
||||||
|
*/
|
||||||
export function getFilterLimit(filter: Filter): number {
|
export function getFilterLimit(filter: Filter): number {
|
||||||
if (filter.ids && !filter.ids.length) return 0
|
if (filter.ids && !filter.ids.length) return 0
|
||||||
if (filter.kinds && !filter.kinds.length) return 0
|
if (filter.kinds && !filter.kinds.length) return 0
|
||||||
if (filter.authors && !filter.authors.length) return 0
|
if (filter.authors && !filter.authors.length) return 0
|
||||||
|
|
||||||
|
for (const [key, value] of Object.entries(filter)) {
|
||||||
|
if (key[0] === '#' && Array.isArray(value) && !value.length) return 0
|
||||||
|
}
|
||||||
|
|
||||||
return Math.min(
|
return Math.min(
|
||||||
|
// The `limit` property creates an artificial limit.
|
||||||
Math.max(0, filter.limit ?? Infinity),
|
Math.max(0, filter.limit ?? Infinity),
|
||||||
|
|
||||||
|
// There can only be one event per `id`.
|
||||||
filter.ids?.length ?? Infinity,
|
filter.ids?.length ?? Infinity,
|
||||||
|
|
||||||
|
// Replaceable events are limited by the number of authors and kinds.
|
||||||
filter.authors?.length && filter.kinds?.every(kind => isReplaceableKind(kind))
|
filter.authors?.length && filter.kinds?.every(kind => isReplaceableKind(kind))
|
||||||
? filter.authors.length * filter.kinds.length
|
? filter.authors.length * filter.kinds.length
|
||||||
: Infinity,
|
: Infinity,
|
||||||
|
|
||||||
|
// Parameterized replaceable events are limited by the number of authors, kinds, and "d" tags.
|
||||||
|
filter.authors?.length && filter.kinds?.every(kind => isAddressableKind(kind)) && filter['#d']?.length
|
||||||
|
? filter.authors.length * filter.kinds.length * filter['#d'].length
|
||||||
|
: Infinity,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
7
index.ts
7
index.ts
@@ -1,7 +1,7 @@
|
|||||||
export * from './pure.ts'
|
export * from './pure.ts'
|
||||||
export * from './relay.ts'
|
export { Relay } from './relay.ts'
|
||||||
export * from './filter.ts'
|
export * from './filter.ts'
|
||||||
export * from './pool.ts'
|
export { SimplePool } from './pool.ts'
|
||||||
export * from './references.ts'
|
export * from './references.ts'
|
||||||
|
|
||||||
export * as nip04 from './nip04.ts'
|
export * as nip04 from './nip04.ts'
|
||||||
@@ -9,6 +9,7 @@ export * as nip05 from './nip05.ts'
|
|||||||
export * as nip10 from './nip10.ts'
|
export * as nip10 from './nip10.ts'
|
||||||
export * as nip11 from './nip11.ts'
|
export * as nip11 from './nip11.ts'
|
||||||
export * as nip13 from './nip13.ts'
|
export * as nip13 from './nip13.ts'
|
||||||
|
export * as nip17 from './nip17.ts'
|
||||||
export * as nip18 from './nip18.ts'
|
export * as nip18 from './nip18.ts'
|
||||||
export * as nip19 from './nip19.ts'
|
export * as nip19 from './nip19.ts'
|
||||||
export * as nip21 from './nip21.ts'
|
export * as nip21 from './nip21.ts'
|
||||||
@@ -20,7 +21,9 @@ export * as nip39 from './nip39.ts'
|
|||||||
export * as nip42 from './nip42.ts'
|
export * as nip42 from './nip42.ts'
|
||||||
export * as nip44 from './nip44.ts'
|
export * as nip44 from './nip44.ts'
|
||||||
export * as nip47 from './nip47.ts'
|
export * as nip47 from './nip47.ts'
|
||||||
|
export * as nip54 from './nip54.ts'
|
||||||
export * as nip57 from './nip57.ts'
|
export * as nip57 from './nip57.ts'
|
||||||
|
export * as nip59 from './nip59.ts'
|
||||||
export * as nip98 from './nip98.ts'
|
export * as nip98 from './nip98.ts'
|
||||||
|
|
||||||
export * as kinds from './kinds.ts'
|
export * as kinds from './kinds.ts'
|
||||||
|
|||||||
12
jsr.json
12
jsr.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@nostr/tools",
|
"name": "@nostr/tools",
|
||||||
"version": "2.3.2",
|
"version": "2.14.3",
|
||||||
"exports": {
|
"exports": {
|
||||||
".": "./index.ts",
|
".": "./index.ts",
|
||||||
"./core": "./core.ts",
|
"./core": "./core.ts",
|
||||||
@@ -16,9 +16,11 @@
|
|||||||
"./nip04": "./nip04.ts",
|
"./nip04": "./nip04.ts",
|
||||||
"./nip05": "./nip05.ts",
|
"./nip05": "./nip05.ts",
|
||||||
"./nip06": "./nip06.ts",
|
"./nip06": "./nip06.ts",
|
||||||
|
"./nip07": "./nip07.ts",
|
||||||
"./nip10": "./nip10.ts",
|
"./nip10": "./nip10.ts",
|
||||||
"./nip11": "./nip11.ts",
|
"./nip11": "./nip11.ts",
|
||||||
"./nip13": "./nip13.ts",
|
"./nip13": "./nip13.ts",
|
||||||
|
"./nip17": "./nip17.ts",
|
||||||
"./nip18": "./nip18.ts",
|
"./nip18": "./nip18.ts",
|
||||||
"./nip19": "./nip19.ts",
|
"./nip19": "./nip19.ts",
|
||||||
"./nip21": "./nip21.ts",
|
"./nip21": "./nip21.ts",
|
||||||
@@ -32,13 +34,17 @@
|
|||||||
"./nip44": "./nip44.ts",
|
"./nip44": "./nip44.ts",
|
||||||
"./nip46": "./nip46.ts",
|
"./nip46": "./nip46.ts",
|
||||||
"./nip49": "./nip49.ts",
|
"./nip49": "./nip49.ts",
|
||||||
|
"./nip54": "./nip54.ts",
|
||||||
"./nip57": "./nip57.ts",
|
"./nip57": "./nip57.ts",
|
||||||
|
"./nip58": "./nip58.ts",
|
||||||
|
"./nip59": "./nip59.ts",
|
||||||
"./nip75": "./nip75.ts",
|
"./nip75": "./nip75.ts",
|
||||||
"./nip94": "./nip94.ts",
|
"./nip94": "./nip94.ts",
|
||||||
"./nip96": "./nip96.ts",
|
|
||||||
"./nip98": "./nip98.ts",
|
"./nip98": "./nip98.ts",
|
||||||
"./nip99": "./nip99.ts",
|
"./nip99": "./nip99.ts",
|
||||||
|
"./nipb7": "./nipb7.ts",
|
||||||
"./fakejson": "./fakejson.ts",
|
"./fakejson": "./fakejson.ts",
|
||||||
"./utils": "./utils.ts"
|
"./utils": "./utils.ts"
|
||||||
|
"./signer": "./signer.ts"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
6
justfile
6
justfile
@@ -12,6 +12,12 @@ test-only file:
|
|||||||
bun test {{file}}
|
bun test {{file}}
|
||||||
|
|
||||||
publish: build
|
publish: build
|
||||||
|
# publish to jsr first because it is more strict and will catch some errors
|
||||||
|
perl -i -0pe "s/},\n \"optionalDependencies\": {\n/,/" package.json
|
||||||
|
jsr publish --allow-dirty
|
||||||
|
git checkout -- package.json
|
||||||
|
|
||||||
|
# then to npm
|
||||||
npm publish
|
npm publish
|
||||||
|
|
||||||
format:
|
format:
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { expect, test } from 'bun:test'
|
||||||
import { classifyKind } from './kinds.ts'
|
import { classifyKind, isKind, Repost, ShortTextNote } from './kinds.ts'
|
||||||
|
import { finalizeEvent, generateSecretKey } from './pure.ts'
|
||||||
|
|
||||||
test('kind classification', () => {
|
test('kind classification', () => {
|
||||||
expect(classifyKind(1)).toBe('regular')
|
expect(classifyKind(1)).toBe('regular')
|
||||||
@@ -19,3 +20,22 @@ test('kind classification', () => {
|
|||||||
expect(classifyKind(40000)).toBe('unknown')
|
expect(classifyKind(40000)).toBe('unknown')
|
||||||
expect(classifyKind(255)).toBe('unknown')
|
expect(classifyKind(255)).toBe('unknown')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('kind type guard', () => {
|
||||||
|
const privateKey = generateSecretKey()
|
||||||
|
const repostedEvent = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [
|
||||||
|
['e', 'replied event id'],
|
||||||
|
['p', 'replied event pubkey'],
|
||||||
|
],
|
||||||
|
content: 'Replied to a post',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(isKind(repostedEvent, ShortTextNote)).toBeTrue()
|
||||||
|
expect(isKind(repostedEvent, Repost)).toBeFalse()
|
||||||
|
})
|
||||||
|
|||||||
95
kinds.ts
95
kinds.ts
@@ -1,3 +1,5 @@
|
|||||||
|
import { NostrEvent, validateEvent } from './pure.ts'
|
||||||
|
|
||||||
/** Events are **regular**, which means they're all expected to be stored by relays. */
|
/** Events are **regular**, which means they're all expected to be stored by relays. */
|
||||||
export function isRegularKind(kind: number): boolean {
|
export function isRegularKind(kind: number): boolean {
|
||||||
return (1000 <= kind && kind < 10000) || [1, 2, 4, 5, 6, 7, 8, 16, 40, 41, 42, 43, 44].includes(kind)
|
return (1000 <= kind && kind < 10000) || [1, 2, 4, 5, 6, 7, 8, 16, 40, 41, 42, 43, 44].includes(kind)
|
||||||
@@ -13,8 +15,8 @@ export function isEphemeralKind(kind: number): boolean {
|
|||||||
return 20000 <= kind && kind < 30000
|
return 20000 <= kind && kind < 30000
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Events are **parameterized replaceable**, which means that, for each combination of `pubkey`, `kind` and the `d` tag, only the latest event is expected to be stored by relays, older versions are expected to be discarded. */
|
/** Events are **addressable**, which means that, for each combination of `pubkey`, `kind` and the `d` tag, only the latest event is expected to be stored by relays, older versions are expected to be discarded. */
|
||||||
export function isParameterizedReplaceableKind(kind: number): boolean {
|
export function isAddressableKind(kind: number): boolean {
|
||||||
return 30000 <= kind && kind < 40000
|
return 30000 <= kind && kind < 40000
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,81 +28,166 @@ export function classifyKind(kind: number): KindClassification {
|
|||||||
if (isRegularKind(kind)) return 'regular'
|
if (isRegularKind(kind)) return 'regular'
|
||||||
if (isReplaceableKind(kind)) return 'replaceable'
|
if (isReplaceableKind(kind)) return 'replaceable'
|
||||||
if (isEphemeralKind(kind)) return 'ephemeral'
|
if (isEphemeralKind(kind)) return 'ephemeral'
|
||||||
if (isParameterizedReplaceableKind(kind)) return 'parameterized'
|
if (isAddressableKind(kind)) return 'parameterized'
|
||||||
return 'unknown'
|
return 'unknown'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function isKind<T extends number>(event: unknown, kind: T | Array<T>): event is NostrEvent & { kind: T } {
|
||||||
|
const kindAsArray: number[] = kind instanceof Array ? kind : [kind]
|
||||||
|
return (validateEvent(event) && kindAsArray.includes(event.kind)) || false
|
||||||
|
}
|
||||||
|
|
||||||
export const Metadata = 0
|
export const Metadata = 0
|
||||||
|
export type Metadata = typeof Metadata
|
||||||
export const ShortTextNote = 1
|
export const ShortTextNote = 1
|
||||||
|
export type ShortTextNote = typeof ShortTextNote
|
||||||
export const RecommendRelay = 2
|
export const RecommendRelay = 2
|
||||||
|
export type RecommendRelay = typeof RecommendRelay
|
||||||
export const Contacts = 3
|
export const Contacts = 3
|
||||||
|
export type Contacts = typeof Contacts
|
||||||
export const EncryptedDirectMessage = 4
|
export const EncryptedDirectMessage = 4
|
||||||
export const EncryptedDirectMessages = 4
|
export type EncryptedDirectMessage = typeof EncryptedDirectMessage
|
||||||
export const EventDeletion = 5
|
export const EventDeletion = 5
|
||||||
|
export type EventDeletion = typeof EventDeletion
|
||||||
export const Repost = 6
|
export const Repost = 6
|
||||||
|
export type Repost = typeof Repost
|
||||||
export const Reaction = 7
|
export const Reaction = 7
|
||||||
|
export type Reaction = typeof Reaction
|
||||||
export const BadgeAward = 8
|
export const BadgeAward = 8
|
||||||
|
export type BadgeAward = typeof BadgeAward
|
||||||
|
export const Seal = 13
|
||||||
|
export type Seal = typeof Seal
|
||||||
|
export const PrivateDirectMessage = 14
|
||||||
|
export type PrivateDirectMessage = typeof PrivateDirectMessage
|
||||||
export const GenericRepost = 16
|
export const GenericRepost = 16
|
||||||
|
export type GenericRepost = typeof GenericRepost
|
||||||
export const ChannelCreation = 40
|
export const ChannelCreation = 40
|
||||||
|
export type ChannelCreation = typeof ChannelCreation
|
||||||
export const ChannelMetadata = 41
|
export const ChannelMetadata = 41
|
||||||
|
export type ChannelMetadata = typeof ChannelMetadata
|
||||||
export const ChannelMessage = 42
|
export const ChannelMessage = 42
|
||||||
|
export type ChannelMessage = typeof ChannelMessage
|
||||||
export const ChannelHideMessage = 43
|
export const ChannelHideMessage = 43
|
||||||
|
export type ChannelHideMessage = typeof ChannelHideMessage
|
||||||
export const ChannelMuteUser = 44
|
export const ChannelMuteUser = 44
|
||||||
|
export type ChannelMuteUser = typeof ChannelMuteUser
|
||||||
export const OpenTimestamps = 1040
|
export const OpenTimestamps = 1040
|
||||||
|
export type OpenTimestamps = typeof OpenTimestamps
|
||||||
|
export const GiftWrap = 1059
|
||||||
|
export type GiftWrap = typeof GiftWrap
|
||||||
export const FileMetadata = 1063
|
export const FileMetadata = 1063
|
||||||
|
export type FileMetadata = typeof FileMetadata
|
||||||
export const LiveChatMessage = 1311
|
export const LiveChatMessage = 1311
|
||||||
|
export type LiveChatMessage = typeof LiveChatMessage
|
||||||
export const ProblemTracker = 1971
|
export const ProblemTracker = 1971
|
||||||
|
export type ProblemTracker = typeof ProblemTracker
|
||||||
export const Report = 1984
|
export const Report = 1984
|
||||||
|
export type Report = typeof Report
|
||||||
export const Reporting = 1984
|
export const Reporting = 1984
|
||||||
|
export type Reporting = typeof Reporting
|
||||||
export const Label = 1985
|
export const Label = 1985
|
||||||
|
export type Label = typeof Label
|
||||||
export const CommunityPostApproval = 4550
|
export const CommunityPostApproval = 4550
|
||||||
|
export type CommunityPostApproval = typeof CommunityPostApproval
|
||||||
export const JobRequest = 5999
|
export const JobRequest = 5999
|
||||||
|
export type JobRequest = typeof JobRequest
|
||||||
export const JobResult = 6999
|
export const JobResult = 6999
|
||||||
|
export type JobResult = typeof JobResult
|
||||||
export const JobFeedback = 7000
|
export const JobFeedback = 7000
|
||||||
|
export type JobFeedback = typeof JobFeedback
|
||||||
export const ZapGoal = 9041
|
export const ZapGoal = 9041
|
||||||
|
export type ZapGoal = typeof ZapGoal
|
||||||
export const ZapRequest = 9734
|
export const ZapRequest = 9734
|
||||||
|
export type ZapRequest = typeof ZapRequest
|
||||||
export const Zap = 9735
|
export const Zap = 9735
|
||||||
|
export type Zap = typeof Zap
|
||||||
export const Highlights = 9802
|
export const Highlights = 9802
|
||||||
|
export type Highlights = typeof Highlights
|
||||||
export const Mutelist = 10000
|
export const Mutelist = 10000
|
||||||
|
export type Mutelist = typeof Mutelist
|
||||||
export const Pinlist = 10001
|
export const Pinlist = 10001
|
||||||
|
export type Pinlist = typeof Pinlist
|
||||||
export const RelayList = 10002
|
export const RelayList = 10002
|
||||||
|
export type RelayList = typeof RelayList
|
||||||
export const BookmarkList = 10003
|
export const BookmarkList = 10003
|
||||||
|
export type BookmarkList = typeof BookmarkList
|
||||||
export const CommunitiesList = 10004
|
export const CommunitiesList = 10004
|
||||||
|
export type CommunitiesList = typeof CommunitiesList
|
||||||
export const PublicChatsList = 10005
|
export const PublicChatsList = 10005
|
||||||
|
export type PublicChatsList = typeof PublicChatsList
|
||||||
export const BlockedRelaysList = 10006
|
export const BlockedRelaysList = 10006
|
||||||
|
export type BlockedRelaysList = typeof BlockedRelaysList
|
||||||
export const SearchRelaysList = 10007
|
export const SearchRelaysList = 10007
|
||||||
|
export type SearchRelaysList = typeof SearchRelaysList
|
||||||
export const InterestsList = 10015
|
export const InterestsList = 10015
|
||||||
|
export type InterestsList = typeof InterestsList
|
||||||
export const UserEmojiList = 10030
|
export const UserEmojiList = 10030
|
||||||
|
export type UserEmojiList = typeof UserEmojiList
|
||||||
|
export const DirectMessageRelaysList = 10050
|
||||||
|
export type DirectMessageRelaysList = typeof DirectMessageRelaysList
|
||||||
export const FileServerPreference = 10096
|
export const FileServerPreference = 10096
|
||||||
|
export type FileServerPreference = typeof FileServerPreference
|
||||||
export const NWCWalletInfo = 13194
|
export const NWCWalletInfo = 13194
|
||||||
|
export type NWCWalletInfo = typeof NWCWalletInfo
|
||||||
export const LightningPubRPC = 21000
|
export const LightningPubRPC = 21000
|
||||||
|
export type LightningPubRPC = typeof LightningPubRPC
|
||||||
export const ClientAuth = 22242
|
export const ClientAuth = 22242
|
||||||
|
export type ClientAuth = typeof ClientAuth
|
||||||
export const NWCWalletRequest = 23194
|
export const NWCWalletRequest = 23194
|
||||||
|
export type NWCWalletRequest = typeof NWCWalletRequest
|
||||||
export const NWCWalletResponse = 23195
|
export const NWCWalletResponse = 23195
|
||||||
|
export type NWCWalletResponse = typeof NWCWalletResponse
|
||||||
export const NostrConnect = 24133
|
export const NostrConnect = 24133
|
||||||
|
export type NostrConnect = typeof NostrConnect
|
||||||
export const HTTPAuth = 27235
|
export const HTTPAuth = 27235
|
||||||
|
export type HTTPAuth = typeof HTTPAuth
|
||||||
export const Followsets = 30000
|
export const Followsets = 30000
|
||||||
|
export type Followsets = typeof Followsets
|
||||||
export const Genericlists = 30001
|
export const Genericlists = 30001
|
||||||
|
export type Genericlists = typeof Genericlists
|
||||||
export const Relaysets = 30002
|
export const Relaysets = 30002
|
||||||
|
export type Relaysets = typeof Relaysets
|
||||||
export const Bookmarksets = 30003
|
export const Bookmarksets = 30003
|
||||||
|
export type Bookmarksets = typeof Bookmarksets
|
||||||
export const Curationsets = 30004
|
export const Curationsets = 30004
|
||||||
|
export type Curationsets = typeof Curationsets
|
||||||
export const ProfileBadges = 30008
|
export const ProfileBadges = 30008
|
||||||
|
export type ProfileBadges = typeof ProfileBadges
|
||||||
export const BadgeDefinition = 30009
|
export const BadgeDefinition = 30009
|
||||||
|
export type BadgeDefinition = typeof BadgeDefinition
|
||||||
export const Interestsets = 30015
|
export const Interestsets = 30015
|
||||||
|
export type Interestsets = typeof Interestsets
|
||||||
export const CreateOrUpdateStall = 30017
|
export const CreateOrUpdateStall = 30017
|
||||||
|
export type CreateOrUpdateStall = typeof CreateOrUpdateStall
|
||||||
export const CreateOrUpdateProduct = 30018
|
export const CreateOrUpdateProduct = 30018
|
||||||
|
export type CreateOrUpdateProduct = typeof CreateOrUpdateProduct
|
||||||
export const LongFormArticle = 30023
|
export const LongFormArticle = 30023
|
||||||
|
export type LongFormArticle = typeof LongFormArticle
|
||||||
export const DraftLong = 30024
|
export const DraftLong = 30024
|
||||||
|
export type DraftLong = typeof DraftLong
|
||||||
export const Emojisets = 30030
|
export const Emojisets = 30030
|
||||||
|
export type Emojisets = typeof Emojisets
|
||||||
export const Application = 30078
|
export const Application = 30078
|
||||||
|
export type Application = typeof Application
|
||||||
export const LiveEvent = 30311
|
export const LiveEvent = 30311
|
||||||
|
export type LiveEvent = typeof LiveEvent
|
||||||
export const UserStatuses = 30315
|
export const UserStatuses = 30315
|
||||||
|
export type UserStatuses = typeof UserStatuses
|
||||||
export const ClassifiedListing = 30402
|
export const ClassifiedListing = 30402
|
||||||
|
export type ClassifiedListing = typeof ClassifiedListing
|
||||||
export const DraftClassifiedListing = 30403
|
export const DraftClassifiedListing = 30403
|
||||||
|
export type DraftClassifiedListing = typeof DraftClassifiedListing
|
||||||
export const Date = 31922
|
export const Date = 31922
|
||||||
|
export type Date = typeof Date
|
||||||
export const Time = 31923
|
export const Time = 31923
|
||||||
|
export type Time = typeof Time
|
||||||
export const Calendar = 31924
|
export const Calendar = 31924
|
||||||
|
export type Calendar = typeof Calendar
|
||||||
export const CalendarEventRSVP = 31925
|
export const CalendarEventRSVP = 31925
|
||||||
|
export type CalendarEventRSVP = typeof CalendarEventRSVP
|
||||||
export const Handlerrecommendation = 31989
|
export const Handlerrecommendation = 31989
|
||||||
|
export type Handlerrecommendation = typeof Handlerrecommendation
|
||||||
export const Handlerinformation = 31990
|
export const Handlerinformation = 31990
|
||||||
|
export type Handlerinformation = typeof Handlerinformation
|
||||||
export const CommunityDefinition = 34550
|
export const CommunityDefinition = 34550
|
||||||
|
export type CommunityDefinition = typeof CommunityDefinition
|
||||||
|
|||||||
4
nip04.ts
4
nip04.ts
@@ -5,7 +5,7 @@ import { base64 } from '@scure/base'
|
|||||||
|
|
||||||
import { utf8Decoder, utf8Encoder } from './utils.ts'
|
import { utf8Decoder, utf8Encoder } from './utils.ts'
|
||||||
|
|
||||||
export async function encrypt(secretKey: string | Uint8Array, pubkey: string, text: string): Promise<string> {
|
export function encrypt(secretKey: string | Uint8Array, pubkey: string, text: string): string {
|
||||||
const privkey: string = secretKey instanceof Uint8Array ? bytesToHex(secretKey) : secretKey
|
const privkey: string = secretKey instanceof Uint8Array ? bytesToHex(secretKey) : secretKey
|
||||||
const key = secp256k1.getSharedSecret(privkey, '02' + pubkey)
|
const key = secp256k1.getSharedSecret(privkey, '02' + pubkey)
|
||||||
const normalizedKey = getNormalizedX(key)
|
const normalizedKey = getNormalizedX(key)
|
||||||
@@ -21,7 +21,7 @@ export async function encrypt(secretKey: string | Uint8Array, pubkey: string, te
|
|||||||
return `${ctb64}?iv=${ivb64}`
|
return `${ctb64}?iv=${ivb64}`
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function decrypt(secretKey: string | Uint8Array, pubkey: string, data: string): Promise<string> {
|
export function decrypt(secretKey: string | Uint8Array, pubkey: string, data: string): string {
|
||||||
const privkey: string = secretKey instanceof Uint8Array ? bytesToHex(secretKey) : secretKey
|
const privkey: string = secretKey instanceof Uint8Array ? bytesToHex(secretKey) : secretKey
|
||||||
let [ctb64, ivb64] = data.split('?iv=')
|
let [ctb64, ivb64] = data.split('?iv=')
|
||||||
let key = secp256k1.getSharedSecret(privkey, '02' + pubkey)
|
let key = secp256k1.getSharedSecret(privkey, '02' + pubkey)
|
||||||
|
|||||||
@@ -1,18 +1,44 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { test, expect } from 'bun:test'
|
||||||
import fetch from 'node-fetch'
|
|
||||||
|
|
||||||
import { useFetchImplementation, queryProfile } from './nip05.ts'
|
import { useFetchImplementation, queryProfile, NIP05_REGEX, isNip05 } from './nip05.ts'
|
||||||
|
|
||||||
|
test('validate NIP05_REGEX', () => {
|
||||||
|
expect(NIP05_REGEX.test('_@bob.com.br')).toBeTrue()
|
||||||
|
expect(NIP05_REGEX.test('bob@bob.com.br')).toBeTrue()
|
||||||
|
expect(NIP05_REGEX.test('b&b@bob.com.br')).toBeFalse()
|
||||||
|
|
||||||
|
expect('b&b@bob.com.br'.match(NIP05_REGEX)).toBeNull()
|
||||||
|
expect(Array.from('bob@bob.com.br'.match(NIP05_REGEX) || [])).toEqual(['bob@bob.com.br', 'bob', 'bob.com.br', '.br'])
|
||||||
|
|
||||||
|
expect(isNip05('bob@bob.com.br')).toBeTrue()
|
||||||
|
expect(isNip05('b&b@bob.com.br')).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
test('fetch nip05 profiles', async () => {
|
test('fetch nip05 profiles', async () => {
|
||||||
useFetchImplementation(fetch)
|
const fetchStub = async (url: string) => ({
|
||||||
|
status: 200,
|
||||||
|
async json() {
|
||||||
|
return {
|
||||||
|
'https://compile-error.net/.well-known/nostr.json?name=_': {
|
||||||
|
names: { _: '2c7cc62a697ea3a7826521f3fd34f0cb273693cbe5e9310f35449f43622a5cdc' },
|
||||||
|
},
|
||||||
|
'https://fiatjaf.com/.well-known/nostr.json?name=_': {
|
||||||
|
names: { _: '3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d' },
|
||||||
|
relays: {
|
||||||
|
'3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d': [
|
||||||
|
'wss://pyramid.fiatjaf.com',
|
||||||
|
'wss://nos.lol',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}[url]
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
let p1 = await queryProfile('jb55.com')
|
useFetchImplementation(fetchStub)
|
||||||
expect(p1!.pubkey).toEqual('32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245')
|
|
||||||
expect(p1!.relays).toEqual(['wss://relay.damus.io'])
|
|
||||||
|
|
||||||
let p2 = await queryProfile('jb55@jb55.com')
|
let p2 = await queryProfile('compile-error.net')
|
||||||
expect(p2!.pubkey).toEqual('32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245')
|
expect(p2!.pubkey).toEqual('2c7cc62a697ea3a7826521f3fd34f0cb273693cbe5e9310f35449f43622a5cdc')
|
||||||
expect(p2!.relays).toEqual(['wss://relay.damus.io'])
|
|
||||||
|
|
||||||
let p3 = await queryProfile('_@fiatjaf.com')
|
let p3 = await queryProfile('_@fiatjaf.com')
|
||||||
expect(p3!.pubkey).toEqual('3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d')
|
expect(p3!.pubkey).toEqual('3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d')
|
||||||
|
|||||||
33
nip05.ts
33
nip05.ts
@@ -1,5 +1,7 @@
|
|||||||
import { ProfilePointer } from './nip19.ts'
|
import { ProfilePointer } from './nip19.ts'
|
||||||
|
|
||||||
|
export type Nip05 = `${string}@${string}`
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* NIP-05 regex. The localpart is optional, and should be assumed to be `_` otherwise.
|
* NIP-05 regex. The localpart is optional, and should be assumed to be `_` otherwise.
|
||||||
*
|
*
|
||||||
@@ -8,21 +10,28 @@ import { ProfilePointer } from './nip19.ts'
|
|||||||
* - 2: domain
|
* - 2: domain
|
||||||
*/
|
*/
|
||||||
export const NIP05_REGEX = /^(?:([\w.+-]+)@)?([\w_-]+(\.[\w_-]+)+)$/
|
export const NIP05_REGEX = /^(?:([\w.+-]+)@)?([\w_-]+(\.[\w_-]+)+)$/
|
||||||
|
export const isNip05 = (value?: string | null): value is Nip05 => NIP05_REGEX.test(value || '')
|
||||||
|
|
||||||
var _fetch: any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
let _fetch: any
|
||||||
|
|
||||||
try {
|
try {
|
||||||
_fetch = fetch
|
_fetch = fetch
|
||||||
} catch {}
|
} catch (_) {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
|
||||||
export function useFetchImplementation(fetchImplementation: any) {
|
export function useFetchImplementation(fetchImplementation: unknown) {
|
||||||
_fetch = fetchImplementation
|
_fetch = fetchImplementation
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function searchDomain(domain: string, query = ''): Promise<{ [name: string]: string }> {
|
export async function searchDomain(domain: string, query = ''): Promise<{ [name: string]: string }> {
|
||||||
try {
|
try {
|
||||||
const url = `https://${domain}/.well-known/nostr.json?name=${query}`
|
const url = `https://${domain}/.well-known/nostr.json?name=${query}`
|
||||||
const res = await _fetch(url, { redirect: 'error' })
|
const res = await _fetch(url, { redirect: 'manual' })
|
||||||
|
if (res.status !== 200) {
|
||||||
|
throw Error('Wrong response code')
|
||||||
|
}
|
||||||
const json = await res.json()
|
const json = await res.json()
|
||||||
return json.names
|
return json.names
|
||||||
} catch (_) {
|
} catch (_) {
|
||||||
@@ -34,20 +43,24 @@ export async function queryProfile(fullname: string): Promise<ProfilePointer | n
|
|||||||
const match = fullname.match(NIP05_REGEX)
|
const match = fullname.match(NIP05_REGEX)
|
||||||
if (!match) return null
|
if (!match) return null
|
||||||
|
|
||||||
const [_, name = '_', domain] = match
|
const [, name = '_', domain] = match
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const url = `https://${domain}/.well-known/nostr.json?name=${name}`
|
const url = `https://${domain}/.well-known/nostr.json?name=${name}`
|
||||||
const res = await (await _fetch(url, { redirect: 'error' })).json()
|
const res = await _fetch(url, { redirect: 'manual' })
|
||||||
|
if (res.status !== 200) {
|
||||||
|
throw Error('Wrong response code')
|
||||||
|
}
|
||||||
|
const json = await res.json()
|
||||||
|
|
||||||
let pubkey = res.names[name]
|
const pubkey = json.names[name]
|
||||||
return pubkey ? { pubkey, relays: res.relays?.[pubkey] } : null
|
return pubkey ? { pubkey, relays: json.relays?.[pubkey] } : null
|
||||||
} catch (_e) {
|
} catch (_e) {
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function isValid(pubkey: string, nip05: string): Promise<boolean> {
|
export async function isValid(pubkey: string, nip05: Nip05): Promise<boolean> {
|
||||||
let res = await queryProfile(nip05)
|
const res = await queryProfile(nip05)
|
||||||
return res ? res.pubkey === pubkey : false
|
return res ? res.pubkey === pubkey : false
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,28 +1,77 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { test, expect } from 'bun:test'
|
||||||
import { privateKeyFromSeedWords } from './nip06.ts'
|
import {
|
||||||
|
privateKeyFromSeedWords,
|
||||||
|
accountFromSeedWords,
|
||||||
|
extendedKeysFromSeedWords,
|
||||||
|
accountFromExtendedKey,
|
||||||
|
} from './nip06.ts'
|
||||||
|
import { hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
test('generate private key from a mnemonic', async () => {
|
test('generate private key from a mnemonic', async () => {
|
||||||
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
||||||
const privateKey = privateKeyFromSeedWords(mnemonic)
|
const privateKey = privateKeyFromSeedWords(mnemonic)
|
||||||
expect(privateKey).toEqual('c26cf31d8ba425b555ca27d00ca71b5008004f2f662470f8c8131822ec129fe2')
|
expect(privateKey).toEqual(hexToBytes('c26cf31d8ba425b555ca27d00ca71b5008004f2f662470f8c8131822ec129fe2'))
|
||||||
})
|
})
|
||||||
|
|
||||||
test('generate private key for account 1 from a mnemonic', async () => {
|
test('generate private key for account 1 from a mnemonic', async () => {
|
||||||
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
||||||
const privateKey = privateKeyFromSeedWords(mnemonic, undefined, 1)
|
const privateKey = privateKeyFromSeedWords(mnemonic, undefined, 1)
|
||||||
expect(privateKey).toEqual('b5fc7f229de3fb5c189063e3b3fc6c921d8f4366cff5bd31c6f063493665eb2b')
|
expect(privateKey).toEqual(hexToBytes('b5fc7f229de3fb5c189063e3b3fc6c921d8f4366cff5bd31c6f063493665eb2b'))
|
||||||
})
|
})
|
||||||
|
|
||||||
test('generate private key from a mnemonic and passphrase', async () => {
|
test('generate private key from a mnemonic and passphrase', async () => {
|
||||||
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
||||||
const passphrase = '123'
|
const passphrase = '123'
|
||||||
const privateKey = privateKeyFromSeedWords(mnemonic, passphrase)
|
const privateKey = privateKeyFromSeedWords(mnemonic, passphrase)
|
||||||
expect(privateKey).toEqual('55a22b8203273d0aaf24c22c8fbe99608e70c524b17265641074281c8b978ae4')
|
expect(privateKey).toEqual(hexToBytes('55a22b8203273d0aaf24c22c8fbe99608e70c524b17265641074281c8b978ae4'))
|
||||||
})
|
})
|
||||||
|
|
||||||
test('generate private key for account 1 from a mnemonic and passphrase', async () => {
|
test('generate private key for account 1 from a mnemonic and passphrase', async () => {
|
||||||
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
||||||
const passphrase = '123'
|
const passphrase = '123'
|
||||||
const privateKey = privateKeyFromSeedWords(mnemonic, passphrase, 1)
|
const privateKey = privateKeyFromSeedWords(mnemonic, passphrase, 1)
|
||||||
expect(privateKey).toEqual('2e0f7bd9e3c3ebcdff1a90fb49c913477e7c055eba1a415d571b6a8c714c7135')
|
expect(privateKey).toEqual(hexToBytes('2e0f7bd9e3c3ebcdff1a90fb49c913477e7c055eba1a415d571b6a8c714c7135'))
|
||||||
|
})
|
||||||
|
|
||||||
|
test('generate private and public key for account 1 from a mnemonic and passphrase', async () => {
|
||||||
|
const mnemonic = 'zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo zoo wrong'
|
||||||
|
const passphrase = '123'
|
||||||
|
const { privateKey, publicKey } = accountFromSeedWords(mnemonic, passphrase, 1)
|
||||||
|
expect(privateKey).toEqual(hexToBytes('2e0f7bd9e3c3ebcdff1a90fb49c913477e7c055eba1a415d571b6a8c714c7135'))
|
||||||
|
expect(publicKey).toEqual('13f55f4f01576570ea342eb7d2b611f9dc78f8dc601aeb512011e4e73b90cf0a')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('generate extended keys from mnemonic', () => {
|
||||||
|
const mnemonic = 'abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'
|
||||||
|
const passphrase = ''
|
||||||
|
const extendedAccountIndex = 0
|
||||||
|
const { privateExtendedKey, publicExtendedKey } = extendedKeysFromSeedWords(
|
||||||
|
mnemonic,
|
||||||
|
passphrase,
|
||||||
|
extendedAccountIndex,
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(privateExtendedKey).toBe(
|
||||||
|
'xprv9z78fizET65qsCaRr1MSutTSGk1fcKfSt1sBqmuWShtkjRJJ4WCKcSnha6EmgNzFSsyom3MWtydHyPtJtSLZQUtictVQtM2vkPcguh6TQCH',
|
||||||
|
)
|
||||||
|
expect(publicExtendedKey).toBe(
|
||||||
|
'xpub6D6V5EX8HTe95getx2tTH2QApmrA1nPJFEnneAK813RjcDdSc3WaAF7BRNpTF7o7zXjVm3DD3VMX66jhQ7wLaZ9sS6NzyfiwfzqDZbxvpDN',
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('generate account from extended private key', () => {
|
||||||
|
const xprv =
|
||||||
|
'xprv9z78fizET65qsCaRr1MSutTSGk1fcKfSt1sBqmuWShtkjRJJ4WCKcSnha6EmgNzFSsyom3MWtydHyPtJtSLZQUtictVQtM2vkPcguh6TQCH'
|
||||||
|
const { privateKey, publicKey } = accountFromExtendedKey(xprv)
|
||||||
|
|
||||||
|
expect(privateKey).toEqual(hexToBytes('5f29af3b9676180290e77a4efad265c4c2ff28a5302461f73597fda26bb25731'))
|
||||||
|
expect(publicKey).toBe('e8bcf3823669444d0b49ad45d65088635d9fd8500a75b5f20b59abefa56a144f')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('generate account from extended public key', () => {
|
||||||
|
const xpub =
|
||||||
|
'xpub6D6V5EX8HTe95getx2tTH2QApmrA1nPJFEnneAK813RjcDdSc3WaAF7BRNpTF7o7zXjVm3DD3VMX66jhQ7wLaZ9sS6NzyfiwfzqDZbxvpDN'
|
||||||
|
const { publicKey } = accountFromExtendedKey(xpub)
|
||||||
|
|
||||||
|
expect(publicKey).toBe('e8bcf3823669444d0b49ad45d65088635d9fd8500a75b5f20b59abefa56a144f')
|
||||||
})
|
})
|
||||||
|
|||||||
62
nip06.ts
62
nip06.ts
@@ -3,11 +3,67 @@ import { wordlist } from '@scure/bip39/wordlists/english'
|
|||||||
import { generateMnemonic, mnemonicToSeedSync, validateMnemonic } from '@scure/bip39'
|
import { generateMnemonic, mnemonicToSeedSync, validateMnemonic } from '@scure/bip39'
|
||||||
import { HDKey } from '@scure/bip32'
|
import { HDKey } from '@scure/bip32'
|
||||||
|
|
||||||
export function privateKeyFromSeedWords(mnemonic: string, passphrase?: string, accountIndex = 0): string {
|
const DERIVATION_PATH = `m/44'/1237'`
|
||||||
|
|
||||||
|
export function privateKeyFromSeedWords(mnemonic: string, passphrase?: string, accountIndex = 0): Uint8Array {
|
||||||
let root = HDKey.fromMasterSeed(mnemonicToSeedSync(mnemonic, passphrase))
|
let root = HDKey.fromMasterSeed(mnemonicToSeedSync(mnemonic, passphrase))
|
||||||
let privateKey = root.derive(`m/44'/1237'/${accountIndex}'/0/0`).privateKey
|
let privateKey = root.derive(`${DERIVATION_PATH}/${accountIndex}'/0/0`).privateKey
|
||||||
if (!privateKey) throw new Error('could not derive private key')
|
if (!privateKey) throw new Error('could not derive private key')
|
||||||
return bytesToHex(privateKey)
|
return privateKey
|
||||||
|
}
|
||||||
|
|
||||||
|
export function accountFromSeedWords(
|
||||||
|
mnemonic: string,
|
||||||
|
passphrase?: string,
|
||||||
|
accountIndex = 0,
|
||||||
|
): {
|
||||||
|
privateKey: Uint8Array
|
||||||
|
publicKey: string
|
||||||
|
} {
|
||||||
|
const root = HDKey.fromMasterSeed(mnemonicToSeedSync(mnemonic, passphrase))
|
||||||
|
const seed = root.derive(`${DERIVATION_PATH}/${accountIndex}'/0/0`)
|
||||||
|
const publicKey = bytesToHex(seed.publicKey!.slice(1))
|
||||||
|
const privateKey = seed.privateKey
|
||||||
|
if (!privateKey || !publicKey) {
|
||||||
|
throw new Error('could not derive key pair')
|
||||||
|
}
|
||||||
|
return { privateKey, publicKey }
|
||||||
|
}
|
||||||
|
|
||||||
|
export function extendedKeysFromSeedWords(
|
||||||
|
mnemonic: string,
|
||||||
|
passphrase?: string,
|
||||||
|
extendedAccountIndex = 0,
|
||||||
|
): {
|
||||||
|
privateExtendedKey: string
|
||||||
|
publicExtendedKey: string
|
||||||
|
} {
|
||||||
|
let root = HDKey.fromMasterSeed(mnemonicToSeedSync(mnemonic, passphrase))
|
||||||
|
let seed = root.derive(`${DERIVATION_PATH}/${extendedAccountIndex}'`)
|
||||||
|
let privateExtendedKey = seed.privateExtendedKey
|
||||||
|
let publicExtendedKey = seed.publicExtendedKey
|
||||||
|
if (!privateExtendedKey && !publicExtendedKey) throw new Error('could not derive extended key pair')
|
||||||
|
return { privateExtendedKey, publicExtendedKey }
|
||||||
|
}
|
||||||
|
|
||||||
|
export function accountFromExtendedKey(
|
||||||
|
base58key: string,
|
||||||
|
accountIndex = 0,
|
||||||
|
): {
|
||||||
|
privateKey?: Uint8Array
|
||||||
|
publicKey: string
|
||||||
|
} {
|
||||||
|
let extendedKey = HDKey.fromExtendedKey(base58key)
|
||||||
|
let version = base58key.slice(0, 4)
|
||||||
|
let child = extendedKey.deriveChild(0).deriveChild(accountIndex)
|
||||||
|
let publicKey = bytesToHex(child.publicKey!.slice(1))
|
||||||
|
if (!publicKey) throw new Error('could not derive public key')
|
||||||
|
if (version === 'xprv') {
|
||||||
|
let privateKey = child.privateKey!
|
||||||
|
if (!privateKey) throw new Error('could not derive private key')
|
||||||
|
return { privateKey, publicKey }
|
||||||
|
}
|
||||||
|
return { publicKey }
|
||||||
}
|
}
|
||||||
|
|
||||||
export function generateSeedWords(): string {
|
export function generateSeedWords(): string {
|
||||||
|
|||||||
14
nip07.ts
Normal file
14
nip07.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { EventTemplate, VerifiedEvent } from './core.ts'
|
||||||
|
|
||||||
|
export interface WindowNostr {
|
||||||
|
getPublicKey(): Promise<string>
|
||||||
|
signEvent(event: EventTemplate): Promise<VerifiedEvent>
|
||||||
|
nip04?: {
|
||||||
|
encrypt(pubkey: string, plaintext: string): Promise<string>
|
||||||
|
decrypt(pubkey: string, ciphertext: string): Promise<string>
|
||||||
|
}
|
||||||
|
nip44?: {
|
||||||
|
encrypt(pubkey: string, plaintext: string): Promise<string>
|
||||||
|
decrypt(pubkey: string, ciphertext: string): Promise<string>
|
||||||
|
}
|
||||||
|
}
|
||||||
187
nip10.test.ts
187
nip10.test.ts
@@ -5,20 +5,21 @@ describe('parse NIP10-referenced events', () => {
|
|||||||
test('legacy + a lot of events', () => {
|
test('legacy + a lot of events', () => {
|
||||||
let event = {
|
let event = {
|
||||||
tags: [
|
tags: [
|
||||||
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c'],
|
|
||||||
['e', 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631'],
|
|
||||||
['e', '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64'],
|
|
||||||
['e', '49aff7ae6daeaaa2777931b90f9bb29f6cb01c5a3d7d88c8ba82d890f264afb4'],
|
|
||||||
['e', '567b7c11f0fe582361e3cea6fcc7609a8942dfe196ee1b98d5604c93fbeea976'],
|
|
||||||
['e', '090c037b2e399ee74d9f134758928948dd9154413ca1a1acb37155046e03a051'],
|
|
||||||
['e', '89f220b63465c93542b1a78caa3a952cf4f196e91a50596493c8093c533ebc4d'],
|
|
||||||
['p', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
|
||||||
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
|
||||||
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0'],
|
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0'],
|
||||||
|
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
||||||
|
['p', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
||||||
|
['e', '89f220b63465c93542b1a78caa3a952cf4f196e91a50596493c8093c533ebc4d'],
|
||||||
|
['e', '090c037b2e399ee74d9f134758928948dd9154413ca1a1acb37155046e03a051'],
|
||||||
|
['e', '567b7c11f0fe582361e3cea6fcc7609a8942dfe196ee1b98d5604c93fbeea976'],
|
||||||
|
['e', '49aff7ae6daeaaa2777931b90f9bb29f6cb01c5a3d7d88c8ba82d890f264afb4'],
|
||||||
|
['e', '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64'],
|
||||||
|
['e', 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631'],
|
||||||
|
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c'],
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(parse(event)).toEqual({
|
expect(parse(event)).toEqual({
|
||||||
|
quotes: [],
|
||||||
mentions: [
|
mentions: [
|
||||||
{
|
{
|
||||||
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
||||||
@@ -55,33 +56,80 @@ describe('parse NIP10-referenced events', () => {
|
|||||||
relays: [],
|
relays: [],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
reply: {
|
root: {
|
||||||
id: '89f220b63465c93542b1a78caa3a952cf4f196e91a50596493c8093c533ebc4d',
|
id: '89f220b63465c93542b1a78caa3a952cf4f196e91a50596493c8093c533ebc4d',
|
||||||
relays: [],
|
relays: [],
|
||||||
},
|
},
|
||||||
root: {
|
reply: {
|
||||||
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
||||||
relays: [],
|
relays: [],
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
test('legacy + 3 events', () => {
|
test('modern', () => {
|
||||||
let event = {
|
let event = {
|
||||||
tags: [
|
tags: [
|
||||||
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c'],
|
|
||||||
['e', 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631'],
|
|
||||||
['e', '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64'],
|
|
||||||
['p', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
|
||||||
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
|
||||||
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0'],
|
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0'],
|
||||||
|
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
||||||
|
['e', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
||||||
|
['e', 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631', '', 'root'],
|
||||||
|
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c', '', 'reply'],
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(parse(event)).toEqual({
|
expect(parse(event)).toEqual({
|
||||||
|
quotes: [],
|
||||||
mentions: [
|
mentions: [
|
||||||
{
|
{
|
||||||
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
id: '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7',
|
||||||
|
relays: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
profiles: [
|
||||||
|
{
|
||||||
|
pubkey: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
||||||
|
relays: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pubkey: '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
||||||
|
relays: [],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
root: {
|
||||||
|
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
||||||
|
relays: [],
|
||||||
|
},
|
||||||
|
reply: {
|
||||||
|
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
||||||
|
relays: [],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('modern, inverted, author hint', () => {
|
||||||
|
let event = {
|
||||||
|
tags: [
|
||||||
|
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0', 'wss://goiaba.com'],
|
||||||
|
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
||||||
|
['p', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
||||||
|
['e', '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64', '', 'reply'],
|
||||||
|
[
|
||||||
|
'e',
|
||||||
|
'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
||||||
|
'wss://banana.com',
|
||||||
|
'root',
|
||||||
|
'4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
||||||
|
],
|
||||||
|
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c'],
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(parse(event)).toEqual({
|
||||||
|
quotes: [],
|
||||||
|
mentions: [
|
||||||
|
{
|
||||||
|
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
||||||
relays: [],
|
relays: [],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@@ -96,98 +144,80 @@ describe('parse NIP10-referenced events', () => {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
pubkey: '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
pubkey: '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
||||||
relays: [],
|
relays: ['wss://banana.com', 'wss://goiaba.com'],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
root: {
|
||||||
|
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
||||||
|
relays: ['wss://banana.com', 'wss://goiaba.com'],
|
||||||
|
author: '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
||||||
|
},
|
||||||
reply: {
|
reply: {
|
||||||
id: '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64',
|
id: '5e081ebb19153357d7c31e8a10b9ceeef29313f58dc8d701f66727fab02aef64',
|
||||||
relays: [],
|
relays: [],
|
||||||
},
|
},
|
||||||
root: {
|
|
||||||
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
|
||||||
relays: [],
|
|
||||||
},
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
test('legacy + 2 events', () => {
|
test('1 event, relay hint from author', () => {
|
||||||
let event = {
|
let event = {
|
||||||
tags: [
|
tags: [
|
||||||
['e', 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c'],
|
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec', 'wss://banana.com'],
|
||||||
['e', 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631'],
|
[
|
||||||
['p', '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7'],
|
'e',
|
||||||
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
'9abbfd9b9ac5ecdab45d14b8bf8d746139ea039e931a1b376d19a239f1946590',
|
||||||
['p', '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0'],
|
'',
|
||||||
|
'root',
|
||||||
|
'534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
||||||
|
],
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(parse(event)).toEqual({
|
expect(parse(event)).toEqual({
|
||||||
|
quotes: [],
|
||||||
mentions: [],
|
mentions: [],
|
||||||
profiles: [
|
profiles: [
|
||||||
{
|
|
||||||
pubkey: '77ce56f89d1228f7ff3743ce1ad1b254857b9008564727ebd5a1f317362f6ca7',
|
|
||||||
relays: [],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
pubkey: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
pubkey: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
||||||
relays: [],
|
relays: ['wss://banana.com'],
|
||||||
},
|
|
||||||
{
|
|
||||||
pubkey: '4ca4f5533e40da5e0508796d409e6bb35a50b26fc304345617ab017183d83ac0',
|
|
||||||
relays: [],
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
reply: {
|
reply: {
|
||||||
id: 'bbd72f0ae14374aa8fb166b483cfcf99b57d7f4cf1600ccbf17c350040834631',
|
author: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
||||||
relays: [],
|
|
||||||
},
|
|
||||||
root: {
|
|
||||||
id: 'b857504288c18a15950dd05b9e8772c62ca6289d5aac373c0a8ee5b132e94e7c',
|
|
||||||
relays: [],
|
|
||||||
},
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
test('legacy + 1 event', () => {
|
|
||||||
let event = {
|
|
||||||
tags: [
|
|
||||||
['e', '9abbfd9b9ac5ecdab45d14b8bf8d746139ea039e931a1b376d19a239f1946590'],
|
|
||||||
['p', '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec'],
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(parse(event)).toEqual({
|
|
||||||
mentions: [],
|
|
||||||
profiles: [
|
|
||||||
{
|
|
||||||
pubkey: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
|
||||||
relays: [],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
reply: undefined,
|
|
||||||
root: {
|
|
||||||
id: '9abbfd9b9ac5ecdab45d14b8bf8d746139ea039e931a1b376d19a239f1946590',
|
id: '9abbfd9b9ac5ecdab45d14b8bf8d746139ea039e931a1b376d19a239f1946590',
|
||||||
relays: [],
|
relays: ['wss://banana.com'],
|
||||||
|
},
|
||||||
|
root: {
|
||||||
|
author: '534780e44da7b494485e85cd4cca6af4f6caa1627472432b6f2a4ece0e9e54ec',
|
||||||
|
id: '9abbfd9b9ac5ecdab45d14b8bf8d746139ea039e931a1b376d19a239f1946590',
|
||||||
|
relays: ['wss://banana.com'],
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
test('recommended + 1 event', () => {
|
test('many p 1 reply', () => {
|
||||||
let event = {
|
let event = {
|
||||||
tags: [
|
tags: [
|
||||||
['p', 'a8c21fcd8aa1f4befba14d72fc7a012397732d30d8b3131af912642f3c726f52', 'wss://relay.mostr.pub'],
|
|
||||||
['p', '003d7fd21fd09ff7f6f63a75daf194dd99feefbe6919cc376b7359d5090aa9a6', 'wss://relay.mostr.pub'],
|
|
||||||
['p', '2f6fbe452edd3987d3c67f3b034c03ec5bcf4d054c521c3a954686f89f03212e', 'wss://relay.mostr.pub'],
|
|
||||||
['p', '44c7c74668ff222b0e0b30579c49fc6e22dafcdeaad091036c947f9856590f1e', 'wss://relay.mostr.pub'],
|
|
||||||
['p', 'c5cf39149caebda4cdd61771c51f6ba91ef5645919004e5c4998a4ea69f00512', 'wss://relay.mostr.pub'],
|
|
||||||
['p', '094d44bb1e812696c57f57ad1c0c707812dedbe72c07e538b80639032c236a9e', 'wss://relay.mostr.pub'],
|
|
||||||
['p', 'a1ba0ac9b6ec098f726a3c11ec654df4a32cbb84b5377e8788395e9c27d9ecda', 'wss://relay.mostr.pub'],
|
['p', 'a1ba0ac9b6ec098f726a3c11ec654df4a32cbb84b5377e8788395e9c27d9ecda', 'wss://relay.mostr.pub'],
|
||||||
['e', 'f9472913904ab7e9da008dcb2d85fd4af2d2993ada483d00c646d0c4481d031d', 'wss://relay.mostr.pub', 'reply'],
|
['p', '094d44bb1e812696c57f57ad1c0c707812dedbe72c07e538b80639032c236a9e', 'wss://relay.mostr.pub'],
|
||||||
|
['p', 'c5cf39149caebda4cdd61771c51f6ba91ef5645919004e5c4998a4ea69f00512', 'wss://relay.mostr.pub'],
|
||||||
|
['p', '44c7c74668ff222b0e0b30579c49fc6e22dafcdeaad091036c947f9856590f1e', 'wss://relay.mostr.pub'],
|
||||||
|
['p', '2f6fbe452edd3987d3c67f3b034c03ec5bcf4d054c521c3a954686f89f03212e', 'wss://relay.mostr.pub'],
|
||||||
|
['p', '003d7fd21fd09ff7f6f63a75daf194dd99feefbe6919cc376b7359d5090aa9a6', 'wss://relay.mostr.pub'],
|
||||||
|
['p', 'a8c21fcd8aa1f4befba14d72fc7a012397732d30d8b3131af912642f3c726f52', 'wss://relay.mostr.pub'],
|
||||||
|
[
|
||||||
|
'e',
|
||||||
|
'f9472913904ab7e9da008dcb2d85fd4af2d2993ada483d00c646d0c4481d031d',
|
||||||
|
'wss://relay.mostr.pub',
|
||||||
|
'reply',
|
||||||
|
'c5cf39149caebda4cdd61771c51f6ba91ef5645919004e5c4998a4ea69f00512',
|
||||||
|
],
|
||||||
['mostr', 'https://poa.st/objects/dc50684b-6364-4264-ab16-49f4622f05ea'],
|
['mostr', 'https://poa.st/objects/dc50684b-6364-4264-ab16-49f4622f05ea'],
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(parse(event)).toEqual({
|
expect(parse(event)).toEqual({
|
||||||
|
quotes: [],
|
||||||
mentions: [],
|
mentions: [],
|
||||||
profiles: [
|
profiles: [
|
||||||
{
|
{
|
||||||
@@ -222,8 +252,13 @@ describe('parse NIP10-referenced events', () => {
|
|||||||
reply: {
|
reply: {
|
||||||
id: 'f9472913904ab7e9da008dcb2d85fd4af2d2993ada483d00c646d0c4481d031d',
|
id: 'f9472913904ab7e9da008dcb2d85fd4af2d2993ada483d00c646d0c4481d031d',
|
||||||
relays: ['wss://relay.mostr.pub'],
|
relays: ['wss://relay.mostr.pub'],
|
||||||
|
author: 'c5cf39149caebda4cdd61771c51f6ba91ef5645919004e5c4998a4ea69f00512',
|
||||||
|
},
|
||||||
|
root: {
|
||||||
|
id: 'f9472913904ab7e9da008dcb2d85fd4af2d2993ada483d00c646d0c4481d031d',
|
||||||
|
relays: ['wss://relay.mostr.pub'],
|
||||||
|
author: 'c5cf39149caebda4cdd61771c51f6ba91ef5645919004e5c4998a4ea69f00512',
|
||||||
},
|
},
|
||||||
root: undefined,
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
152
nip10.ts
152
nip10.ts
@@ -1,7 +1,7 @@
|
|||||||
import type { Event } from './core.ts'
|
import type { Event } from './core.ts'
|
||||||
import type { EventPointer, ProfilePointer } from './nip19.ts'
|
import type { EventPointer, ProfilePointer } from './nip19.ts'
|
||||||
|
|
||||||
export type NIP10Result = {
|
export function parse(event: Pick<Event, 'tags'>): {
|
||||||
/**
|
/**
|
||||||
* Pointer to the root of the thread.
|
* Pointer to the root of the thread.
|
||||||
*/
|
*/
|
||||||
@@ -13,29 +13,80 @@ export type NIP10Result = {
|
|||||||
reply: EventPointer | undefined
|
reply: EventPointer | undefined
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Pointers to events which may or may not be in the reply chain.
|
* Pointers to events that may or may not be in the reply chain.
|
||||||
*/
|
*/
|
||||||
mentions: EventPointer[]
|
mentions: EventPointer[]
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pointers to events that were directly quoted.
|
||||||
|
*/
|
||||||
|
quotes: EventPointer[]
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* List of pubkeys that are involved in the thread in no particular order.
|
* List of pubkeys that are involved in the thread in no particular order.
|
||||||
*/
|
*/
|
||||||
profiles: ProfilePointer[]
|
profiles: ProfilePointer[]
|
||||||
}
|
} {
|
||||||
|
const result: ReturnType<typeof parse> = {
|
||||||
export function parse(event: Pick<Event, 'tags'>): NIP10Result {
|
|
||||||
const result: NIP10Result = {
|
|
||||||
reply: undefined,
|
reply: undefined,
|
||||||
root: undefined,
|
root: undefined,
|
||||||
mentions: [],
|
mentions: [],
|
||||||
profiles: [],
|
profiles: [],
|
||||||
|
quotes: [],
|
||||||
}
|
}
|
||||||
|
|
||||||
const eTags: string[][] = []
|
let maybeParent: EventPointer | undefined
|
||||||
|
let maybeRoot: EventPointer | undefined
|
||||||
|
|
||||||
|
for (let i = event.tags.length - 1; i >= 0; i--) {
|
||||||
|
const tag = event.tags[i]
|
||||||
|
|
||||||
for (const tag of event.tags) {
|
|
||||||
if (tag[0] === 'e' && tag[1]) {
|
if (tag[0] === 'e' && tag[1]) {
|
||||||
eTags.push(tag)
|
const [_, eTagEventId, eTagRelayUrl, eTagMarker, eTagAuthor] = tag as [
|
||||||
|
string,
|
||||||
|
string,
|
||||||
|
undefined | string,
|
||||||
|
undefined | string,
|
||||||
|
undefined | string,
|
||||||
|
]
|
||||||
|
|
||||||
|
const eventPointer: EventPointer = {
|
||||||
|
id: eTagEventId,
|
||||||
|
relays: eTagRelayUrl ? [eTagRelayUrl] : [],
|
||||||
|
author: eTagAuthor,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eTagMarker === 'root') {
|
||||||
|
result.root = eventPointer
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eTagMarker === 'reply') {
|
||||||
|
result.reply = eventPointer
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eTagMarker === 'mention') {
|
||||||
|
result.mentions.push(eventPointer)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!maybeParent) {
|
||||||
|
maybeParent = eventPointer
|
||||||
|
} else {
|
||||||
|
maybeRoot = eventPointer
|
||||||
|
}
|
||||||
|
|
||||||
|
result.mentions.push(eventPointer)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tag[0] === 'q' && tag[1]) {
|
||||||
|
const [_, eTagEventId, eTagRelayUrl] = tag as [string, string, undefined | string]
|
||||||
|
result.quotes.push({
|
||||||
|
id: eTagEventId,
|
||||||
|
relays: eTagRelayUrl ? [eTagRelayUrl] : [],
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tag[0] === 'p' && tag[1]) {
|
if (tag[0] === 'p' && tag[1]) {
|
||||||
@@ -43,49 +94,54 @@ export function parse(event: Pick<Event, 'tags'>): NIP10Result {
|
|||||||
pubkey: tag[1],
|
pubkey: tag[1],
|
||||||
relays: tag[2] ? [tag[2]] : [],
|
relays: tag[2] ? [tag[2]] : [],
|
||||||
})
|
})
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let eTagIndex = 0; eTagIndex < eTags.length; eTagIndex++) {
|
// get legacy (positional) markers, set reply to root and vice-versa if one of them is missing
|
||||||
const eTag = eTags[eTagIndex]
|
if (!result.root) {
|
||||||
|
result.root = maybeRoot || maybeParent || result.reply
|
||||||
const [_, eTagEventId, eTagRelayUrl, eTagMarker] = eTag as [string, string, undefined | string, undefined | string]
|
|
||||||
|
|
||||||
const eventPointer: EventPointer = {
|
|
||||||
id: eTagEventId,
|
|
||||||
relays: eTagRelayUrl ? [eTagRelayUrl] : [],
|
|
||||||
}
|
|
||||||
|
|
||||||
const isFirstETag = eTagIndex === 0
|
|
||||||
const isLastETag = eTagIndex === eTags.length - 1
|
|
||||||
|
|
||||||
if (eTagMarker === 'root') {
|
|
||||||
result.root = eventPointer
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if (eTagMarker === 'reply') {
|
|
||||||
result.reply = eventPointer
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if (eTagMarker === 'mention') {
|
|
||||||
result.mentions.push(eventPointer)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isFirstETag) {
|
|
||||||
result.root = eventPointer
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isLastETag) {
|
|
||||||
result.reply = eventPointer
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
result.mentions.push(eventPointer)
|
|
||||||
}
|
}
|
||||||
|
if (!result.reply) {
|
||||||
|
result.reply = maybeParent || result.root
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove root and reply from mentions, inherit relay hints from authors if any
|
||||||
|
;[result.reply, result.root].forEach(ref => {
|
||||||
|
if (!ref) return
|
||||||
|
|
||||||
|
let idx = result.mentions.indexOf(ref)
|
||||||
|
if (idx !== -1) {
|
||||||
|
result.mentions.splice(idx, 1)
|
||||||
|
}
|
||||||
|
if (ref.author) {
|
||||||
|
let author = result.profiles.find(p => p.pubkey === ref.author)
|
||||||
|
if (author && author.relays) {
|
||||||
|
if (!ref.relays) {
|
||||||
|
ref.relays = []
|
||||||
|
}
|
||||||
|
author.relays.forEach(url => {
|
||||||
|
if (ref.relays!?.indexOf(url) === -1) ref.relays!.push(url)
|
||||||
|
})
|
||||||
|
author.relays = ref.relays
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
result.mentions.forEach(ref => {
|
||||||
|
if (ref!.author) {
|
||||||
|
let author = result.profiles.find(p => p.pubkey === ref.author)
|
||||||
|
if (author && author.relays) {
|
||||||
|
if (!ref.relays) {
|
||||||
|
ref.relays = []
|
||||||
|
}
|
||||||
|
author.relays.forEach(url => {
|
||||||
|
if (ref.relays!.indexOf(url) === -1) ref.relays!.push(url)
|
||||||
|
})
|
||||||
|
author.relays = ref.relays
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,9 @@ describe('requesting relay as for NIP11', () => {
|
|||||||
const info = await fetchRelayInformation('wss://nos.lol')
|
const info = await fetchRelayInformation('wss://nos.lol')
|
||||||
expect(info.name).toEqual('nos.lol')
|
expect(info.name).toEqual('nos.lol')
|
||||||
expect(info.description).toContain('Generally accepts notes, except spammy ones.')
|
expect(info.description).toContain('Generally accepts notes, except spammy ones.')
|
||||||
expect(info.supported_nips).toEqual([1, 2, 4, 9, 11, 12, 16, 20, 22, 28, 33, 40])
|
expect(info.supported_nips).toContain(1)
|
||||||
|
expect(info.supported_nips).toContain(11)
|
||||||
|
expect(info.supported_nips).toContain(70)
|
||||||
expect(info.software).toEqual('git+https://github.com/hoytech/strfry.git')
|
expect(info.software).toEqual('git+https://github.com/hoytech/strfry.git')
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
14
nip11.ts
14
nip11.ts
@@ -68,7 +68,7 @@ export interface BasicRelayInformation {
|
|||||||
* from `[` to `]` and is after UTF-8 serialization (so some
|
* from `[` to `]` and is after UTF-8 serialization (so some
|
||||||
* unicode characters will cost 2-3 bytes). It is equal to
|
* unicode characters will cost 2-3 bytes). It is equal to
|
||||||
* the maximum size of the WebSocket message frame.
|
* the maximum size of the WebSocket message frame.
|
||||||
* @param max_subscription total number of subscriptions
|
* @param max_subscriptions total number of subscriptions
|
||||||
* that may be active on a single websocket connection to
|
* that may be active on a single websocket connection to
|
||||||
* this relay. It's possible that authenticated clients with
|
* this relay. It's possible that authenticated clients with
|
||||||
* a (paid) relationship to the relay may have higher limits.
|
* a (paid) relationship to the relay may have higher limits.
|
||||||
@@ -101,12 +101,17 @@ export interface BasicRelayInformation {
|
|||||||
* authentication to happen before a new connection may
|
* authentication to happen before a new connection may
|
||||||
* perform any other action. Even if set to False,
|
* perform any other action. Even if set to False,
|
||||||
* authentication may be required for specific actions.
|
* authentication may be required for specific actions.
|
||||||
|
* @param restricted_writes: this relay requires some kind
|
||||||
|
* of condition to be fulfilled in order to accept events
|
||||||
|
* (not necessarily, but including
|
||||||
* @param payment_required this relay requires payment
|
* @param payment_required this relay requires payment
|
||||||
* before a new connection may perform any action.
|
* before a new connection may perform any action.
|
||||||
|
* @param created_at_lower_limit: 'created_at' lower limit
|
||||||
|
* @param created_at_upper_limit: 'created_at' upper limit
|
||||||
*/
|
*/
|
||||||
export interface Limitations {
|
export interface Limitations {
|
||||||
max_message_length: number
|
max_message_length: number
|
||||||
max_subscription: number
|
max_subscriptions: number
|
||||||
max_filters: number
|
max_filters: number
|
||||||
max_limit: number
|
max_limit: number
|
||||||
max_subid_length: number
|
max_subid_length: number
|
||||||
@@ -116,9 +121,12 @@ export interface Limitations {
|
|||||||
min_pow_difficulty: number
|
min_pow_difficulty: number
|
||||||
auth_required: boolean
|
auth_required: boolean
|
||||||
payment_required: boolean
|
payment_required: boolean
|
||||||
|
created_at_lower_limit: number
|
||||||
|
created_at_upper_limit: number
|
||||||
|
restricted_writes: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
interface RetentionDetails {
|
export interface RetentionDetails {
|
||||||
kinds: (number | number[])[]
|
kinds: (number | number[])[]
|
||||||
time?: number | null
|
time?: number | null
|
||||||
count?: number | null
|
count?: number | null
|
||||||
|
|||||||
@@ -2,9 +2,14 @@ import { test, expect } from 'bun:test'
|
|||||||
import { getPow, minePow } from './nip13.ts'
|
import { getPow, minePow } from './nip13.ts'
|
||||||
|
|
||||||
test('identifies proof-of-work difficulty', async () => {
|
test('identifies proof-of-work difficulty', async () => {
|
||||||
const id = '000006d8c378af1779d2feebc7603a125d99eca0ccf1085959b307f64e5dd358'
|
;[
|
||||||
const difficulty = getPow(id)
|
['000006d8c378af1779d2feebc7603a125d99eca0ccf1085959b307f64e5dd358', 21],
|
||||||
expect(difficulty).toEqual(21)
|
['6bf5b4f434813c64b523d2b0e6efe18f3bd0cbbd0a5effd8ece9e00fd2531996', 1],
|
||||||
|
['00003479309ecdb46b1c04ce129d2709378518588bed6776e60474ebde3159ae', 18],
|
||||||
|
['01a76167d41add96be4959d9e618b7a35f26551d62c43c11e5e64094c6b53c83', 7],
|
||||||
|
['ac4f44bae06a45ebe88cfbd3c66358750159650a26c0d79e8ccaa92457fca4f6', 0],
|
||||||
|
['0000000000000000006cfbd3c66358750159650a26c0d79e8ccaa92457fca4f6', 73],
|
||||||
|
].forEach(([id, diff]) => expect(getPow(id as string)).toEqual(diff as number))
|
||||||
})
|
})
|
||||||
|
|
||||||
test('mines POW for an event', async () => {
|
test('mines POW for an event', async () => {
|
||||||
|
|||||||
24
nip13.ts
24
nip13.ts
@@ -1,15 +1,19 @@
|
|||||||
import { type UnsignedEvent, type Event, getEventHash } from './pure.ts'
|
import { bytesToHex } from '@noble/hashes/utils'
|
||||||
|
import { type UnsignedEvent, type Event } from './pure.ts'
|
||||||
|
import { sha256 } from '@noble/hashes/sha256'
|
||||||
|
|
||||||
|
import { utf8Encoder } from './utils.ts'
|
||||||
|
|
||||||
/** Get POW difficulty from a Nostr hex ID. */
|
/** Get POW difficulty from a Nostr hex ID. */
|
||||||
export function getPow(hex: string): number {
|
export function getPow(hex: string): number {
|
||||||
let count = 0
|
let count = 0
|
||||||
|
|
||||||
for (let i = 0; i < hex.length; i++) {
|
for (let i = 0; i < 64; i += 8) {
|
||||||
const nibble = parseInt(hex[i], 16)
|
const nibble = parseInt(hex.substring(i, i + 8), 16)
|
||||||
if (nibble === 0) {
|
if (nibble === 0) {
|
||||||
count += 4
|
count += 32
|
||||||
} else {
|
} else {
|
||||||
count += Math.clz32(nibble) - 28
|
count += Math.clz32(nibble)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -20,8 +24,6 @@ export function getPow(hex: string): number {
|
|||||||
/**
|
/**
|
||||||
* Mine an event with the desired POW. This function mutates the event.
|
* Mine an event with the desired POW. This function mutates the event.
|
||||||
* Note that this operation is synchronous and should be run in a worker context to avoid blocking the main thread.
|
* Note that this operation is synchronous and should be run in a worker context to avoid blocking the main thread.
|
||||||
*
|
|
||||||
* Adapted from Snort: https://git.v0l.io/Kieran/snort/src/commit/4df6c19248184218c4c03728d61e94dae5f2d90c/packages/system/src/pow-util.ts#L14-L36
|
|
||||||
*/
|
*/
|
||||||
export function minePow(unsigned: UnsignedEvent, difficulty: number): Omit<Event, 'sig'> {
|
export function minePow(unsigned: UnsignedEvent, difficulty: number): Omit<Event, 'sig'> {
|
||||||
let count = 0
|
let count = 0
|
||||||
@@ -41,7 +43,7 @@ export function minePow(unsigned: UnsignedEvent, difficulty: number): Omit<Event
|
|||||||
|
|
||||||
tag[1] = (++count).toString()
|
tag[1] = (++count).toString()
|
||||||
|
|
||||||
event.id = getEventHash(event)
|
event.id = fastEventHash(event)
|
||||||
|
|
||||||
if (getPow(event.id) >= difficulty) {
|
if (getPow(event.id) >= difficulty) {
|
||||||
break
|
break
|
||||||
@@ -50,3 +52,9 @@ export function minePow(unsigned: UnsignedEvent, difficulty: number): Omit<Event
|
|||||||
|
|
||||||
return event
|
return event
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function fastEventHash(evt: UnsignedEvent): string {
|
||||||
|
return bytesToHex(
|
||||||
|
sha256(utf8Encoder.encode(JSON.stringify([0, evt.pubkey, evt.created_at, evt.kind, evt.tags, evt.content]))),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
97
nip17.test.ts
Normal file
97
nip17.test.ts
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
import { test, expect } from 'bun:test'
|
||||||
|
import { getPublicKey } from './pure.ts'
|
||||||
|
import { decode } from './nip19.ts'
|
||||||
|
import { wrapEvent, wrapManyEvents, unwrapEvent } from './nip17.ts'
|
||||||
|
import { hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
|
const senderPrivateKey = decode(`nsec1p0ht6p3wepe47sjrgesyn4m50m6avk2waqudu9rl324cg2c4ufesyp6rdg`).data
|
||||||
|
|
||||||
|
const sk1 = hexToBytes('f09ac9b695d0a4c6daa418fe95b977eea20f54d9545592bc36a4f9e14f3eb840')
|
||||||
|
const sk2 = hexToBytes('5393a825e5892d8e18d4a5ea61ced105e8bb2a106f42876be3a40522e0b13747')
|
||||||
|
|
||||||
|
const recipients = [
|
||||||
|
{ publicKey: getPublicKey(sk1), relayUrl: 'wss://relay1.com' },
|
||||||
|
{ publicKey: getPublicKey(sk2) }, // No relay URL for this recipient
|
||||||
|
]
|
||||||
|
const message = 'Hello, this is a direct message!'
|
||||||
|
const conversationTitle = 'Private Group Conversation' // Optional
|
||||||
|
const replyTo = { eventId: 'previousEventId123' } // Optional, for replies
|
||||||
|
|
||||||
|
const wrappedEvent = wrapEvent(senderPrivateKey, recipients[0], message, conversationTitle, replyTo)
|
||||||
|
|
||||||
|
test('wrapEvent', () => {
|
||||||
|
const expected = {
|
||||||
|
content: '',
|
||||||
|
id: '',
|
||||||
|
created_at: 1728537932,
|
||||||
|
kind: 1059,
|
||||||
|
pubkey: '',
|
||||||
|
sig: '',
|
||||||
|
tags: [['p', 'b60849e5aae4113b236f9deb34f6f85605b4c53930651309a0d60c7ea721aad0']],
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(wrappedEvent.kind).toEqual(expected.kind)
|
||||||
|
expect(wrappedEvent.tags).toEqual(expected.tags)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('wrapManyEvents', () => {
|
||||||
|
const expected = [
|
||||||
|
{
|
||||||
|
kind: 1059,
|
||||||
|
content: '',
|
||||||
|
created_at: 1729581521,
|
||||||
|
tags: [['p', '611df01bfcf85c26ae65453b772d8f1dfd25c264621c0277e1fc1518686faef9']],
|
||||||
|
pubkey: '',
|
||||||
|
id: '',
|
||||||
|
sig: '',
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
kind: 1059,
|
||||||
|
content: '',
|
||||||
|
created_at: 1729594619,
|
||||||
|
tags: [['p', 'b60849e5aae4113b236f9deb34f6f85605b4c53930651309a0d60c7ea721aad0']],
|
||||||
|
pubkey: '',
|
||||||
|
id: '',
|
||||||
|
sig: '',
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
kind: 1059,
|
||||||
|
content: '',
|
||||||
|
created_at: 1729560014,
|
||||||
|
tags: [['p', '36f7288c84d85ca6aa189dc3581d63ce140b7eeef5ae759421c5b5a3627312db']],
|
||||||
|
pubkey: '',
|
||||||
|
id: '',
|
||||||
|
sig: '',
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
const wrappedEvents = wrapManyEvents(senderPrivateKey, recipients, message, conversationTitle, replyTo)
|
||||||
|
|
||||||
|
wrappedEvents.forEach((event, index) => {
|
||||||
|
expect(event.kind).toEqual(expected[index].kind)
|
||||||
|
expect(event.tags).toEqual(expected[index].tags)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('unwrapEvent', () => {
|
||||||
|
const expected = {
|
||||||
|
kind: 14,
|
||||||
|
content: 'Hello, this is a direct message!',
|
||||||
|
pubkey: '611df01bfcf85c26ae65453b772d8f1dfd25c264621c0277e1fc1518686faef9',
|
||||||
|
tags: [
|
||||||
|
['p', 'b60849e5aae4113b236f9deb34f6f85605b4c53930651309a0d60c7ea721aad0', 'wss://relay1.com'],
|
||||||
|
['e', 'previousEventId123', '', 'reply'],
|
||||||
|
['subject', 'Private Group Conversation'],
|
||||||
|
],
|
||||||
|
}
|
||||||
|
const result = unwrapEvent(wrappedEvent, sk1)
|
||||||
|
|
||||||
|
expect(result.kind).toEqual(expected.kind)
|
||||||
|
expect(result.content).toEqual(expected.content)
|
||||||
|
expect(result.pubkey).toEqual(expected.pubkey)
|
||||||
|
expect(result.tags).toEqual(expected.tags)
|
||||||
|
})
|
||||||
77
nip17.ts
Normal file
77
nip17.ts
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import { PrivateDirectMessage } from './kinds.ts'
|
||||||
|
import { EventTemplate, NostrEvent, getPublicKey } from './pure.ts'
|
||||||
|
import * as nip59 from './nip59.ts'
|
||||||
|
|
||||||
|
type Recipient = {
|
||||||
|
publicKey: string
|
||||||
|
relayUrl?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type ReplyTo = {
|
||||||
|
eventId: string
|
||||||
|
relayUrl?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
function createEvent(
|
||||||
|
recipients: Recipient | Recipient[],
|
||||||
|
message: string,
|
||||||
|
conversationTitle?: string,
|
||||||
|
replyTo?: ReplyTo,
|
||||||
|
): EventTemplate {
|
||||||
|
const baseEvent: EventTemplate = {
|
||||||
|
created_at: Math.ceil(Date.now() / 1000),
|
||||||
|
kind: PrivateDirectMessage,
|
||||||
|
tags: [],
|
||||||
|
content: message,
|
||||||
|
}
|
||||||
|
|
||||||
|
const recipientsArray = Array.isArray(recipients) ? recipients : [recipients]
|
||||||
|
|
||||||
|
recipientsArray.forEach(({ publicKey, relayUrl }) => {
|
||||||
|
baseEvent.tags.push(relayUrl ? ['p', publicKey, relayUrl] : ['p', publicKey])
|
||||||
|
})
|
||||||
|
|
||||||
|
if (replyTo) {
|
||||||
|
baseEvent.tags.push(['e', replyTo.eventId, replyTo.relayUrl || '', 'reply'])
|
||||||
|
}
|
||||||
|
|
||||||
|
if (conversationTitle) {
|
||||||
|
baseEvent.tags.push(['subject', conversationTitle])
|
||||||
|
}
|
||||||
|
|
||||||
|
return baseEvent
|
||||||
|
}
|
||||||
|
|
||||||
|
export function wrapEvent(
|
||||||
|
senderPrivateKey: Uint8Array,
|
||||||
|
recipient: Recipient,
|
||||||
|
message: string,
|
||||||
|
conversationTitle?: string,
|
||||||
|
replyTo?: ReplyTo,
|
||||||
|
): NostrEvent {
|
||||||
|
const event = createEvent(recipient, message, conversationTitle, replyTo)
|
||||||
|
return nip59.wrapEvent(event, senderPrivateKey, recipient.publicKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function wrapManyEvents(
|
||||||
|
senderPrivateKey: Uint8Array,
|
||||||
|
recipients: Recipient[],
|
||||||
|
message: string,
|
||||||
|
conversationTitle?: string,
|
||||||
|
replyTo?: ReplyTo,
|
||||||
|
): NostrEvent[] {
|
||||||
|
if (!recipients || recipients.length === 0) {
|
||||||
|
throw new Error('At least one recipient is required.')
|
||||||
|
}
|
||||||
|
|
||||||
|
const senderPublicKey = getPublicKey(senderPrivateKey)
|
||||||
|
|
||||||
|
// wrap the event for the sender and then for each recipient
|
||||||
|
return [{ publicKey: senderPublicKey }, ...recipients].map(recipient =>
|
||||||
|
wrapEvent(senderPrivateKey, recipient, message, conversationTitle, replyTo),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const unwrapEvent = nip59.unwrapEvent
|
||||||
|
|
||||||
|
export const unwrapManyEvents = nip59.unwrapManyEvents
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { describe, test, expect } from 'bun:test'
|
import { describe, test, expect } from 'bun:test'
|
||||||
import { hexToBytes } from '@noble/hashes/utils'
|
import { hexToBytes } from '@noble/hashes/utils'
|
||||||
import { finalizeEvent, getPublicKey } from './pure.ts'
|
import { EventTemplate, finalizeEvent, getPublicKey } from './pure.ts'
|
||||||
import { Repost, ShortTextNote } from './kinds.ts'
|
import { GenericRepost, Repost, ShortTextNote, BadgeDefinition as BadgeDefinitionKind } from './kinds.ts'
|
||||||
import { finishRepostEvent, getRepostedEventPointer, getRepostedEvent } from './nip18.ts'
|
import { finishRepostEvent, getRepostedEventPointer, getRepostedEvent } from './nip18.ts'
|
||||||
import { buildEvent } from './test-helpers.ts'
|
import { buildEvent } from './test-helpers.ts'
|
||||||
|
|
||||||
@@ -86,6 +86,51 @@ describe('finishRepostEvent + getRepostedEventPointer + getRepostedEvent', () =>
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('GenericRepost', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const eventTemplate: EventTemplate = {
|
||||||
|
content: '',
|
||||||
|
created_at: 1617932114,
|
||||||
|
kind: BadgeDefinitionKind,
|
||||||
|
tags: [
|
||||||
|
['d', 'badge-id'],
|
||||||
|
['name', 'Badge Name'],
|
||||||
|
['description', 'Badge Description'],
|
||||||
|
['image', 'https://example.com/badge.png', '1024x1024'],
|
||||||
|
['thumb', 'https://example.com/thumb.png', '100x100'],
|
||||||
|
['thumb', 'https://example.com/thumb2.png', '200x200'],
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
const repostedEvent = finalizeEvent(eventTemplate, privateKey)
|
||||||
|
test('should create a generic reposted event', () => {
|
||||||
|
const template = { created_at: 1617932115 }
|
||||||
|
const event = finishRepostEvent(template, repostedEvent, relayUrl, privateKey)
|
||||||
|
|
||||||
|
expect(event.kind).toEqual(GenericRepost)
|
||||||
|
expect(event.tags).toEqual([
|
||||||
|
['e', repostedEvent.id, relayUrl],
|
||||||
|
['p', repostedEvent.pubkey],
|
||||||
|
['k', '30009'],
|
||||||
|
])
|
||||||
|
expect(event.content).toEqual(JSON.stringify(repostedEvent))
|
||||||
|
expect(event.created_at).toEqual(template.created_at)
|
||||||
|
expect(event.pubkey).toEqual(publicKey)
|
||||||
|
|
||||||
|
const repostedEventPointer = getRepostedEventPointer(event)
|
||||||
|
|
||||||
|
expect(repostedEventPointer!.id).toEqual(repostedEvent.id)
|
||||||
|
expect(repostedEventPointer!.author).toEqual(repostedEvent.pubkey)
|
||||||
|
expect(repostedEventPointer!.relays).toEqual([relayUrl])
|
||||||
|
|
||||||
|
const repostedEventFromContent = getRepostedEvent(event)
|
||||||
|
|
||||||
|
expect(repostedEventFromContent).toEqual(repostedEvent)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('getRepostedEventPointer', () => {
|
describe('getRepostedEventPointer', () => {
|
||||||
test('should parse an event with only an `e` tag', () => {
|
test('should parse an event with only an `e` tag', () => {
|
||||||
const event = buildEvent({
|
const event = buildEvent({
|
||||||
@@ -100,3 +145,26 @@ describe('getRepostedEventPointer', () => {
|
|||||||
expect(repostedEventPointer!.relays).toEqual([relayUrl])
|
expect(repostedEventPointer!.relays).toEqual([relayUrl])
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('finishRepostEvent', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
|
||||||
|
test('should create an event with empty content if the reposted event is protected', () => {
|
||||||
|
const repostedEvent = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [['-']],
|
||||||
|
content: 'Replied to a post',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
const template = {
|
||||||
|
created_at: 1617932115,
|
||||||
|
}
|
||||||
|
|
||||||
|
const event = finishRepostEvent(template, repostedEvent, relayUrl, privateKey)
|
||||||
|
|
||||||
|
expect(event.content).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|||||||
21
nip18.ts
21
nip18.ts
@@ -1,6 +1,6 @@
|
|||||||
import { Event, finalizeEvent, verifyEvent } from './pure.ts'
|
import { GenericRepost, Repost, ShortTextNote } from './kinds.ts'
|
||||||
import { Repost } from './kinds.ts'
|
|
||||||
import { EventPointer } from './nip19.ts'
|
import { EventPointer } from './nip19.ts'
|
||||||
|
import { Event, finalizeEvent, verifyEvent } from './pure.ts'
|
||||||
|
|
||||||
export type RepostEventTemplate = {
|
export type RepostEventTemplate = {
|
||||||
/**
|
/**
|
||||||
@@ -25,11 +25,20 @@ export function finishRepostEvent(
|
|||||||
relayUrl: string,
|
relayUrl: string,
|
||||||
privateKey: Uint8Array,
|
privateKey: Uint8Array,
|
||||||
): Event {
|
): Event {
|
||||||
|
let kind: Repost | GenericRepost
|
||||||
|
const tags = [...(t.tags ?? []), ['e', reposted.id, relayUrl], ['p', reposted.pubkey]]
|
||||||
|
if (reposted.kind === ShortTextNote) {
|
||||||
|
kind = Repost
|
||||||
|
} else {
|
||||||
|
kind = GenericRepost
|
||||||
|
tags.push(['k', String(reposted.kind)])
|
||||||
|
}
|
||||||
|
|
||||||
return finalizeEvent(
|
return finalizeEvent(
|
||||||
{
|
{
|
||||||
kind: Repost,
|
kind,
|
||||||
tags: [...(t.tags ?? []), ['e', reposted.id, relayUrl], ['p', reposted.pubkey]],
|
tags,
|
||||||
content: t.content === '' ? '' : JSON.stringify(reposted),
|
content: t.content === '' || reposted.tags?.find(tag => tag[0] === '-') ? '' : JSON.stringify(reposted),
|
||||||
created_at: t.created_at,
|
created_at: t.created_at,
|
||||||
},
|
},
|
||||||
privateKey,
|
privateKey,
|
||||||
@@ -37,7 +46,7 @@ export function finishRepostEvent(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function getRepostedEventPointer(event: Event): undefined | EventPointer {
|
export function getRepostedEventPointer(event: Event): undefined | EventPointer {
|
||||||
if (event.kind !== Repost) {
|
if (![Repost, GenericRepost].includes(event.kind)) {
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
164
nip19.test.ts
164
nip19.test.ts
@@ -1,17 +1,14 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { describe, expect, test } from 'bun:test'
|
||||||
import { generateSecretKey, getPublicKey } from './pure.ts'
|
|
||||||
import {
|
import {
|
||||||
decode,
|
decode,
|
||||||
naddrEncode,
|
naddrEncode,
|
||||||
|
neventEncode,
|
||||||
|
NostrTypeGuard,
|
||||||
nprofileEncode,
|
nprofileEncode,
|
||||||
npubEncode,
|
npubEncode,
|
||||||
nrelayEncode,
|
nsecEncode
|
||||||
nsecEncode,
|
|
||||||
neventEncode,
|
|
||||||
type AddressPointer,
|
|
||||||
type ProfilePointer,
|
|
||||||
EventPointer,
|
|
||||||
} from './nip19.ts'
|
} from './nip19.ts'
|
||||||
|
import { generateSecretKey, getPublicKey } from './pure.ts'
|
||||||
|
|
||||||
test('encode and decode nsec', () => {
|
test('encode and decode nsec', () => {
|
||||||
let sk = generateSecretKey()
|
let sk = generateSecretKey()
|
||||||
@@ -38,7 +35,7 @@ test('encode and decode nprofile', () => {
|
|||||||
expect(nprofile).toMatch(/nprofile1\w+/)
|
expect(nprofile).toMatch(/nprofile1\w+/)
|
||||||
let { type, data } = decode(nprofile)
|
let { type, data } = decode(nprofile)
|
||||||
expect(type).toEqual('nprofile')
|
expect(type).toEqual('nprofile')
|
||||||
const pointer = data as ProfilePointer
|
const pointer = data
|
||||||
expect(pointer.pubkey).toEqual(pk)
|
expect(pointer.pubkey).toEqual(pk)
|
||||||
expect(pointer.relays).toContain(relays[0])
|
expect(pointer.relays).toContain(relays[0])
|
||||||
expect(pointer.relays).toContain(relays[1])
|
expect(pointer.relays).toContain(relays[1])
|
||||||
@@ -67,7 +64,7 @@ test('encode and decode naddr', () => {
|
|||||||
expect(naddr).toMatch(/naddr1\w+/)
|
expect(naddr).toMatch(/naddr1\w+/)
|
||||||
let { type, data } = decode(naddr)
|
let { type, data } = decode(naddr)
|
||||||
expect(type).toEqual('naddr')
|
expect(type).toEqual('naddr')
|
||||||
const pointer = data as AddressPointer
|
const pointer = data
|
||||||
expect(pointer.pubkey).toEqual(pk)
|
expect(pointer.pubkey).toEqual(pk)
|
||||||
expect(pointer.relays).toContain(relays[0])
|
expect(pointer.relays).toContain(relays[0])
|
||||||
expect(pointer.relays).toContain(relays[1])
|
expect(pointer.relays).toContain(relays[1])
|
||||||
@@ -86,7 +83,7 @@ test('encode and decode nevent', () => {
|
|||||||
expect(nevent).toMatch(/nevent1\w+/)
|
expect(nevent).toMatch(/nevent1\w+/)
|
||||||
let { type, data } = decode(nevent)
|
let { type, data } = decode(nevent)
|
||||||
expect(type).toEqual('nevent')
|
expect(type).toEqual('nevent')
|
||||||
const pointer = data as EventPointer
|
const pointer = data
|
||||||
expect(pointer.id).toEqual(pk)
|
expect(pointer.id).toEqual(pk)
|
||||||
expect(pointer.relays).toContain(relays[0])
|
expect(pointer.relays).toContain(relays[0])
|
||||||
expect(pointer.kind).toEqual(30023)
|
expect(pointer.kind).toEqual(30023)
|
||||||
@@ -103,7 +100,7 @@ test('encode and decode nevent with kind 0', () => {
|
|||||||
expect(nevent).toMatch(/nevent1\w+/)
|
expect(nevent).toMatch(/nevent1\w+/)
|
||||||
let { type, data } = decode(nevent)
|
let { type, data } = decode(nevent)
|
||||||
expect(type).toEqual('nevent')
|
expect(type).toEqual('nevent')
|
||||||
const pointer = data as EventPointer
|
const pointer = data
|
||||||
expect(pointer.id).toEqual(pk)
|
expect(pointer.id).toEqual(pk)
|
||||||
expect(pointer.relays).toContain(relays[0])
|
expect(pointer.relays).toContain(relays[0])
|
||||||
expect(pointer.kind).toEqual(0)
|
expect(pointer.kind).toEqual(0)
|
||||||
@@ -121,7 +118,7 @@ test('encode and decode naddr with empty "d"', () => {
|
|||||||
expect(naddr).toMatch(/naddr\w+/)
|
expect(naddr).toMatch(/naddr\w+/)
|
||||||
let { type, data } = decode(naddr)
|
let { type, data } = decode(naddr)
|
||||||
expect(type).toEqual('naddr')
|
expect(type).toEqual('naddr')
|
||||||
const pointer = data as AddressPointer
|
const pointer = data
|
||||||
expect(pointer.identifier).toEqual('')
|
expect(pointer.identifier).toEqual('')
|
||||||
expect(pointer.relays).toContain(relays[0])
|
expect(pointer.relays).toContain(relays[0])
|
||||||
expect(pointer.kind).toEqual(3)
|
expect(pointer.kind).toEqual(3)
|
||||||
@@ -133,7 +130,7 @@ test('decode naddr from habla.news', () => {
|
|||||||
'naddr1qq98yetxv4ex2mnrv4esygrl54h466tz4v0re4pyuavvxqptsejl0vxcmnhfl60z3rth2xkpjspsgqqqw4rsf34vl5',
|
'naddr1qq98yetxv4ex2mnrv4esygrl54h466tz4v0re4pyuavvxqptsejl0vxcmnhfl60z3rth2xkpjspsgqqqw4rsf34vl5',
|
||||||
)
|
)
|
||||||
expect(type).toEqual('naddr')
|
expect(type).toEqual('naddr')
|
||||||
const pointer = data as AddressPointer
|
const pointer = data
|
||||||
expect(pointer.pubkey).toEqual('7fa56f5d6962ab1e3cd424e758c3002b8665f7b0d8dcee9fe9e288d7751ac194')
|
expect(pointer.pubkey).toEqual('7fa56f5d6962ab1e3cd424e758c3002b8665f7b0d8dcee9fe9e288d7751ac194')
|
||||||
expect(pointer.kind).toEqual(30023)
|
expect(pointer.kind).toEqual(30023)
|
||||||
expect(pointer.identifier).toEqual('references')
|
expect(pointer.identifier).toEqual('references')
|
||||||
@@ -145,7 +142,7 @@ test('decode naddr from go-nostr with different TLV ordering', () => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
expect(type).toEqual('naddr')
|
expect(type).toEqual('naddr')
|
||||||
const pointer = data as AddressPointer
|
const pointer = data
|
||||||
expect(pointer.pubkey).toEqual('3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d')
|
expect(pointer.pubkey).toEqual('3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d')
|
||||||
expect(pointer.relays).toContain('wss://relay.nostr.example.mydomain.example.com')
|
expect(pointer.relays).toContain('wss://relay.nostr.example.mydomain.example.com')
|
||||||
expect(pointer.relays).toContain('wss://nostr.banana.com')
|
expect(pointer.relays).toContain('wss://nostr.banana.com')
|
||||||
@@ -153,11 +150,134 @@ test('decode naddr from go-nostr with different TLV ordering', () => {
|
|||||||
expect(pointer.identifier).toEqual('banana')
|
expect(pointer.identifier).toEqual('banana')
|
||||||
})
|
})
|
||||||
|
|
||||||
test('encode and decode nrelay', () => {
|
describe('NostrTypeGuard', () => {
|
||||||
let url = 'wss://relay.nostr.example'
|
test('isNProfile', () => {
|
||||||
let nrelay = nrelayEncode(url)
|
const is = NostrTypeGuard.isNProfile('nprofile1qqsvc6ulagpn7kwrcwdqgp797xl7usumqa6s3kgcelwq6m75x8fe8yc5usxdg')
|
||||||
expect(nrelay).toMatch(/nrelay1\w+/)
|
|
||||||
let { type, data } = decode(nrelay)
|
expect(is).toBeTrue()
|
||||||
expect(type).toEqual('nrelay')
|
})
|
||||||
expect(data).toEqual(url)
|
|
||||||
|
test('isNProfile invalid nprofile', () => {
|
||||||
|
const is = NostrTypeGuard.isNProfile('nprofile1qqsvc6ulagpn7kwrcwdqgp797xl7usumqa6s3kgcelwq6m75x8fe8yc5usxãg')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNProfile with invalid nprofile', () => {
|
||||||
|
const is = NostrTypeGuard.isNProfile('nsec1lqw6zqyanj9mz8gwhdam6tqge42vptz4zg93qsfej440xm5h5esqya0juv')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNEvent', () => {
|
||||||
|
const is = NostrTypeGuard.isNEvent(
|
||||||
|
'nevent1qqst8cujky046negxgwwm5ynqwn53t8aqjr6afd8g59nfqwxpdhylpcpzamhxue69uhhyetvv9ujuetcv9khqmr99e3k7mg8arnc9',
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNEvent with invalid nevent', () => {
|
||||||
|
const is = NostrTypeGuard.isNEvent(
|
||||||
|
'nevent1qqst8cujky046negxgwwm5ynqwn53t8aqjr6afd8g59nfqwxpdhylpcpzamhxue69uhhyetvv9ujuetcv9khqmr99e3k7mg8ãrnc9',
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNEvent with invalid nevent', () => {
|
||||||
|
const is = NostrTypeGuard.isNEvent('nprofile1qqsvc6ulagpn7kwrcwdqgp797xl7usumqa6s3kgcelwq6m75x8fe8yc5usxdg')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNAddr', () => {
|
||||||
|
const is = NostrTypeGuard.isNAddr(
|
||||||
|
'naddr1qqxnzdesxqmnxvpexqunzvpcqyt8wumn8ghj7un9d3shjtnwdaehgu3wvfskueqzypve7elhmamff3sr5mgxxms4a0rppkmhmn7504h96pfcdkpplvl2jqcyqqq823cnmhuld',
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNAddr with invalid nadress', () => {
|
||||||
|
const is = NostrTypeGuard.isNAddr('nsec1lqw6zqyanj9mz8gwhdam6tqge42vptz4zg93qsfej440xm5h5esqya0juv')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNSec', () => {
|
||||||
|
const is = NostrTypeGuard.isNSec('nsec1lqw6zqyanj9mz8gwhdam6tqge42vptz4zg93qsfej440xm5h5esqya0juv')
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNSec with invalid nsec', () => {
|
||||||
|
const is = NostrTypeGuard.isNSec('nsec1lqw6zqyanj9mz8gwhdam6tqge42vptz4zg93qsfej440xm5h5esqya0juã')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNSec with invalid nsec', () => {
|
||||||
|
const is = NostrTypeGuard.isNSec('nprofile1qqsvc6ulagpn7kwrcwdqgp797xl7usumqa6s3kgcelwq6m75x8fe8yc5usxdg')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNPub', () => {
|
||||||
|
const is = NostrTypeGuard.isNPub('npub1jz5mdljkmffmqjshpyjgqgrhdkuxd9ztzasv8xeh5q92fv33sjgqy4pats')
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNPub with invalid npub', () => {
|
||||||
|
const is = NostrTypeGuard.isNPub('npub1jz5mdljkmffmqjshpyjgqgrhdkuxd9ztzãsv8xeh5q92fv33sjgqy4pats')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNPub with invalid npub', () => {
|
||||||
|
const is = NostrTypeGuard.isNPub('nsec1lqw6zqyanj9mz8gwhdam6tqge42vptz4zg93qsfej440xm5h5esqya0juv')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNote', () => {
|
||||||
|
const is = NostrTypeGuard.isNote('note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky')
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNote with invalid note', () => {
|
||||||
|
const is = NostrTypeGuard.isNote('note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sçlreky')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNote with invalid note', () => {
|
||||||
|
const is = NostrTypeGuard.isNote('npub1jz5mdljkmffmqjshpyjgqgrhdkuxd9ztzasv8xeh5q92fv33sjgqy4pats')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNcryptsec', () => {
|
||||||
|
const is = NostrTypeGuard.isNcryptsec(
|
||||||
|
'ncryptsec1qgg9947rlpvqu76pj5ecreduf9jxhselq2nae2kghhvd5g7dgjtcxfqtd67p9m0w57lspw8gsq6yphnm8623nsl8xn9j4jdzz84zm3frztj3z7s35vpzmqf6ksu8r89qk5z2zxfmu5gv8th8wclt0h4p',
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(is).toBeTrue()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNcryptsec with invalid ncrytpsec', () => {
|
||||||
|
const is = NostrTypeGuard.isNcryptsec(
|
||||||
|
'ncryptsec1qgg9947rlpvqu76pj5ecreduf9jxhselq2nae2kghhvd5g7dgjtcxfqtd67p9m0w57lspw8gsq6yphnm8623nsã8xn9j4jdzz84zm3frztj3z7s35vpzmqf6ksu8r89qk5z2zxfmu5gv8th8wclt0h4p',
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('isNcryptsec with invalid ncrytpsec', () => {
|
||||||
|
const is = NostrTypeGuard.isNcryptsec('note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sçlreky')
|
||||||
|
|
||||||
|
expect(is).toBeFalse()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
110
nip19.ts
110
nip19.ts
@@ -3,6 +3,24 @@ import { bech32 } from '@scure/base'
|
|||||||
|
|
||||||
import { utf8Decoder, utf8Encoder } from './utils.ts'
|
import { utf8Decoder, utf8Encoder } from './utils.ts'
|
||||||
|
|
||||||
|
export type NProfile = `nprofile1${string}`
|
||||||
|
export type NEvent = `nevent1${string}`
|
||||||
|
export type NAddr = `naddr1${string}`
|
||||||
|
export type NSec = `nsec1${string}`
|
||||||
|
export type NPub = `npub1${string}`
|
||||||
|
export type Note = `note1${string}`
|
||||||
|
export type Ncryptsec = `ncryptsec1${string}`
|
||||||
|
|
||||||
|
export const NostrTypeGuard = {
|
||||||
|
isNProfile: (value?: string | null): value is NProfile => /^nprofile1[a-z\d]+$/.test(value || ''),
|
||||||
|
isNEvent: (value?: string | null): value is NEvent => /^nevent1[a-z\d]+$/.test(value || ''),
|
||||||
|
isNAddr: (value?: string | null): value is NAddr => /^naddr1[a-z\d]+$/.test(value || ''),
|
||||||
|
isNSec: (value?: string | null): value is NSec => /^nsec1[a-z\d]{58}$/.test(value || ''),
|
||||||
|
isNPub: (value?: string | null): value is NPub => /^npub1[a-z\d]{58}$/.test(value || ''),
|
||||||
|
isNote: (value?: string | null): value is Note => /^note1[a-z\d]+$/.test(value || ''),
|
||||||
|
isNcryptsec: (value?: string | null): value is Ncryptsec => /^ncryptsec1[a-z\d]+$/.test(value || ''),
|
||||||
|
}
|
||||||
|
|
||||||
export const Bech32MaxSize = 5000
|
export const Bech32MaxSize = 5000
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -43,29 +61,56 @@ export type AddressPointer = {
|
|||||||
relays?: string[]
|
relays?: string[]
|
||||||
}
|
}
|
||||||
|
|
||||||
type Prefixes = {
|
export function decodeNostrURI(nip19code: string): ReturnType<typeof decode> | { type: 'invalid'; data: null } {
|
||||||
nprofile: ProfilePointer
|
try {
|
||||||
nrelay: string
|
if (nip19code.startsWith('nostr:')) nip19code = nip19code.substring(6)
|
||||||
nevent: EventPointer
|
return decode(nip19code)
|
||||||
naddr: AddressPointer
|
} catch (_err) {
|
||||||
nsec: Uint8Array
|
return { type: 'invalid', data: null }
|
||||||
npub: string
|
}
|
||||||
note: string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type DecodeValue<Prefix extends keyof Prefixes> = {
|
export type DecodedNevent = {
|
||||||
type: Prefix
|
type: 'nevent'
|
||||||
data: Prefixes[Prefix]
|
data: EventPointer
|
||||||
}
|
}
|
||||||
|
|
||||||
export type DecodeResult = {
|
export type DecodedNprofile = {
|
||||||
[P in keyof Prefixes]: DecodeValue<P>
|
type: 'nprofile'
|
||||||
}[keyof Prefixes]
|
data: ProfilePointer
|
||||||
|
}
|
||||||
|
|
||||||
export function decode<Prefix extends keyof Prefixes>(nip19: `${Prefix}1${string}`): DecodeValue<Prefix>
|
export type DecodedNaddr = {
|
||||||
export function decode(nip19: string): DecodeResult
|
type: 'naddr'
|
||||||
export function decode(nip19: string): DecodeResult {
|
data: AddressPointer
|
||||||
let { prefix, words } = bech32.decode(nip19, Bech32MaxSize)
|
}
|
||||||
|
|
||||||
|
export type DecodedNsec = {
|
||||||
|
type: 'nsec'
|
||||||
|
data: Uint8Array
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DecodedNpub = {
|
||||||
|
type: 'npub'
|
||||||
|
data: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DecodedNote = {
|
||||||
|
type: 'note'
|
||||||
|
data: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DecodedResult = DecodedNevent | DecodedNprofile | DecodedNaddr | DecodedNpub | DecodedNsec | DecodedNote
|
||||||
|
|
||||||
|
export function decode(nip19: NEvent): DecodedNevent
|
||||||
|
export function decode(nip19: NProfile): DecodedNprofile
|
||||||
|
export function decode(nip19: NAddr): DecodedNaddr
|
||||||
|
export function decode(nip19: NSec): DecodedNsec
|
||||||
|
export function decode(nip19: NPub): DecodedNpub
|
||||||
|
export function decode(nip19: Note): DecodedNote
|
||||||
|
export function decode(code: string): DecodedResult
|
||||||
|
export function decode(code: string): DecodedResult {
|
||||||
|
let { prefix, words } = bech32.decode(code, Bech32MaxSize)
|
||||||
let data = new Uint8Array(bech32.fromWords(words))
|
let data = new Uint8Array(bech32.fromWords(words))
|
||||||
|
|
||||||
switch (prefix) {
|
switch (prefix) {
|
||||||
@@ -119,16 +164,6 @@ export function decode(nip19: string): DecodeResult {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case 'nrelay': {
|
|
||||||
let tlv = parseTLV(data)
|
|
||||||
if (!tlv[0]?.[0]) throw new Error('missing TLV 0 for nrelay')
|
|
||||||
|
|
||||||
return {
|
|
||||||
type: 'nrelay',
|
|
||||||
data: utf8Decoder.decode(tlv[0][0]),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'nsec':
|
case 'nsec':
|
||||||
return { type: prefix, data }
|
return { type: prefix, data }
|
||||||
|
|
||||||
@@ -158,15 +193,15 @@ function parseTLV(data: Uint8Array): TLV {
|
|||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
export function nsecEncode(key: Uint8Array): `nsec1${string}` {
|
export function nsecEncode(key: Uint8Array): NSec {
|
||||||
return encodeBytes('nsec', key)
|
return encodeBytes('nsec', key)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function npubEncode(hex: string): `npub1${string}` {
|
export function npubEncode(hex: string): NPub {
|
||||||
return encodeBytes('npub', hexToBytes(hex))
|
return encodeBytes('npub', hexToBytes(hex))
|
||||||
}
|
}
|
||||||
|
|
||||||
export function noteEncode(hex: string): `note1${string}` {
|
export function noteEncode(hex: string): Note {
|
||||||
return encodeBytes('note', hexToBytes(hex))
|
return encodeBytes('note', hexToBytes(hex))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -179,7 +214,7 @@ export function encodeBytes<Prefix extends string>(prefix: Prefix, bytes: Uint8A
|
|||||||
return encodeBech32(prefix, bytes)
|
return encodeBech32(prefix, bytes)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function nprofileEncode(profile: ProfilePointer): `nprofile1${string}` {
|
export function nprofileEncode(profile: ProfilePointer): NProfile {
|
||||||
let data = encodeTLV({
|
let data = encodeTLV({
|
||||||
0: [hexToBytes(profile.pubkey)],
|
0: [hexToBytes(profile.pubkey)],
|
||||||
1: (profile.relays || []).map(url => utf8Encoder.encode(url)),
|
1: (profile.relays || []).map(url => utf8Encoder.encode(url)),
|
||||||
@@ -187,7 +222,7 @@ export function nprofileEncode(profile: ProfilePointer): `nprofile1${string}` {
|
|||||||
return encodeBech32('nprofile', data)
|
return encodeBech32('nprofile', data)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function neventEncode(event: EventPointer): `nevent1${string}` {
|
export function neventEncode(event: EventPointer): NEvent {
|
||||||
let kindArray
|
let kindArray
|
||||||
if (event.kind !== undefined) {
|
if (event.kind !== undefined) {
|
||||||
kindArray = integerToUint8Array(event.kind)
|
kindArray = integerToUint8Array(event.kind)
|
||||||
@@ -203,7 +238,7 @@ export function neventEncode(event: EventPointer): `nevent1${string}` {
|
|||||||
return encodeBech32('nevent', data)
|
return encodeBech32('nevent', data)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function naddrEncode(addr: AddressPointer): `naddr1${string}` {
|
export function naddrEncode(addr: AddressPointer): NAddr {
|
||||||
let kind = new ArrayBuffer(4)
|
let kind = new ArrayBuffer(4)
|
||||||
new DataView(kind).setUint32(0, addr.kind, false)
|
new DataView(kind).setUint32(0, addr.kind, false)
|
||||||
|
|
||||||
@@ -216,13 +251,6 @@ export function naddrEncode(addr: AddressPointer): `naddr1${string}` {
|
|||||||
return encodeBech32('naddr', data)
|
return encodeBech32('naddr', data)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function nrelayEncode(url: string): `nrelay1${string}` {
|
|
||||||
let data = encodeTLV({
|
|
||||||
0: [utf8Encoder.encode(url)],
|
|
||||||
})
|
|
||||||
return encodeBech32('nrelay', data)
|
|
||||||
}
|
|
||||||
|
|
||||||
function encodeTLV(tlv: TLV): Uint8Array {
|
function encodeTLV(tlv: TLV): Uint8Array {
|
||||||
let entries: Uint8Array[] = []
|
let entries: Uint8Array[] = []
|
||||||
|
|
||||||
|
|||||||
28
nip21.ts
28
nip21.ts
@@ -1,4 +1,4 @@
|
|||||||
import { BECH32_REGEX, decode, type DecodeResult } from './nip19.ts'
|
import { AddressPointer, BECH32_REGEX, decode, EventPointer, ProfilePointer } from './nip19.ts'
|
||||||
|
|
||||||
/** Nostr URI regex, eg `nostr:npub1...` */
|
/** Nostr URI regex, eg `nostr:npub1...` */
|
||||||
export const NOSTR_URI_REGEX: RegExp = new RegExp(`nostr:(${BECH32_REGEX.source})`)
|
export const NOSTR_URI_REGEX: RegExp = new RegExp(`nostr:(${BECH32_REGEX.source})`)
|
||||||
@@ -15,7 +15,31 @@ export interface NostrURI {
|
|||||||
/** The bech32-encoded data (eg `npub1...`). */
|
/** The bech32-encoded data (eg `npub1...`). */
|
||||||
value: string
|
value: string
|
||||||
/** Decoded bech32 string, according to NIP-19. */
|
/** Decoded bech32 string, according to NIP-19. */
|
||||||
decoded: DecodeResult
|
decoded:
|
||||||
|
| {
|
||||||
|
type: 'nevent'
|
||||||
|
data: EventPointer
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'nprofile'
|
||||||
|
data: ProfilePointer
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'naddr'
|
||||||
|
data: AddressPointer
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'npub'
|
||||||
|
data: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'nsec'
|
||||||
|
data: Uint8Array
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'note'
|
||||||
|
data: string
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Parse and decode a Nostr URI. */
|
/** Parse and decode a Nostr URI. */
|
||||||
|
|||||||
117
nip27.test.ts
117
nip27.test.ts
@@ -1,68 +1,77 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { test, expect } from 'bun:test'
|
||||||
import { matchAll, replaceAll } from './nip27.ts'
|
import { parse } from './nip27.ts'
|
||||||
|
|
||||||
test('matchAll', () => {
|
test('first: parse simple content with 1 url and 1 nostr uri', () => {
|
||||||
const result = matchAll(
|
const content = `nostr:npub1hpslpc8c5sp3e2nhm2fr7swsfqpys5vyjar5dwpn7e7decps6r8qkcln63 check out my profile:nostr:npub1xtscya34g58tk0z605fvr788k263gsu6cy9x0mhnm87echrgufzsevkk5s; and this cool image https://images.com/image.jpg`
|
||||||
'Hello nostr:npub108pv4cg5ag52nq082kd5leu9ffrn2gdg6g4xdwatn73y36uzplmq9uyev6!\n\nnostr:note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
const blocks = Array.from(parse(content))
|
||||||
)
|
|
||||||
|
|
||||||
expect([...result]).toEqual([
|
expect(blocks).toEqual([
|
||||||
{
|
{ type: 'reference', pointer: { pubkey: 'b861f0e0f8a4031caa77da923f41d04802485184974746b833f67cdce030d0ce' } },
|
||||||
uri: 'nostr:npub108pv4cg5ag52nq082kd5leu9ffrn2gdg6g4xdwatn73y36uzplmq9uyev6',
|
{ type: 'text', text: ' check out my profile:' },
|
||||||
value: 'npub108pv4cg5ag52nq082kd5leu9ffrn2gdg6g4xdwatn73y36uzplmq9uyev6',
|
{ type: 'reference', pointer: { pubkey: '32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245' } },
|
||||||
decoded: {
|
{ type: 'text', text: '; and this cool image ' },
|
||||||
type: 'npub',
|
{ type: 'image', url: 'https://images.com/image.jpg' },
|
||||||
data: '79c2cae114ea28a981e7559b4fe7854a473521a8d22a66bbab9fa248eb820ff6',
|
|
||||||
},
|
|
||||||
start: 6,
|
|
||||||
end: 75,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
uri: 'nostr:note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
|
||||||
value: 'note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
|
||||||
decoded: {
|
|
||||||
type: 'note',
|
|
||||||
data: '46d731680add2990efe1cc619dc9b8014feeb23261ab9dee50e9d11814de5a2b',
|
|
||||||
},
|
|
||||||
start: 78,
|
|
||||||
end: 147,
|
|
||||||
},
|
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
|
||||||
test('matchAll with an invalid nip19', () => {
|
test('second: parse content with 3 urls of different types', () => {
|
||||||
const result = matchAll(
|
const content = `:wss://oa.ao; this was a relay and now here's a video -> https://videos.com/video.mp4! and some music:
|
||||||
'Hello nostr:npub129tvj896hqqkljerxkccpj9flshwnw999v9uwn9lfmwlj8vnzwgq9y5llnpub1rujdpkd8mwezrvpqd2rx2zphfaztqrtsfg6w3vdnlj!\n\nnostr:note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
http://music.com/song.mp3
|
||||||
)
|
and a regular link: https://regular.com/page?ok=true. and now a broken link: https://kjxkxk and a broken nostr ref: nostr:nevent1qqsr0f9w78uyy09qwmjt0kv63j4l7sxahq33725lqyyp79whlfjurwspz4mhxue69uhh56nzv34hxcfwv9ehw6nyddhq0ag9xg and a fake nostr ref: nostr:llll ok but finally https://ok.com!`
|
||||||
|
const blocks = Array.from(parse(content))
|
||||||
|
|
||||||
expect([...result]).toEqual([
|
expect(blocks).toEqual([
|
||||||
|
{ type: 'text', text: ':' },
|
||||||
|
{ type: 'relay', url: 'wss://oa.ao/' },
|
||||||
|
{ type: 'text', text: "; this was a relay and now here's a video -> " },
|
||||||
|
{ type: 'video', url: 'https://videos.com/video.mp4' },
|
||||||
|
{ type: 'text', text: '! and some music:\n' },
|
||||||
|
{ type: 'audio', url: 'http://music.com/song.mp3' },
|
||||||
|
{ type: 'text', text: '\nand a regular link: ' },
|
||||||
|
{ type: 'url', url: 'https://regular.com/page?ok=true' },
|
||||||
{
|
{
|
||||||
decoded: {
|
type: 'text',
|
||||||
data: '46d731680add2990efe1cc619dc9b8014feeb23261ab9dee50e9d11814de5a2b',
|
text: '. and now a broken link: https://kjxkxk and a broken nostr ref: nostr:nevent1qqsr0f9w78uyy09qwmjt0kv63j4l7sxahq33725lqyyp79whlfjurwspz4mhxue69uhh56nzv34hxcfwv9ehw6nyddhq0ag9xg and a fake nostr ref: nostr:llll ok but finally ',
|
||||||
type: 'note',
|
|
||||||
},
|
|
||||||
end: 193,
|
|
||||||
start: 124,
|
|
||||||
uri: 'nostr:note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
|
||||||
value: 'note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky',
|
|
||||||
},
|
},
|
||||||
|
{ type: 'url', url: 'https://ok.com/' },
|
||||||
|
{ type: 'text', text: '!' },
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
|
||||||
test('replaceAll', () => {
|
test('third: parse complex content with 4 nostr uris and 3 urls', () => {
|
||||||
const content =
|
const content = `Look at these profiles nostr:npub1xtscya34g58tk0z605fvr788k263gsu6cy9x0mhnm87echrgufzsevkk5s nostr:nprofile1qqs8z4gwdjp6jwqlxhzk35dgpcgl50swljtal58q796f9ghdkexr02gppamhxue69uhhzamfv46jucm0d574e4uy check this event nostr:nevent1qqsr0f9w78uyy09qwmjt0kv63j4l7sxahq33725lqyyp79whlfjurwspz4mhxue69uhh56nzv34hxcfwv9ehw6nyddhq0ag9xl
|
||||||
'Hello nostr:npub108pv4cg5ag52nq082kd5leu9ffrn2gdg6g4xdwatn73y36uzplmq9uyev6!\n\nnostr:note1gmtnz6q2m55epmlpe3semjdcq987av3jvx4emmjsa8g3s9x7tg4sclreky'
|
here's an image https://example.com/pic.png and another profile nostr:npub1xtscya34g58tk0z605fvr788k263gsu6cy9x0mhnm87echrgufzsevkk5s
|
||||||
|
with a video https://example.com/vid.webm and finally https://example.com/docs`
|
||||||
|
const blocks = Array.from(parse(content))
|
||||||
|
|
||||||
const result = replaceAll(content, ({ decoded, value }) => {
|
expect(blocks).toEqual([
|
||||||
switch (decoded.type) {
|
{ type: 'text', text: 'Look at these profiles ' },
|
||||||
case 'npub':
|
{ type: 'reference', pointer: { pubkey: '32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245' } },
|
||||||
return '@alex'
|
{ type: 'text', text: ' ' },
|
||||||
case 'note':
|
{
|
||||||
return '!1234'
|
type: 'reference',
|
||||||
default:
|
pointer: {
|
||||||
return value
|
pubkey: '71550e6c83a9381f35c568d1a80e11fa3e0efc97dfd0e0f17492a2edb64c37a9',
|
||||||
}
|
relays: ['wss://qwieu.com'],
|
||||||
})
|
},
|
||||||
|
},
|
||||||
expect(result).toEqual('Hello @alex!\n\n!1234')
|
{ type: 'text', text: ' check this event ' },
|
||||||
|
{
|
||||||
|
type: 'reference',
|
||||||
|
pointer: {
|
||||||
|
id: '37a4aef1f8423ca076e4b7d99a8cabff40ddb8231f2a9f01081f15d7fa65c1ba',
|
||||||
|
relays: ['wss://zjbdksa.aswjdkn'],
|
||||||
|
author: undefined,
|
||||||
|
kind: undefined,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ type: 'text', text: "\n here's an image " },
|
||||||
|
{ type: 'image', url: 'https://example.com/pic.png' },
|
||||||
|
{ type: 'text', text: ' and another profile ' },
|
||||||
|
{ type: 'reference', pointer: { pubkey: '32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245' } },
|
||||||
|
{ type: 'text', text: '\n with a video ' },
|
||||||
|
{ type: 'video', url: 'https://example.com/vid.webm' },
|
||||||
|
{ type: 'text', text: ' and finally ' },
|
||||||
|
{ type: 'url', url: 'https://example.com/docs' },
|
||||||
|
])
|
||||||
})
|
})
|
||||||
|
|||||||
212
nip27.ts
212
nip27.ts
@@ -1,63 +1,169 @@
|
|||||||
import { decode } from './nip19.ts'
|
import { AddressPointer, EventPointer, ProfilePointer, decode } from './nip19.ts'
|
||||||
import { NOSTR_URI_REGEX, type NostrURI } from './nip21.ts'
|
|
||||||
|
|
||||||
/** Regex to find NIP-21 URIs inside event content. */
|
export type Block =
|
||||||
export const regex = (): RegExp => new RegExp(`\\b${NOSTR_URI_REGEX.source}\\b`, 'g')
|
| {
|
||||||
|
type: 'text'
|
||||||
|
text: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'reference'
|
||||||
|
pointer: ProfilePointer | AddressPointer | EventPointer
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'url'
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'relay'
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'image'
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'video'
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: 'audio'
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
|
||||||
/** Match result for a Nostr URI in event content. */
|
const noCharacter = /\W/m
|
||||||
export interface NostrURIMatch extends NostrURI {
|
const noURLCharacter = /\W |\W$|$|,| /m
|
||||||
/** Index where the URI begins in the event content. */
|
|
||||||
start: number
|
|
||||||
/** Index where the URI ends in the event content. */
|
|
||||||
end: number
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Find and decode all NIP-21 URIs. */
|
export function* parse(content: string): Iterable<Block> {
|
||||||
export function* matchAll(content: string): Iterable<NostrURIMatch> {
|
const max = content.length
|
||||||
const matches = content.matchAll(regex())
|
let prevIndex = 0
|
||||||
|
let index = 0
|
||||||
|
while (index < max) {
|
||||||
|
let u = content.indexOf(':', index)
|
||||||
|
if (u === -1) {
|
||||||
|
// reached end
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
for (const match of matches) {
|
if (content.substring(u - 5, u) === 'nostr') {
|
||||||
try {
|
const m = content.substring(u + 60).match(noCharacter)
|
||||||
const [uri, value] = match
|
const end = m ? u + 60 + m.index! : max
|
||||||
|
try {
|
||||||
|
let pointer: ProfilePointer | AddressPointer | EventPointer
|
||||||
|
let { data, type } = decode(content.substring(u + 1, end))
|
||||||
|
|
||||||
yield {
|
switch (type) {
|
||||||
uri: uri as `nostr:${string}`,
|
case 'npub':
|
||||||
value,
|
pointer = { pubkey: data } as ProfilePointer
|
||||||
decoded: decode(value),
|
break
|
||||||
start: match.index!,
|
case 'nsec':
|
||||||
end: match.index! + uri.length,
|
case 'note':
|
||||||
|
// ignore this, treat it as not a valid uri
|
||||||
|
index = end + 1
|
||||||
|
continue
|
||||||
|
default:
|
||||||
|
pointer = data as any
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prevIndex !== u - 5) {
|
||||||
|
yield { type: 'text', text: content.substring(prevIndex, u - 5) }
|
||||||
|
}
|
||||||
|
yield { type: 'reference', pointer }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
} catch (_err) {
|
||||||
|
// ignore this, not a valid nostr uri
|
||||||
|
index = u + 1
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
} catch (_e) {
|
} else if (content.substring(u - 5, u) === 'https' || content.substring(u - 4, u) === 'http') {
|
||||||
// do nothing
|
const m = content.substring(u + 4).match(noURLCharacter)
|
||||||
|
const end = m ? u + 4 + m.index! : max
|
||||||
|
const prefixLen = content[u - 1] === 's' ? 5 : 4
|
||||||
|
try {
|
||||||
|
let url = new URL(content.substring(u - prefixLen, end))
|
||||||
|
if (url.hostname.indexOf('.') === -1) {
|
||||||
|
throw new Error('invalid url')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prevIndex !== u - prefixLen) {
|
||||||
|
yield { type: 'text', text: content.substring(prevIndex, u - prefixLen) }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
url.pathname.endsWith('.png') ||
|
||||||
|
url.pathname.endsWith('.jpg') ||
|
||||||
|
url.pathname.endsWith('.jpeg') ||
|
||||||
|
url.pathname.endsWith('.gif') ||
|
||||||
|
url.pathname.endsWith('.webp')
|
||||||
|
) {
|
||||||
|
yield { type: 'image', url: url.toString() }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
url.pathname.endsWith('.mp4') ||
|
||||||
|
url.pathname.endsWith('.avi') ||
|
||||||
|
url.pathname.endsWith('.webm') ||
|
||||||
|
url.pathname.endsWith('.mkv')
|
||||||
|
) {
|
||||||
|
yield { type: 'video', url: url.toString() }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
url.pathname.endsWith('.mp3') ||
|
||||||
|
url.pathname.endsWith('.aac') ||
|
||||||
|
url.pathname.endsWith('.ogg') ||
|
||||||
|
url.pathname.endsWith('.opus')
|
||||||
|
) {
|
||||||
|
yield { type: 'audio', url: url.toString() }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
yield { type: 'url', url: url.toString() }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
} catch (_err) {
|
||||||
|
// ignore this, not a valid url
|
||||||
|
index = end + 1
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else if (content.substring(u - 3, u) === 'wss' || content.substring(u - 2, u) === 'ws') {
|
||||||
|
const m = content.substring(u + 4).match(noURLCharacter)
|
||||||
|
const end = m ? u + 4 + m.index! : max
|
||||||
|
const prefixLen = content[u - 1] === 's' ? 3 : 2
|
||||||
|
try {
|
||||||
|
let url = new URL(content.substring(u - prefixLen, end))
|
||||||
|
if (url.hostname.indexOf('.') === -1) {
|
||||||
|
throw new Error('invalid ws url')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prevIndex !== u - prefixLen) {
|
||||||
|
yield { type: 'text', text: content.substring(prevIndex, u - prefixLen) }
|
||||||
|
}
|
||||||
|
yield { type: 'relay', url: url.toString() }
|
||||||
|
index = end
|
||||||
|
prevIndex = index
|
||||||
|
continue
|
||||||
|
} catch (_err) {
|
||||||
|
// ignore this, not a valid url
|
||||||
|
index = end + 1
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// ignore this, it is nothing
|
||||||
|
index = u + 1
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
if (prevIndex !== max) {
|
||||||
* Replace all occurrences of Nostr URIs in the text.
|
yield { type: 'text', text: content.substring(prevIndex) }
|
||||||
*
|
}
|
||||||
* WARNING: using this on an HTML string is potentially unsafe!
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* ```ts
|
|
||||||
* nip27.replaceAll(event.content, ({ decoded, value }) => {
|
|
||||||
* switch(decoded.type) {
|
|
||||||
* case 'npub':
|
|
||||||
* return renderMention(decoded)
|
|
||||||
* case 'note':
|
|
||||||
* return renderNote(decoded)
|
|
||||||
* default:
|
|
||||||
* return value
|
|
||||||
* }
|
|
||||||
* })
|
|
||||||
* ```
|
|
||||||
*/
|
|
||||||
export function replaceAll(content: string, replacer: (match: NostrURI) => string): string {
|
|
||||||
return content.replaceAll(regex(), (uri, value: string) => {
|
|
||||||
return replacer({
|
|
||||||
uri: uri as `nostr:${string}`,
|
|
||||||
value,
|
|
||||||
decoded: decode(value),
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|||||||
10
nip28.ts
10
nip28.ts
@@ -1,5 +1,11 @@
|
|||||||
import { Event, finalizeEvent } from './pure.ts'
|
import { Event, finalizeEvent } from './pure.ts'
|
||||||
import { ChannelCreation, ChannelHideMessage, ChannelMessage, ChannelMetadata, ChannelMuteUser } from './kinds.ts'
|
import {
|
||||||
|
ChannelCreation,
|
||||||
|
ChannelHideMessage,
|
||||||
|
ChannelMessage,
|
||||||
|
ChannelMetadata as KindChannelMetadata,
|
||||||
|
ChannelMuteUser,
|
||||||
|
} from './kinds.ts'
|
||||||
|
|
||||||
export interface ChannelMetadata {
|
export interface ChannelMetadata {
|
||||||
name: string
|
name: string
|
||||||
@@ -78,7 +84,7 @@ export const channelMetadataEvent = (t: ChannelMetadataEventTemplate, privateKey
|
|||||||
|
|
||||||
return finalizeEvent(
|
return finalizeEvent(
|
||||||
{
|
{
|
||||||
kind: ChannelMetadata,
|
kind: KindChannelMetadata,
|
||||||
tags: [['e', t.channel_create_event_id], ...(t.tags ?? [])],
|
tags: [['e', t.channel_create_event_id], ...(t.tags ?? [])],
|
||||||
content: content,
|
content: content,
|
||||||
created_at: t.created_at,
|
created_at: t.created_at,
|
||||||
|
|||||||
696
nip29.ts
696
nip29.ts
@@ -1,86 +1,528 @@
|
|||||||
import { AbstractSimplePool } from './abstract-pool.ts'
|
import { AbstractSimplePool } from './abstract-pool.ts'
|
||||||
import { Subscription } from './abstract-relay.ts'
|
import { Subscription } from './abstract-relay.ts'
|
||||||
import { decode } from './nip19.ts'
|
import type { Event, EventTemplate } from './core.ts'
|
||||||
import type { Event } from './core.ts'
|
import { fetchRelayInformation, RelayInformation } from './nip11.ts'
|
||||||
import { fetchRelayInformation } from './nip11.ts'
|
import { AddressPointer, decode, NostrTypeGuard } from './nip19.ts'
|
||||||
import { normalizeURL } from './utils.ts'
|
import { normalizeURL } from './utils.ts'
|
||||||
import { AddressPointer } from './nip19.ts'
|
|
||||||
|
|
||||||
export function subscribeRelayGroups(
|
/**
|
||||||
pool: AbstractSimplePool,
|
* Represents a NIP29 group.
|
||||||
url: string,
|
*/
|
||||||
params: {
|
export type Group = {
|
||||||
ongroups: (_: Group[]) => void
|
relay: string
|
||||||
onerror: (_: Error) => void
|
metadata: GroupMetadata
|
||||||
onconnect?: () => void
|
admins?: GroupAdmin[]
|
||||||
},
|
members?: GroupMember[]
|
||||||
): () => void {
|
reference: GroupReference
|
||||||
let normalized = normalizeURL(url)
|
|
||||||
let sub: Subscription
|
|
||||||
let groups: Group[] = []
|
|
||||||
|
|
||||||
fetchRelayInformation(normalized)
|
|
||||||
.then(async info => {
|
|
||||||
let rl = await pool.ensureRelay(normalized)
|
|
||||||
params.onconnect?.()
|
|
||||||
sub = rl.prepareSubscription(
|
|
||||||
[
|
|
||||||
{
|
|
||||||
kinds: [39000],
|
|
||||||
limit: 50,
|
|
||||||
authors: [info.pubkey],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
{
|
|
||||||
onevent(event: Event) {
|
|
||||||
groups.push(parseGroup(event, normalized))
|
|
||||||
},
|
|
||||||
oneose() {
|
|
||||||
params.ongroups(groups)
|
|
||||||
sub.onevent = (event: Event) => {
|
|
||||||
groups.push(parseGroup(event, normalized))
|
|
||||||
params.ongroups(groups)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
)
|
|
||||||
sub.fire()
|
|
||||||
})
|
|
||||||
.catch(params.onerror)
|
|
||||||
|
|
||||||
return () => sub.close()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function loadGroup(pool: AbstractSimplePool, gr: GroupReference): Promise<Group> {
|
/**
|
||||||
let normalized = normalizeURL(gr.host)
|
* Represents the metadata for a NIP29 group.
|
||||||
|
*/
|
||||||
let info = await fetchRelayInformation(normalized)
|
export type GroupMetadata = {
|
||||||
let event = await pool.get([normalized], {
|
id: string
|
||||||
kinds: [39000],
|
pubkey: string
|
||||||
authors: [info.pubkey],
|
name?: string
|
||||||
'#d': [gr.id],
|
picture?: string
|
||||||
})
|
about?: string
|
||||||
if (!event) throw new Error(`group '${gr.id}' not found on ${gr.host}`)
|
isPublic?: boolean
|
||||||
return parseGroup(event, normalized)
|
isOpen?: boolean
|
||||||
}
|
|
||||||
|
|
||||||
export async function loadGroupFromCode(pool: AbstractSimplePool, code: string): Promise<Group> {
|
|
||||||
let gr = parseGroupCode(code)
|
|
||||||
if (!gr) throw new Error(`code "${code}" does not identify a group`)
|
|
||||||
return loadGroup(pool, gr)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a NIP29 group reference.
|
||||||
|
*/
|
||||||
export type GroupReference = {
|
export type GroupReference = {
|
||||||
id: string
|
id: string
|
||||||
host: string
|
host: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a NIP29 group member.
|
||||||
|
*/
|
||||||
|
export type GroupMember = {
|
||||||
|
pubkey: string
|
||||||
|
label?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a NIP29 group admin.
|
||||||
|
*/
|
||||||
|
export type GroupAdmin = {
|
||||||
|
pubkey: string
|
||||||
|
label?: string
|
||||||
|
permissions: GroupAdminPermission[]
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the permissions that a NIP29 group admin can have.
|
||||||
|
*/
|
||||||
|
export enum GroupAdminPermission {
|
||||||
|
/** @deprecated use PutUser instead */
|
||||||
|
AddUser = 'add-user',
|
||||||
|
EditMetadata = 'edit-metadata',
|
||||||
|
DeleteEvent = 'delete-event',
|
||||||
|
RemoveUser = 'remove-user',
|
||||||
|
/** @deprecated removed from NIP */
|
||||||
|
AddPermission = 'add-permission',
|
||||||
|
/** @deprecated removed from NIP */
|
||||||
|
RemovePermission = 'remove-permission',
|
||||||
|
/** @deprecated removed from NIP */
|
||||||
|
EditGroupStatus = 'edit-group-status',
|
||||||
|
PutUser = 'put-user',
|
||||||
|
CreateGroup = 'create-group',
|
||||||
|
DeleteGroup = 'delete-group',
|
||||||
|
CreateInvite = 'create-invite',
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates a group metadata event template.
|
||||||
|
*
|
||||||
|
* @param group - The group object.
|
||||||
|
* @returns An event template with the generated group metadata that can be signed later.
|
||||||
|
*/
|
||||||
|
export function generateGroupMetadataEventTemplate(group: Group): EventTemplate {
|
||||||
|
const tags: string[][] = [['d', group.metadata.id]]
|
||||||
|
group.metadata.name && tags.push(['name', group.metadata.name])
|
||||||
|
group.metadata.picture && tags.push(['picture', group.metadata.picture])
|
||||||
|
group.metadata.about && tags.push(['about', group.metadata.about])
|
||||||
|
group.metadata.isPublic && tags.push(['public'])
|
||||||
|
group.metadata.isOpen && tags.push(['open'])
|
||||||
|
|
||||||
|
return {
|
||||||
|
content: '',
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
kind: 39000,
|
||||||
|
tags,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates a group metadata event.
|
||||||
|
*
|
||||||
|
* @param event - The event to validate.
|
||||||
|
* @returns A boolean indicating whether the event is valid.
|
||||||
|
*/
|
||||||
|
export function validateGroupMetadataEvent(event: Event): boolean {
|
||||||
|
if (event.kind !== 39000) return false
|
||||||
|
|
||||||
|
if (!event.pubkey) return false
|
||||||
|
|
||||||
|
const requiredTags = ['d'] as const
|
||||||
|
for (const tag of requiredTags) {
|
||||||
|
if (!event.tags.find(([t]) => t == tag)) return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates an event template for group admins.
|
||||||
|
*
|
||||||
|
* @param group - The group object.
|
||||||
|
* @param admins - An array of group admins.
|
||||||
|
* @returns The generated event template with the group admins that can be signed later.
|
||||||
|
*/
|
||||||
|
export function generateGroupAdminsEventTemplate(group: Group, admins: GroupAdmin[]): EventTemplate {
|
||||||
|
const tags: string[][] = [['d', group.metadata.id]]
|
||||||
|
for (const admin of admins) {
|
||||||
|
tags.push(['p', admin.pubkey, admin.label || '', ...admin.permissions])
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
content: '',
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
kind: 39001,
|
||||||
|
tags,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates a group admins event.
|
||||||
|
*
|
||||||
|
* @param event - The event to validate.
|
||||||
|
* @returns True if the event is valid, false otherwise.
|
||||||
|
*/
|
||||||
|
export function validateGroupAdminsEvent(event: Event): boolean {
|
||||||
|
if (event.kind !== 39001) return false
|
||||||
|
|
||||||
|
const requiredTags = ['d'] as const
|
||||||
|
for (const tag of requiredTags) {
|
||||||
|
if (!event.tags.find(([t]) => t == tag)) return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// validate permissions
|
||||||
|
for (const [tag, _value, _label, ...permissions] of event.tags) {
|
||||||
|
if (tag !== 'p') continue
|
||||||
|
|
||||||
|
for (let i = 0; i < permissions.length; i += 1) {
|
||||||
|
if (typeof permissions[i] !== 'string') return false
|
||||||
|
|
||||||
|
// validate permission name from the GroupAdminPermission enum
|
||||||
|
if (!Object.values(GroupAdminPermission).includes(permissions[i] as GroupAdminPermission)) return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates an event template for a group with its members.
|
||||||
|
*
|
||||||
|
* @param group - The group object.
|
||||||
|
* @param members - An array of group members.
|
||||||
|
* @returns The generated event template with the group members that can be signed later.
|
||||||
|
*/
|
||||||
|
export function generateGroupMembersEventTemplate(group: Group, members: GroupMember[]): EventTemplate {
|
||||||
|
const tags: string[][] = [['d', group.metadata.id]]
|
||||||
|
for (const member of members) {
|
||||||
|
tags.push(['p', member.pubkey, member.label || ''])
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
content: '',
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
kind: 39002,
|
||||||
|
tags,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates a group members event.
|
||||||
|
*
|
||||||
|
* @param event - The event to validate.
|
||||||
|
* @returns Returns `true` if the event is a valid group members event, `false` otherwise.
|
||||||
|
*/
|
||||||
|
export function validateGroupMembersEvent(event: Event): boolean {
|
||||||
|
if (event.kind !== 39002) return false
|
||||||
|
|
||||||
|
const requiredTags = ['d'] as const
|
||||||
|
for (const tag of requiredTags) {
|
||||||
|
if (!event.tags.find(([t]) => t == tag)) return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the normalized relay URL based on the provided group reference.
|
||||||
|
*
|
||||||
|
* @param groupReference - The group reference object containing the host.
|
||||||
|
* @returns The normalized relay URL.
|
||||||
|
*/
|
||||||
|
export function getNormalizedRelayURLByGroupReference(groupReference: GroupReference): string {
|
||||||
|
return normalizeURL(groupReference.host)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches relay information by group reference.
|
||||||
|
*
|
||||||
|
* @param groupReference The group reference.
|
||||||
|
* @returns A promise that resolves to the relay information.
|
||||||
|
*/
|
||||||
|
export async function fetchRelayInformationByGroupReference(groupReference: GroupReference): Promise<RelayInformation> {
|
||||||
|
const normalizedRelayURL = getNormalizedRelayURLByGroupReference(groupReference)
|
||||||
|
|
||||||
|
return fetchRelayInformation(normalizedRelayURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches the group metadata event from the specified pool.
|
||||||
|
* If the normalizedRelayURL is not provided, it will be obtained using the groupReference.
|
||||||
|
* If the relayInformation is not provided, it will be fetched using the normalizedRelayURL.
|
||||||
|
*
|
||||||
|
* @param {Object} options - The options object.
|
||||||
|
* @param {AbstractSimplePool} options.pool - The pool to fetch the group metadata event from.
|
||||||
|
* @param {GroupReference} options.groupReference - The reference to the group.
|
||||||
|
* @param {string} [options.normalizedRelayURL] - The normalized URL of the relay.
|
||||||
|
* @param {RelayInformation} [options.relayInformation] - The relay information object.
|
||||||
|
* @returns {Promise<Event>} The group metadata event that can be parsed later to get the group metadata object.
|
||||||
|
* @throws {Error} If the group is not found on the specified relay.
|
||||||
|
*/
|
||||||
|
export async function fetchGroupMetadataEvent({
|
||||||
|
pool,
|
||||||
|
groupReference,
|
||||||
|
relayInformation,
|
||||||
|
normalizedRelayURL,
|
||||||
|
}: {
|
||||||
|
pool: AbstractSimplePool
|
||||||
|
groupReference: GroupReference
|
||||||
|
normalizedRelayURL?: string
|
||||||
|
relayInformation?: RelayInformation
|
||||||
|
}): Promise<Event> {
|
||||||
|
if (!normalizedRelayURL) {
|
||||||
|
normalizedRelayURL = getNormalizedRelayURLByGroupReference(groupReference)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!relayInformation) {
|
||||||
|
relayInformation = await fetchRelayInformation(normalizedRelayURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
const groupMetadataEvent = await pool.get([normalizedRelayURL], {
|
||||||
|
kinds: [39000],
|
||||||
|
authors: [relayInformation.pubkey],
|
||||||
|
'#d': [groupReference.id],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!groupMetadataEvent) throw new Error(`group '${groupReference.id}' not found on ${normalizedRelayURL}`)
|
||||||
|
|
||||||
|
return groupMetadataEvent
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a group metadata event and returns the corresponding GroupMetadata object.
|
||||||
|
*
|
||||||
|
* @param event - The event to parse.
|
||||||
|
* @returns The parsed GroupMetadata object.
|
||||||
|
* @throws An error if the group metadata event is invalid.
|
||||||
|
*/
|
||||||
|
export function parseGroupMetadataEvent(event: Event): GroupMetadata {
|
||||||
|
if (!validateGroupMetadataEvent(event)) throw new Error('invalid group metadata event')
|
||||||
|
|
||||||
|
const metadata: GroupMetadata = {
|
||||||
|
id: '',
|
||||||
|
pubkey: event.pubkey,
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const [tag, value] of event.tags) {
|
||||||
|
switch (tag) {
|
||||||
|
case 'd':
|
||||||
|
metadata.id = value
|
||||||
|
break
|
||||||
|
case 'name':
|
||||||
|
metadata.name = value
|
||||||
|
break
|
||||||
|
case 'picture':
|
||||||
|
metadata.picture = value
|
||||||
|
break
|
||||||
|
case 'about':
|
||||||
|
metadata.about = value
|
||||||
|
break
|
||||||
|
case 'public':
|
||||||
|
metadata.isPublic = true
|
||||||
|
break
|
||||||
|
case 'open':
|
||||||
|
metadata.isOpen = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return metadata
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches the group admins event from the specified pool.
|
||||||
|
* If the normalizedRelayURL is not provided, it will be obtained from the groupReference.
|
||||||
|
* If the relayInformation is not provided, it will be fetched using the normalizedRelayURL.
|
||||||
|
*
|
||||||
|
* @param {Object} options - The options object.
|
||||||
|
* @param {AbstractSimplePool} options.pool - The pool to fetch the group admins event from.
|
||||||
|
* @param {GroupReference} options.groupReference - The reference to the group.
|
||||||
|
* @param {string} [options.normalizedRelayURL] - The normalized relay URL.
|
||||||
|
* @param {RelayInformation} [options.relayInformation] - The relay information.
|
||||||
|
* @returns {Promise<Event>} The group admins event that can be parsed later to get the group admins object.
|
||||||
|
* @throws {Error} If the group admins event is not found on the specified relay.
|
||||||
|
*/
|
||||||
|
export async function fetchGroupAdminsEvent({
|
||||||
|
pool,
|
||||||
|
groupReference,
|
||||||
|
relayInformation,
|
||||||
|
normalizedRelayURL,
|
||||||
|
}: {
|
||||||
|
pool: AbstractSimplePool
|
||||||
|
groupReference: GroupReference
|
||||||
|
normalizedRelayURL?: string
|
||||||
|
relayInformation?: RelayInformation
|
||||||
|
}): Promise<Event> {
|
||||||
|
if (!normalizedRelayURL) {
|
||||||
|
normalizedRelayURL = getNormalizedRelayURLByGroupReference(groupReference)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!relayInformation) {
|
||||||
|
relayInformation = await fetchRelayInformation(normalizedRelayURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
const groupAdminsEvent = await pool.get([normalizedRelayURL], {
|
||||||
|
kinds: [39001],
|
||||||
|
authors: [relayInformation.pubkey],
|
||||||
|
'#d': [groupReference.id],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!groupAdminsEvent) throw new Error(`admins for group '${groupReference.id}' not found on ${normalizedRelayURL}`)
|
||||||
|
|
||||||
|
return groupAdminsEvent
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a group admins event and returns an array of GroupAdmin objects.
|
||||||
|
*
|
||||||
|
* @param event - The event to parse.
|
||||||
|
* @returns An array of GroupAdmin objects.
|
||||||
|
* @throws Throws an error if the group admins event is invalid.
|
||||||
|
*/
|
||||||
|
export function parseGroupAdminsEvent(event: Event): GroupAdmin[] {
|
||||||
|
if (!validateGroupAdminsEvent(event)) throw new Error('invalid group admins event')
|
||||||
|
|
||||||
|
const admins: GroupAdmin[] = []
|
||||||
|
|
||||||
|
for (const [tag, value, label, ...permissions] of event.tags) {
|
||||||
|
if (tag !== 'p') continue
|
||||||
|
|
||||||
|
admins.push({
|
||||||
|
pubkey: value,
|
||||||
|
label,
|
||||||
|
permissions: permissions as GroupAdminPermission[],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return admins
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches the group members event from the specified relay.
|
||||||
|
* If the normalizedRelayURL is not provided, it will be obtained using the groupReference.
|
||||||
|
* If the relayInformation is not provided, it will be fetched using the normalizedRelayURL.
|
||||||
|
*
|
||||||
|
* @param {Object} options - The options object.
|
||||||
|
* @param {AbstractSimplePool} options.pool - The pool object.
|
||||||
|
* @param {GroupReference} options.groupReference - The group reference object.
|
||||||
|
* @param {string} [options.normalizedRelayURL] - The normalized relay URL.
|
||||||
|
* @param {RelayInformation} [options.relayInformation] - The relay information object.
|
||||||
|
* @returns {Promise<Event>} The group members event that can be parsed later to get the group members object.
|
||||||
|
* @throws {Error} If the group members event is not found.
|
||||||
|
*/
|
||||||
|
export async function fetchGroupMembersEvent({
|
||||||
|
pool,
|
||||||
|
groupReference,
|
||||||
|
relayInformation,
|
||||||
|
normalizedRelayURL,
|
||||||
|
}: {
|
||||||
|
pool: AbstractSimplePool
|
||||||
|
groupReference: GroupReference
|
||||||
|
normalizedRelayURL?: string
|
||||||
|
relayInformation?: RelayInformation
|
||||||
|
}): Promise<Event> {
|
||||||
|
if (!normalizedRelayURL) {
|
||||||
|
normalizedRelayURL = getNormalizedRelayURLByGroupReference(groupReference)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!relayInformation) {
|
||||||
|
relayInformation = await fetchRelayInformation(normalizedRelayURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
const groupMembersEvent = await pool.get([normalizedRelayURL], {
|
||||||
|
kinds: [39002],
|
||||||
|
authors: [relayInformation.pubkey],
|
||||||
|
'#d': [groupReference.id],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!groupMembersEvent) throw new Error(`members for group '${groupReference.id}' not found on ${normalizedRelayURL}`)
|
||||||
|
|
||||||
|
return groupMembersEvent
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a group members event and returns an array of GroupMember objects.
|
||||||
|
* @param event - The event to parse.
|
||||||
|
* @returns An array of GroupMember objects.
|
||||||
|
* @throws Throws an error if the group members event is invalid.
|
||||||
|
*/
|
||||||
|
export function parseGroupMembersEvent(event: Event): GroupMember[] {
|
||||||
|
if (!validateGroupMembersEvent(event)) throw new Error('invalid group members event')
|
||||||
|
|
||||||
|
const members: GroupMember[] = []
|
||||||
|
|
||||||
|
for (const [tag, value, label] of event.tags) {
|
||||||
|
if (tag !== 'p') continue
|
||||||
|
|
||||||
|
members.push({
|
||||||
|
pubkey: value,
|
||||||
|
label,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return members
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetches and parses the group metadata event, group admins event, and group members event from the specified pool.
|
||||||
|
* If the normalized relay URL is not provided, it will be obtained using the group reference.
|
||||||
|
* If the relay information is not provided, it will be fetched using the normalized relay URL.
|
||||||
|
*
|
||||||
|
* @param {Object} options - The options for loading the group.
|
||||||
|
* @param {AbstractSimplePool} options.pool - The pool to load the group from.
|
||||||
|
* @param {GroupReference} options.groupReference - The reference of the group to load.
|
||||||
|
* @param {string} [options.normalizedRelayURL] - The normalized URL of the relay to use.
|
||||||
|
* @param {RelayInformation} [options.relayInformation] - The relay information to use.
|
||||||
|
* @returns {Promise<Group>} A promise that resolves to the loaded group.
|
||||||
|
*/
|
||||||
|
export async function loadGroup({
|
||||||
|
pool,
|
||||||
|
groupReference,
|
||||||
|
normalizedRelayURL,
|
||||||
|
relayInformation,
|
||||||
|
}: {
|
||||||
|
pool: AbstractSimplePool
|
||||||
|
groupReference: GroupReference
|
||||||
|
normalizedRelayURL?: string
|
||||||
|
relayInformation?: RelayInformation
|
||||||
|
}): Promise<Group> {
|
||||||
|
if (!normalizedRelayURL) {
|
||||||
|
normalizedRelayURL = getNormalizedRelayURLByGroupReference(groupReference)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!relayInformation) {
|
||||||
|
relayInformation = await fetchRelayInformation(normalizedRelayURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
const metadataEvent = await fetchGroupMetadataEvent({ pool, groupReference, normalizedRelayURL, relayInformation })
|
||||||
|
const metadata = parseGroupMetadataEvent(metadataEvent)
|
||||||
|
|
||||||
|
const adminsEvent = await fetchGroupAdminsEvent({ pool, groupReference, normalizedRelayURL, relayInformation })
|
||||||
|
const admins = parseGroupAdminsEvent(adminsEvent)
|
||||||
|
|
||||||
|
const membersEvent = await fetchGroupMembersEvent({ pool, groupReference, normalizedRelayURL, relayInformation })
|
||||||
|
const members = parseGroupMembersEvent(membersEvent)
|
||||||
|
|
||||||
|
const group: Group = {
|
||||||
|
relay: normalizedRelayURL,
|
||||||
|
metadata,
|
||||||
|
admins,
|
||||||
|
members,
|
||||||
|
reference: groupReference,
|
||||||
|
}
|
||||||
|
|
||||||
|
return group
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads a group from the specified pool using the provided group code.
|
||||||
|
*
|
||||||
|
* @param {AbstractSimplePool} pool - The pool to load the group from.
|
||||||
|
* @param {string} code - The code representing the group.
|
||||||
|
* @returns {Promise<Group>} - A promise that resolves to the loaded group.
|
||||||
|
* @throws {Error} - If the group code is invalid.
|
||||||
|
*/
|
||||||
|
export async function loadGroupFromCode(pool: AbstractSimplePool, code: string): Promise<Group> {
|
||||||
|
const groupReference = parseGroupCode(code)
|
||||||
|
|
||||||
|
if (!groupReference) throw new Error('invalid group code')
|
||||||
|
|
||||||
|
return loadGroup({ pool, groupReference })
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a group code and returns a GroupReference object.
|
||||||
|
*
|
||||||
|
* @param code The group code to parse.
|
||||||
|
* @returns A GroupReference object if the code is valid, otherwise null.
|
||||||
|
*/
|
||||||
export function parseGroupCode(code: string): null | GroupReference {
|
export function parseGroupCode(code: string): null | GroupReference {
|
||||||
if (code.startsWith('naddr1')) {
|
if (NostrTypeGuard.isNAddr(code)) {
|
||||||
try {
|
try {
|
||||||
let { data } = decode(code)
|
let { data } = decode(code)
|
||||||
|
|
||||||
let { relays, identifier } = data as AddressPointer
|
let { relays, identifier } = data
|
||||||
if (!relays || relays.length === 0) return null
|
if (!relays || relays.length === 0) return null
|
||||||
|
|
||||||
let host = relays![0]
|
let host = relays![0]
|
||||||
@@ -99,68 +541,74 @@ export function parseGroupCode(code: string): null | GroupReference {
|
|||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encodes a group reference into a string.
|
||||||
|
*
|
||||||
|
* @param gr - The group reference to encode.
|
||||||
|
* @returns The encoded group reference as a string.
|
||||||
|
*/
|
||||||
export function encodeGroupReference(gr: GroupReference): string {
|
export function encodeGroupReference(gr: GroupReference): string {
|
||||||
if (gr.host.startsWith('https://')) gr.host = gr.host.slice(8)
|
const { host, id } = gr
|
||||||
if (gr.host.startsWith('wss://')) gr.host = gr.host.slice(6)
|
const normalizedHost = host.replace(/^(https?:\/\/|wss?:\/\/)/, '')
|
||||||
return `${gr.host}'${gr.id}`
|
|
||||||
|
return `${normalizedHost}'${id}`
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Group = {
|
/**
|
||||||
id: string
|
* Subscribes to relay groups metadata events and calls the provided event handler function
|
||||||
relay: string
|
* when an event is received.
|
||||||
pubkey: string
|
*
|
||||||
name?: string
|
* @param {Object} options - The options for subscribing to relay groups metadata events.
|
||||||
picture?: string
|
* @param {AbstractSimplePool} options.pool - The pool to subscribe to.
|
||||||
about?: string
|
* @param {string} options.relayURL - The URL of the relay.
|
||||||
public?: boolean
|
* @param {Function} options.onError - The error handler function.
|
||||||
open?: boolean
|
* @param {Function} options.onEvent - The event handler function.
|
||||||
}
|
* @param {Function} [options.onConnect] - The connect handler function.
|
||||||
|
* @returns {Function} - A function to close the subscription
|
||||||
|
*/
|
||||||
|
export function subscribeRelayGroupsMetadataEvents({
|
||||||
|
pool,
|
||||||
|
relayURL,
|
||||||
|
onError,
|
||||||
|
onEvent,
|
||||||
|
onConnect,
|
||||||
|
}: {
|
||||||
|
pool: AbstractSimplePool
|
||||||
|
relayURL: string
|
||||||
|
onError: (err: Error) => void
|
||||||
|
onEvent: (event: Event) => void
|
||||||
|
onConnect?: () => void
|
||||||
|
}): () => void {
|
||||||
|
let sub: Subscription
|
||||||
|
|
||||||
export function parseGroup(event: Event, relay: string): Group {
|
const normalizedRelayURL = normalizeURL(relayURL)
|
||||||
const group: Partial<Group> = { relay, pubkey: event.pubkey }
|
|
||||||
for (let i = 0; i < event.tags.length; i++) {
|
|
||||||
const tag = event.tags[i]
|
|
||||||
switch (tag[0]) {
|
|
||||||
case 'd':
|
|
||||||
group.id = tag[1] || ''
|
|
||||||
break
|
|
||||||
case 'name':
|
|
||||||
group.name = tag[1] || ''
|
|
||||||
break
|
|
||||||
case 'about':
|
|
||||||
group.about = tag[1] || ''
|
|
||||||
break
|
|
||||||
case 'picture':
|
|
||||||
group.picture = tag[1] || ''
|
|
||||||
break
|
|
||||||
case 'open':
|
|
||||||
group.open = true
|
|
||||||
break
|
|
||||||
case 'public':
|
|
||||||
group.public = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return group as Group
|
|
||||||
}
|
|
||||||
|
|
||||||
export type Member = {
|
fetchRelayInformation(normalizedRelayURL)
|
||||||
pubkey: string
|
.then(async info => {
|
||||||
label?: string
|
const abstractedRelay = await pool.ensureRelay(normalizedRelayURL)
|
||||||
permissions: string[]
|
|
||||||
}
|
|
||||||
|
|
||||||
export function parseMembers(event: Event): Member[] {
|
onConnect?.()
|
||||||
const members = []
|
|
||||||
for (let i = 0; i < event.tags.length; i++) {
|
sub = abstractedRelay.prepareSubscription(
|
||||||
const tag = event.tags[i]
|
[
|
||||||
if (tag.length < 2) continue
|
{
|
||||||
if (tag[0] !== 'p') continue
|
kinds: [39000],
|
||||||
if (!tag[1].match(/^[0-9a-f]{64}$/)) continue
|
limit: 50,
|
||||||
const member: Member = { pubkey: tag[1], permissions: [] }
|
authors: [info.pubkey],
|
||||||
if (tag.length > 2) member.label = tag[2]
|
},
|
||||||
if (tag.length > 3) member.permissions = tag.slice(3)
|
],
|
||||||
members.push(member)
|
{
|
||||||
}
|
onevent(event: Event) {
|
||||||
return members
|
onEvent(event)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
sub.close()
|
||||||
|
|
||||||
|
onError(err)
|
||||||
|
})
|
||||||
|
|
||||||
|
return () => sub.close()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import { test, expect } from 'bun:test'
|
import { test, expect } from 'bun:test'
|
||||||
import { v2 } from './nip44.js'
|
import { v2 } from './nip44.js'
|
||||||
import { bytesToHex, hexToBytes } from '@noble/hashes/utils'
|
import { bytesToHex, hexToBytes } from '@noble/hashes/utils'
|
||||||
import { default as vec } from './nip44.vectors.json' assert { type: 'json' }
|
import { default as vec } from './nip44.vectors.json' with { type: 'json' }
|
||||||
import { schnorr } from '@noble/curves/secp256k1'
|
import { schnorr } from '@noble/curves/secp256k1'
|
||||||
const v2vec = vec.v2
|
const v2vec = vec.v2
|
||||||
|
|
||||||
test('get_conversation_key', () => {
|
test('get_conversation_key', () => {
|
||||||
for (const v of v2vec.valid.get_conversation_key) {
|
for (const v of v2vec.valid.get_conversation_key) {
|
||||||
const key = v2.utils.getConversationKey(v.sec1, v.pub2)
|
const key = v2.utils.getConversationKey(hexToBytes(v.sec1), v.pub2)
|
||||||
expect(bytesToHex(key)).toEqual(v.conversation_key)
|
expect(bytesToHex(key)).toEqual(v.conversation_key)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -15,7 +15,7 @@ test('get_conversation_key', () => {
|
|||||||
test('encrypt_decrypt', () => {
|
test('encrypt_decrypt', () => {
|
||||||
for (const v of v2vec.valid.encrypt_decrypt) {
|
for (const v of v2vec.valid.encrypt_decrypt) {
|
||||||
const pub2 = bytesToHex(schnorr.getPublicKey(v.sec2))
|
const pub2 = bytesToHex(schnorr.getPublicKey(v.sec2))
|
||||||
const key = v2.utils.getConversationKey(v.sec1, pub2)
|
const key = v2.utils.getConversationKey(hexToBytes(v.sec1), pub2)
|
||||||
expect(bytesToHex(key)).toEqual(v.conversation_key)
|
expect(bytesToHex(key)).toEqual(v.conversation_key)
|
||||||
const ciphertext = v2.encrypt(v.plaintext, key, hexToBytes(v.nonce))
|
const ciphertext = v2.encrypt(v.plaintext, key, hexToBytes(v.nonce))
|
||||||
expect(ciphertext).toEqual(v.payload)
|
expect(ciphertext).toEqual(v.payload)
|
||||||
@@ -39,6 +39,8 @@ test('decrypt', async () => {
|
|||||||
|
|
||||||
test('get_conversation_key', async () => {
|
test('get_conversation_key', async () => {
|
||||||
for (const v of v2vec.invalid.get_conversation_key) {
|
for (const v of v2vec.invalid.get_conversation_key) {
|
||||||
expect(() => v2.utils.getConversationKey(v.sec1, v.pub2)).toThrow(/(Point is not on curve|Cannot find square root)/)
|
expect(() => v2.utils.getConversationKey(hexToBytes(v.sec1), v.pub2)).toThrow(
|
||||||
|
/(Point is not on curve|Cannot find square root)/,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
221
nip44.ts
221
nip44.ts
@@ -4,129 +4,124 @@ import { secp256k1 } from '@noble/curves/secp256k1'
|
|||||||
import { extract as hkdf_extract, expand as hkdf_expand } from '@noble/hashes/hkdf'
|
import { extract as hkdf_extract, expand as hkdf_expand } from '@noble/hashes/hkdf'
|
||||||
import { hmac } from '@noble/hashes/hmac'
|
import { hmac } from '@noble/hashes/hmac'
|
||||||
import { sha256 } from '@noble/hashes/sha256'
|
import { sha256 } from '@noble/hashes/sha256'
|
||||||
import { concatBytes, randomBytes, utf8ToBytes } from '@noble/hashes/utils'
|
import { concatBytes, randomBytes } from '@noble/hashes/utils'
|
||||||
import { base64 } from '@scure/base'
|
import { base64 } from '@scure/base'
|
||||||
|
|
||||||
const decoder = new TextDecoder()
|
import { utf8Decoder, utf8Encoder } from './utils.ts'
|
||||||
|
|
||||||
class u {
|
const minPlaintextSize = 0x0001 // 1b msg => padded to 32b
|
||||||
static minPlaintextSize = 0x0001 // 1b msg => padded to 32b
|
const maxPlaintextSize = 0xffff // 65535 (64kb-1) => padded to 64kb
|
||||||
static maxPlaintextSize = 0xffff // 65535 (64kb-1) => padded to 64kb
|
|
||||||
|
|
||||||
static utf8Encode = utf8ToBytes
|
export function getConversationKey(privkeyA: Uint8Array, pubkeyB: string): Uint8Array {
|
||||||
|
const sharedX = secp256k1.getSharedSecret(privkeyA, '02' + pubkeyB).subarray(1, 33)
|
||||||
|
return hkdf_extract(sha256, sharedX, 'nip44-v2')
|
||||||
|
}
|
||||||
|
|
||||||
static utf8Decode(bytes: Uint8Array): string {
|
function getMessageKeys(
|
||||||
return decoder.decode(bytes)
|
conversationKey: Uint8Array,
|
||||||
}
|
nonce: Uint8Array,
|
||||||
|
): { chacha_key: Uint8Array; chacha_nonce: Uint8Array; hmac_key: Uint8Array } {
|
||||||
static getConversationKey(privkeyA: string, pubkeyB: string): Uint8Array {
|
const keys = hkdf_expand(sha256, conversationKey, nonce, 76)
|
||||||
const sharedX = secp256k1.getSharedSecret(privkeyA, '02' + pubkeyB).subarray(1, 33)
|
return {
|
||||||
return hkdf_extract(sha256, sharedX, 'nip44-v2')
|
chacha_key: keys.subarray(0, 32),
|
||||||
}
|
chacha_nonce: keys.subarray(32, 44),
|
||||||
|
hmac_key: keys.subarray(44, 76),
|
||||||
static getMessageKeys(
|
|
||||||
conversationKey: Uint8Array,
|
|
||||||
nonce: Uint8Array,
|
|
||||||
): { chacha_key: Uint8Array; chacha_nonce: Uint8Array; hmac_key: Uint8Array } {
|
|
||||||
const keys = hkdf_expand(sha256, conversationKey, nonce, 76)
|
|
||||||
return {
|
|
||||||
chacha_key: keys.subarray(0, 32),
|
|
||||||
chacha_nonce: keys.subarray(32, 44),
|
|
||||||
hmac_key: keys.subarray(44, 76),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static calcPaddedLen(len: number): number {
|
|
||||||
if (!Number.isSafeInteger(len) || len < 1) throw new Error('expected positive integer')
|
|
||||||
if (len <= 32) return 32
|
|
||||||
const nextPower = 1 << (Math.floor(Math.log2(len - 1)) + 1)
|
|
||||||
const chunk = nextPower <= 256 ? 32 : nextPower / 8
|
|
||||||
return chunk * (Math.floor((len - 1) / chunk) + 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
static writeU16BE(num: number): Uint8Array {
|
|
||||||
if (!Number.isSafeInteger(num) || num < u.minPlaintextSize || num > u.maxPlaintextSize)
|
|
||||||
throw new Error('invalid plaintext size: must be between 1 and 65535 bytes')
|
|
||||||
const arr = new Uint8Array(2)
|
|
||||||
new DataView(arr.buffer).setUint16(0, num, false)
|
|
||||||
return arr
|
|
||||||
}
|
|
||||||
|
|
||||||
static pad(plaintext: string): Uint8Array {
|
|
||||||
const unpadded = u.utf8Encode(plaintext)
|
|
||||||
const unpaddedLen = unpadded.length
|
|
||||||
const prefix = u.writeU16BE(unpaddedLen)
|
|
||||||
const suffix = new Uint8Array(u.calcPaddedLen(unpaddedLen) - unpaddedLen)
|
|
||||||
return concatBytes(prefix, unpadded, suffix)
|
|
||||||
}
|
|
||||||
|
|
||||||
static unpad(padded: Uint8Array): string {
|
|
||||||
const unpaddedLen = new DataView(padded.buffer).getUint16(0)
|
|
||||||
const unpadded = padded.subarray(2, 2 + unpaddedLen)
|
|
||||||
if (
|
|
||||||
unpaddedLen < u.minPlaintextSize ||
|
|
||||||
unpaddedLen > u.maxPlaintextSize ||
|
|
||||||
unpadded.length !== unpaddedLen ||
|
|
||||||
padded.length !== 2 + u.calcPaddedLen(unpaddedLen)
|
|
||||||
)
|
|
||||||
throw new Error('invalid padding')
|
|
||||||
return u.utf8Decode(unpadded)
|
|
||||||
}
|
|
||||||
|
|
||||||
static hmacAad(key: Uint8Array, message: Uint8Array, aad: Uint8Array): Uint8Array {
|
|
||||||
if (aad.length !== 32) throw new Error('AAD associated data must be 32 bytes')
|
|
||||||
const combined = concatBytes(aad, message)
|
|
||||||
return hmac(sha256, key, combined)
|
|
||||||
}
|
|
||||||
|
|
||||||
// metadata: always 65b (version: 1b, nonce: 32b, max: 32b)
|
|
||||||
// plaintext: 1b to 0xffff
|
|
||||||
// padded plaintext: 32b to 0xffff
|
|
||||||
// ciphertext: 32b+2 to 0xffff+2
|
|
||||||
// raw payload: 99 (65+32+2) to 65603 (65+0xffff+2)
|
|
||||||
// compressed payload (base64): 132b to 87472b
|
|
||||||
static decodePayload(payload: string): { nonce: Uint8Array; ciphertext: Uint8Array; mac: Uint8Array } {
|
|
||||||
if (typeof payload !== 'string') throw new Error('payload must be a valid string')
|
|
||||||
const plen = payload.length
|
|
||||||
if (plen < 132 || plen > 87472) throw new Error('invalid payload length: ' + plen)
|
|
||||||
if (payload[0] === '#') throw new Error('unknown encryption version')
|
|
||||||
let data: Uint8Array
|
|
||||||
try {
|
|
||||||
data = base64.decode(payload)
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error('invalid base64: ' + (error as any).message)
|
|
||||||
}
|
|
||||||
const dlen = data.length
|
|
||||||
if (dlen < 99 || dlen > 65603) throw new Error('invalid data length: ' + dlen)
|
|
||||||
const vers = data[0]
|
|
||||||
if (vers !== 2) throw new Error('unknown encryption version ' + vers)
|
|
||||||
return {
|
|
||||||
nonce: data.subarray(1, 33),
|
|
||||||
ciphertext: data.subarray(33, -32),
|
|
||||||
mac: data.subarray(-32),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class v2 {
|
function calcPaddedLen(len: number): number {
|
||||||
static utils = u
|
if (!Number.isSafeInteger(len) || len < 1) throw new Error('expected positive integer')
|
||||||
|
if (len <= 32) return 32
|
||||||
|
const nextPower = 1 << (Math.floor(Math.log2(len - 1)) + 1)
|
||||||
|
const chunk = nextPower <= 256 ? 32 : nextPower / 8
|
||||||
|
return chunk * (Math.floor((len - 1) / chunk) + 1)
|
||||||
|
}
|
||||||
|
|
||||||
static encrypt(plaintext: string, conversationKey: Uint8Array, nonce: Uint8Array = randomBytes(32)): string {
|
function writeU16BE(num: number): Uint8Array {
|
||||||
const { chacha_key, chacha_nonce, hmac_key } = u.getMessageKeys(conversationKey, nonce)
|
if (!Number.isSafeInteger(num) || num < minPlaintextSize || num > maxPlaintextSize)
|
||||||
const padded = u.pad(plaintext)
|
throw new Error('invalid plaintext size: must be between 1 and 65535 bytes')
|
||||||
const ciphertext = chacha20(chacha_key, chacha_nonce, padded)
|
const arr = new Uint8Array(2)
|
||||||
const mac = u.hmacAad(hmac_key, ciphertext, nonce)
|
new DataView(arr.buffer).setUint16(0, num, false)
|
||||||
return base64.encode(concatBytes(new Uint8Array([2]), nonce, ciphertext, mac))
|
return arr
|
||||||
|
}
|
||||||
|
|
||||||
|
function pad(plaintext: string): Uint8Array {
|
||||||
|
const unpadded = utf8Encoder.encode(plaintext)
|
||||||
|
const unpaddedLen = unpadded.length
|
||||||
|
const prefix = writeU16BE(unpaddedLen)
|
||||||
|
const suffix = new Uint8Array(calcPaddedLen(unpaddedLen) - unpaddedLen)
|
||||||
|
return concatBytes(prefix, unpadded, suffix)
|
||||||
|
}
|
||||||
|
|
||||||
|
function unpad(padded: Uint8Array): string {
|
||||||
|
const unpaddedLen = new DataView(padded.buffer).getUint16(0)
|
||||||
|
const unpadded = padded.subarray(2, 2 + unpaddedLen)
|
||||||
|
if (
|
||||||
|
unpaddedLen < minPlaintextSize ||
|
||||||
|
unpaddedLen > maxPlaintextSize ||
|
||||||
|
unpadded.length !== unpaddedLen ||
|
||||||
|
padded.length !== 2 + calcPaddedLen(unpaddedLen)
|
||||||
|
)
|
||||||
|
throw new Error('invalid padding')
|
||||||
|
return utf8Decoder.decode(unpadded)
|
||||||
|
}
|
||||||
|
|
||||||
|
function hmacAad(key: Uint8Array, message: Uint8Array, aad: Uint8Array): Uint8Array {
|
||||||
|
if (aad.length !== 32) throw new Error('AAD associated data must be 32 bytes')
|
||||||
|
const combined = concatBytes(aad, message)
|
||||||
|
return hmac(sha256, key, combined)
|
||||||
|
}
|
||||||
|
|
||||||
|
// metadata: always 65b (version: 1b, nonce: 32b, max: 32b)
|
||||||
|
// plaintext: 1b to 0xffff
|
||||||
|
// padded plaintext: 32b to 0xffff
|
||||||
|
// ciphertext: 32b+2 to 0xffff+2
|
||||||
|
// raw payload: 99 (65+32+2) to 65603 (65+0xffff+2)
|
||||||
|
// compressed payload (base64): 132b to 87472b
|
||||||
|
function decodePayload(payload: string): { nonce: Uint8Array; ciphertext: Uint8Array; mac: Uint8Array } {
|
||||||
|
if (typeof payload !== 'string') throw new Error('payload must be a valid string')
|
||||||
|
const plen = payload.length
|
||||||
|
if (plen < 132 || plen > 87472) throw new Error('invalid payload length: ' + plen)
|
||||||
|
if (payload[0] === '#') throw new Error('unknown encryption version')
|
||||||
|
let data: Uint8Array
|
||||||
|
try {
|
||||||
|
data = base64.decode(payload)
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error('invalid base64: ' + (error as any).message)
|
||||||
}
|
}
|
||||||
|
const dlen = data.length
|
||||||
static decrypt(payload: string, conversationKey: Uint8Array): string {
|
if (dlen < 99 || dlen > 65603) throw new Error('invalid data length: ' + dlen)
|
||||||
const { nonce, ciphertext, mac } = u.decodePayload(payload)
|
const vers = data[0]
|
||||||
const { chacha_key, chacha_nonce, hmac_key } = u.getMessageKeys(conversationKey, nonce)
|
if (vers !== 2) throw new Error('unknown encryption version ' + vers)
|
||||||
const calculatedMac = u.hmacAad(hmac_key, ciphertext, nonce)
|
return {
|
||||||
if (!equalBytes(calculatedMac, mac)) throw new Error('invalid MAC')
|
nonce: data.subarray(1, 33),
|
||||||
const padded = chacha20(chacha_key, chacha_nonce, ciphertext)
|
ciphertext: data.subarray(33, -32),
|
||||||
return u.unpad(padded)
|
mac: data.subarray(-32),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default { v2 }
|
export function encrypt(plaintext: string, conversationKey: Uint8Array, nonce: Uint8Array = randomBytes(32)): string {
|
||||||
|
const { chacha_key, chacha_nonce, hmac_key } = getMessageKeys(conversationKey, nonce)
|
||||||
|
const padded = pad(plaintext)
|
||||||
|
const ciphertext = chacha20(chacha_key, chacha_nonce, padded)
|
||||||
|
const mac = hmacAad(hmac_key, ciphertext, nonce)
|
||||||
|
return base64.encode(concatBytes(new Uint8Array([2]), nonce, ciphertext, mac))
|
||||||
|
}
|
||||||
|
|
||||||
|
export function decrypt(payload: string, conversationKey: Uint8Array): string {
|
||||||
|
const { nonce, ciphertext, mac } = decodePayload(payload)
|
||||||
|
const { chacha_key, chacha_nonce, hmac_key } = getMessageKeys(conversationKey, nonce)
|
||||||
|
const calculatedMac = hmacAad(hmac_key, ciphertext, nonce)
|
||||||
|
if (!equalBytes(calculatedMac, mac)) throw new Error('invalid MAC')
|
||||||
|
const padded = chacha20(chacha_key, chacha_nonce, ciphertext)
|
||||||
|
return unpad(padded)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const v2 = {
|
||||||
|
utils: {
|
||||||
|
getConversationKey,
|
||||||
|
calcPaddedLen,
|
||||||
|
},
|
||||||
|
encrypt,
|
||||||
|
decrypt,
|
||||||
|
}
|
||||||
|
|||||||
92
nip46.ts
92
nip46.ts
@@ -1,11 +1,12 @@
|
|||||||
import { NostrEvent, UnsignedEvent, VerifiedEvent } from './core.ts'
|
import { EventTemplate, NostrEvent, VerifiedEvent } from './core.ts'
|
||||||
import { generateSecretKey, finalizeEvent, getPublicKey, verifyEvent } from './pure.ts'
|
import { generateSecretKey, finalizeEvent, getPublicKey, verifyEvent } from './pure.ts'
|
||||||
import { AbstractSimplePool, SubCloser } from './abstract-pool.ts'
|
import { AbstractSimplePool, SubCloser } from './abstract-pool.ts'
|
||||||
import { decrypt, encrypt } from './nip04.ts'
|
import { getConversationKey, decrypt, encrypt } from './nip44.ts'
|
||||||
import { NIP05_REGEX } from './nip05.ts'
|
import { NIP05_REGEX } from './nip05.ts'
|
||||||
import { SimplePool } from './pool.ts'
|
import { SimplePool } from './pool.ts'
|
||||||
import { Handlerinformation, NostrConnect } from './kinds.ts'
|
import { Handlerinformation, NostrConnect } from './kinds.ts'
|
||||||
import { hexToBytes } from '@noble/hashes/utils'
|
import type { RelayRecord } from './relay.ts'
|
||||||
|
import { Signer } from './signer.ts'
|
||||||
|
|
||||||
var _fetch: any
|
var _fetch: any
|
||||||
|
|
||||||
@@ -17,7 +18,7 @@ export function useFetchImplementation(fetchImplementation: any) {
|
|||||||
_fetch = fetchImplementation
|
_fetch = fetchImplementation
|
||||||
}
|
}
|
||||||
|
|
||||||
export const BUNKER_REGEX = /^bunker:\/\/([0-9a-f]{64})\??([?\/\w:.=&%]*)$/
|
export const BUNKER_REGEX = /^bunker:\/\/([0-9a-f]{64})\??([?\/\w:.=&%-]*)$/
|
||||||
const EMAIL_REGEX = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
|
const EMAIL_REGEX = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
|
||||||
|
|
||||||
export type BunkerPointer = {
|
export type BunkerPointer = {
|
||||||
@@ -26,6 +27,17 @@ export type BunkerPointer = {
|
|||||||
secret: null | string
|
secret: null | string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function toBunkerURL(bunkerPointer: BunkerPointer): string {
|
||||||
|
let bunkerURL = new URL(`bunker://${bunkerPointer.pubkey}`)
|
||||||
|
bunkerPointer.relays.forEach(relay => {
|
||||||
|
bunkerURL.searchParams.append('relay', relay)
|
||||||
|
})
|
||||||
|
if (bunkerPointer.secret) {
|
||||||
|
bunkerURL.searchParams.set('secret', bunkerPointer.secret)
|
||||||
|
}
|
||||||
|
return bunkerURL.toString()
|
||||||
|
}
|
||||||
|
|
||||||
/** This takes either a bunker:// URL or a name@domain.com NIP-05 identifier
|
/** This takes either a bunker:// URL or a name@domain.com NIP-05 identifier
|
||||||
and returns a BunkerPointer -- or null in case of error */
|
and returns a BunkerPointer -- or null in case of error */
|
||||||
export async function parseBunkerInput(input: string): Promise<BunkerPointer | null> {
|
export async function parseBunkerInput(input: string): Promise<BunkerPointer | null> {
|
||||||
@@ -47,7 +59,7 @@ export async function parseBunkerInput(input: string): Promise<BunkerPointer | n
|
|||||||
return queryBunkerProfile(input)
|
return queryBunkerProfile(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function queryBunkerProfile(nip05: string): Promise<BunkerPointer | null> {
|
export async function queryBunkerProfile(nip05: string): Promise<BunkerPointer | null> {
|
||||||
const match = nip05.match(NIP05_REGEX)
|
const match = nip05.match(NIP05_REGEX)
|
||||||
if (!match) return null
|
if (!match) return null
|
||||||
|
|
||||||
@@ -71,9 +83,10 @@ export type BunkerSignerParams = {
|
|||||||
onauth?: (url: string) => void
|
onauth?: (url: string) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export class BunkerSigner {
|
export class BunkerSigner implements Signer {
|
||||||
|
private params: BunkerSignerParams
|
||||||
private pool: AbstractSimplePool
|
private pool: AbstractSimplePool
|
||||||
private subCloser: SubCloser
|
private subCloser: SubCloser | undefined
|
||||||
private isOpen: boolean
|
private isOpen: boolean
|
||||||
private serial: number
|
private serial: number
|
||||||
private idPrefix: string
|
private idPrefix: string
|
||||||
@@ -85,8 +98,11 @@ export class BunkerSigner {
|
|||||||
}
|
}
|
||||||
private waitingForAuth: { [id: string]: boolean }
|
private waitingForAuth: { [id: string]: boolean }
|
||||||
private secretKey: Uint8Array
|
private secretKey: Uint8Array
|
||||||
|
private conversationKey: Uint8Array
|
||||||
public bp: BunkerPointer
|
public bp: BunkerPointer
|
||||||
|
|
||||||
|
private cachedPubKey: string | undefined
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new instance of the Nip46 class.
|
* Creates a new instance of the Nip46 class.
|
||||||
* @param relays - An array of relay addresses.
|
* @param relays - An array of relay addresses.
|
||||||
@@ -98,8 +114,10 @@ export class BunkerSigner {
|
|||||||
throw new Error('no relays are specified for this bunker')
|
throw new Error('no relays are specified for this bunker')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.params = params
|
||||||
this.pool = params.pool || new SimplePool()
|
this.pool = params.pool || new SimplePool()
|
||||||
this.secretKey = clientSecretKey
|
this.secretKey = clientSecretKey
|
||||||
|
this.conversationKey = getConversationKey(clientSecretKey, bp.pubkey)
|
||||||
this.bp = bp
|
this.bp = bp
|
||||||
this.isOpen = false
|
this.isOpen = false
|
||||||
this.idPrefix = Math.random().toString(36).substring(7)
|
this.idPrefix = Math.random().toString(36).substring(7)
|
||||||
@@ -107,15 +125,21 @@ export class BunkerSigner {
|
|||||||
this.listeners = {}
|
this.listeners = {}
|
||||||
this.waitingForAuth = {}
|
this.waitingForAuth = {}
|
||||||
|
|
||||||
|
this.setupSubscription(params)
|
||||||
|
}
|
||||||
|
|
||||||
|
private setupSubscription(params: BunkerSignerParams) {
|
||||||
const listeners = this.listeners
|
const listeners = this.listeners
|
||||||
const waitingForAuth = this.waitingForAuth
|
const waitingForAuth = this.waitingForAuth
|
||||||
|
const convKey = this.conversationKey
|
||||||
|
|
||||||
this.subCloser = this.pool.subscribeMany(
|
this.subCloser = this.pool.subscribe(
|
||||||
this.bp.relays,
|
this.bp.relays,
|
||||||
[{ kinds: [NostrConnect], '#p': [getPublicKey(this.secretKey)] }],
|
{ kinds: [NostrConnect], authors: [this.bp.pubkey], '#p': [getPublicKey(this.secretKey)] },
|
||||||
{
|
{
|
||||||
async onevent(event: NostrEvent) {
|
onevent: async (event: NostrEvent) => {
|
||||||
const { id, result, error } = JSON.parse(await decrypt(clientSecretKey, event.pubkey, event.content))
|
const o = JSON.parse(decrypt(event.content, convKey))
|
||||||
|
const { id, result, error } = o
|
||||||
|
|
||||||
if (result === 'auth_url' && waitingForAuth[id]) {
|
if (result === 'auth_url' && waitingForAuth[id]) {
|
||||||
delete waitingForAuth[id]
|
delete waitingForAuth[id]
|
||||||
@@ -124,7 +148,7 @@ export class BunkerSigner {
|
|||||||
params.onauth(error)
|
params.onauth(error)
|
||||||
} else {
|
} else {
|
||||||
console.warn(
|
console.warn(
|
||||||
`nostr-tools/nip46: remote signer ${bp.pubkey} tried to send an "auth_url"='${error}' but there was no onauth() callback configured.`,
|
`nostr-tools/nip46: remote signer ${this.bp.pubkey} tried to send an "auth_url"='${error}' but there was no onauth() callback configured.`,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
@@ -137,6 +161,9 @@ export class BunkerSigner {
|
|||||||
delete listeners[id]
|
delete listeners[id]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
onclose: () => {
|
||||||
|
this.subCloser = undefined
|
||||||
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
this.isOpen = true
|
this.isOpen = true
|
||||||
@@ -145,17 +172,19 @@ export class BunkerSigner {
|
|||||||
// closes the subscription -- this object can't be used anymore after this
|
// closes the subscription -- this object can't be used anymore after this
|
||||||
async close() {
|
async close() {
|
||||||
this.isOpen = false
|
this.isOpen = false
|
||||||
this.subCloser.close()
|
this.subCloser!.close()
|
||||||
}
|
}
|
||||||
|
|
||||||
async sendRequest(method: string, params: string[]): Promise<string> {
|
async sendRequest(method: string, params: string[]): Promise<string> {
|
||||||
return new Promise(async (resolve, reject) => {
|
return new Promise(async (resolve, reject) => {
|
||||||
try {
|
try {
|
||||||
if (!this.isOpen) throw new Error('this signer is not open anymore, create a new one')
|
if (!this.isOpen) throw new Error('this signer is not open anymore, create a new one')
|
||||||
|
if (!this.subCloser) this.setupSubscription(this.params)
|
||||||
|
|
||||||
this.serial++
|
this.serial++
|
||||||
const id = `${this.idPrefix}-${this.serial}`
|
const id = `${this.idPrefix}-${this.serial}`
|
||||||
|
|
||||||
const encryptedContent = await encrypt(this.secretKey, this.bp.pubkey, JSON.stringify({ id, method, params }))
|
const encryptedContent = encrypt(JSON.stringify({ id, method, params }), this.conversationKey)
|
||||||
|
|
||||||
// the request event
|
// the request event
|
||||||
const verifiedEvent: VerifiedEvent = finalizeEvent(
|
const verifiedEvent: VerifiedEvent = finalizeEvent(
|
||||||
@@ -197,17 +226,22 @@ export class BunkerSigner {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This was supposed to call the "get_public_key" method on the bunker,
|
* Calls the "get_public_key" method on the bunker.
|
||||||
* but instead we just returns the public key we already know.
|
* (before we would return the public key hardcoded in the bunker parameters, but
|
||||||
|
* that is not correct as that may be the bunker pubkey and the actual signer
|
||||||
|
* pubkey may be different.)
|
||||||
*/
|
*/
|
||||||
async getPublicKey(): Promise<string> {
|
async getPublicKey(): Promise<string> {
|
||||||
return this.bp.pubkey
|
if (!this.cachedPubKey) {
|
||||||
|
this.cachedPubKey = await this.sendRequest('get_public_key', [])
|
||||||
|
}
|
||||||
|
return this.cachedPubKey
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calls the "get_relays" method on the bunker.
|
* @deprecated removed from NIP
|
||||||
*/
|
*/
|
||||||
async getRelays(): Promise<{ [relay: string]: { read: boolean; write: boolean } }> {
|
async getRelays(): Promise<RelayRecord> {
|
||||||
return JSON.parse(await this.sendRequest('get_relays', []))
|
return JSON.parse(await this.sendRequest('get_relays', []))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -216,10 +250,10 @@ export class BunkerSigner {
|
|||||||
* @param event - The event to sign.
|
* @param event - The event to sign.
|
||||||
* @returns A Promise that resolves to the signed event.
|
* @returns A Promise that resolves to the signed event.
|
||||||
*/
|
*/
|
||||||
async signEvent(event: UnsignedEvent): Promise<VerifiedEvent> {
|
async signEvent(event: EventTemplate): Promise<VerifiedEvent> {
|
||||||
let resp = await this.sendRequest('sign_event', [JSON.stringify(event)])
|
let resp = await this.sendRequest('sign_event', [JSON.stringify(event)])
|
||||||
let signed: NostrEvent = JSON.parse(resp)
|
let signed: NostrEvent = JSON.parse(resp)
|
||||||
if (signed.pubkey === this.bp.pubkey && verifyEvent(signed)) {
|
if (verifyEvent(signed)) {
|
||||||
return signed
|
return signed
|
||||||
} else {
|
} else {
|
||||||
throw new Error(`event returned from bunker is improperly signed: ${JSON.stringify(signed)}`)
|
throw new Error(`event returned from bunker is improperly signed: ${JSON.stringify(signed)}`)
|
||||||
@@ -234,17 +268,12 @@ export class BunkerSigner {
|
|||||||
return await this.sendRequest('nip04_decrypt', [thirdPartyPubkey, ciphertext])
|
return await this.sendRequest('nip04_decrypt', [thirdPartyPubkey, ciphertext])
|
||||||
}
|
}
|
||||||
|
|
||||||
async nip44GetKey(thirdPartyPubkey: string): Promise<Uint8Array> {
|
|
||||||
let resp = await this.sendRequest('nip44_get_key', [thirdPartyPubkey])
|
|
||||||
return hexToBytes(resp)
|
|
||||||
}
|
|
||||||
|
|
||||||
async nip44Encrypt(thirdPartyPubkey: string, plaintext: string): Promise<string> {
|
async nip44Encrypt(thirdPartyPubkey: string, plaintext: string): Promise<string> {
|
||||||
return await this.sendRequest('nip44_encrypt', [thirdPartyPubkey, plaintext])
|
return await this.sendRequest('nip44_encrypt', [thirdPartyPubkey, plaintext])
|
||||||
}
|
}
|
||||||
|
|
||||||
async nip44Decrypt(thirdPartyPubkey: string, ciphertext: string): Promise<string> {
|
async nip44Decrypt(thirdPartyPubkey: string, ciphertext: string): Promise<string> {
|
||||||
return await this.sendRequest('nip44_encrypt', [thirdPartyPubkey, ciphertext])
|
return await this.sendRequest('nip44_decrypt', [thirdPartyPubkey, ciphertext])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -254,6 +283,8 @@ export class BunkerSigner {
|
|||||||
* @param username - The username for the account.
|
* @param username - The username for the account.
|
||||||
* @param domain - The domain for the account.
|
* @param domain - The domain for the account.
|
||||||
* @param email - The optional email for the account.
|
* @param email - The optional email for the account.
|
||||||
|
* @param localSecretKey - Optionally pass a local secret key that will be used to communicate with the bunker,
|
||||||
|
this will default to generating a random key.
|
||||||
* @throws Error if the email is present but invalid.
|
* @throws Error if the email is present but invalid.
|
||||||
* @returns A Promise that resolves to the auth_url that the client should follow to create an account.
|
* @returns A Promise that resolves to the auth_url that the client should follow to create an account.
|
||||||
*/
|
*/
|
||||||
@@ -263,11 +294,11 @@ export async function createAccount(
|
|||||||
username: string,
|
username: string,
|
||||||
domain: string,
|
domain: string,
|
||||||
email?: string,
|
email?: string,
|
||||||
|
localSecretKey: Uint8Array = generateSecretKey(),
|
||||||
): Promise<BunkerSigner> {
|
): Promise<BunkerSigner> {
|
||||||
if (email && !EMAIL_REGEX.test(email)) throw new Error('Invalid email')
|
if (email && !EMAIL_REGEX.test(email)) throw new Error('Invalid email')
|
||||||
|
|
||||||
let sk = generateSecretKey()
|
let rpc = new BunkerSigner(localSecretKey, bunker.bunkerPointer, params)
|
||||||
let rpc = new BunkerSigner(sk, bunker.bunkerPointer, params)
|
|
||||||
|
|
||||||
let pubkey = await rpc.sendRequest('create_account', [username, domain, email || ''])
|
let pubkey = await rpc.sendRequest('create_account', [username, domain, email || ''])
|
||||||
|
|
||||||
@@ -279,9 +310,6 @@ export async function createAccount(
|
|||||||
return rpc
|
return rpc
|
||||||
}
|
}
|
||||||
|
|
||||||
// @deprecated use fetchBunkerProviders instead
|
|
||||||
export const fetchCustodialBunkers = fetchBunkerProviders
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetches info on available providers that announce themselves using NIP-89 events.
|
* Fetches info on available providers that announce themselves using NIP-89 events.
|
||||||
* @returns A promise that resolves to an array of available bunker objects.
|
* @returns A promise that resolves to an array of available bunker objects.
|
||||||
|
|||||||
2
nip47.ts
2
nip47.ts
@@ -32,7 +32,7 @@ export async function makeNwcRequestEvent(
|
|||||||
invoice,
|
invoice,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
const encryptedContent = await encrypt(secretKey, pubkey, JSON.stringify(content))
|
const encryptedContent = encrypt(secretKey, pubkey, JSON.stringify(content))
|
||||||
const eventTemplate = {
|
const eventTemplate = {
|
||||||
kind: NWCWalletRequest,
|
kind: NWCWalletRequest,
|
||||||
created_at: Math.round(Date.now() / 1000),
|
created_at: Math.round(Date.now() / 1000),
|
||||||
|
|||||||
9
nip49.ts
9
nip49.ts
@@ -1,10 +1,15 @@
|
|||||||
import { scrypt } from '@noble/hashes/scrypt'
|
import { scrypt } from '@noble/hashes/scrypt'
|
||||||
import { xchacha20poly1305 } from '@noble/ciphers/chacha'
|
import { xchacha20poly1305 } from '@noble/ciphers/chacha'
|
||||||
import { concatBytes, randomBytes } from '@noble/hashes/utils'
|
import { concatBytes, randomBytes } from '@noble/hashes/utils'
|
||||||
import { Bech32MaxSize, encodeBytes } from './nip19.ts'
|
import { Bech32MaxSize, Ncryptsec, encodeBytes } from './nip19.ts'
|
||||||
import { bech32 } from '@scure/base'
|
import { bech32 } from '@scure/base'
|
||||||
|
|
||||||
export function encrypt(sec: Uint8Array, password: string, logn: number = 16, ksb: 0x00 | 0x01 | 0x02 = 0x02): string {
|
export function encrypt(
|
||||||
|
sec: Uint8Array,
|
||||||
|
password: string,
|
||||||
|
logn: number = 16,
|
||||||
|
ksb: 0x00 | 0x01 | 0x02 = 0x02,
|
||||||
|
): Ncryptsec {
|
||||||
let salt = randomBytes(16)
|
let salt = randomBytes(16)
|
||||||
let n = 2 ** logn
|
let n = 2 ** logn
|
||||||
let key = scrypt(password.normalize('NFKC'), salt, { N: n, r: 8, p: 1, dkLen: 32 })
|
let key = scrypt(password.normalize('NFKC'), salt, { N: n, r: 8, p: 1, dkLen: 32 })
|
||||||
|
|||||||
42
nip54.test.ts
Normal file
42
nip54.test.ts
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import { describe, test, expect } from 'bun:test'
|
||||||
|
import { normalizeIdentifier } from './nip54.ts'
|
||||||
|
|
||||||
|
describe('normalizeIdentifier', () => {
|
||||||
|
test('converts to lowercase', () => {
|
||||||
|
expect(normalizeIdentifier('HELLO')).toBe('hello')
|
||||||
|
expect(normalizeIdentifier('MixedCase')).toBe('mixedcase')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('trims whitespace', () => {
|
||||||
|
expect(normalizeIdentifier(' hello ')).toBe('hello')
|
||||||
|
expect(normalizeIdentifier('\thello\n')).toBe('hello')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('normalizes Unicode to NFKC form', () => {
|
||||||
|
// é can be represented as single char é (U+00E9) or e + ´ (U+0065 U+0301)
|
||||||
|
expect(normalizeIdentifier('café')).toBe('café')
|
||||||
|
expect(normalizeIdentifier('cafe\u0301')).toBe('café')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('replaces non-alphanumeric characters with hyphens', () => {
|
||||||
|
expect(normalizeIdentifier('hello world')).toBe('hello-world')
|
||||||
|
expect(normalizeIdentifier('user@example.com')).toBe('user-example-com')
|
||||||
|
expect(normalizeIdentifier('$special#chars!')).toBe('-special-chars-')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('preserves numbers', () => {
|
||||||
|
expect(normalizeIdentifier('user123')).toBe('user123')
|
||||||
|
expect(normalizeIdentifier('2fast4you')).toBe('2fast4you')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('handles multiple consecutive special characters', () => {
|
||||||
|
expect(normalizeIdentifier('hello!!!world')).toBe('hello---world')
|
||||||
|
expect(normalizeIdentifier('multiple spaces')).toBe('multiple---spaces')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('handles Unicode letters from different scripts', () => {
|
||||||
|
expect(normalizeIdentifier('привет')).toBe('привет')
|
||||||
|
expect(normalizeIdentifier('こんにちは')).toBe('こんにちは')
|
||||||
|
expect(normalizeIdentifier('مرحبا')).toBe('مرحبا')
|
||||||
|
})
|
||||||
|
})
|
||||||
19
nip54.ts
Normal file
19
nip54.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
export function normalizeIdentifier(name: string): string {
|
||||||
|
// Trim and lowercase
|
||||||
|
name = name.trim().toLowerCase()
|
||||||
|
|
||||||
|
// Normalize Unicode to NFKC form
|
||||||
|
name = name.normalize('NFKC')
|
||||||
|
|
||||||
|
// Convert to array of characters and map each one
|
||||||
|
return Array.from(name)
|
||||||
|
.map(char => {
|
||||||
|
// Check if character is letter or number using Unicode ranges
|
||||||
|
if (/\p{Letter}/u.test(char) || /\p{Number}/u.test(char)) {
|
||||||
|
return char
|
||||||
|
}
|
||||||
|
|
||||||
|
return '-'
|
||||||
|
})
|
||||||
|
.join('')
|
||||||
|
}
|
||||||
166
nip55.test.ts
Normal file
166
nip55.test.ts
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
import { test, expect } from 'bun:test'
|
||||||
|
import * as nip55 from './nip55.js'
|
||||||
|
|
||||||
|
// Function to parse the NostrSigner URI
|
||||||
|
function parseNostrSignerUri(uri: string) {
|
||||||
|
const [base, query] = uri.split('?')
|
||||||
|
const basePart = base.replace('nostrsigner:', '')
|
||||||
|
|
||||||
|
let jsonObject = null
|
||||||
|
if (basePart) {
|
||||||
|
try {
|
||||||
|
jsonObject = JSON.parse(decodeURIComponent(basePart))
|
||||||
|
} catch (e) {
|
||||||
|
console.warn('Failed to parse base JSON:', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const urlSearchParams = new URLSearchParams(query)
|
||||||
|
const queryParams = Object.fromEntries(urlSearchParams.entries())
|
||||||
|
if (queryParams.permissions) {
|
||||||
|
queryParams.permissions = JSON.parse(decodeURIComponent(queryParams.permissions))
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
base: jsonObject,
|
||||||
|
...queryParams,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test cases
|
||||||
|
test('Get Public Key URI', () => {
|
||||||
|
const permissions = [{ type: 'sign_event', kind: 22242 }, { type: 'nip44_decrypt' }]
|
||||||
|
const callbackUrl = 'https://example.com/?event='
|
||||||
|
|
||||||
|
const uri = nip55.getPublicKeyUri({
|
||||||
|
permissions,
|
||||||
|
callbackUrl,
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'get_public_key')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('callbackUrl', 'https://example.com/?event=')
|
||||||
|
expect(jsonObject).toHaveProperty('permissions[0].type', 'sign_event')
|
||||||
|
expect(jsonObject).toHaveProperty('permissions[0].kind', 22242)
|
||||||
|
expect(jsonObject).toHaveProperty('permissions[1].type', 'nip44_decrypt')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Sign Event URI', () => {
|
||||||
|
const eventJson = { kind: 1, content: 'test' }
|
||||||
|
|
||||||
|
const uri = nip55.signEventUri({
|
||||||
|
eventJson,
|
||||||
|
id: 'some_id',
|
||||||
|
currentUser: 'hex_pub_key',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('base.kind', 1)
|
||||||
|
expect(jsonObject).toHaveProperty('base.content', 'test')
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'sign_event')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('id', 'some_id')
|
||||||
|
expect(jsonObject).toHaveProperty('current_user', 'hex_pub_key')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Encrypt NIP-04 URI', () => {
|
||||||
|
const callbackUrl = 'https://example.com/?event='
|
||||||
|
|
||||||
|
const uri = nip55.encryptNip04Uri({
|
||||||
|
callbackUrl,
|
||||||
|
pubKey: 'hex_pub_key',
|
||||||
|
content: 'plainText',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'nip04_encrypt')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('callbackUrl', callbackUrl)
|
||||||
|
expect(jsonObject).toHaveProperty('pubKey', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('plainText', 'plainText')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Decrypt NIP-04 URI', () => {
|
||||||
|
const uri = nip55.decryptNip04Uri({
|
||||||
|
id: 'some_id',
|
||||||
|
currentUser: 'hex_pub_key',
|
||||||
|
pubKey: 'hex_pub_key',
|
||||||
|
content: 'encryptedText',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'nip04_decrypt')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('id', 'some_id')
|
||||||
|
expect(jsonObject).toHaveProperty('current_user', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('pubKey', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('encryptedText', 'encryptedText')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Encrypt NIP-44 URI', () => {
|
||||||
|
const uri = nip55.encryptNip44Uri({
|
||||||
|
id: 'some_id',
|
||||||
|
currentUser: 'hex_pub_key',
|
||||||
|
pubKey: 'hex_pub_key',
|
||||||
|
content: 'plainText',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'nip44_encrypt')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('id', 'some_id')
|
||||||
|
expect(jsonObject).toHaveProperty('current_user', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('pubKey', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('plainText', 'plainText')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Decrypt NIP-44 URI', () => {
|
||||||
|
const uri = nip55.decryptNip44Uri({
|
||||||
|
id: 'some_id',
|
||||||
|
currentUser: 'hex_pub_key',
|
||||||
|
pubKey: 'hex_pub_key',
|
||||||
|
content: 'encryptedText',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'nip44_decrypt')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'none')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'signature')
|
||||||
|
expect(jsonObject).toHaveProperty('id', 'some_id')
|
||||||
|
expect(jsonObject).toHaveProperty('current_user', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('pubKey', 'hex_pub_key')
|
||||||
|
expect(jsonObject).toHaveProperty('encryptedText', 'encryptedText')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Decrypt Zap Event URI', () => {
|
||||||
|
const eventJson = { kind: 1, content: 'test' }
|
||||||
|
|
||||||
|
const uri = nip55.decryptZapEventUri({
|
||||||
|
eventJson,
|
||||||
|
id: 'some_id',
|
||||||
|
currentUser: 'hex_pub_key',
|
||||||
|
returnType: 'event',
|
||||||
|
compressionType: 'gzip',
|
||||||
|
})
|
||||||
|
|
||||||
|
const jsonObject = parseNostrSignerUri(uri)
|
||||||
|
|
||||||
|
expect(jsonObject).toHaveProperty('type', 'decrypt_zap_event')
|
||||||
|
expect(jsonObject).toHaveProperty('compressionType', 'gzip')
|
||||||
|
expect(jsonObject).toHaveProperty('returnType', 'event')
|
||||||
|
expect(jsonObject).toHaveProperty('base.kind', 1)
|
||||||
|
expect(jsonObject).toHaveProperty('id', 'some_id')
|
||||||
|
expect(jsonObject).toHaveProperty('current_user', 'hex_pub_key')
|
||||||
|
})
|
||||||
123
nip55.ts
Normal file
123
nip55.ts
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
type BaseParams = {
|
||||||
|
callbackUrl?: string
|
||||||
|
returnType?: 'signature' | 'event'
|
||||||
|
compressionType?: 'none' | 'gzip'
|
||||||
|
}
|
||||||
|
|
||||||
|
type PermissionsParams = BaseParams & {
|
||||||
|
permissions?: { type: string; kind?: number }[]
|
||||||
|
}
|
||||||
|
|
||||||
|
type EventUriParams = BaseParams & {
|
||||||
|
eventJson: Record<string, unknown>
|
||||||
|
id?: string
|
||||||
|
currentUser?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type EncryptDecryptParams = BaseParams & {
|
||||||
|
pubKey: string
|
||||||
|
content: string
|
||||||
|
id?: string
|
||||||
|
currentUser?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type UriParams = BaseParams & {
|
||||||
|
base: string
|
||||||
|
type: string
|
||||||
|
id?: string
|
||||||
|
currentUser?: string
|
||||||
|
permissions?: { type: string; kind?: number }[]
|
||||||
|
pubKey?: string
|
||||||
|
plainText?: string
|
||||||
|
encryptedText?: string
|
||||||
|
appName?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
function encodeParams(params: Record<string, unknown>): string {
|
||||||
|
return new URLSearchParams(params as Record<string, string>).toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterUndefined<T extends Record<string, unknown>>(obj: T): T {
|
||||||
|
return Object.fromEntries(Object.entries(obj).filter(([, value]) => value !== undefined)) as T
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildUri({
|
||||||
|
base,
|
||||||
|
type,
|
||||||
|
callbackUrl,
|
||||||
|
returnType = 'signature',
|
||||||
|
compressionType = 'none',
|
||||||
|
...params
|
||||||
|
}: UriParams): string {
|
||||||
|
const baseParams = {
|
||||||
|
type,
|
||||||
|
compressionType,
|
||||||
|
returnType,
|
||||||
|
callbackUrl,
|
||||||
|
id: params.id,
|
||||||
|
current_user: params.currentUser,
|
||||||
|
permissions:
|
||||||
|
params.permissions && params.permissions.length > 0
|
||||||
|
? encodeURIComponent(JSON.stringify(params.permissions))
|
||||||
|
: undefined,
|
||||||
|
pubKey: params.pubKey,
|
||||||
|
plainText: params.plainText,
|
||||||
|
encryptedText: params.encryptedText,
|
||||||
|
appName: params.appName,
|
||||||
|
}
|
||||||
|
|
||||||
|
const filteredParams = filterUndefined(baseParams)
|
||||||
|
return `${base}?${encodeParams(filteredParams)}`
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildDefaultUri(type: string, params: Partial<UriParams>): string {
|
||||||
|
return buildUri({
|
||||||
|
base: 'nostrsigner:',
|
||||||
|
type,
|
||||||
|
...params,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPublicKeyUri({ permissions = [], ...params }: PermissionsParams): string {
|
||||||
|
return buildDefaultUri('get_public_key', { permissions, ...params })
|
||||||
|
}
|
||||||
|
|
||||||
|
export function signEventUri({ eventJson, ...params }: EventUriParams): string {
|
||||||
|
return buildUri({
|
||||||
|
base: `nostrsigner:${encodeURIComponent(JSON.stringify(eventJson))}`,
|
||||||
|
type: 'sign_event',
|
||||||
|
...params,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function encryptUri(type: 'nip44_encrypt' | 'nip04_encrypt', params: EncryptDecryptParams): string {
|
||||||
|
return buildDefaultUri(type, { ...params, plainText: params.content })
|
||||||
|
}
|
||||||
|
|
||||||
|
function decryptUri(type: 'nip44_decrypt' | 'nip04_decrypt', params: EncryptDecryptParams): string {
|
||||||
|
return buildDefaultUri(type, { ...params, encryptedText: params.content })
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptNip04Uri(params: EncryptDecryptParams): string {
|
||||||
|
return encryptUri('nip04_encrypt', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function decryptNip04Uri(params: EncryptDecryptParams): string {
|
||||||
|
return decryptUri('nip04_decrypt', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptNip44Uri(params: EncryptDecryptParams): string {
|
||||||
|
return encryptUri('nip44_encrypt', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function decryptNip44Uri(params: EncryptDecryptParams): string {
|
||||||
|
return decryptUri('nip44_decrypt', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function decryptZapEventUri({ eventJson, ...params }: EventUriParams): string {
|
||||||
|
return buildUri({
|
||||||
|
base: `nostrsigner:${encodeURIComponent(JSON.stringify(eventJson))}`,
|
||||||
|
type: 'decrypt_zap_event',
|
||||||
|
...params,
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -1,7 +1,14 @@
|
|||||||
import { describe, test, expect, mock } from 'bun:test'
|
import { describe, test, expect, mock } from 'bun:test'
|
||||||
import { finalizeEvent } from './pure.ts'
|
import { finalizeEvent } from './pure.ts'
|
||||||
import { getPublicKey, generateSecretKey } from './pure.ts'
|
import { getPublicKey, generateSecretKey } from './pure.ts'
|
||||||
import { getZapEndpoint, makeZapReceipt, makeZapRequest, useFetchImplementation, validateZapRequest } from './nip57.ts'
|
import {
|
||||||
|
getSatoshisAmountFromBolt11,
|
||||||
|
getZapEndpoint,
|
||||||
|
makeZapReceipt,
|
||||||
|
makeZapRequest,
|
||||||
|
useFetchImplementation,
|
||||||
|
validateZapRequest,
|
||||||
|
} from './nip57.ts'
|
||||||
import { buildEvent } from './test-helpers.ts'
|
import { buildEvent } from './test-helpers.ts'
|
||||||
|
|
||||||
describe('getZapEndpoint', () => {
|
describe('getZapEndpoint', () => {
|
||||||
@@ -317,3 +324,26 @@ describe('makeZapReceipt', () => {
|
|||||||
expect(JSON.stringify(result.tags)).not.toContain('preimage')
|
expect(JSON.stringify(result.tags)).not.toContain('preimage')
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('parses the amount from bolt11 invoices', () => {
|
||||||
|
expect(
|
||||||
|
getSatoshisAmountFromBolt11(
|
||||||
|
'lnbc4u1p5zcarnpp5djng98r73nxu66nxp6gndjkw24q7rdzgp7p80lt0gk4z3h3krkssdq9tfpygcqzzsxqzjcsp58hz3v5qefdm70g5fnm2cn6q9thzpu6m4f5wjqurhur5xzmf9vl3s9qxpqysgq9v6qv86xaruzeak9jjyz54fygrkn526z7xhm0llh8wl44gcgh0rznhjqdswd4cjurzdgh0pgzrfj4sd7f3mf89jd6kadse008ex7kxgqqa5xrk',
|
||||||
|
),
|
||||||
|
).toEqual(400)
|
||||||
|
expect(
|
||||||
|
getSatoshisAmountFromBolt11(
|
||||||
|
'lnbc8400u1p5zcaz5pp5ltvyhtg4ed7sd8jurj28ugmavezkmqsadpe3t9npufpcrd0uet0scqzyssp5l3hz4ayt5ee0p83ma4a96l2rruhx33eyycewldu2ffa5pk2qx7jq9q7sqqqqqqqqqqqqqqqqqqqsqqqqqysgqdq8w3jhxaqmqz9gxqyjw5qrzjqwryaup9lh50kkranzgcdnn2fgvx390wgj5jd07rwr3vxeje0glclll8qkt3np4rqyqqqqlgqqqqqeqqjqhuhjk5u9r850ncxngne7cfp9s08s2nm6c2rkz7jhl8gjmlx0fga5tlncgeuh4avlsrkq6ljyyhgq8rrxprga03esqhd0gf5455x6tdcqahhw9q',
|
||||||
|
),
|
||||||
|
).toEqual(840000)
|
||||||
|
expect(
|
||||||
|
getSatoshisAmountFromBolt11(
|
||||||
|
'lnbc210n1p5zcuaxpp52nn778cfk46md4ld0hdj2juuzvfrsrdaf4ek2k0yeensae07x2cqdq9tfpygcqzzsxqzjcsp5768c4k79jtnq92pgppan8rjnujcpcqhnqwqwk3lm5dfr7e0k2a7s9qxpqysgqt8lnh9l7ple27t73x7gty570ltas2s33uahc7egke5tdmhxr3ezn590wf2utxyt7d3afnk2lxc2u0enc6n53ck4mxwpmzpxa7ws05aqp0c5x3r',
|
||||||
|
),
|
||||||
|
).toEqual(21)
|
||||||
|
expect(
|
||||||
|
getSatoshisAmountFromBolt11(
|
||||||
|
'lnbc899640n1p5zcuavpp5w72fqrf09286lq33vw364qryrq5nw60z4dhdx56f8w05xkx4massdq9tfpygcqzzsxqzjcsp5qrqn4kpvem5jwpl63kj5pfdlqxg2plaffz0prz7vaqjy29uc66us9qxpqysgqlhzzqmn2jxd2476404krm8nvrarymwq7nj2zecl92xug54ek0mfntdxvxwslf756m8kq0r7jtpantm52fmewc72r5lfmd85505jnemgqw5j0pc',
|
||||||
|
),
|
||||||
|
).toEqual(89964)
|
||||||
|
})
|
||||||
|
|||||||
67
nip57.ts
67
nip57.ts
@@ -2,6 +2,7 @@ import { bech32 } from '@scure/base'
|
|||||||
|
|
||||||
import { validateEvent, verifyEvent, type Event, type EventTemplate } from './pure.ts'
|
import { validateEvent, verifyEvent, type Event, type EventTemplate } from './pure.ts'
|
||||||
import { utf8Decoder } from './utils.ts'
|
import { utf8Decoder } from './utils.ts'
|
||||||
|
import { isReplaceableKind, isAddressableKind } from './kinds.ts'
|
||||||
|
|
||||||
var _fetch: any
|
var _fetch: any
|
||||||
|
|
||||||
@@ -49,7 +50,7 @@ export function makeZapRequest({
|
|||||||
comment = '',
|
comment = '',
|
||||||
}: {
|
}: {
|
||||||
profile: string
|
profile: string
|
||||||
event: string | null
|
event: string | Event | null
|
||||||
amount: number
|
amount: number
|
||||||
comment: string
|
comment: string
|
||||||
relays: string[]
|
relays: string[]
|
||||||
@@ -68,9 +69,22 @@ export function makeZapRequest({
|
|||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
if (event) {
|
if (event && typeof event === 'string') {
|
||||||
zr.tags.push(['e', event])
|
zr.tags.push(['e', event])
|
||||||
}
|
}
|
||||||
|
if (event && typeof event === 'object') {
|
||||||
|
// replacable event
|
||||||
|
if (isReplaceableKind(event.kind)) {
|
||||||
|
const a = ['a', `${event.kind}:${event.pubkey}:`]
|
||||||
|
zr.tags.push(a)
|
||||||
|
// addressable event
|
||||||
|
} else if (isAddressableKind(event.kind)) {
|
||||||
|
let d = event.tags.find(([t, v]) => t === 'd' && v)
|
||||||
|
if (!d) throw new Error('d tag not found or is empty')
|
||||||
|
const a = ['a', `${event.kind}:${event.pubkey}:${d[1]}`]
|
||||||
|
zr.tags.push(a)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return zr
|
return zr
|
||||||
}
|
}
|
||||||
@@ -128,3 +142,52 @@ export function makeZapReceipt({
|
|||||||
|
|
||||||
return zap
|
return zap
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getSatoshisAmountFromBolt11(bolt11: string): number {
|
||||||
|
if (bolt11.length < 50) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
bolt11 = bolt11.substring(0, 50)
|
||||||
|
const idx = bolt11.lastIndexOf('1')
|
||||||
|
if (idx === -1) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
const hrp = bolt11.substring(0, idx)
|
||||||
|
if (!hrp.startsWith('lnbc')) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
const amount = hrp.substring(4) // equivalent to strings.CutPrefix
|
||||||
|
|
||||||
|
if (amount.length < 1) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// if last character is a digit, then the amount can just be interpreted as BTC
|
||||||
|
const char = amount[amount.length - 1]
|
||||||
|
const digit = char.charCodeAt(0) - '0'.charCodeAt(0)
|
||||||
|
const isDigit = digit >= 0 && digit <= 9
|
||||||
|
|
||||||
|
let cutPoint = amount.length - 1
|
||||||
|
if (isDigit) {
|
||||||
|
cutPoint++
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cutPoint < 1) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
const num = parseInt(amount.substring(0, cutPoint))
|
||||||
|
|
||||||
|
switch (char) {
|
||||||
|
case 'm':
|
||||||
|
return num * 100000
|
||||||
|
case 'u':
|
||||||
|
return num * 100
|
||||||
|
case 'n':
|
||||||
|
return num / 10
|
||||||
|
case 'p':
|
||||||
|
return num / 10000
|
||||||
|
default:
|
||||||
|
return num * 100000000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
113
nip59.test.ts
Normal file
113
nip59.test.ts
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
import { test, expect } from 'bun:test'
|
||||||
|
import { wrapEvent, wrapManyEvents, unwrapEvent, unwrapManyEvents } from './nip59.ts'
|
||||||
|
import { decode } from './nip19.ts'
|
||||||
|
import { NostrEvent, getPublicKey } from './pure.ts'
|
||||||
|
import { SimplePool } from './pool.ts'
|
||||||
|
import { GiftWrap } from './kinds.ts'
|
||||||
|
import { hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
|
const senderPrivateKey = decode(`nsec1p0ht6p3wepe47sjrgesyn4m50m6avk2waqudu9rl324cg2c4ufesyp6rdg`).data as Uint8Array
|
||||||
|
const recipientPrivateKey = decode(`nsec1uyyrnx7cgfp40fcskcr2urqnzekc20fj0er6de0q8qvhx34ahazsvs9p36`).data as Uint8Array
|
||||||
|
const recipientPublicKey = getPublicKey(recipientPrivateKey)
|
||||||
|
const event = {
|
||||||
|
kind: 1,
|
||||||
|
content: 'Are you going to the party tonight?',
|
||||||
|
}
|
||||||
|
|
||||||
|
const wrappedEvent = wrapEvent(event, senderPrivateKey, recipientPublicKey)
|
||||||
|
|
||||||
|
test('wrapEvent', () => {
|
||||||
|
const expected = {
|
||||||
|
content: '',
|
||||||
|
id: '',
|
||||||
|
created_at: 1728537932,
|
||||||
|
kind: 1059,
|
||||||
|
pubkey: '',
|
||||||
|
sig: '',
|
||||||
|
tags: [['p', '166bf3765ebd1fc55decfe395beff2ea3b2a4e0a8946e7eb578512b555737c99']],
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
}
|
||||||
|
const result = wrapEvent(event, senderPrivateKey, recipientPublicKey)
|
||||||
|
|
||||||
|
expect(result.kind).toEqual(expected.kind)
|
||||||
|
expect(result.tags).toEqual(expected.tags)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('wrapManyEvent', () => {
|
||||||
|
const expected = [
|
||||||
|
{
|
||||||
|
kind: 1059,
|
||||||
|
content: '',
|
||||||
|
created_at: 1729581521,
|
||||||
|
tags: [['p', '611df01bfcf85c26ae65453b772d8f1dfd25c264621c0277e1fc1518686faef9']],
|
||||||
|
pubkey: '',
|
||||||
|
id: '',
|
||||||
|
sig: '',
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
kind: 1059,
|
||||||
|
content: '',
|
||||||
|
created_at: 1729594619,
|
||||||
|
tags: [['p', '166bf3765ebd1fc55decfe395beff2ea3b2a4e0a8946e7eb578512b555737c99']],
|
||||||
|
pubkey: '',
|
||||||
|
id: '',
|
||||||
|
sig: '',
|
||||||
|
[Symbol('verified')]: true,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
const wrappedEvents = wrapManyEvents(event, senderPrivateKey, [recipientPublicKey])
|
||||||
|
|
||||||
|
wrappedEvents.forEach((event, index) => {
|
||||||
|
expect(event.kind).toEqual(expected[index].kind)
|
||||||
|
expect(event.tags).toEqual(expected[index].tags)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('unwrapEvent', () => {
|
||||||
|
const expected = {
|
||||||
|
kind: 1,
|
||||||
|
content: 'Are you going to the party tonight?',
|
||||||
|
pubkey: '611df01bfcf85c26ae65453b772d8f1dfd25c264621c0277e1fc1518686faef9',
|
||||||
|
tags: [],
|
||||||
|
}
|
||||||
|
const result = unwrapEvent(wrappedEvent, recipientPrivateKey)
|
||||||
|
|
||||||
|
expect(result.kind).toEqual(expected.kind)
|
||||||
|
expect(result.content).toEqual(expected.content)
|
||||||
|
expect(result.pubkey).toEqual(expected.pubkey)
|
||||||
|
expect(result.tags).toEqual(expected.tags)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('getWrappedEvents and unwrapManyEvents', async () => {
|
||||||
|
const expected = [
|
||||||
|
{
|
||||||
|
created_at: 1729721879,
|
||||||
|
content: 'Hello!',
|
||||||
|
tags: [['p', '33d6bb037bf2e8c4571708e480e42d141bedc5a562b4884ec233b22d6fdea6aa']],
|
||||||
|
kind: 14,
|
||||||
|
pubkey: 'c0f56665e73eedc90b9565ecb34d961a2eb7ac1e2747899e4f73a813f940bc22',
|
||||||
|
id: 'aee0a3e6487b2ac8c1851cc84f3ae0fca9af8a9bdad85c4ba5fdf45d3ee817c3',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
created_at: 1729722025,
|
||||||
|
content: 'How are you?',
|
||||||
|
tags: [['p', '33d6bb037bf2e8c4571708e480e42d141bedc5a562b4884ec233b22d6fdea6aa']],
|
||||||
|
kind: 14,
|
||||||
|
pubkey: 'c0f56665e73eedc90b9565ecb34d961a2eb7ac1e2747899e4f73a813f940bc22',
|
||||||
|
id: '212387ec5efee7d6eb20b747121e9fc1adb798de6c3185e932335bb1bcc61a77',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
const relays = ['wss://relay.damus.io', 'wss://nos.lol']
|
||||||
|
const privateKey = hexToBytes('582c3e7902c10c84d1cfe899a102e56bde628972d58d63011163ce0cdf4279b6')
|
||||||
|
const publicKey = '33d6bb037bf2e8c4571708e480e42d141bedc5a562b4884ec233b22d6fdea6aa'
|
||||||
|
|
||||||
|
const pool = new SimplePool()
|
||||||
|
const wrappedEvents: NostrEvent[] = await pool.querySync(relays, { kinds: [GiftWrap], '#p': [publicKey] })
|
||||||
|
const unwrappedEvents = unwrapManyEvents(wrappedEvents, privateKey)
|
||||||
|
|
||||||
|
unwrappedEvents.forEach((event, index) => {
|
||||||
|
expect(event).toEqual(expected[index])
|
||||||
|
})
|
||||||
|
})
|
||||||
107
nip59.ts
Normal file
107
nip59.ts
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
import { EventTemplate, UnsignedEvent, NostrEvent } from './core.ts'
|
||||||
|
import { getConversationKey, decrypt, encrypt } from './nip44.ts'
|
||||||
|
import { getEventHash, generateSecretKey, finalizeEvent, getPublicKey } from './pure.ts'
|
||||||
|
import { Seal, GiftWrap } from './kinds.ts'
|
||||||
|
|
||||||
|
type Rumor = UnsignedEvent & { id: string }
|
||||||
|
|
||||||
|
const TWO_DAYS = 2 * 24 * 60 * 60
|
||||||
|
|
||||||
|
const now = () => Math.round(Date.now() / 1000)
|
||||||
|
const randomNow = () => Math.round(now() - Math.random() * TWO_DAYS)
|
||||||
|
|
||||||
|
const nip44ConversationKey = (privateKey: Uint8Array, publicKey: string) => getConversationKey(privateKey, publicKey)
|
||||||
|
|
||||||
|
const nip44Encrypt = (data: EventTemplate, privateKey: Uint8Array, publicKey: string) =>
|
||||||
|
encrypt(JSON.stringify(data), nip44ConversationKey(privateKey, publicKey))
|
||||||
|
|
||||||
|
const nip44Decrypt = (data: NostrEvent, privateKey: Uint8Array) =>
|
||||||
|
JSON.parse(decrypt(data.content, nip44ConversationKey(privateKey, data.pubkey)))
|
||||||
|
|
||||||
|
export function createRumor(event: Partial<UnsignedEvent>, privateKey: Uint8Array): Rumor {
|
||||||
|
const rumor = {
|
||||||
|
created_at: now(),
|
||||||
|
content: '',
|
||||||
|
tags: [],
|
||||||
|
...event,
|
||||||
|
pubkey: getPublicKey(privateKey),
|
||||||
|
} as any
|
||||||
|
|
||||||
|
rumor.id = getEventHash(rumor)
|
||||||
|
|
||||||
|
return rumor as Rumor
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createSeal(rumor: Rumor, privateKey: Uint8Array, recipientPublicKey: string): NostrEvent {
|
||||||
|
return finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: Seal,
|
||||||
|
content: nip44Encrypt(rumor, privateKey, recipientPublicKey),
|
||||||
|
created_at: randomNow(),
|
||||||
|
tags: [],
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createWrap(seal: NostrEvent, recipientPublicKey: string): NostrEvent {
|
||||||
|
const randomKey = generateSecretKey()
|
||||||
|
|
||||||
|
return finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: GiftWrap,
|
||||||
|
content: nip44Encrypt(seal, randomKey, recipientPublicKey),
|
||||||
|
created_at: randomNow(),
|
||||||
|
tags: [['p', recipientPublicKey]],
|
||||||
|
},
|
||||||
|
randomKey,
|
||||||
|
) as NostrEvent
|
||||||
|
}
|
||||||
|
|
||||||
|
export function wrapEvent(
|
||||||
|
event: Partial<UnsignedEvent>,
|
||||||
|
senderPrivateKey: Uint8Array,
|
||||||
|
recipientPublicKey: string,
|
||||||
|
): NostrEvent {
|
||||||
|
const rumor = createRumor(event, senderPrivateKey)
|
||||||
|
|
||||||
|
const seal = createSeal(rumor, senderPrivateKey, recipientPublicKey)
|
||||||
|
return createWrap(seal, recipientPublicKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function wrapManyEvents(
|
||||||
|
event: Partial<UnsignedEvent>,
|
||||||
|
senderPrivateKey: Uint8Array,
|
||||||
|
recipientsPublicKeys: string[],
|
||||||
|
): NostrEvent[] {
|
||||||
|
if (!recipientsPublicKeys || recipientsPublicKeys.length === 0) {
|
||||||
|
throw new Error('At least one recipient is required.')
|
||||||
|
}
|
||||||
|
|
||||||
|
const senderPublicKey = getPublicKey(senderPrivateKey)
|
||||||
|
|
||||||
|
const wrappeds = [wrapEvent(event, senderPrivateKey, senderPublicKey)]
|
||||||
|
|
||||||
|
recipientsPublicKeys.forEach(recipientPublicKey => {
|
||||||
|
wrappeds.push(wrapEvent(event, senderPrivateKey, recipientPublicKey))
|
||||||
|
})
|
||||||
|
|
||||||
|
return wrappeds
|
||||||
|
}
|
||||||
|
|
||||||
|
export function unwrapEvent(wrap: NostrEvent, recipientPrivateKey: Uint8Array): Rumor {
|
||||||
|
const unwrappedSeal = nip44Decrypt(wrap, recipientPrivateKey)
|
||||||
|
return nip44Decrypt(unwrappedSeal, recipientPrivateKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function unwrapManyEvents(wrappedEvents: NostrEvent[], recipientPrivateKey: Uint8Array): Rumor[] {
|
||||||
|
let unwrappedEvents: Rumor[] = []
|
||||||
|
|
||||||
|
wrappedEvents.forEach(e => {
|
||||||
|
unwrappedEvents.push(unwrapEvent(e, recipientPrivateKey))
|
||||||
|
})
|
||||||
|
|
||||||
|
unwrappedEvents.sort((a, b) => a.created_at - b.created_at)
|
||||||
|
|
||||||
|
return unwrappedEvents
|
||||||
|
}
|
||||||
@@ -21,6 +21,7 @@ describe('generateEventTemplate', () => {
|
|||||||
image: 'https://example.com/image.jpg',
|
image: 'https://example.com/image.jpg',
|
||||||
summary: 'Lorem ipsum',
|
summary: 'Lorem ipsum',
|
||||||
alt: 'Image alt text',
|
alt: 'Image alt text',
|
||||||
|
fallback: ['https://fallback1.example.com/image.jpg', 'https://fallback2.example.com/image.jpg'],
|
||||||
}
|
}
|
||||||
|
|
||||||
const expectedEventTemplate: EventTemplate = {
|
const expectedEventTemplate: EventTemplate = {
|
||||||
@@ -40,6 +41,8 @@ describe('generateEventTemplate', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback1.example.com/image.jpg'],
|
||||||
|
['fallback', 'https://fallback2.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -71,6 +74,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -100,6 +104,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -129,6 +134,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -158,6 +164,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -181,6 +188,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -204,6 +212,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -227,6 +236,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -259,6 +269,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -288,6 +299,7 @@ describe('validateEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -319,6 +331,8 @@ describe('parseEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback1.example.com/image.jpg'],
|
||||||
|
['fallback', 'https://fallback2.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
@@ -340,6 +354,7 @@ describe('parseEvent', () => {
|
|||||||
image: 'https://example.com/image.jpg',
|
image: 'https://example.com/image.jpg',
|
||||||
summary: 'Lorem ipsum',
|
summary: 'Lorem ipsum',
|
||||||
alt: 'Image alt text',
|
alt: 'Image alt text',
|
||||||
|
fallback: ['https://fallback1.example.com/image.jpg', 'https://fallback2.example.com/image.jpg'],
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -364,6 +379,7 @@ describe('parseEvent', () => {
|
|||||||
['image', 'https://example.com/image.jpg'],
|
['image', 'https://example.com/image.jpg'],
|
||||||
['summary', 'Lorem ipsum'],
|
['summary', 'Lorem ipsum'],
|
||||||
['alt', 'Image alt text'],
|
['alt', 'Image alt text'],
|
||||||
|
['fallback', 'https://fallback.example.com/image.jpg'],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
|
|||||||
10
nip94.ts
10
nip94.ts
@@ -75,6 +75,11 @@ export type FileMetadataObject = {
|
|||||||
* Optional: A description for accessibility, providing context or a brief description of the file.
|
* Optional: A description for accessibility, providing context or a brief description of the file.
|
||||||
*/
|
*/
|
||||||
alt?: string
|
alt?: string
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional: fallback URLs in case url fails.
|
||||||
|
*/
|
||||||
|
fallback?: string[]
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -104,6 +109,7 @@ export function generateEventTemplate(fileMetadata: FileMetadataObject): EventTe
|
|||||||
if (fileMetadata.image) eventTemplate.tags.push(['image', fileMetadata.image])
|
if (fileMetadata.image) eventTemplate.tags.push(['image', fileMetadata.image])
|
||||||
if (fileMetadata.summary) eventTemplate.tags.push(['summary', fileMetadata.summary])
|
if (fileMetadata.summary) eventTemplate.tags.push(['summary', fileMetadata.summary])
|
||||||
if (fileMetadata.alt) eventTemplate.tags.push(['alt', fileMetadata.alt])
|
if (fileMetadata.alt) eventTemplate.tags.push(['alt', fileMetadata.alt])
|
||||||
|
if (fileMetadata.fallback) fileMetadata.fallback.forEach(url => eventTemplate.tags.push(['fallback', url]))
|
||||||
|
|
||||||
return eventTemplate
|
return eventTemplate
|
||||||
}
|
}
|
||||||
@@ -194,6 +200,10 @@ export function parseEvent(event: Event): FileMetadataObject {
|
|||||||
case 'alt':
|
case 'alt':
|
||||||
fileMetadata.alt = value
|
fileMetadata.alt = value
|
||||||
break
|
break
|
||||||
|
case 'fallback':
|
||||||
|
fileMetadata.fallback ??= []
|
||||||
|
fileMetadata.fallback.push(value)
|
||||||
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
654
nip96.test.ts
654
nip96.test.ts
@@ -1,654 +0,0 @@
|
|||||||
import { describe, expect, it } from 'bun:test'
|
|
||||||
import { HttpResponse, http } from 'msw'
|
|
||||||
import { setupServer } from 'msw/node'
|
|
||||||
|
|
||||||
import { FileServerPreference } from './kinds.ts'
|
|
||||||
import {
|
|
||||||
calculateFileHash,
|
|
||||||
checkFileProcessingStatus,
|
|
||||||
deleteFile,
|
|
||||||
generateDownloadUrl,
|
|
||||||
generateFSPEventTemplate,
|
|
||||||
readServerConfig,
|
|
||||||
uploadFile,
|
|
||||||
validateDelayedProcessingResponse,
|
|
||||||
validateFileUploadResponse,
|
|
||||||
validateServerConfiguration,
|
|
||||||
type DelayedProcessingResponse,
|
|
||||||
type FileUploadResponse,
|
|
||||||
type ServerConfiguration,
|
|
||||||
} from './nip96.ts'
|
|
||||||
|
|
||||||
describe('validateServerConfiguration', () => {
|
|
||||||
it("should return true if 'api_url' is valid URL", () => {
|
|
||||||
const config: ServerConfiguration = {
|
|
||||||
api_url: 'http://example.com',
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(validateServerConfiguration(config)).toBe(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
it("should return false if 'api_url' is empty", () => {
|
|
||||||
const config: ServerConfiguration = {
|
|
||||||
api_url: '',
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(validateServerConfiguration(config)).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it("should return false if both 'api_url' and 'delegated_to_url' are provided", () => {
|
|
||||||
const config: ServerConfiguration = {
|
|
||||||
api_url: 'http://example.com',
|
|
||||||
delegated_to_url: 'http://example.com',
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(validateServerConfiguration(config)).toBe(false)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('readServerConfig', () => {
|
|
||||||
it('should return a valid ServerConfiguration object', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const HTTPROUTE = '/.well-known/nostr/nip96.json' as const
|
|
||||||
const validConfig: ServerConfiguration = {
|
|
||||||
api_url: 'http://example.com',
|
|
||||||
}
|
|
||||||
const handler = http.get(`http://example.com${HTTPROUTE}`, () => {
|
|
||||||
return HttpResponse.json(validConfig)
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const result = await readServerConfig('http://example.com/')
|
|
||||||
|
|
||||||
expect(result).toEqual(validConfig)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error if response is not valid', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const HTTPROUTE = '/.well-known/nostr/nip96.json' as const
|
|
||||||
const invalidConfig = {
|
|
||||||
// missing api_url
|
|
||||||
}
|
|
||||||
const handler = http.get(`http://example.com${HTTPROUTE}`, () => {
|
|
||||||
return HttpResponse.json(invalidConfig)
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
expect(readServerConfig('http://example.com/')).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error if response is not proper json', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const HTTPROUTE = '/.well-known/nostr/nip96.json' as const
|
|
||||||
const handler = http.get(`http://example.com${HTTPROUTE}`, () => {
|
|
||||||
return HttpResponse.json(null)
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
expect(readServerConfig('http://example.com/')).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error if response status is not 200', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const HTTPROUTE = '/.well-known/nostr/nip96.json' as const
|
|
||||||
const handler = http.get(`http://example.com${HTTPROUTE}`, () => {
|
|
||||||
return new HttpResponse(null, { status: 400 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
expect(readServerConfig('http://example.com/')).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error if input url is not valid', async () => {
|
|
||||||
expect(readServerConfig('invalid-url')).rejects.toThrow()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('validateFileUploadResponse', () => {
|
|
||||||
it('should return true if response is valid', () => {
|
|
||||||
const mockResponse: FileUploadResponse = {
|
|
||||||
status: 'error',
|
|
||||||
message: 'File uploaded failed',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if status is undefined', () => {
|
|
||||||
const mockResponse: Omit<FileUploadResponse, 'status'> = {
|
|
||||||
// status: 'error',
|
|
||||||
message: 'File upload failed',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if message is undefined', () => {
|
|
||||||
const mockResponse: Omit<FileUploadResponse, 'message'> = {
|
|
||||||
status: 'error',
|
|
||||||
// message: 'message',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if status is not valid', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'something else',
|
|
||||||
message: 'message',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if "message" is not a string', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'error',
|
|
||||||
message: 123,
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if status is "processing" and "processing_url" is undefined', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'processing',
|
|
||||||
message: 'message',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if status is "processing" and "processing_url" is not a string', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'processing',
|
|
||||||
message: 'message',
|
|
||||||
processing_url: 123,
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if status is "success" and "nip94_event" is undefined', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if "nip94_event" tags are invalid', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
nip94_event: {
|
|
||||||
tags: [
|
|
||||||
// missing url
|
|
||||||
['ox', '719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b'],
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false if "nip94_event" tags are empty', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
nip94_event: {
|
|
||||||
tags: [],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return true if "nip94_event" tags are valid', () => {
|
|
||||||
const mockResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
nip94_event: {
|
|
||||||
tags: [
|
|
||||||
['url', 'http://example.com'],
|
|
||||||
['ox', '719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b'],
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = validateFileUploadResponse(mockResponse)
|
|
||||||
|
|
||||||
expect(result).toBe(true)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('uploadFile', () => {
|
|
||||||
it('should return a valid FileUploadResponse object', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const validFileUploadResponse: FileUploadResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
nip94_event: {
|
|
||||||
content: '',
|
|
||||||
tags: [
|
|
||||||
['url', 'http://example.com'],
|
|
||||||
['ox', '719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b'],
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return HttpResponse.json(validFileUploadResponse, { status: 200 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
const result = await uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)
|
|
||||||
|
|
||||||
expect(result).toEqual(validFileUploadResponse)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw a proper error if response status is 413', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return new HttpResponse(null, { status: 413 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)).rejects.toThrow('File too large!')
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw a proper error if response status is 400', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return new HttpResponse(null, { status: 400 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)).rejects.toThrow(
|
|
||||||
'Bad request! Some fields are missing or invalid!',
|
|
||||||
)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw a proper error if response status is 403', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return new HttpResponse(null, { status: 403 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)).rejects.toThrow(
|
|
||||||
'Forbidden! Payload tag does not match the requested file!',
|
|
||||||
)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw a proper error if response status is 402', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return new HttpResponse(null, { status: 402 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)).rejects.toThrow('Payment required!')
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw a proper error if response status is not 200, 400, 402, 403, 413', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.post('http://example.com/upload', () => {
|
|
||||||
return new HttpResponse(null, { status: 500 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const serverUploadUrl = 'http://example.com/upload'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(uploadFile(file, serverUploadUrl, nip98AuthorizationHeader)).rejects.toThrow(
|
|
||||||
'Unknown error in uploading file!',
|
|
||||||
)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('generateDownloadUrl', () => {
|
|
||||||
it('should generate a download URL without file extension', () => {
|
|
||||||
const fileHash = 'abc123'
|
|
||||||
const serverDownloadUrl = 'http://example.com/download'
|
|
||||||
const expectedUrl = 'http://example.com/download/abc123'
|
|
||||||
|
|
||||||
const result = generateDownloadUrl(fileHash, serverDownloadUrl)
|
|
||||||
|
|
||||||
expect(result).toBe(expectedUrl)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should generate a download URL with file extension', () => {
|
|
||||||
const fileHash = 'abc123'
|
|
||||||
const serverDownloadUrl = 'http://example.com/download'
|
|
||||||
const fileExtension = '.jpg'
|
|
||||||
const expectedUrl = 'http://example.com/download/abc123.jpg'
|
|
||||||
|
|
||||||
const result = generateDownloadUrl(fileHash, serverDownloadUrl, fileExtension)
|
|
||||||
|
|
||||||
expect(result).toBe(expectedUrl)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('deleteFile', () => {
|
|
||||||
it('should return a basic json response for successful delete', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.delete('http://example.com/delete/abc123', () => {
|
|
||||||
return HttpResponse.json({ status: 'success', message: 'File deleted.' }, { status: 200 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const fileHash = 'abc123'
|
|
||||||
const serverDeleteUrl = 'http://example.com/delete'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
const result = await deleteFile(fileHash, serverDeleteUrl, nip98AuthorizationHeader)
|
|
||||||
|
|
||||||
expect(result).toEqual({ status: 'success', message: 'File deleted.' })
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error for unsuccessful delete', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.delete('http://example.com/delete/abc123', () => {
|
|
||||||
return new HttpResponse(null, { status: 400 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const fileHash = 'abc123'
|
|
||||||
const serverDeleteUrl = 'http://example.com/delete'
|
|
||||||
const nip98AuthorizationHeader = 'Nostr abcabc'
|
|
||||||
|
|
||||||
expect(deleteFile(fileHash, serverDeleteUrl, nip98AuthorizationHeader)).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('validateDelayedProcessingResponse', () => {
|
|
||||||
it('should return false for non-object input', () => {
|
|
||||||
expect(validateDelayedProcessingResponse('not an object')).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for null input', () => {
|
|
||||||
expect(validateDelayedProcessingResponse(null)).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for object missing required properties', () => {
|
|
||||||
const missingStatus: Omit<DelayedProcessingResponse, 'status'> = {
|
|
||||||
// missing status
|
|
||||||
message: 'test',
|
|
||||||
percentage: 50,
|
|
||||||
}
|
|
||||||
const missingMessage: Omit<DelayedProcessingResponse, 'message'> = {
|
|
||||||
status: 'processing',
|
|
||||||
// missing message
|
|
||||||
percentage: 50,
|
|
||||||
}
|
|
||||||
const missingPercentage: Omit<DelayedProcessingResponse, 'percentage'> = {
|
|
||||||
status: 'processing',
|
|
||||||
message: 'test',
|
|
||||||
// missing percentage
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(validateDelayedProcessingResponse(missingStatus)).toBe(false)
|
|
||||||
expect(validateDelayedProcessingResponse(missingMessage)).toBe(false)
|
|
||||||
expect(validateDelayedProcessingResponse(missingPercentage)).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for invalid status', () => {
|
|
||||||
expect(validateDelayedProcessingResponse({ status: 'invalid', message: 'test', percentage: 50 })).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for non-string message', () => {
|
|
||||||
expect(validateDelayedProcessingResponse({ status: 'processing', message: 123, percentage: 50 })).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for non-number percentage', () => {
|
|
||||||
expect(validateDelayedProcessingResponse({ status: 'processing', message: 'test', percentage: '50' })).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for percentage out of range', () => {
|
|
||||||
expect(validateDelayedProcessingResponse({ status: 'processing', message: 'test', percentage: 150 })).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return true for valid input', () => {
|
|
||||||
expect(validateDelayedProcessingResponse({ status: 'processing', message: 'test', percentage: 50 })).toBe(true)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('checkFileProcessingStatus', () => {
|
|
||||||
it('should throw an error if response is not ok', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.get('http://example.com/status/abc123', () => {
|
|
||||||
return new HttpResponse(null, { status: 400 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const processingUrl = 'http://example.com/status/abc123'
|
|
||||||
|
|
||||||
expect(checkFileProcessingStatus(processingUrl)).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should throw an error if response is not a valid json', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const handler = http.get('http://example.com/status/abc123', () => {
|
|
||||||
return HttpResponse.text('not a json', { status: 200 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const processingUrl = 'http://example.com/status/abc123'
|
|
||||||
|
|
||||||
expect(checkFileProcessingStatus(processingUrl)).rejects.toThrow()
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return a valid DelayedProcessingResponse object if response status is 200', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const validDelayedProcessingResponse: DelayedProcessingResponse = {
|
|
||||||
status: 'processing',
|
|
||||||
message: 'test',
|
|
||||||
percentage: 50,
|
|
||||||
}
|
|
||||||
const handler = http.get('http://example.com/status/abc123', () => {
|
|
||||||
return HttpResponse.json(validDelayedProcessingResponse, { status: 200 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const processingUrl = 'http://example.com/status/abc123'
|
|
||||||
|
|
||||||
const result = await checkFileProcessingStatus(processingUrl)
|
|
||||||
|
|
||||||
expect(result).toEqual(validDelayedProcessingResponse)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return a valid FileUploadResponse object if response status is 201', async () => {
|
|
||||||
// setup mock server
|
|
||||||
const validFileUploadResponse: FileUploadResponse = {
|
|
||||||
status: 'success',
|
|
||||||
message: 'message',
|
|
||||||
nip94_event: {
|
|
||||||
content: '',
|
|
||||||
tags: [
|
|
||||||
['url', 'http://example.com'],
|
|
||||||
['ox', '719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b'],
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
const handler = http.get('http://example.com/status/abc123', () => {
|
|
||||||
return HttpResponse.json(validFileUploadResponse, { status: 201 })
|
|
||||||
})
|
|
||||||
const server = setupServer(handler)
|
|
||||||
server.listen()
|
|
||||||
|
|
||||||
const processingUrl = 'http://example.com/status/abc123'
|
|
||||||
|
|
||||||
const result = await checkFileProcessingStatus(processingUrl)
|
|
||||||
|
|
||||||
expect(result).toEqual(validFileUploadResponse)
|
|
||||||
|
|
||||||
// cleanup mock server
|
|
||||||
server.resetHandlers()
|
|
||||||
server.close()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('generateFSPEventTemplate', () => {
|
|
||||||
it('should generate FSP event template', () => {
|
|
||||||
const serverUrls = ['http://example.com', 'https://example.org']
|
|
||||||
const eventTemplate = generateFSPEventTemplate(serverUrls)
|
|
||||||
|
|
||||||
expect(eventTemplate.kind).toBe(FileServerPreference)
|
|
||||||
expect(eventTemplate.content).toBe('')
|
|
||||||
expect(eventTemplate.tags).toEqual([
|
|
||||||
['server', 'http://example.com'],
|
|
||||||
['server', 'https://example.org'],
|
|
||||||
])
|
|
||||||
expect(typeof eventTemplate.created_at).toBe('number')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should filter invalid server URLs', () => {
|
|
||||||
const serverUrls = ['http://example.com', 'invalid-url', 'https://example.org']
|
|
||||||
const eventTemplate = generateFSPEventTemplate(serverUrls)
|
|
||||||
|
|
||||||
expect(eventTemplate.tags).toEqual([
|
|
||||||
['server', 'http://example.com'],
|
|
||||||
['server', 'https://example.org'],
|
|
||||||
])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should handle empty server URLs', () => {
|
|
||||||
const serverUrls: string[] = []
|
|
||||||
const eventTemplate = generateFSPEventTemplate(serverUrls)
|
|
||||||
|
|
||||||
expect(eventTemplate.tags).toEqual([])
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('calculateFileHash', () => {
|
|
||||||
it('should calculate file hash', async () => {
|
|
||||||
const file = new File(['hello world'], 'hello.txt')
|
|
||||||
const hash = await calculateFileHash(file)
|
|
||||||
|
|
||||||
expect(hash).toBe('b94d27b9934d3e08a52e52d7da7dabfac484efe37a5380ee9088f7ace2efcde9')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should calculate file hash with empty file', async () => {
|
|
||||||
const file = new File([], 'empty.txt')
|
|
||||||
const hash = await calculateFileHash(file)
|
|
||||||
|
|
||||||
expect(hash).toBe('e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855')
|
|
||||||
})
|
|
||||||
})
|
|
||||||
582
nip96.ts
582
nip96.ts
@@ -1,582 +0,0 @@
|
|||||||
import { sha256 } from '@noble/hashes/sha256'
|
|
||||||
import { EventTemplate } from './core.ts'
|
|
||||||
import { FileServerPreference } from './kinds.ts'
|
|
||||||
import { bytesToHex } from '@noble/hashes/utils'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents the configuration for a server compliant with NIP-96.
|
|
||||||
*/
|
|
||||||
export type ServerConfiguration = {
|
|
||||||
/**
|
|
||||||
* The base URL from which file upload and deletion operations are served.
|
|
||||||
* Also used for downloads if "download_url" is not specified.
|
|
||||||
*/
|
|
||||||
api_url: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. The base URL from which files are downloaded.
|
|
||||||
* Used if different from the "api_url".
|
|
||||||
*/
|
|
||||||
download_url?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. URL of another HTTP file storage server's configuration.
|
|
||||||
* Used by nostr relays to delegate to another server.
|
|
||||||
* In this case, "api_url" must be an empty string.
|
|
||||||
*/
|
|
||||||
delegated_to_url?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. An array of NIP numbers that this server supports.
|
|
||||||
*/
|
|
||||||
supported_nips?: number[]
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. URL to the server's Terms of Service.
|
|
||||||
*/
|
|
||||||
tos_url?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. An array of MIME types supported by the server.
|
|
||||||
*/
|
|
||||||
content_types?: string[]
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. Defines various storage plans offered by the server.
|
|
||||||
*/
|
|
||||||
plans?: {
|
|
||||||
[planKey: string]: {
|
|
||||||
/**
|
|
||||||
* The name of the storage plan.
|
|
||||||
*/
|
|
||||||
name: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. Indicates whether NIP-98 is required for uploads in this plan.
|
|
||||||
*/
|
|
||||||
is_nip98_required?: boolean
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. URL to a landing page providing more information about the plan.
|
|
||||||
*/
|
|
||||||
url?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. The maximum file size allowed under this plan, in bytes.
|
|
||||||
*/
|
|
||||||
max_byte_size?: number
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. Defines the range of file expiration in days.
|
|
||||||
* The first value indicates the minimum expiration time, and the second value indicates the maximum.
|
|
||||||
* A value of 0 indicates no expiration.
|
|
||||||
*/
|
|
||||||
file_expiration?: [number, number]
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. Specifies the types of media transformations supported under this plan.
|
|
||||||
* Currently, only image transformations are considered.
|
|
||||||
*/
|
|
||||||
media_transformations?: {
|
|
||||||
/**
|
|
||||||
* Optional. An array of supported image transformation types.
|
|
||||||
*/
|
|
||||||
image?: string[]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents the optional form data fields for file upload in accordance with NIP-96.
|
|
||||||
*/
|
|
||||||
export type OptionalFormDataFields = {
|
|
||||||
/**
|
|
||||||
* Specifies the desired expiration time of the file on the server.
|
|
||||||
* It should be a string representing a UNIX timestamp in seconds.
|
|
||||||
* An empty string indicates that the file should be stored indefinitely.
|
|
||||||
*/
|
|
||||||
expiration?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Indicates the size of the file in bytes.
|
|
||||||
* This field can be used by the server to pre-validate the file size before processing the upload.
|
|
||||||
*/
|
|
||||||
size?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides a strict description of the file for accessibility purposes,
|
|
||||||
* particularly useful for visibility-impaired users.
|
|
||||||
*/
|
|
||||||
alt?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A loose, more descriptive caption for the file.
|
|
||||||
* This can be used for additional context or commentary about the file.
|
|
||||||
*/
|
|
||||||
caption?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Specifies the intended use of the file.
|
|
||||||
* Can be either 'avatar' or 'banner', indicating if the file is to be used as an avatar or a banner.
|
|
||||||
* Absence of this field suggests standard file upload without special treatment.
|
|
||||||
*/
|
|
||||||
media_type?: 'avatar' | 'banner'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The MIME type of the file being uploaded.
|
|
||||||
* This can be used for early rejection by the server if the file type isn't supported.
|
|
||||||
*/
|
|
||||||
content_type?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Other custom form data fields.
|
|
||||||
*/
|
|
||||||
[key: string]: string | undefined
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Type representing the response from a NIP-96 compliant server after a file upload request.
|
|
||||||
*/
|
|
||||||
export type FileUploadResponse = {
|
|
||||||
/**
|
|
||||||
* The status of the upload request.
|
|
||||||
* - 'success': Indicates the file was successfully uploaded.
|
|
||||||
* - 'error': Indicates there was an error in the upload process.
|
|
||||||
* - 'processing': Indicates the file is still being processed (used in cases of delayed processing).
|
|
||||||
*/
|
|
||||||
status: 'success' | 'error' | 'processing'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A message provided by the server, which could be a success message, error description, or processing status.
|
|
||||||
*/
|
|
||||||
message: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. A URL provided by the server where the upload processing status can be checked.
|
|
||||||
* This is relevant in cases where the file upload involves delayed processing.
|
|
||||||
*/
|
|
||||||
processing_url?: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Optional. An event object conforming to NIP-94, which includes details about the uploaded file.
|
|
||||||
* This object is typically provided in the response for a successful upload and contains
|
|
||||||
* essential information such as the download URL and file metadata.
|
|
||||||
*/
|
|
||||||
nip94_event?: {
|
|
||||||
/**
|
|
||||||
* A collection of key-value pairs (tags) providing metadata about the uploaded file.
|
|
||||||
* Standard tags include:
|
|
||||||
* - 'url': The URL where the file can be accessed.
|
|
||||||
* - 'ox': The SHA-256 hash of the original file before any server-side transformations.
|
|
||||||
* Additional optional tags might include file dimensions, MIME type, etc.
|
|
||||||
*/
|
|
||||||
tags: Array<[string, string]>
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A content field, which is typically empty for file upload events but included for consistency with the NIP-94 structure.
|
|
||||||
*/
|
|
||||||
content: string
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Type representing the response from a NIP-96 compliant server after a delayed processing request.
|
|
||||||
*/
|
|
||||||
export type DelayedProcessingResponse = {
|
|
||||||
/**
|
|
||||||
* The status of the delayed processing request.
|
|
||||||
* - 'processing': Indicates the file is still being processed.
|
|
||||||
* - 'error': Indicates there was an error in the processing.
|
|
||||||
*/
|
|
||||||
status: 'processing' | 'error'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A message provided by the server, which could be a success message or error description.
|
|
||||||
*/
|
|
||||||
message: string
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The percentage of the file that has been processed. This is a number between 0 and 100.
|
|
||||||
*/
|
|
||||||
percentage: number
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Validates the server configuration.
|
|
||||||
*
|
|
||||||
* @param config - The server configuration object.
|
|
||||||
* @returns True if the configuration is valid, false otherwise.
|
|
||||||
*/
|
|
||||||
export function validateServerConfiguration(config: ServerConfiguration): boolean {
|
|
||||||
if (Boolean(config.api_url) == false) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Boolean(config.delegated_to_url) && Boolean(config.api_url)) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetches, parses, and validates the server configuration from the given URL.
|
|
||||||
*
|
|
||||||
* @param serverUrl The URL of the server.
|
|
||||||
* @returns The server configuration, or an error if the configuration could not be fetched or parsed.
|
|
||||||
*/
|
|
||||||
export async function readServerConfig(serverUrl: string): Promise<ServerConfiguration> {
|
|
||||||
const HTTPROUTE = '/.well-known/nostr/nip96.json' as const
|
|
||||||
let fetchUrl = ''
|
|
||||||
|
|
||||||
try {
|
|
||||||
const { origin } = new URL(serverUrl)
|
|
||||||
fetchUrl = origin + HTTPROUTE
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error('Invalid URL')
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch(fetchUrl)
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(`Error fetching ${fetchUrl}: ${response.statusText}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
const data: any = await response.json()
|
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
throw new Error('No data')
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!validateServerConfiguration(data)) {
|
|
||||||
throw new Error('Invalid configuration data')
|
|
||||||
}
|
|
||||||
|
|
||||||
return data
|
|
||||||
} catch (_) {
|
|
||||||
throw new Error(`Error fetching.`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Validates if the given object is a valid FileUploadResponse.
|
|
||||||
*
|
|
||||||
* @param response - The object to validate.
|
|
||||||
* @returns true if the object is a valid FileUploadResponse, otherwise false.
|
|
||||||
*/
|
|
||||||
export function validateFileUploadResponse(response: any): response is FileUploadResponse {
|
|
||||||
if (typeof response !== 'object' || response === null) return false
|
|
||||||
|
|
||||||
if (!response.status || !response.message) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.status !== 'success' && response.status !== 'error' && response.status !== 'processing') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof response.message !== 'string') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.status === 'processing' && !response.processing_url) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.processing_url) {
|
|
||||||
if (typeof response.processing_url !== 'string') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.status === 'success' && !response.nip94_event) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.nip94_event) {
|
|
||||||
if (
|
|
||||||
!response.nip94_event.tags ||
|
|
||||||
!Array.isArray(response.nip94_event.tags) ||
|
|
||||||
response.nip94_event.tags.length === 0
|
|
||||||
) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const tag of response.nip94_event.tags) {
|
|
||||||
if (!Array.isArray(tag) || tag.length !== 2) return false
|
|
||||||
|
|
||||||
if (typeof tag[0] !== 'string' || typeof tag[1] !== 'string') return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(response.nip94_event.tags as string[]).find(t => t[0] === 'url')) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(response.nip94_event.tags as string[]).find(t => t[0] === 'ox')) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uploads a file to a NIP-96 compliant server.
|
|
||||||
*
|
|
||||||
* @param file - The file to be uploaded.
|
|
||||||
* @param serverApiUrl - The API URL of the server, retrieved from the server's configuration.
|
|
||||||
* @param nip98AuthorizationHeader - The authorization header from NIP-98.
|
|
||||||
* @param optionalFormDataFields - Optional form data fields.
|
|
||||||
* @returns A promise that resolves to the server's response.
|
|
||||||
*/
|
|
||||||
export async function uploadFile(
|
|
||||||
file: File,
|
|
||||||
serverApiUrl: string,
|
|
||||||
nip98AuthorizationHeader: string,
|
|
||||||
optionalFormDataFields?: OptionalFormDataFields,
|
|
||||||
): Promise<FileUploadResponse> {
|
|
||||||
// Create FormData object
|
|
||||||
const formData = new FormData()
|
|
||||||
|
|
||||||
// Append the authorization header to HTML Form Data
|
|
||||||
formData.append('Authorization', nip98AuthorizationHeader)
|
|
||||||
|
|
||||||
// Append optional fields to FormData
|
|
||||||
optionalFormDataFields &&
|
|
||||||
Object.entries(optionalFormDataFields).forEach(([key, value]) => {
|
|
||||||
if (value) {
|
|
||||||
formData.append(key, value)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Append the file to FormData as the last field
|
|
||||||
formData.append('file', file)
|
|
||||||
|
|
||||||
// Make the POST request to the server
|
|
||||||
const response = await fetch(serverApiUrl, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
Authorization: nip98AuthorizationHeader,
|
|
||||||
'Content-Type': 'multipart/form-data',
|
|
||||||
},
|
|
||||||
body: formData,
|
|
||||||
})
|
|
||||||
|
|
||||||
if (response.ok === false) {
|
|
||||||
// 413 Payload Too Large
|
|
||||||
if (response.status === 413) {
|
|
||||||
throw new Error('File too large!')
|
|
||||||
}
|
|
||||||
|
|
||||||
// 400 Bad Request
|
|
||||||
if (response.status === 400) {
|
|
||||||
throw new Error('Bad request! Some fields are missing or invalid!')
|
|
||||||
}
|
|
||||||
|
|
||||||
// 403 Forbidden
|
|
||||||
if (response.status === 403) {
|
|
||||||
throw new Error('Forbidden! Payload tag does not match the requested file!')
|
|
||||||
}
|
|
||||||
|
|
||||||
// 402 Payment Required
|
|
||||||
if (response.status === 402) {
|
|
||||||
throw new Error('Payment required!')
|
|
||||||
}
|
|
||||||
|
|
||||||
// unknown error
|
|
||||||
throw new Error('Unknown error in uploading file!')
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const parsedResponse = await response.json()
|
|
||||||
|
|
||||||
if (!validateFileUploadResponse(parsedResponse)) {
|
|
||||||
throw new Error('Invalid response from the server!')
|
|
||||||
}
|
|
||||||
|
|
||||||
return parsedResponse
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error('Error parsing JSON response!')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generates the URL for downloading a file from a NIP-96 compliant server.
|
|
||||||
*
|
|
||||||
* @param fileHash - The SHA-256 hash of the original file.
|
|
||||||
* @param serverDownloadUrl - The base URL provided by the server, retrieved from the server's configuration.
|
|
||||||
* @param fileExtension - An optional parameter that specifies the file extension (e.g., '.jpg', '.png').
|
|
||||||
* @returns A string representing the complete URL to download the file.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
export function generateDownloadUrl(fileHash: string, serverDownloadUrl: string, fileExtension?: string): string {
|
|
||||||
// Construct the base download URL using the file hash
|
|
||||||
let downloadUrl = `${serverDownloadUrl}/${fileHash}`
|
|
||||||
|
|
||||||
// Append the file extension if provided
|
|
||||||
if (fileExtension) {
|
|
||||||
downloadUrl += fileExtension
|
|
||||||
}
|
|
||||||
|
|
||||||
return downloadUrl
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends a request to delete a file from a NIP-96 compliant server.
|
|
||||||
*
|
|
||||||
* @param fileHash - The SHA-256 hash of the original file.
|
|
||||||
* @param serverApiUrl - The base API URL of the server, retrieved from the server's configuration.
|
|
||||||
* @param nip98AuthorizationHeader - The authorization header from NIP-98.
|
|
||||||
* @returns A promise that resolves to the server's response to the deletion request.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
export async function deleteFile(
|
|
||||||
fileHash: string,
|
|
||||||
serverApiUrl: string,
|
|
||||||
nip98AuthorizationHeader: string,
|
|
||||||
): Promise<any> {
|
|
||||||
// make sure the serverApiUrl ends with a slash
|
|
||||||
if (!serverApiUrl.endsWith('/')) {
|
|
||||||
serverApiUrl += '/'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Construct the URL for the delete request
|
|
||||||
const deleteUrl = `${serverApiUrl}${fileHash}`
|
|
||||||
|
|
||||||
// Send the DELETE request
|
|
||||||
const response = await fetch(deleteUrl, {
|
|
||||||
method: 'DELETE',
|
|
||||||
headers: {
|
|
||||||
Authorization: nip98AuthorizationHeader,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Handle the response
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error('Error deleting file!')
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the response from the server
|
|
||||||
try {
|
|
||||||
return await response.json()
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error('Error parsing JSON response!')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Validates the server's response to a delayed processing request.
|
|
||||||
*
|
|
||||||
* @param response - The server's response to a delayed processing request.
|
|
||||||
* @returns A boolean indicating whether the response is valid.
|
|
||||||
*/
|
|
||||||
export function validateDelayedProcessingResponse(response: any): response is DelayedProcessingResponse {
|
|
||||||
if (typeof response !== 'object' || response === null) return false
|
|
||||||
|
|
||||||
if (!response.status || !response.message || !response.percentage) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.status !== 'processing' && response.status !== 'error') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof response.message !== 'string') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof response.percentage !== 'number') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Number(response.percentage) < 0 || Number(response.percentage) > 100) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks the processing status of a file when delayed processing is used.
|
|
||||||
*
|
|
||||||
* @param processingUrl - The URL provided by the server where the processing status can be checked.
|
|
||||||
* @returns A promise that resolves to an object containing the processing status and other relevant information.
|
|
||||||
*/
|
|
||||||
export async function checkFileProcessingStatus(
|
|
||||||
processingUrl: string,
|
|
||||||
): Promise<FileUploadResponse | DelayedProcessingResponse> {
|
|
||||||
// Make the GET request to the processing URL
|
|
||||||
const response = await fetch(processingUrl)
|
|
||||||
|
|
||||||
// Handle the response
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(`Failed to retrieve processing status. Server responded with status: ${response.status}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse the response
|
|
||||||
try {
|
|
||||||
const parsedResponse = await response.json()
|
|
||||||
|
|
||||||
// 201 Created: Indicates the processing is over.
|
|
||||||
if (response.status === 201) {
|
|
||||||
// Validate the response
|
|
||||||
if (!validateFileUploadResponse(parsedResponse)) {
|
|
||||||
throw new Error('Invalid response from the server!')
|
|
||||||
}
|
|
||||||
|
|
||||||
return parsedResponse
|
|
||||||
}
|
|
||||||
|
|
||||||
// 200 OK: Indicates the processing is still ongoing.
|
|
||||||
if (response.status === 200) {
|
|
||||||
// Validate the response
|
|
||||||
if (!validateDelayedProcessingResponse(parsedResponse)) {
|
|
||||||
throw new Error('Invalid response from the server!')
|
|
||||||
}
|
|
||||||
|
|
||||||
return parsedResponse
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Error('Invalid response from the server!')
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error('Error parsing JSON response!')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generates an event template to indicate a user's File Server Preferences.
|
|
||||||
* This event is of kind 10096 and is used to specify one or more preferred servers for file uploads.
|
|
||||||
*
|
|
||||||
* @param serverUrls - An array of URLs representing the user's preferred file storage servers.
|
|
||||||
* @returns An object representing a Nostr event template for setting file server preferences.
|
|
||||||
*/
|
|
||||||
export function generateFSPEventTemplate(serverUrls: string[]): EventTemplate {
|
|
||||||
serverUrls = serverUrls.filter(serverUrl => {
|
|
||||||
try {
|
|
||||||
new URL(serverUrl)
|
|
||||||
return true
|
|
||||||
} catch (error) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return {
|
|
||||||
kind: FileServerPreference,
|
|
||||||
content: '',
|
|
||||||
tags: serverUrls.map(serverUrl => ['server', serverUrl]),
|
|
||||||
created_at: Math.floor(Date.now() / 1000),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Calculates the SHA-256 hash of a given file. This hash is used in various NIP-96 operations,
|
|
||||||
* such as file upload, download, and deletion, to uniquely identify files.
|
|
||||||
*
|
|
||||||
* @param file - The file for which the SHA-256 hash needs to be calculated.
|
|
||||||
* @returns A promise that resolves to the SHA-256 hash of the file.
|
|
||||||
*/
|
|
||||||
export async function calculateFileHash(file: Blob): Promise<string> {
|
|
||||||
return bytesToHex(sha256(new Uint8Array(await file.arrayBuffer())))
|
|
||||||
}
|
|
||||||
55
nipb7.test.ts
Normal file
55
nipb7.test.ts
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { test, expect } from 'bun:test'
|
||||||
|
import { BlossomClient } from './nipb7.ts'
|
||||||
|
import { sha256 } from '@noble/hashes/sha256'
|
||||||
|
import { bytesToHex } from './utils.ts'
|
||||||
|
import { PlainKeySigner } from './signer.ts'
|
||||||
|
import { generateSecretKey } from './pure.ts'
|
||||||
|
|
||||||
|
test('blossom', async () => {
|
||||||
|
const BLOSSOM_SERVER = 'blossom.primal.net'
|
||||||
|
const TEST_CONTENT = 'hello world'
|
||||||
|
const TEST_BLOB = new Blob([TEST_CONTENT], { type: 'text/plain' })
|
||||||
|
|
||||||
|
const expectedHash = bytesToHex(sha256(new TextEncoder().encode(TEST_CONTENT)))
|
||||||
|
|
||||||
|
const signer = new PlainKeySigner(generateSecretKey())
|
||||||
|
const client = new BlossomClient(BLOSSOM_SERVER, signer)
|
||||||
|
expect(client).toBeDefined()
|
||||||
|
|
||||||
|
// check for non-existent file should throw
|
||||||
|
const invalidHash = expectedHash.slice(0, 62) + 'ba'
|
||||||
|
let hasThrown = false
|
||||||
|
try {
|
||||||
|
await client.check(invalidHash)
|
||||||
|
} catch (err) {
|
||||||
|
hasThrown = true
|
||||||
|
}
|
||||||
|
expect(hasThrown).toBeTrue()
|
||||||
|
|
||||||
|
// upload hello world blob
|
||||||
|
const descriptor = await client.uploadBlob(TEST_BLOB, 'text/plain')
|
||||||
|
expect(descriptor).toBeDefined()
|
||||||
|
expect(descriptor.sha256).toBe(expectedHash)
|
||||||
|
expect(descriptor.size).toBe(TEST_CONTENT.length)
|
||||||
|
expect(descriptor.type).toBe('text/plain')
|
||||||
|
expect(descriptor.url).toContain(expectedHash)
|
||||||
|
expect(descriptor.uploaded).toBeGreaterThan(0)
|
||||||
|
await client.check(expectedHash)
|
||||||
|
|
||||||
|
// download and verify
|
||||||
|
const downloadedBuffer = await client.download(expectedHash)
|
||||||
|
const downloadedContent = new TextDecoder().decode(downloadedBuffer)
|
||||||
|
expect(downloadedContent).toBe(TEST_CONTENT)
|
||||||
|
|
||||||
|
// list blobs should include our uploaded file
|
||||||
|
const blobs = await client.list()
|
||||||
|
|
||||||
|
expect(Array.isArray(blobs)).toBe(true)
|
||||||
|
const ourBlob = blobs.find(blob => blob.sha256 === expectedHash)
|
||||||
|
expect(ourBlob).toBeDefined()
|
||||||
|
expect(ourBlob?.type).toBe('text/plain')
|
||||||
|
expect(ourBlob?.size).toBe(TEST_CONTENT.length)
|
||||||
|
|
||||||
|
// delete
|
||||||
|
await client.delete(expectedHash)
|
||||||
|
})
|
||||||
203
nipb7.ts
Normal file
203
nipb7.ts
Normal file
@@ -0,0 +1,203 @@
|
|||||||
|
import { sha256 } from '@noble/hashes/sha256'
|
||||||
|
import { EventTemplate } from './core.ts'
|
||||||
|
import { Signer } from './signer.ts'
|
||||||
|
import { bytesToHex } from './utils.ts'
|
||||||
|
|
||||||
|
export type BlobDescriptor = {
|
||||||
|
url: string
|
||||||
|
sha256: string
|
||||||
|
size: number
|
||||||
|
type: string
|
||||||
|
uploaded: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export class BlossomClient {
|
||||||
|
private mediaserver: string
|
||||||
|
private signer: Signer
|
||||||
|
|
||||||
|
constructor(mediaserver: string, signer: Signer) {
|
||||||
|
if (!mediaserver.startsWith('http')) {
|
||||||
|
mediaserver = 'https://' + mediaserver
|
||||||
|
}
|
||||||
|
this.mediaserver = mediaserver.replace(/\/$/, '') + '/'
|
||||||
|
this.signer = signer
|
||||||
|
}
|
||||||
|
|
||||||
|
private async httpCall(
|
||||||
|
method: string,
|
||||||
|
url: string,
|
||||||
|
contentType?: string,
|
||||||
|
addAuthorization?: () => Promise<string>,
|
||||||
|
body?: File | Blob,
|
||||||
|
result?: any,
|
||||||
|
): Promise<any> {
|
||||||
|
const headers: { [_: string]: string } = {}
|
||||||
|
|
||||||
|
if (contentType) {
|
||||||
|
headers['Content-Type'] = contentType
|
||||||
|
}
|
||||||
|
|
||||||
|
if (addAuthorization) {
|
||||||
|
const auth = await addAuthorization()
|
||||||
|
if (auth) {
|
||||||
|
headers['Authorization'] = auth
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(this.mediaserver + url, {
|
||||||
|
method,
|
||||||
|
headers,
|
||||||
|
body,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (response.status >= 300) {
|
||||||
|
const reason = response.headers.get('X-Reason') || response.statusText
|
||||||
|
throw new Error(`${url} returned an error (${response.status}): ${reason}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result !== null && response.headers.get('content-type')?.includes('application/json')) {
|
||||||
|
return await response.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
private async authorizationHeader(modify?: (event: EventTemplate) => void): Promise<string> {
|
||||||
|
const now = Math.floor(Date.now() / 1000)
|
||||||
|
const event: EventTemplate = {
|
||||||
|
created_at: now,
|
||||||
|
kind: 24242,
|
||||||
|
content: 'blossom stuff',
|
||||||
|
tags: [['expiration', String(now + 60)]],
|
||||||
|
}
|
||||||
|
|
||||||
|
if (modify) {
|
||||||
|
modify(event)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const signedEvent = await this.signer.signEvent(event)
|
||||||
|
const eventJson = JSON.stringify(signedEvent)
|
||||||
|
return 'Nostr ' + btoa(eventJson)
|
||||||
|
} catch (error) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private isValid32ByteHex(hash: string): boolean {
|
||||||
|
return /^[a-f0-9]{64}$/i.test(hash)
|
||||||
|
}
|
||||||
|
|
||||||
|
async check(hash: string): Promise<void> {
|
||||||
|
if (!this.isValid32ByteHex(hash)) {
|
||||||
|
throw new Error(`${hash} is not a valid 32-byte hex string`)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.httpCall('HEAD', hash)
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(`failed to check for ${hash}: ${error}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async uploadBlob(file: File | Blob, contentType?: string): Promise<BlobDescriptor> {
|
||||||
|
const hash = bytesToHex(sha256(new Uint8Array(await file.arrayBuffer())))
|
||||||
|
const actualContentType = contentType || file.type || 'application/octet-stream'
|
||||||
|
|
||||||
|
const bd = await this.httpCall(
|
||||||
|
'PUT',
|
||||||
|
'upload',
|
||||||
|
actualContentType,
|
||||||
|
() =>
|
||||||
|
this.authorizationHeader(evt => {
|
||||||
|
evt.tags.push(['t', 'upload'])
|
||||||
|
evt.tags.push(['x', hash])
|
||||||
|
}),
|
||||||
|
file,
|
||||||
|
{},
|
||||||
|
)
|
||||||
|
|
||||||
|
return bd
|
||||||
|
}
|
||||||
|
|
||||||
|
async uploadFile(file: File): Promise<BlobDescriptor> {
|
||||||
|
return this.uploadBlob(file, file.type)
|
||||||
|
}
|
||||||
|
|
||||||
|
async download(hash: string): Promise<ArrayBuffer> {
|
||||||
|
if (!this.isValid32ByteHex(hash)) {
|
||||||
|
throw new Error(`${hash} is not a valid 32-byte hex string`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const authHeader = await this.authorizationHeader(evt => {
|
||||||
|
evt.tags.push(['t', 'get'])
|
||||||
|
evt.tags.push(['x', hash])
|
||||||
|
})
|
||||||
|
|
||||||
|
const response = await fetch(this.mediaserver + hash, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
Authorization: authHeader,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (response.status >= 300) {
|
||||||
|
throw new Error(`${hash} is not present in ${this.mediaserver}: ${response.status}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return await response.arrayBuffer()
|
||||||
|
}
|
||||||
|
|
||||||
|
async downloadAsBlob(hash: string): Promise<Blob> {
|
||||||
|
const arrayBuffer = await this.download(hash)
|
||||||
|
return new Blob([arrayBuffer])
|
||||||
|
}
|
||||||
|
|
||||||
|
async list(): Promise<BlobDescriptor[]> {
|
||||||
|
const pubkey = await this.signer.getPublicKey()
|
||||||
|
|
||||||
|
if (!this.isValid32ByteHex(pubkey)) {
|
||||||
|
throw new Error(`pubkey ${pubkey} is not valid`)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const bds = await this.httpCall(
|
||||||
|
'GET',
|
||||||
|
`list/${pubkey}`,
|
||||||
|
undefined,
|
||||||
|
() =>
|
||||||
|
this.authorizationHeader(evt => {
|
||||||
|
evt.tags.push(['t', 'list'])
|
||||||
|
}),
|
||||||
|
undefined,
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
return bds
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(`failed to list blobs: ${error}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async delete(hash: string): Promise<void> {
|
||||||
|
if (!this.isValid32ByteHex(hash)) {
|
||||||
|
throw new Error(`${hash} is not a valid 32-byte hex string`)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.httpCall(
|
||||||
|
'DELETE',
|
||||||
|
hash,
|
||||||
|
undefined,
|
||||||
|
() =>
|
||||||
|
this.authorizationHeader(evt => {
|
||||||
|
evt.tags.push(['t', 'delete'])
|
||||||
|
evt.tags.push(['x', hash])
|
||||||
|
}),
|
||||||
|
undefined,
|
||||||
|
null,
|
||||||
|
)
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(`failed to delete ${hash}: ${error}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
47
package.json
47
package.json
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"name": "nostr-tools",
|
"name": "nostr-tools",
|
||||||
"version": "2.4.0",
|
"version": "2.14.3",
|
||||||
"description": "Tools for making a Nostr client.",
|
"description": "Tools for making a Nostr client.",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@@ -85,6 +85,9 @@
|
|||||||
"require": "./lib/cjs/nip06.js",
|
"require": "./lib/cjs/nip06.js",
|
||||||
"types": "./lib/types/nip06.d.ts"
|
"types": "./lib/types/nip06.d.ts"
|
||||||
},
|
},
|
||||||
|
"./nip07": {
|
||||||
|
"types": "./lib/types/nip07.d.ts"
|
||||||
|
},
|
||||||
"./nip10": {
|
"./nip10": {
|
||||||
"import": "./lib/esm/nip10.js",
|
"import": "./lib/esm/nip10.js",
|
||||||
"require": "./lib/cjs/nip10.js",
|
"require": "./lib/cjs/nip10.js",
|
||||||
@@ -100,6 +103,11 @@
|
|||||||
"require": "./lib/cjs/nip13.js",
|
"require": "./lib/cjs/nip13.js",
|
||||||
"types": "./lib/types/nip13.d.ts"
|
"types": "./lib/types/nip13.d.ts"
|
||||||
},
|
},
|
||||||
|
"./nip17": {
|
||||||
|
"import": "./lib/esm/nip17.js",
|
||||||
|
"require": "./lib/cjs/nip17.js",
|
||||||
|
"types": "./lib/types/nip17.d.ts"
|
||||||
|
},
|
||||||
"./nip18": {
|
"./nip18": {
|
||||||
"import": "./lib/esm/nip18.js",
|
"import": "./lib/esm/nip18.js",
|
||||||
"require": "./lib/cjs/nip18.js",
|
"require": "./lib/cjs/nip18.js",
|
||||||
@@ -165,11 +173,21 @@
|
|||||||
"require": "./lib/cjs/nip49.js",
|
"require": "./lib/cjs/nip49.js",
|
||||||
"types": "./lib/types/nip49.d.ts"
|
"types": "./lib/types/nip49.d.ts"
|
||||||
},
|
},
|
||||||
|
"./nip54": {
|
||||||
|
"import": "./lib/esm/nip54.js",
|
||||||
|
"require": "./lib/cjs/nip54.js",
|
||||||
|
"types": "./lib/types/nip54.d.ts"
|
||||||
|
},
|
||||||
"./nip57": {
|
"./nip57": {
|
||||||
"import": "./lib/esm/nip57.js",
|
"import": "./lib/esm/nip57.js",
|
||||||
"require": "./lib/cjs/nip57.js",
|
"require": "./lib/cjs/nip57.js",
|
||||||
"types": "./lib/types/nip57.d.ts"
|
"types": "./lib/types/nip57.d.ts"
|
||||||
},
|
},
|
||||||
|
"./nip59": {
|
||||||
|
"import": "./lib/esm/nip59.js",
|
||||||
|
"require": "./lib/cjs/nip59.js",
|
||||||
|
"types": "./lib/types/nip59.d.ts"
|
||||||
|
},
|
||||||
"./nip58": {
|
"./nip58": {
|
||||||
"import": "./lib/esm/nip58.js",
|
"import": "./lib/esm/nip58.js",
|
||||||
"require": "./lib/cjs/nip58.js",
|
"require": "./lib/cjs/nip58.js",
|
||||||
@@ -185,11 +203,6 @@
|
|||||||
"require": "./lib/cjs/nip94.js",
|
"require": "./lib/cjs/nip94.js",
|
||||||
"types": "./lib/types/nip94.d.ts"
|
"types": "./lib/types/nip94.d.ts"
|
||||||
},
|
},
|
||||||
"./nip96": {
|
|
||||||
"import": "./lib/esm/nip96.js",
|
|
||||||
"require": "./lib/cjs/nip96.js",
|
|
||||||
"types": "./lib/types/nip96.d.ts"
|
|
||||||
},
|
|
||||||
"./nip98": {
|
"./nip98": {
|
||||||
"import": "./lib/esm/nip98.js",
|
"import": "./lib/esm/nip98.js",
|
||||||
"require": "./lib/cjs/nip98.js",
|
"require": "./lib/cjs/nip98.js",
|
||||||
@@ -200,11 +213,21 @@
|
|||||||
"require": "./lib/cjs/nip99.js",
|
"require": "./lib/cjs/nip99.js",
|
||||||
"types": "./lib/types/nip99.d.ts"
|
"types": "./lib/types/nip99.d.ts"
|
||||||
},
|
},
|
||||||
|
"./nipb7": {
|
||||||
|
"import": "./lib/esm/nipb7.js",
|
||||||
|
"require": "./lib/cjs/nipb7.js",
|
||||||
|
"types": "./lib/types/nipb7.d.ts"
|
||||||
|
},
|
||||||
"./fakejson": {
|
"./fakejson": {
|
||||||
"import": "./lib/esm/fakejson.js",
|
"import": "./lib/esm/fakejson.js",
|
||||||
"require": "./lib/cjs/fakejson.js",
|
"require": "./lib/cjs/fakejson.js",
|
||||||
"types": "./lib/types/fakejson.d.ts"
|
"types": "./lib/types/fakejson.d.ts"
|
||||||
},
|
},
|
||||||
|
"./signer": {
|
||||||
|
"import": "./lib/esm/signer.js",
|
||||||
|
"require": "./lib/cjs/signer.js",
|
||||||
|
"types": "./lib/types/signer.d.ts"
|
||||||
|
},
|
||||||
"./utils": {
|
"./utils": {
|
||||||
"import": "./lib/esm/utils.js",
|
"import": "./lib/esm/utils.js",
|
||||||
"require": "./lib/cjs/utils.js",
|
"require": "./lib/cjs/utils.js",
|
||||||
@@ -218,10 +241,8 @@
|
|||||||
"@noble/hashes": "1.3.1",
|
"@noble/hashes": "1.3.1",
|
||||||
"@scure/base": "1.1.1",
|
"@scure/base": "1.1.1",
|
||||||
"@scure/bip32": "1.3.1",
|
"@scure/bip32": "1.3.1",
|
||||||
"@scure/bip39": "1.2.1"
|
"@scure/bip39": "1.2.1",
|
||||||
},
|
"nostr-wasm": "0.1.0"
|
||||||
"optionalDependencies": {
|
|
||||||
"nostr-wasm": "v0.1.0"
|
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"typescript": ">=5.0.0"
|
"typescript": ">=5.0.0"
|
||||||
@@ -245,18 +266,14 @@
|
|||||||
"@typescript-eslint/parser": "^6.5.0",
|
"@typescript-eslint/parser": "^6.5.0",
|
||||||
"bun-types": "^1.0.18",
|
"bun-types": "^1.0.18",
|
||||||
"esbuild": "0.16.9",
|
"esbuild": "0.16.9",
|
||||||
"esbuild-plugin-alias": "^0.2.1",
|
|
||||||
"eslint": "^8.56.0",
|
"eslint": "^8.56.0",
|
||||||
"eslint-config-prettier": "^9.0.0",
|
"eslint-config-prettier": "^9.0.0",
|
||||||
"eslint-plugin-babel": "^5.3.1",
|
|
||||||
"esm-loader-typescript": "^1.0.3",
|
|
||||||
"events": "^3.3.0",
|
"events": "^3.3.0",
|
||||||
"mitata": "^0.1.6",
|
"mitata": "^0.1.6",
|
||||||
"mock-socket": "^9.3.1",
|
"mock-socket": "^9.3.1",
|
||||||
"msw": "^2.1.4",
|
|
||||||
"node-fetch": "^2.6.9",
|
"node-fetch": "^2.6.9",
|
||||||
"prettier": "^3.0.3",
|
"prettier": "^3.0.3",
|
||||||
"typescript": "^5.0.4"
|
"typescript": "^5.8.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"prepublish": "just build"
|
"prepublish": "just build"
|
||||||
|
|||||||
114
pool.test.ts
114
pool.test.ts
@@ -1,9 +1,9 @@
|
|||||||
import { afterEach, beforeEach, expect, test } from 'bun:test'
|
import { afterEach, beforeEach, expect, test } from 'bun:test'
|
||||||
|
|
||||||
import { SimplePool } from './pool.ts'
|
import { SimplePool, useWebSocketImplementation } from './pool.ts'
|
||||||
import { finalizeEvent, generateSecretKey, getPublicKey, type Event } from './pure.ts'
|
import { finalizeEvent, generateSecretKey, getPublicKey, type Event } from './pure.ts'
|
||||||
import { useWebSocketImplementation } from './relay.ts'
|
|
||||||
import { MockRelay, MockWebSocketClient } from './test-helpers.ts'
|
import { MockRelay, MockWebSocketClient } from './test-helpers.ts'
|
||||||
|
import { hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
useWebSocketImplementation(MockWebSocketClient)
|
useWebSocketImplementation(MockWebSocketClient)
|
||||||
|
|
||||||
@@ -84,6 +84,86 @@ test('same with double subs', async () => {
|
|||||||
expect(received).toHaveLength(2)
|
expect(received).toHaveLength(2)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('subscribe many map', async () => {
|
||||||
|
let priv = hexToBytes('8ea002840d413ccdd5be98df5dd89d799eaa566355ede83ca0bbdbb4b145e0d3')
|
||||||
|
let pub = getPublicKey(priv)
|
||||||
|
|
||||||
|
let received: Event[] = []
|
||||||
|
let event1 = finalizeEvent(
|
||||||
|
{
|
||||||
|
created_at: Math.round(Date.now() / 1000),
|
||||||
|
content: 'test1',
|
||||||
|
kind: 20001,
|
||||||
|
tags: [],
|
||||||
|
},
|
||||||
|
priv,
|
||||||
|
)
|
||||||
|
let event2 = finalizeEvent(
|
||||||
|
{
|
||||||
|
created_at: Math.round(Date.now() / 1000),
|
||||||
|
content: 'test2',
|
||||||
|
kind: 20002,
|
||||||
|
tags: [['t', 'biloba']],
|
||||||
|
},
|
||||||
|
priv,
|
||||||
|
)
|
||||||
|
let event3 = finalizeEvent(
|
||||||
|
{
|
||||||
|
created_at: Math.round(Date.now() / 1000),
|
||||||
|
content: 'test3',
|
||||||
|
kind: 20003,
|
||||||
|
tags: [['t', 'biloba']],
|
||||||
|
},
|
||||||
|
priv,
|
||||||
|
)
|
||||||
|
|
||||||
|
const [relayA, relayB, relayC] = relayURLs
|
||||||
|
|
||||||
|
pool.subscribeMap(
|
||||||
|
[
|
||||||
|
{ url: relayA, filter: { authors: [pub], kinds: [20001] } },
|
||||||
|
{ url: relayB, filter: { authors: [pub], kinds: [20002] } },
|
||||||
|
{ url: relayC, filter: { kinds: [20003], '#t': ['biloba'] } },
|
||||||
|
],
|
||||||
|
{
|
||||||
|
onevent(event: Event) {
|
||||||
|
received.push(event)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// publish the first
|
||||||
|
await Promise.all(pool.publish([relayA, relayB], event1))
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 100))
|
||||||
|
|
||||||
|
expect(received).toHaveLength(1)
|
||||||
|
expect(received[0]).toEqual(event1)
|
||||||
|
|
||||||
|
// publish the second
|
||||||
|
await pool.publish([relayB], event2)[0]
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 100))
|
||||||
|
|
||||||
|
expect(received).toHaveLength(2)
|
||||||
|
expect(received[1]).toEqual(event2)
|
||||||
|
|
||||||
|
// publish a events that shouldn't match our filters
|
||||||
|
await Promise.all([
|
||||||
|
...pool.publish([relayA, relayB], event3),
|
||||||
|
...pool.publish([relayA, relayB, relayC], event1),
|
||||||
|
pool.publish([relayA, relayB, relayC], event2),
|
||||||
|
])
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 100))
|
||||||
|
|
||||||
|
expect(received).toHaveLength(2)
|
||||||
|
|
||||||
|
// publsih the third
|
||||||
|
await pool.publish([relayC], event3)[0]
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 100))
|
||||||
|
|
||||||
|
expect(received).toHaveLength(3)
|
||||||
|
expect(received[2]).toEqual(event3)
|
||||||
|
})
|
||||||
|
|
||||||
test('query a bunch of events and cancel on eose', async () => {
|
test('query a bunch of events and cancel on eose', async () => {
|
||||||
let events = new Set<string>()
|
let events = new Set<string>()
|
||||||
|
|
||||||
@@ -125,3 +205,33 @@ test('get()', async () => {
|
|||||||
expect(event).not.toBeNull()
|
expect(event).not.toBeNull()
|
||||||
expect(event).toHaveProperty('id', ids[0])
|
expect(event).toHaveProperty('id', ids[0])
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('track relays when publishing', async () => {
|
||||||
|
let event1 = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: 1,
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
tags: [],
|
||||||
|
content: 'hello',
|
||||||
|
},
|
||||||
|
generateSecretKey(),
|
||||||
|
)
|
||||||
|
let event2 = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: 1,
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
tags: [],
|
||||||
|
content: 'hello',
|
||||||
|
},
|
||||||
|
generateSecretKey(),
|
||||||
|
)
|
||||||
|
|
||||||
|
pool.trackRelays = true
|
||||||
|
await Promise.all(pool.publish(relayURLs, event1))
|
||||||
|
expect(pool.seenOn.get(event1.id)).toBeDefined()
|
||||||
|
expect(Array.from(pool.seenOn.get(event1.id)!).map(r => r.url)).toEqual(expect.arrayContaining(relayURLs))
|
||||||
|
|
||||||
|
pool.trackRelays = false
|
||||||
|
await Promise.all(pool.publish(relayURLs, event2))
|
||||||
|
expect(pool.seenOn.get(event2.id)).toBeUndefined()
|
||||||
|
})
|
||||||
|
|||||||
14
pool.ts
14
pool.ts
@@ -1,9 +1,21 @@
|
|||||||
|
/* global WebSocket */
|
||||||
|
|
||||||
import { verifyEvent } from './pure.ts'
|
import { verifyEvent } from './pure.ts'
|
||||||
import { AbstractSimplePool } from './abstract-pool.ts'
|
import { AbstractSimplePool } from './abstract-pool.ts'
|
||||||
|
|
||||||
|
var _WebSocket: typeof WebSocket
|
||||||
|
|
||||||
|
try {
|
||||||
|
_WebSocket = WebSocket
|
||||||
|
} catch {}
|
||||||
|
|
||||||
|
export function useWebSocketImplementation(websocketImplementation: any) {
|
||||||
|
_WebSocket = websocketImplementation
|
||||||
|
}
|
||||||
|
|
||||||
export class SimplePool extends AbstractSimplePool {
|
export class SimplePool extends AbstractSimplePool {
|
||||||
constructor() {
|
constructor() {
|
||||||
super({ verifyEvent })
|
super({ verifyEvent, websocketImplementation: _WebSocket })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
293
pure.test.ts
Normal file
293
pure.test.ts
Normal file
@@ -0,0 +1,293 @@
|
|||||||
|
import { describe, test, expect } from 'bun:test'
|
||||||
|
|
||||||
|
import {
|
||||||
|
finalizeEvent,
|
||||||
|
serializeEvent,
|
||||||
|
getEventHash,
|
||||||
|
validateEvent,
|
||||||
|
verifyEvent,
|
||||||
|
verifiedSymbol,
|
||||||
|
getPublicKey,
|
||||||
|
generateSecretKey,
|
||||||
|
} from './pure.ts'
|
||||||
|
import { ShortTextNote } from './kinds.ts'
|
||||||
|
import { bytesToHex, hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
|
test('private key generation', () => {
|
||||||
|
expect(bytesToHex(generateSecretKey())).toMatch(/[a-f0-9]{64}/)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('public key generation', () => {
|
||||||
|
expect(getPublicKey(generateSecretKey())).toMatch(/[a-f0-9]{64}/)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('public key from private key deterministic', () => {
|
||||||
|
let sk = generateSecretKey()
|
||||||
|
let pk = getPublicKey(sk)
|
||||||
|
|
||||||
|
for (let i = 0; i < 5; i++) {
|
||||||
|
expect(getPublicKey(sk)).toEqual(pk)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('finalizeEvent', () => {
|
||||||
|
test('should create a signed event from a template', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const template = {
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
}
|
||||||
|
|
||||||
|
const event = finalizeEvent(template, privateKey)
|
||||||
|
|
||||||
|
expect(event.kind).toEqual(template.kind)
|
||||||
|
expect(event.tags).toEqual(template.tags)
|
||||||
|
expect(event.content).toEqual(template.content)
|
||||||
|
expect(event.created_at).toEqual(template.created_at)
|
||||||
|
expect(event.pubkey).toEqual(publicKey)
|
||||||
|
expect(typeof event.id).toEqual('string')
|
||||||
|
expect(typeof event.sig).toEqual('string')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('serializeEvent', () => {
|
||||||
|
test('should serialize a valid event object', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const unsignedEvent = {
|
||||||
|
pubkey: publicKey,
|
||||||
|
created_at: 1617932115,
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
}
|
||||||
|
|
||||||
|
const serializedEvent = serializeEvent(unsignedEvent)
|
||||||
|
|
||||||
|
expect(serializedEvent).toEqual(
|
||||||
|
JSON.stringify([
|
||||||
|
0,
|
||||||
|
publicKey,
|
||||||
|
unsignedEvent.created_at,
|
||||||
|
unsignedEvent.kind,
|
||||||
|
unsignedEvent.tags,
|
||||||
|
unsignedEvent.content,
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should throw an error for an invalid event object', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const invalidEvent = {
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
created_at: 1617932115,
|
||||||
|
pubkey: publicKey, // missing content
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
// @ts-expect-error
|
||||||
|
serializeEvent(invalidEvent)
|
||||||
|
}).toThrow("can't serialize event with wrong or missing properties")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('getEventHash', () => {
|
||||||
|
test('should return the correct event hash', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const unsignedEvent = {
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
pubkey: publicKey,
|
||||||
|
}
|
||||||
|
|
||||||
|
const eventHash = getEventHash(unsignedEvent)
|
||||||
|
|
||||||
|
expect(typeof eventHash).toEqual('string')
|
||||||
|
expect(eventHash.length).toEqual(64)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('validateEvent', () => {
|
||||||
|
test('should return true for a valid event object', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const unsignedEvent = {
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
pubkey: publicKey,
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValid = validateEvent(unsignedEvent)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for a non object event', () => {
|
||||||
|
const nonObjectEvent = ''
|
||||||
|
const isValid = validateEvent(nonObjectEvent)
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an event object with missing properties', () => {
|
||||||
|
const invalidEvent = {
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
created_at: 1617932115, // missing content and pubkey
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValid = validateEvent(invalidEvent)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an empty object', () => {
|
||||||
|
const emptyObj = {}
|
||||||
|
|
||||||
|
const isValid = validateEvent(emptyObj)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an object with invalid properties', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const invalidEvent = {
|
||||||
|
kind: 1,
|
||||||
|
tags: [],
|
||||||
|
created_at: '1617932115', // should be a number
|
||||||
|
pubkey: publicKey,
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValid = validateEvent(invalidEvent)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an object with an invalid public key', () => {
|
||||||
|
const invalidEvent = {
|
||||||
|
kind: 1,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
pubkey: 'invalid_pubkey',
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValid = validateEvent(invalidEvent)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an object with invalid tags', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const publicKey = getPublicKey(privateKey)
|
||||||
|
|
||||||
|
const invalidEvent = {
|
||||||
|
kind: 1,
|
||||||
|
tags: {}, // should be an array
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
pubkey: publicKey,
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValid = validateEvent(invalidEvent)
|
||||||
|
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('verifyEvent', () => {
|
||||||
|
test('should return true for a valid event signature', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const event = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
|
||||||
|
const isValid = verifyEvent(event)
|
||||||
|
expect(isValid).toEqual(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an invalid event signature', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
|
||||||
|
// tamper with the signature
|
||||||
|
event.sig = event.sig.replace(/^.{3}/g, '666')
|
||||||
|
|
||||||
|
const isValid = verifyEvent(event)
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false when verifying an event with a different private key', () => {
|
||||||
|
const privateKey1 = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
|
||||||
|
const privateKey2 = hexToBytes('5b4a34f4e4b23c63ad55a35e3f84a3b53d96dbf266edf521a8358f71d19cbf67')
|
||||||
|
const publicKey2 = getPublicKey(privateKey2)
|
||||||
|
|
||||||
|
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: ShortTextNote,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey1,
|
||||||
|
)
|
||||||
|
|
||||||
|
// verify with different private key
|
||||||
|
const isValid = verifyEvent({
|
||||||
|
...event,
|
||||||
|
pubkey: publicKey2,
|
||||||
|
})
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should return false for an invalid event id', () => {
|
||||||
|
const privateKey = hexToBytes('d217c1ff2f8a65c3e3a1740db3b9f58b8c848bb45e26d00ed4714e4a0f4ceecf')
|
||||||
|
|
||||||
|
const { [verifiedSymbol]: _, ...event } = finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: 1,
|
||||||
|
tags: [],
|
||||||
|
content: 'Hello, world!',
|
||||||
|
created_at: 1617932115,
|
||||||
|
},
|
||||||
|
privateKey,
|
||||||
|
)
|
||||||
|
|
||||||
|
// tamper with the id
|
||||||
|
event.id = event.id.replace(/^.{3}/g, '666')
|
||||||
|
|
||||||
|
const isValid = verifyEvent(event)
|
||||||
|
expect(isValid).toEqual(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { expect, test } from 'bun:test'
|
import { expect, test } from 'bun:test'
|
||||||
|
import { Server } from 'mock-socket'
|
||||||
import { finalizeEvent, generateSecretKey, getPublicKey } from './pure.ts'
|
import { finalizeEvent, generateSecretKey, getPublicKey } from './pure.ts'
|
||||||
import { Relay, useWebSocketImplementation } from './relay.ts'
|
import { Relay, useWebSocketImplementation } from './relay.ts'
|
||||||
import { MockRelay, MockWebSocketClient } from './test-helpers.ts'
|
import { MockRelay, MockWebSocketClient } from './test-helpers.ts'
|
||||||
@@ -92,3 +92,28 @@ test('listening and publishing and closing', async done => {
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('publish timeout', async () => {
|
||||||
|
const url = 'wss://relay.example.com'
|
||||||
|
new Server(url)
|
||||||
|
|
||||||
|
const relay = new Relay(url)
|
||||||
|
relay.publishTimeout = 100
|
||||||
|
await relay.connect()
|
||||||
|
|
||||||
|
setTimeout(() => relay.close(), 20000) // close the relay to fail the test on timeout
|
||||||
|
|
||||||
|
expect(
|
||||||
|
relay.publish(
|
||||||
|
finalizeEvent(
|
||||||
|
{
|
||||||
|
kind: 1,
|
||||||
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
|
tags: [],
|
||||||
|
content: 'hello',
|
||||||
|
},
|
||||||
|
generateSecretKey(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
).rejects.toThrow('publish timed out')
|
||||||
|
})
|
||||||
|
|||||||
19
relay.ts
19
relay.ts
@@ -1,16 +1,21 @@
|
|||||||
|
/* global WebSocket */
|
||||||
|
|
||||||
import { verifyEvent } from './pure.ts'
|
import { verifyEvent } from './pure.ts'
|
||||||
import { AbstractRelay } from './abstract-relay.ts'
|
import { AbstractRelay } from './abstract-relay.ts'
|
||||||
|
|
||||||
/**
|
var _WebSocket: typeof WebSocket
|
||||||
* @deprecated use Relay.connect() instead.
|
|
||||||
*/
|
try {
|
||||||
export function relayConnect(url: string): Promise<Relay> {
|
_WebSocket = WebSocket
|
||||||
return Relay.connect(url)
|
} catch {}
|
||||||
|
|
||||||
|
export function useWebSocketImplementation(websocketImplementation: any) {
|
||||||
|
_WebSocket = websocketImplementation
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Relay extends AbstractRelay {
|
export class Relay extends AbstractRelay {
|
||||||
constructor(url: string) {
|
constructor(url: string) {
|
||||||
super(url, { verifyEvent })
|
super(url, { verifyEvent, websocketImplementation: _WebSocket })
|
||||||
}
|
}
|
||||||
|
|
||||||
static async connect(url: string): Promise<Relay> {
|
static async connect(url: string): Promise<Relay> {
|
||||||
@@ -20,4 +25,6 @@ export class Relay extends AbstractRelay {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type RelayRecord = Record<string, { read: boolean; write: boolean }>
|
||||||
|
|
||||||
export * from './abstract-relay.ts'
|
export * from './abstract-relay.ts'
|
||||||
|
|||||||
23
signer.ts
Normal file
23
signer.ts
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { EventTemplate, VerifiedEvent } from './core.ts'
|
||||||
|
import { finalizeEvent, getPublicKey } from './pure.ts'
|
||||||
|
|
||||||
|
export interface Signer {
|
||||||
|
getPublicKey(): Promise<string>
|
||||||
|
signEvent(event: EventTemplate): Promise<VerifiedEvent>
|
||||||
|
}
|
||||||
|
|
||||||
|
export class PlainKeySigner implements Signer {
|
||||||
|
private secretKey: Uint8Array
|
||||||
|
|
||||||
|
constructor(secretKey: Uint8Array) {
|
||||||
|
this.secretKey = secretKey
|
||||||
|
}
|
||||||
|
|
||||||
|
async getPublicKey(): Promise<string> {
|
||||||
|
return getPublicKey(this.secretKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
async signEvent(event: EventTemplate): Promise<VerifiedEvent> {
|
||||||
|
return finalizeEvent(event, this.secretKey)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -35,9 +35,9 @@ export class MockRelay {
|
|||||||
finalizeEvent(
|
finalizeEvent(
|
||||||
{
|
{
|
||||||
kind: 1,
|
kind: 1,
|
||||||
content: '',
|
content: 'autogenerated by relay',
|
||||||
created_at: Math.floor(Date.now() / 1000),
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
tags: [],
|
tags: [['t', 'auto']],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
),
|
),
|
||||||
@@ -68,9 +68,9 @@ export class MockRelay {
|
|||||||
const event = finalizeEvent(
|
const event = finalizeEvent(
|
||||||
{
|
{
|
||||||
kind,
|
kind,
|
||||||
content: '',
|
content: 'kind-aware autogenerated by relay',
|
||||||
created_at: Math.floor(Date.now() / 1000),
|
created_at: Math.floor(Date.now() / 1000),
|
||||||
tags: [],
|
tags: [['t', 'auto']],
|
||||||
},
|
},
|
||||||
sk,
|
sk,
|
||||||
)
|
)
|
||||||
|
|||||||
25
utils.ts
25
utils.ts
@@ -3,15 +3,21 @@ import type { Event } from './core.ts'
|
|||||||
export const utf8Decoder: TextDecoder = new TextDecoder('utf-8')
|
export const utf8Decoder: TextDecoder = new TextDecoder('utf-8')
|
||||||
export const utf8Encoder: TextEncoder = new TextEncoder()
|
export const utf8Encoder: TextEncoder = new TextEncoder()
|
||||||
|
|
||||||
|
export { bytesToHex, hexToBytes } from '@noble/hashes/utils'
|
||||||
|
|
||||||
export function normalizeURL(url: string): string {
|
export function normalizeURL(url: string): string {
|
||||||
if (url.indexOf('://') === -1) url = 'wss://' + url
|
try {
|
||||||
let p = new URL(url)
|
if (url.indexOf('://') === -1) url = 'wss://' + url
|
||||||
p.pathname = p.pathname.replace(/\/+/g, '/')
|
let p = new URL(url)
|
||||||
if (p.pathname.endsWith('/')) p.pathname = p.pathname.slice(0, -1)
|
p.pathname = p.pathname.replace(/\/+/g, '/')
|
||||||
if ((p.port === '80' && p.protocol === 'ws:') || (p.port === '443' && p.protocol === 'wss:')) p.port = ''
|
if (p.pathname.endsWith('/')) p.pathname = p.pathname.slice(0, -1)
|
||||||
p.searchParams.sort()
|
if ((p.port === '80' && p.protocol === 'ws:') || (p.port === '443' && p.protocol === 'wss:')) p.port = ''
|
||||||
p.hash = ''
|
p.searchParams.sort()
|
||||||
return p.toString()
|
p.hash = ''
|
||||||
|
return p.toString()
|
||||||
|
} catch (e) {
|
||||||
|
throw new Error(`Invalid URL: ${url}`)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function insertEventIntoDescendingList(sortedArray: Event[], event: Event): Event[] {
|
export function insertEventIntoDescendingList(sortedArray: Event[], event: Event): Event[] {
|
||||||
@@ -111,6 +117,9 @@ export class Queue<V> {
|
|||||||
|
|
||||||
const target = this.first
|
const target = this.first
|
||||||
this.first = target.next
|
this.first = target.next
|
||||||
|
if (this.first) {
|
||||||
|
this.first.prev = null // fix: clean up prev pointer
|
||||||
|
}
|
||||||
|
|
||||||
return target.value
|
return target.value
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user