atom/link.go

101 lines
2.5 KiB
Go
Raw Normal View History

2024-10-16 21:28:04 +02:00
package atom
2024-10-13 17:19:40 +02:00
import (
"errors"
"fmt"
"strings"
2024-10-13 17:19:40 +02:00
)
type Link struct {
*CommonAttributes
Title string `xml:"title,attr,omitempty"`
2024-10-17 18:15:41 +02:00
Content string `xml:",chardata"` // undefinedContent in RFC4287
Href IRI `xml:"href,attr"`
2024-10-13 17:19:40 +02:00
Rel string `xml:"rel,attr,omitempty"`
Type MediaType `xml:"type,attr,omitempty"`
HrefLang LanguageTag `xml:"hreflang,attr,omitempty"`
Length uint `xml:"length,attr,omitempty"`
}
2024-10-16 19:59:28 +02:00
// NewLink creates a new Link. It returns a *Link and an error.
func NewLink(href, content string) (*Link, error) {
2024-10-17 18:40:52 +02:00
if content != "" {
if !isValidXML(content) {
return nil, fmt.Errorf("%v not valid XML", content)
}
}
return &Link{Href: IRI(href), Content: content}, nil
2024-10-16 17:17:41 +02:00
}
2024-10-16 19:59:28 +02:00
// Check checks the Link for incompatibilities with RFC4287. It returns an
// error.
2024-10-13 17:19:40 +02:00
func (l *Link) Check() error {
if l.Href == "" {
return errors.New("href attribute of link empty")
} else {
if !isValidIRI(l.Href) {
return fmt.Errorf("href attribute %v of link not correctly formatted", l.Href)
}
2024-10-13 17:19:40 +02:00
}
if l.Rel != "" {
if strings.Contains(l.Rel, ":") && !isValidIRI(IRI(l.Rel)) {
return fmt.Errorf("rel attribute %v of link %v not correctly formatted", l.Rel, l.Href)
}
}
if l.Type != "" {
if !isValidMediaType(string(l.Type)) {
return fmt.Errorf("type attribute %v of link %v invalid media type", l.Type, l.Href)
}
}
if l.HrefLang != "" {
if !isValidLanguageTag(l.HrefLang) {
return fmt.Errorf("hreflang attribute %v of link %v invalid language tag", l.Type, l.HrefLang)
}
2024-10-13 17:19:40 +02:00
}
if l.Content != "" {
if !isValidXML(l.Content) {
return fmt.Errorf("content element %v of link not valid XML", l.Content)
}
2024-10-13 17:19:40 +02:00
}
return nil
}
2024-10-16 19:59:28 +02:00
// hasAlternateDuplicateLinks checks whether multiple Links with Rel
// "alternate" also have Type and HrefLang in common. It returns a bool.
// atom:feed/entry elements MUST NOT contain more than one atom:link element
// with a rel attribute value of "alternate" that has the same combination of
// type and hreflang attribute values.
func hasAlternateDuplicateLinks(l []*Link) bool {
linkMap := make(map[string]bool)
for _, link := range l {
if link.Rel == "alternate" {
key := fmt.Sprint(link.Type, "|", link.HrefLang)
if linkMap[key] {
return true
}
linkMap[key] = true
}
}
return false
}
2024-10-16 19:59:28 +02:00
// alternateRelExists checks whether multiple Links with Rel "alternate" exist.
// It returns a bool.
func alternateRelExists(l []*Link) bool {
for _, link := range l {
if link.Rel == "alternate" {
return true
}
}
return false
}