package main import ( "context" "fmt" "io" "net/http" "regexp" "strings" "time" "github.com/gomarkdown/markdown" "github.com/gomarkdown/markdown/ast" "github.com/gomarkdown/markdown/html" mdhtml "github.com/gomarkdown/markdown/html" "github.com/gomarkdown/markdown/parser" "github.com/microcosm-cc/bluemonday" "mvdan.cc/xurls/v2" "github.com/nbd-wtf/go-nostr" "github.com/nbd-wtf/go-nostr/nip10" "github.com/nbd-wtf/go-nostr/nip19" ) const XML_HEADER = "\n" var ( urlSuffixMatcher = regexp.MustCompile(`[\w-_.]+\.[\w-_.]+(\/[\/\w]*)?$`) nostrEveryMatcher = regexp.MustCompile(`nostr:((npub|note|nevent|nprofile|naddr)1[a-z0-9]+)\b`) nostrNoteNeventMatcher = regexp.MustCompile(`nostr:((note|nevent)1[a-z0-9]+)\b`) nostrNpubNprofileMatcher = regexp.MustCompile(`nostr:((npub|nprofile)1[a-z0-9]+)\b`) urlMatcher = func() *regexp.Regexp { // hack to only allow these schemes while still using this library xurls.Schemes = []string{"https"} xurls.SchemesNoAuthority = []string{"blob"} xurls.SchemesUnofficial = []string{"http"} return xurls.Strict() }() imageExtensionMatcher = regexp.MustCompile(`.*\.(png|jpg|jpeg|gif|webp)(\?.*)?$`) videoExtensionMatcher = regexp.MustCompile(`.*\.(mp4|ogg|webm|mov)(\?.*)?$`) ) var kindNames = map[int]string{ 0: "Metadata", 1: "Short Text Note", 2: "Recommend Relay", 3: "Contacts", 4: "Encrypted Direct Messages", 5: "Event Deletion", 6: "Reposts", 7: "Reaction", 8: "Badge Award", 40: "Channel Creation", 41: "Channel Metadata", 42: "Channel Message", 43: "Channel Hide Message", 44: "Channel Mute User", 1063: "File Metadata", 1984: "Reporting", 9734: "Zap Request", 9735: "Zap", 10000: "Mute List", 10001: "Pin List", 10002: "Relay List Metadata", 13194: "Wallet Info", 22242: "Client Authentication", 23194: "Wallet Request", 23195: "Wallet Response", 24133: "Nostr Connect", 30000: "Categorized People List", 30001: "Categorized Bookmark List", 30008: "Profile Badges", 30009: "Badge Definition", 30017: "Create or update a stall", 30018: "Create or update a product", 30023: "Long-form Content", 30078: "Application-specific Data", } var kindNIPs = map[int]string{ 0: "01", 1: "01", 2: "01", 3: "02", 4: "04", 5: "09", 6: "18", 7: "25", 8: "58", 40: "28", 41: "28", 42: "28", 43: "28", 44: "28", 1063: "94", 1984: "56", 9734: "57", 9735: "57", 10000: "51", 10001: "51", 10002: "65", 13194: "47", 22242: "42", 23194: "47", 23195: "47", 24133: "46", 30000: "51", 30001: "51", 30008: "58", 30009: "58", 30017: "15", 30018: "15", 30023: "23", 30078: "78", } type ClientReference struct { ID string Name string URL string } func generateClientList(code string, event *nostr.Event) []ClientReference { if event.Kind == 1 || event.Kind == 6 { return []ClientReference{ {ID: "native", Name: "your native client", URL: "nostr:" + code}, {ID: "snort", Name: "Snort", URL: "https://Snort.social/e/" + code}, {ID: "nostrudel", Name: "Nostrudel", URL: "https://nostrudel.ninja/#/n/" + code}, {ID: "satellite", Name: "Satellite", URL: "https://satellite.earth/thread/" + event.ID}, {ID: "coracle", Name: "Coracle", URL: "https://coracle.social/" + code}, {ID: "primal", Name: "Primal", URL: "https://primal.net/thread/" + event.ID}, {ID: "nostter", Name: "Nostter", URL: "https://nostter.vercel.app/" + code}, {ID: "highlighter", Name: "Highlighter", URL: "https://highlighter.com/a/" + code}, {ID: "iris", Name: "Iris", URL: "https://iris.to/" + code}, {ID: "yosup", Name: "Yosup", URL: "https://yosup.app/thread/" + event.ID}, } } else if event.Kind == 0 { return []ClientReference{ {ID: "native", Name: "your native client", URL: "nostr:" + code}, {ID: "nosta", Name: "Nosta", URL: "https://nosta.me/" + code}, {ID: "snort", Name: "Snort", URL: "https://snort.social/p/" + code}, {ID: "satellite", Name: "Satellite", URL: "https://satellite.earth/@" + code}, {ID: "coracle", Name: "Coracle", URL: "https://coracle.social/" + code}, {ID: "primal", Name: "Primal", URL: "https://primal.net/profile/" + event.PubKey}, {ID: "nostrudel", Name: "Nostrudel", URL: "https://nostrudel.ninja/#/u/" + code}, {ID: "nostter", Name: "Nostter", URL: "https://nostter.vercel.app/" + code}, {ID: "highlighter", Name: "Highlighter", URL: "https://highlighter.com/p/" + event.PubKey}, {ID: "iris", Name: "Iris", URL: "https://iris.to/" + code}, {ID: "yosup", Name: "Yosup", URL: "https://yosup.app/profile/" + event.PubKey}, {ID: "nosotros", Name: "Nosotros", URL: "https://nosotros.app/" + code}, } } else if event.Kind == 30023 || event.Kind == 30024 { return []ClientReference{ {ID: "native", Name: "your native client", URL: "nostr:" + code}, {ID: "yakihonne", Name: "YakiHonne", URL: "https://yakihonne.com/article/" + code}, {ID: "habla", Name: "Habla", URL: "https://habla.news/a/" + code}, {ID: "highlighter", Name: "Highlighter", URL: "https://highlighter.com/a/" + code}, {ID: "blogstack", Name: "Blogstack", URL: "https://blogstack.io/" + code}, } } return nil } func generateRelayBrowserClientList(host string) []ClientReference { return []ClientReference{ {ID: "coracle", Name: "Coracle", URL: "https://coracle.social/relays/" + host}, } } func getPreviewStyle(r *http.Request) string { if style := r.URL.Query().Get("style"); style != "" { // debug mode return style } ua := strings.ToLower(r.Header.Get("User-Agent")) accept := r.Header.Get("Accept") switch { case strings.Contains(ua, "telegrambot"): return "telegram" case strings.Contains(ua, "twitterbot"): return "twitter" case strings.Contains(ua, "mattermost"): return "mattermost" case strings.Contains(ua, "slack"): return "slack" case strings.Contains(ua, "discord"): return "discord" case strings.Contains(ua, "whatsapp"): return "whatsapp" case strings.Contains(ua, "iframely"): return "iframely" case strings.Contains(accept, "text/html"): return "normal" default: return "unknown" } } func getParentNevent(event *nostr.Event) string { parentNevent := "" replyTag := nip10.GetImmediateReply(event.Tags) if replyTag != nil { relay := "" if len(*replyTag) > 2 { relay = (*replyTag)[2] } else { relay = "" } parentNevent, _ = nip19.EncodeEvent((*replyTag)[1], []string{relay}, "") } return parentNevent } // Rendering functions // ### ### ### ### ### ### ### ### ### ### ### func replaceURLsWithTags(input string, imageReplacementTemplate, videoReplacementTemplate string) string { return urlMatcher.ReplaceAllStringFunc(input, func(match string) string { switch { case imageExtensionMatcher.MatchString(match): // Match and replace image URLs with a custom replacement // Usually is html => ` ` // or markdown !()[...] tags for further processing => `![](%s)` return fmt.Sprintf(imageReplacementTemplate, match) case videoExtensionMatcher.MatchString(match): // Match and replace video URLs with a custom replacement // Usually is html