WIP adding guards to webmention

This commit is contained in:
Niko Abeler 2022-09-09 21:14:49 +02:00
parent a8998068ad
commit d66c1a6817
2 changed files with 140 additions and 26 deletions

63
post.go
View File

@ -8,6 +8,7 @@ import (
"os" "os"
"path" "path"
"sort" "sort"
"sync"
"time" "time"
"github.com/yuin/goldmark" "github.com/yuin/goldmark"
@ -23,6 +24,7 @@ type Post struct {
title string title string
metaLoaded bool metaLoaded bool
meta PostMeta meta PostMeta
wmLock sync.Mutex
} }
type PostMeta struct { type PostMeta struct {
@ -37,39 +39,39 @@ type PostWebmetions struct {
Outgoing []WebmentionOut `ymal:"outgoing"` Outgoing []WebmentionOut `ymal:"outgoing"`
} }
func (post Post) Id() string { func (post *Post) Id() string {
return post.id return post.id
} }
func (post Post) Dir() string { func (post *Post) Dir() string {
return path.Join(post.user.Dir(), "public", post.id) return path.Join(post.user.Dir(), "public", post.id)
} }
func (post Post) WebmentionsFile() string { func (post *Post) WebmentionsFile() string {
return path.Join(post.Dir(), "webmentions.yml") return path.Join(post.Dir(), "webmentions.yml")
} }
func (post Post) MediaDir() string { func (post *Post) MediaDir() string {
return path.Join(post.Dir(), "media") return path.Join(post.Dir(), "media")
} }
func (post Post) UrlPath() string { func (post *Post) UrlPath() string {
return post.user.UrlPath() + "posts/" + post.id + "/" return post.user.UrlPath() + "posts/" + post.id + "/"
} }
func (post Post) FullUrl() string { func (post *Post) FullUrl() string {
return post.user.FullUrl() + "posts/" + post.id + "/" return post.user.FullUrl() + "posts/" + post.id + "/"
} }
func (post Post) UrlMediaPath(filename string) string { func (post *Post) UrlMediaPath(filename string) string {
return post.UrlPath() + "media/" + filename return post.UrlPath() + "media/" + filename
} }
func (post Post) Title() string { func (post *Post) Title() string {
return post.title return post.title
} }
func (post Post) ContentFile() string { func (post *Post) ContentFile() string {
return path.Join(post.Dir(), "index.md") return path.Join(post.Dir(), "index.md")
} }
@ -80,13 +82,13 @@ func (post *Post) Meta() PostMeta {
return post.meta return post.meta
} }
func (post Post) Content() []byte { func (post *Post) Content() []byte {
// read file // read file
data, _ := ioutil.ReadFile(post.ContentFile()) data, _ := ioutil.ReadFile(post.ContentFile())
return data return data
} }
func (post Post) Webmentions() PostWebmetions { func (post *Post) Webmentions() PostWebmetions {
// read status file // read status file
// return parsed webmentions // return parsed webmentions
fileName := post.WebmentionsFile() fileName := post.WebmentionsFile()
@ -108,7 +110,7 @@ func (post Post) Webmentions() PostWebmetions {
return webmentions return webmentions
} }
func (post Post) PersistWebmentions(webmentions PostWebmetions) error { func (post *Post) PersistIncomingWebmentions(webmentions PostWebmetions) error {
data, err := yaml.Marshal(webmentions) data, err := yaml.Marshal(webmentions)
if err != nil { if err != nil {
return err return err
@ -122,7 +124,7 @@ func (post Post) PersistWebmentions(webmentions PostWebmetions) error {
return nil return nil
} }
func (post Post) RenderedContent() bytes.Buffer { func (post *Post) RenderedContent() bytes.Buffer {
data := post.Content() data := post.Content()
// trim yaml block // trim yaml block
@ -162,7 +164,7 @@ func (post Post) RenderedContent() bytes.Buffer {
} }
func (post Post) Aliases() []string { func (post *Post) Aliases() []string {
return post.Meta().Aliases return post.Meta().Aliases
} }
@ -207,10 +209,10 @@ func (post *Post) PersistIncomingWebmention(webmention WebmentionIn) error {
wms.Incoming = append(wms.Incoming, webmention) wms.Incoming = append(wms.Incoming, webmention)
} }
return post.PersistWebmentions(wms) return post.PersistIncomingWebmentions(wms)
} }
func (post *Post) Webmention(source string) (WebmentionIn, error) { func (post *Post) getIncomingWebmention(source string) (WebmentionIn, error) {
wms := post.Webmentions() wms := post.Webmentions()
for _, wm := range wms.Incoming { for _, wm := range wms.Incoming {
if wm.Source == source { if wm.Source == source {
@ -221,20 +223,25 @@ func (post *Post) Webmention(source string) (WebmentionIn, error) {
} }
func (post *Post) AddIncomingWebmention(source string) error { func (post *Post) AddIncomingWebmention(source string) error {
post.wmLock.Lock()
defer post.wmLock.Unlock()
// Check if file already exists // Check if file already exists
_, err := post.Webmention(source) _, err := post.getIncomingWebmention(source)
if err != nil { if err != nil {
wms := post.Webmentions() wms := post.Webmentions()
wms.Incoming = append(wms.Incoming, WebmentionIn{ wms.Incoming = append(wms.Incoming, WebmentionIn{
Source: source, Source: source,
}) })
defer post.EnrichWebmention(source) defer func() {
return post.PersistWebmentions(wms) go post.EnrichWebmention(source)
}()
return post.PersistIncomingWebmentions(wms)
} }
return nil return nil
} }
func (post *Post) AddOutgoingWebmention(target string) error { func (post *Post) addOutgoingWebmention(target string) error {
wms := post.Webmentions() wms := post.Webmentions()
// Check if file already exists // Check if file already exists
@ -248,7 +255,7 @@ func (post *Post) AddOutgoingWebmention(target string) error {
Target: target, Target: target,
} }
wms.Outgoing = append(wms.Outgoing, webmention) wms.Outgoing = append(wms.Outgoing, webmention)
return post.PersistWebmentions(wms) return post.PersistIncomingWebmentions(wms)
} }
func (post *Post) UpdateOutgoingWebmention(webmention *WebmentionOut) error { func (post *Post) UpdateOutgoingWebmention(webmention *WebmentionOut) error {
@ -268,13 +275,16 @@ func (post *Post) UpdateOutgoingWebmention(webmention *WebmentionOut) error {
wms.Outgoing = append(wms.Outgoing, *webmention) wms.Outgoing = append(wms.Outgoing, *webmention)
} }
return post.PersistWebmentions(wms) return post.PersistIncomingWebmentions(wms)
} }
func (post *Post) EnrichWebmention(source string) error { func (post *Post) EnrichWebmention(source string) error {
post.wmLock.Lock()
defer post.wmLock.Unlock()
resp, err := post.user.repo.HttpClient.Get(source) resp, err := post.user.repo.HttpClient.Get(source)
if err == nil { if err == nil {
webmention, err := post.Webmention(source) webmention, err := post.getIncomingWebmention(source)
if err != nil { if err != nil {
return err return err
} }
@ -317,14 +327,17 @@ func (post *Post) ScanForLinks() error {
// this could be done in markdown parsing, but I don't want to // this could be done in markdown parsing, but I don't want to
// rely on goldmark for this (yet) // rely on goldmark for this (yet)
postHtml := post.RenderedContent() postHtml := post.RenderedContent()
links, _ := post.user.repo.Parser.ParseLinksFromString(string(postHtml.Bytes())) links, _ := post.user.repo.Parser.ParseLinksFromString(postHtml.String())
for _, link := range links { for _, link := range links {
post.AddOutgoingWebmention(link) post.addOutgoingWebmention(link)
} }
return nil return nil
} }
func (post *Post) SendWebmention(webmention WebmentionOut) error { func (post *Post) SendWebmention(webmention WebmentionOut) error {
post.wmLock.Lock()
defer post.wmLock.Unlock()
defer post.UpdateOutgoingWebmention(&webmention) defer post.UpdateOutgoingWebmention(&webmention)
webmention.ScannedAt = time.Now() webmention.ScannedAt = time.Now()

View File

@ -4,7 +4,9 @@ import (
"h4kor/owl-blogs" "h4kor/owl-blogs"
"os" "os"
"path" "path"
"strconv"
"strings" "strings"
"sync"
"testing" "testing"
"time" "time"
) )
@ -230,7 +232,7 @@ func TestAddIncomingWebmentionNotOverwritingWebmention(t *testing.T) {
} }
} }
func TestAddIncomingWebmentionAddsParsedTitle(t *testing.T) { func TestEnrichAddsTitle(t *testing.T) {
repo := getTestRepo(owl.RepoConfig{}) repo := getTestRepo(owl.RepoConfig{})
repo.HttpClient = &MockHttpClient{} repo.HttpClient = &MockHttpClient{}
repo.Parser = &MockHtmlParser{} repo.Parser = &MockHtmlParser{}
@ -238,6 +240,7 @@ func TestAddIncomingWebmentionAddsParsedTitle(t *testing.T) {
post, _ := user.CreateNewPost("testpost") post, _ := user.CreateNewPost("testpost")
post.AddIncomingWebmention("https://example.com") post.AddIncomingWebmention("https://example.com")
post.EnrichWebmention("https://example.com")
mentions := post.IncomingWebmentions() mentions := post.IncomingWebmentions()
if len(mentions) != 1 { if len(mentions) != 1 {
@ -371,3 +374,101 @@ func TestCanSendWebmention(t *testing.T) {
t.Errorf("Expected LastSentAt to be set") t.Errorf("Expected LastSentAt to be set")
} }
} }
func TestSendingMultipleWebmentions(t *testing.T) {
repo := getTestRepo(owl.RepoConfig{})
repo.HttpClient = &MockHttpClient{}
repo.Parser = &MockHtmlParser{}
user, _ := repo.CreateUser("testuser")
post, _ := user.CreateNewPost("testpost")
wg := sync.WaitGroup{}
wg.Add(20)
for i := 0; i < 20; i++ {
go func(k int) {
webmention := owl.WebmentionOut{
Target: "http://example.com" + strconv.Itoa(k),
}
post.SendWebmention(webmention)
wg.Done()
}(i)
}
wg.Wait()
webmentions := post.OutgoingWebmentions()
if len(webmentions) != 20 {
t.Errorf("Expected 20 webmentions, got %d", len(webmentions))
}
}
func TestReceivingMultipleWebmentions(t *testing.T) {
repo := getTestRepo(owl.RepoConfig{})
repo.HttpClient = &MockHttpClient{}
repo.Parser = &MockHtmlParser{}
user, _ := repo.CreateUser("testuser")
post, _ := user.CreateNewPost("testpost")
wg := sync.WaitGroup{}
wg.Add(20)
for i := 0; i < 20; i++ {
go func(k int) {
post.AddIncomingWebmention("http://example.com" + strconv.Itoa(k))
wg.Done()
}(i)
}
wg.Wait()
webmentions := post.IncomingWebmentions()
if len(webmentions) != 20 {
t.Errorf("Expected 20 webmentions, got %d", len(webmentions))
}
}
func TestSendingAndReceivingMultipleWebmentions(t *testing.T) {
repo := getTestRepo(owl.RepoConfig{})
repo.HttpClient = &MockHttpClient{}
repo.Parser = &MockHtmlParser{}
user, _ := repo.CreateUser("testuser")
post, _ := user.CreateNewPost("testpost")
wg := sync.WaitGroup{}
wg.Add(40)
for i := 0; i < 20; i++ {
go func(k int) {
post.AddIncomingWebmention("http://example.com" + strconv.Itoa(k))
wg.Done()
}(i)
}
for i := 0; i < 20; i++ {
go func(k int) {
webmention := owl.WebmentionOut{
Target: "http://example.com" + strconv.Itoa(k),
}
post.SendWebmention(webmention)
wg.Done()
}(i)
}
wg.Wait()
ins := post.IncomingWebmentions()
if len(ins) != 20 {
t.Errorf("Expected 20 webmentions, got %d", len(ins))
}
outs := post.OutgoingWebmentions()
if len(outs) != 20 {
t.Errorf("Expected 20 webmentions, got %d", len(outs))
}
}