go-nostr/nip19/nip19.go

229 lines
5.2 KiB
Go
Raw Permalink Normal View History

2022-10-31 14:57:33 -03:00
package nip19
import (
2022-12-27 07:49:26 -03:00
"bytes"
2023-02-27 16:15:04 -03:00
"encoding/binary"
2022-10-31 14:57:33 -03:00
"encoding/hex"
"fmt"
2022-12-27 07:49:26 -03:00
"github.com/btcsuite/btcd/btcutil/bech32"
2023-02-05 16:25:00 -03:00
"github.com/nbd-wtf/go-nostr"
)
2022-12-27 07:49:26 -03:00
func Decode(bech32string string) (prefix string, value any, err error) {
prefix, bits5, err := bech32.DecodeNoLimit(bech32string)
2022-12-27 07:49:26 -03:00
if err != nil {
return "", nil, err
}
data, err := bech32.ConvertBits(bits5, 5, 8, false)
2022-12-27 07:49:26 -03:00
if err != nil {
return prefix, nil, fmt.Errorf("failed translating data into 8 bits: %s", err.Error())
}
switch prefix {
case "npub", "nsec", "note":
if len(data) < 32 {
return prefix, nil, fmt.Errorf("data is less than 32 bytes (%d)", len(data))
}
return prefix, hex.EncodeToString(data[0:32]), nil
case "nprofile":
2023-02-05 16:25:00 -03:00
var result nostr.ProfilePointer
2022-12-27 07:49:26 -03:00
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:
result.PublicKey = hex.EncodeToString(v)
case TLVRelay:
result.Relays = append(result.Relays, string(v))
default:
// ignore
}
curr = curr + 2 + len(v)
}
case "nevent":
2023-02-05 16:25:00 -03:00
var result nostr.EventPointer
2022-12-27 07:49:26 -03:00
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:
result.ID = hex.EncodeToString(v)
case TLVRelay:
result.Relays = append(result.Relays, string(v))
2023-03-17 09:09:36 -03:00
case TLVAuthor:
result.Author = hex.EncodeToString(v)
2022-12-27 07:49:26 -03:00
default:
// ignore
}
2023-02-27 16:15:04 -03:00
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:
result.PublicKey = hex.EncodeToString(v)
case TLVKind:
result.Kind = int(binary.BigEndian.Uint32(v))
default:
// ignore
}
2022-12-27 07:49:26 -03:00
curr = curr + 2 + len(v)
}
}
return prefix, data, fmt.Errorf("unknown tag %s", prefix)
}
2022-10-31 14:57:33 -03:00
func EncodePrivateKey(privateKeyHex string) (string, error) {
b, err := hex.DecodeString(privateKeyHex)
2022-12-27 07:49:26 -03:00
if err != nil {
return "", fmt.Errorf("failed to decode private key hex: %w", err)
}
bits5, err := bech32.ConvertBits(b, 8, 5, true)
2022-10-31 14:57:33 -03:00
if err != nil {
return "", err
}
return bech32.Encode("nsec", bits5)
2022-10-31 14:57:33 -03:00
}
2022-12-19 15:10:23 -03:00
func EncodePublicKey(publicKeyHex string) (string, error) {
2022-10-31 14:57:33 -03:00
b, err := hex.DecodeString(publicKeyHex)
if err != nil {
return "", fmt.Errorf("failed to decode public key hex: %w", err)
2022-10-31 14:57:33 -03:00
}
bits5, err := bech32.ConvertBits(b, 8, 5, true)
if err != nil {
return "", err
}
return bech32.Encode("npub", bits5)
2022-10-31 14:57:33 -03:00
}
func EncodeNote(eventIdHex string) (string, error) {
b, err := hex.DecodeString(eventIdHex)
2022-12-27 07:49:26 -03:00
if err != nil {
return "", fmt.Errorf("failed to decode event id hex: %w", err)
}
bits5, err := bech32.ConvertBits(b, 8, 5, true)
2022-10-31 14:57:33 -03:00
if err != nil {
return "", err
}
return bech32.Encode("note", bits5)
2022-10-31 14:57:33 -03:00
}
2022-12-27 07:49:26 -03:00
func EncodeProfile(publicKeyHex string, relays []string) (string, error) {
buf := &bytes.Buffer{}
pubkey, err := hex.DecodeString(publicKeyHex)
2022-10-31 14:57:33 -03:00
if err != nil {
2022-12-27 07:49:26 -03:00
return "", fmt.Errorf("invalid pubkey '%s': %w", publicKeyHex, err)
2022-10-31 14:57:33 -03:00
}
2022-12-27 07:49:26 -03:00
writeTLVEntry(buf, TLVDefault, pubkey)
2022-10-31 14:57:33 -03:00
2022-12-27 07:49:26 -03:00
for _, url := range relays {
writeTLVEntry(buf, TLVRelay, []byte(url))
}
bits5, err := bech32.ConvertBits(buf.Bytes(), 8, 5, true)
if err != nil {
2022-12-27 07:49:26 -03:00
return "", fmt.Errorf("failed to convert bits: %w", err)
}
return bech32.Encode("nprofile", bits5)
2022-12-27 07:49:26 -03:00
}
2023-03-17 09:09:36 -03:00
func EncodeEvent(eventIdHex string, relays []string, author string) (string, error) {
2022-12-27 07:49:26 -03:00
buf := &bytes.Buffer{}
2023-02-27 16:15:04 -03:00
id, err := hex.DecodeString(eventIdHex)
2023-03-17 09:09:36 -03:00
if err != nil || len(id) != 32 {
2022-12-27 07:49:26 -03:00
return "", fmt.Errorf("invalid id '%s': %w", eventIdHex, err)
}
2023-02-27 16:15:04 -03:00
writeTLVEntry(buf, TLVDefault, id)
2022-12-27 07:49:26 -03:00
for _, url := range relays {
writeTLVEntry(buf, TLVRelay, []byte(url))
}
2023-03-17 09:09:36 -03:00
if pubkey, _ := hex.DecodeString(author); len(pubkey) == 32 {
writeTLVEntry(buf, TLVAuthor, pubkey)
}
bits5, err := bech32.ConvertBits(buf.Bytes(), 8, 5, true)
2022-12-27 07:49:26 -03:00
if err != nil {
return "", fmt.Errorf("failed to convert bits: %w", err)
2022-10-31 14:57:33 -03:00
}
return bech32.Encode("nevent", bits5)
2022-10-31 14:57:33 -03:00
}
2023-02-27 16:15:04 -03:00
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)
2023-02-27 16:15:04 -03:00
if err != nil {
return "", fmt.Errorf("failed to convert bits: %w", err)
}
return bech32.Encode("naddr", bits5)
2023-02-27 16:15:04 -03:00
}