package main import ( "context" "fmt" "html" "html/template" "regexp" "strconv" "strings" "time" "github.com/nbd-wtf/go-nostr" "github.com/nbd-wtf/go-nostr/nip10" "github.com/nbd-wtf/go-nostr/nip19" "github.com/nbd-wtf/go-nostr/sdk" "github.com/texttheater/golang-levenshtein/levenshtein" ) type EnhancedEvent struct { *nostr.Event relays []string subject string summary string author sdk.ProfileMetadata } func NewEnhancedEvent( ctx context.Context, event *nostr.Event, ) EnhancedEvent { ee := EnhancedEvent{Event: event} for _, tag := range event.Tags { if len(tag) < 2 { continue } if tag[0] == "subject" || tag[0] == "title" { ee.subject = tag[1] } if tag[0] == "summary" { ee.summary = tag[1] } } if event.Kind == 0 { spm, _ := sdk.ParseMetadata(event) ee.author = spm } else { ctx, cancel := context.WithTimeout(ctx, time.Second*3) defer cancel() ee.author = sys.FetchProfileMetadata(ctx, event.PubKey) } return ee } func (ee EnhancedEvent) authorLong() string { if ee.author.Name != "" { return fmt.Sprintf("%s (%s)", ee.author.Name, ee.author.NpubShort()) } return ee.author.Npub() } func (ee EnhancedEvent) getParentNevent() string { parentNevent := "" switch ee.Kind { case 1, 1063: parent := nip10.GetImmediateParent(ee.Tags) if parent != nil { parentNevent = nip19.EncodePointer(parent) } case 1311: if atag := ee.Tags.Find("a"); atag != nil { parts := strings.Split(atag[1], ":") kind, _ := strconv.Atoi(parts[0]) var relays []string if (len(atag) > 2) && (atag[2] != "") { relays = []string{atag[2]} } parentNevent, _ = nip19.EncodeEntity(parts[1], kind, parts[2], relays) } } return parentNevent } func (ee EnhancedEvent) isReply() bool { return nip10.GetImmediateParent(ee.Event.Tags) != nil } func (ee EnhancedEvent) Preview() template.HTML { lines := strings.Split(html.EscapeString(ee.Event.Content), "\n") var processedLines []string for _, line := range lines { if strings.TrimSpace(line) == "" { continue } processedLine := shortenNostrURLs(line) processedLines = append(processedLines, processedLine) } return template.HTML(strings.Join(processedLines, "
")) } func (ee EnhancedEvent) RssTitle() string { regex := regexp.MustCompile(`(?i)`) replacedString := regex.ReplaceAllString(string(ee.Preview()), " ") words := strings.Fields(replacedString) title := "" for i, word := range words { if len(title)+len(word)+1 <= 65 { // +1 for space if title != "" { title += " " } title += word } else { if i > 1 { // the first word len is > 65 title = title + " ..." } else { title = "" } break } } content := ee.RssContent() distance := levenshtein.DistanceForStrings([]rune(title), []rune(content), levenshtein.DefaultOptions) similarityThreshold := 5 if distance <= similarityThreshold { return "" } else { return title } } func (ee EnhancedEvent) RssContent() string { content := ee.Event.Content content = basicFormatting(html.EscapeString(content), true, false, false) content = renderQuotesAsHTML(context.Background(), content, false) if nevent := ee.getParentNevent(); nevent != "" { neventShort := nevent[:8] + "…" + nevent[len(nevent)-4:] content = "In reply to " + neventShort + "
_________________________

" + content } return content } func (ee EnhancedEvent) Thumb() string { imgRegex := regexp.MustCompile(`(https?://[^\s]+\.(?:png|jpe?g|gif|bmp|svg)(?:/[^\s]*)?)`) matches := imgRegex.FindAllStringSubmatch(ee.Event.Content, -1) if len(matches) > 0 { // The first match group captures the image URL return matches[0][1] } return "" } func (ee EnhancedEvent) Npub() string { npub, _ := nip19.EncodePublicKey(ee.Event.PubKey) return npub } func (ee EnhancedEvent) NpubShort() string { npub := ee.Npub() return npub[:8] + "…" + npub[len(npub)-4:] } func (ee EnhancedEvent) Nevent() string { nevent, _ := nip19.EncodeEvent(ee.Event.ID, ee.relays, ee.Event.PubKey) return nevent } func (ee EnhancedEvent) CreatedAtStr() string { return time.Unix(int64(ee.Event.CreatedAt), 0).Format("2006-01-02 15:04:05 MST") } func (ee EnhancedEvent) ModifiedAtStr() string { return time.Unix(int64(ee.Event.CreatedAt), 0).Format("2006-01-02T15:04:05Z07:00") }