mirror of
https://github.com/nbd-wtf/go-nostr.git
synced 2025-03-17 13:22:56 +01:00
246 lines
5.8 KiB
Go
246 lines
5.8 KiB
Go
package nip19
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"encoding/hex"
|
|
"fmt"
|
|
|
|
"github.com/btcsuite/btcd/btcutil/bech32"
|
|
"github.com/nbd-wtf/go-nostr"
|
|
)
|
|
|
|
func Decode(bech32string string) (prefix string, value any, err error) {
|
|
prefix, bits5, err := bech32.DecodeNoLimit(bech32string)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
data, err := bech32.ConvertBits(bits5, 5, 8, false)
|
|
if err != nil {
|
|
return prefix, nil, fmt.Errorf("failed to translate data into 8 bits: %s", err.Error())
|
|
}
|
|
|
|
switch prefix {
|
|
case "npub", "nsec", "note":
|
|
if len(data) != 32 {
|
|
return prefix, nil, fmt.Errorf("data should be 32 bytes (%d)", len(data))
|
|
}
|
|
|
|
return prefix, hex.EncodeToString(data[0:32]), nil
|
|
case "nprofile":
|
|
var result nostr.ProfilePointer
|
|
curr := 0
|
|
for {
|
|
t, v := readTLVEntry(data[curr:])
|
|
if v == nil {
|
|
// end here
|
|
if result.PublicKey == "" {
|
|
return prefix, result, fmt.Errorf("no pubkey found for nprofile")
|
|
}
|
|
|
|
return prefix, result, nil
|
|
}
|
|
|
|
switch t {
|
|
case TLVDefault:
|
|
if len(v) != 32 {
|
|
return prefix, nil, fmt.Errorf("pubkey should be 32 bytes (%d)", len(v))
|
|
}
|
|
result.PublicKey = hex.EncodeToString(v)
|
|
case TLVRelay:
|
|
result.Relays = append(result.Relays, string(v))
|
|
default:
|
|
// ignore
|
|
}
|
|
|
|
curr = curr + 2 + len(v)
|
|
}
|
|
case "nevent":
|
|
var result nostr.EventPointer
|
|
curr := 0
|
|
for {
|
|
t, v := readTLVEntry(data[curr:])
|
|
if v == nil {
|
|
// end here
|
|
if result.ID == "" {
|
|
return prefix, result, fmt.Errorf("no id found for nevent")
|
|
}
|
|
|
|
return prefix, result, nil
|
|
}
|
|
|
|
switch t {
|
|
case TLVDefault:
|
|
if len(v) != 32 {
|
|
return prefix, nil, fmt.Errorf("id should be 32 bytes (%d)", len(v))
|
|
}
|
|
result.ID = hex.EncodeToString(v)
|
|
case TLVRelay:
|
|
result.Relays = append(result.Relays, string(v))
|
|
case TLVAuthor:
|
|
if len(v) != 32 {
|
|
return prefix, nil, fmt.Errorf("author should be 32 bytes (%d)", len(v))
|
|
}
|
|
result.Author = hex.EncodeToString(v)
|
|
case TLVKind:
|
|
if len(v) != 4 {
|
|
return prefix, nil, fmt.Errorf("invalid uint32 value for integer (%v)", v)
|
|
}
|
|
result.Kind = int(binary.BigEndian.Uint32(v))
|
|
default:
|
|
// ignore
|
|
}
|
|
|
|
curr = curr + 2 + len(v)
|
|
}
|
|
case "naddr":
|
|
var result nostr.EntityPointer
|
|
curr := 0
|
|
for {
|
|
t, v := readTLVEntry(data[curr:])
|
|
if v == nil {
|
|
// end here
|
|
if result.Kind == 0 || result.Identifier == "" || result.PublicKey == "" {
|
|
return prefix, result, fmt.Errorf("incomplete naddr")
|
|
}
|
|
|
|
return prefix, result, nil
|
|
}
|
|
|
|
switch t {
|
|
case TLVDefault:
|
|
result.Identifier = string(v)
|
|
case TLVRelay:
|
|
result.Relays = append(result.Relays, string(v))
|
|
case TLVAuthor:
|
|
if len(v) != 32 {
|
|
return prefix, nil, fmt.Errorf("author should be 32 bytes (%d)", len(v))
|
|
}
|
|
result.PublicKey = hex.EncodeToString(v)
|
|
case TLVKind:
|
|
result.Kind = int(binary.BigEndian.Uint32(v))
|
|
default:
|
|
// ignore
|
|
}
|
|
|
|
curr = curr + 2 + len(v)
|
|
}
|
|
}
|
|
|
|
return prefix, data, fmt.Errorf("unknown tag %s", prefix)
|
|
}
|
|
|
|
func EncodePrivateKey(privateKeyHex string) (string, error) {
|
|
b, err := hex.DecodeString(privateKeyHex)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to decode private key hex: %w", err)
|
|
}
|
|
|
|
bits5, err := bech32.ConvertBits(b, 8, 5, true)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return bech32.Encode("nsec", bits5)
|
|
}
|
|
|
|
func EncodePublicKey(publicKeyHex string) (string, error) {
|
|
b, err := hex.DecodeString(publicKeyHex)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to decode public key hex: %w", err)
|
|
}
|
|
|
|
bits5, err := bech32.ConvertBits(b, 8, 5, true)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return bech32.Encode("npub", bits5)
|
|
}
|
|
|
|
func EncodeNote(eventIDHex string) (string, error) {
|
|
b, err := hex.DecodeString(eventIDHex)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to decode event id hex: %w", err)
|
|
}
|
|
|
|
bits5, err := bech32.ConvertBits(b, 8, 5, true)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return bech32.Encode("note", bits5)
|
|
}
|
|
|
|
func EncodeProfile(publicKeyHex string, relays []string) (string, error) {
|
|
buf := &bytes.Buffer{}
|
|
pubkey, err := hex.DecodeString(publicKeyHex)
|
|
if err != nil {
|
|
return "", fmt.Errorf("invalid pubkey '%s': %w", publicKeyHex, err)
|
|
}
|
|
writeTLVEntry(buf, TLVDefault, pubkey)
|
|
|
|
for _, url := range relays {
|
|
writeTLVEntry(buf, TLVRelay, []byte(url))
|
|
}
|
|
|
|
bits5, err := bech32.ConvertBits(buf.Bytes(), 8, 5, true)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to convert bits: %w", err)
|
|
}
|
|
|
|
return bech32.Encode("nprofile", bits5)
|
|
}
|
|
|
|
func EncodeEvent(eventIDHex string, relays []string, author string) (string, error) {
|
|
buf := &bytes.Buffer{}
|
|
id, err := hex.DecodeString(eventIDHex)
|
|
if err != nil || len(id) != 32 {
|
|
return "", fmt.Errorf("invalid id '%s': %w", eventIDHex, err)
|
|
}
|
|
writeTLVEntry(buf, TLVDefault, id)
|
|
|
|
for _, url := range relays {
|
|
writeTLVEntry(buf, TLVRelay, []byte(url))
|
|
}
|
|
|
|
if pubkey, _ := hex.DecodeString(author); len(pubkey) == 32 {
|
|
writeTLVEntry(buf, TLVAuthor, pubkey)
|
|
}
|
|
|
|
bits5, err := bech32.ConvertBits(buf.Bytes(), 8, 5, true)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to convert bits: %w", err)
|
|
}
|
|
|
|
return bech32.Encode("nevent", bits5)
|
|
}
|
|
|
|
func EncodeEntity(publicKey string, kind int, identifier string, relays []string) (string, error) {
|
|
buf := &bytes.Buffer{}
|
|
|
|
writeTLVEntry(buf, TLVDefault, []byte(identifier))
|
|
|
|
for _, url := range relays {
|
|
writeTLVEntry(buf, TLVRelay, []byte(url))
|
|
}
|
|
|
|
pubkey, err := hex.DecodeString(publicKey)
|
|
if err != nil {
|
|
return "", fmt.Errorf("invalid pubkey '%s': %w", pubkey, err)
|
|
}
|
|
writeTLVEntry(buf, TLVAuthor, pubkey)
|
|
|
|
kindBytes := make([]byte, 4)
|
|
binary.BigEndian.PutUint32(kindBytes, uint32(kind))
|
|
writeTLVEntry(buf, TLVKind, kindBytes)
|
|
|
|
bits5, err := bech32.ConvertBits(buf.Bytes(), 8, 5, true)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to convert bits: %w", err)
|
|
}
|
|
|
|
return bech32.Encode("naddr", bits5)
|
|
}
|