WIP adding guards to webmention
This commit is contained in:
parent
a8998068ad
commit
d66c1a6817
63
post.go
63
post.go
|
@ -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()
|
||||||
|
|
||||||
|
|
103
post_test.go
103
post_test.go
|
@ -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))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue