Compare commits
5 Commits
b76e529ca3
...
f11dee630d
Author | SHA1 | Date | |
---|---|---|---|
f11dee630d | |||
f8c36a7045 | |||
4fe133a394 | |||
39bd0776c5 | |||
082e71a698 |
47
atom.go
47
atom.go
@ -2,30 +2,41 @@ package atomfeed
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"mime"
|
"mime"
|
||||||
"net/url"
|
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"golang.org/x/text/language"
|
||||||
)
|
)
|
||||||
|
|
||||||
type (
|
type (
|
||||||
EmailAddress string
|
EmailAddress string
|
||||||
LanguageTag string
|
LanguageTag string
|
||||||
MediaType string
|
MediaType string
|
||||||
URI string
|
// URI string
|
||||||
|
IRI string
|
||||||
)
|
)
|
||||||
|
|
||||||
func isValidURL(uri URI) bool {
|
// func isValidURL(uri URI) bool {
|
||||||
_, err := url.ParseRequestURI(string(uri))
|
// _, err := url.ParseRequestURI(string(uri))
|
||||||
return err == nil
|
// return err == nil
|
||||||
}
|
// }
|
||||||
|
//
|
||||||
|
// func isValidURN(uri URI) bool {
|
||||||
|
// pattern := `\A(?i:urn:(?!urn:)(?<nid>[a-z0-9][a-z0-9-]{1,31}):(?<nss>(?:[-a-z0-9()+,.:=@;$_!*'&~\/]|%[0-9a-f]{2})+)(?:\?\+(?<rcomponent>.*?))?(?:\?=(?<qcomponent>.*?))?(?:#(?<fcomponent>.*?))?)\z`
|
||||||
|
// return regexp.MustCompile(pattern).MatchString(string(uri))
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // isValidURI checks whether an URI is valid or not.
|
||||||
|
// func isValidURI(uri URI) bool {
|
||||||
|
// return isValidURL(uri) || isValidURN(uri)
|
||||||
|
// }
|
||||||
|
|
||||||
func isValidURN(uri URI) bool {
|
// isValidIRI checks whether an IRI is valid or not.
|
||||||
pattern := `\A(?i:urn:(?!urn:)(?<nid>[a-z0-9][a-z0-9-]{1,31}):(?<nss>(?:[-a-z0-9()+,.:=@;$_!*'&~\/]|%[0-9a-f]{2})+)(?:\?\+(?<rcomponent>.*?))?(?:\?=(?<qcomponent>.*?))?(?:#(?<fcomponent>.*?))?)\z`
|
// The used pattern stems from
|
||||||
return regexp.MustCompile(pattern).MatchString(string(uri))
|
// https://www.w3.org/2011/04/XMLSchema/TypeLibrary-IRI-RFC3987.xsd
|
||||||
}
|
func isValidIRI(iri IRI) bool {
|
||||||
|
pattern := `((([A-Za-z])[A-Za-z0-9+\-\.]*):((//(((([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------!$&'()*+,;=:]|(%[0-9A-Fa-f][0-9A-Fa-f]))*@))?((\[((((([0-9A-Fa-f]{0,4}:)){6}(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|(::(([0-9A-Fa-f]{0,4}:)){5}(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|(([0-9A-Fa-f]{0,4})?::(([0-9A-Fa-f]{0,4}:)){4}(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|((((([0-9A-Fa-f]{0,4}:))?[0-9A-Fa-f]{0,4}))?::(([0-9A-Fa-f]{0,4}:)){3}(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|((((([0-9A-Fa-f]{0,4}:)){0,2}[0-9A-Fa-f]{0,4}))?::(([0-9A-Fa-f]{0,4}:)){2}(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|((((([0-9A-Fa-f]{0,4}:)){0,3}[0-9A-Fa-f]{0,4}))?::[0-9A-Fa-f]{0,4}:(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|((((([0-9A-Fa-f]{0,4}:)){0,4}[0-9A-Fa-f]{0,4}))?::(([0-9A-Fa-f]{0,4}:[0-9A-Fa-f]{0,4})|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))))|((((([0-9A-Fa-f]{0,4}:)){0,5}[0-9A-Fa-f]{0,4}))?::[0-9A-Fa-f]{0,4})|((((([0-9A-Fa-f]{0,4}:)){0,6}[0-9A-Fa-f]{0,4}))?::))|(v[0-9A-Fa-f]+\.[A-Za-z0-9\-\._~!$&'()*+,;=:]+))\])|(([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5]))\.([0-9]|([1-9][0-9])|(1([0-9]){2})|(2[0-4][0-9])|(25[0-5])))|(([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=]))*)((:[0-9]*))?)((/(([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@]))*))*)|(/(((([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@]))+((/(([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@]))*))*))?)|((([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@]))+((/(([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@]))*))*)|)((\?(([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@])|[---/?])*))?((#((([A-Za-z0-9\-\._~ -豈-﷏ﷰ-𐀀-𠀀-𰀀------------]|(%[0-9A-Fa-f][0-9A-Fa-f])|[!$&'()*+,;=:@])|/|\?))*))?)`
|
||||||
func isValidURI(uri URI) bool {
|
return regexp.MustCompile(pattern).MatchString(string(iri))
|
||||||
return isValidURL(uri) || isValidURN(uri)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func isCorrectlyEscaped(text string) bool {
|
func isCorrectlyEscaped(text string) bool {
|
||||||
@ -57,3 +68,13 @@ func isXMLMediaType(mediaType string) bool {
|
|||||||
|
|
||||||
return strings.HasSuffix(mediaType, "/xml") || strings.HasSuffix(mediaType, "+xml")
|
return strings.HasSuffix(mediaType, "/xml") || strings.HasSuffix(mediaType, "+xml")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func isValidMediaType(mediaType string) bool {
|
||||||
|
_, _, err := mime.ParseMediaType(mediaType)
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func isValidLanguageTag(tag LanguageTag) bool {
|
||||||
|
_, err := language.Parse(string(tag))
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
@ -10,7 +10,7 @@ type Category struct {
|
|||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
Content Content `xml:"content"` // undefinedContent in RFC4287
|
Content Content `xml:"content"` // undefinedContent in RFC4287
|
||||||
Term string `xml:"term,attr"`
|
Term string `xml:"term,attr"`
|
||||||
Scheme URI `xml:"scheme,attr,omitempty"`
|
Scheme IRI `xml:"scheme,attr,omitempty"`
|
||||||
Label string `xml:"label,attr,omitempty"`
|
Label string `xml:"label,attr,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -33,7 +33,7 @@ func (c *Category) Check() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if c.Scheme != "" {
|
if c.Scheme != "" {
|
||||||
if !isValidURI(c.Scheme) {
|
if !isValidIRI(c.Scheme) {
|
||||||
return fmt.Errorf("scheme attribute %v of category not correctly formatted", c.Scheme)
|
return fmt.Errorf("scheme attribute %v of category not correctly formatted", c.Scheme)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,7 @@ package atomfeed
|
|||||||
import "fmt"
|
import "fmt"
|
||||||
|
|
||||||
type CommonAttributes struct {
|
type CommonAttributes struct {
|
||||||
Base URI `xml:"base,attr,omitempty"`
|
Base IRI `xml:"base,attr,omitempty"`
|
||||||
Lang LanguageTag `xml:"lang,attr,omitempty"`
|
Lang LanguageTag `xml:"lang,attr,omitempty"`
|
||||||
UndefinedAttributes []*ExtensionAttribute `xml:",any"`
|
UndefinedAttributes []*ExtensionAttribute `xml:",any"`
|
||||||
}
|
}
|
||||||
|
14
entry.go
14
entry.go
@ -1,10 +1,10 @@
|
|||||||
package atomfeed
|
package atomfeed
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/xml"
|
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// It is advisable that each atom:entry element contain a non-empty atom:title
|
// It is advisable that each atom:entry element contain a non-empty atom:title
|
||||||
@ -58,8 +58,16 @@ func alternateRelExists(l []*Link) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *Entry) AddExtension(name string, value any) {
|
// AddExtension adds the extension to the entry
|
||||||
e.Extensions = append(e.Extensions, &ExtensionElement{XMLName: xml.Name{Local: name}, Value: value})
|
func (e *Entry) AddExtension(x *ExtensionElement) {
|
||||||
|
if e.Extensions == nil {
|
||||||
|
e.Extensions = make([]*ExtensionElement, 1)
|
||||||
|
e.Extensions[0] = x
|
||||||
|
} else {
|
||||||
|
e.Extensions = append(e.Extensions, x)
|
||||||
|
}
|
||||||
|
|
||||||
|
e.Updated.DateTime = DateTime(time.Now())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *Entry) Check() error {
|
func (e *Entry) Check() error {
|
||||||
|
@ -8,7 +8,7 @@ import (
|
|||||||
|
|
||||||
type Generator struct {
|
type Generator struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
URI URI `xml:"uri,attr,omitempty"`
|
URI IRI `xml:"uri,attr,omitempty"`
|
||||||
Version string `xml:"version,attr,omitempty"`
|
Version string `xml:"version,attr,omitempty"`
|
||||||
Text string `xml:"text"`
|
Text string `xml:"text"`
|
||||||
}
|
}
|
||||||
@ -19,7 +19,7 @@ func NewGenerator(text string) *Generator {
|
|||||||
|
|
||||||
func (g *Generator) Check() error {
|
func (g *Generator) Check() error {
|
||||||
if g.URI != "" {
|
if g.URI != "" {
|
||||||
if !isValidURI(g.URI) {
|
if !isValidIRI(g.URI) {
|
||||||
return fmt.Errorf("uri attribute %v of generator not correctly formatted", g.URI)
|
return fmt.Errorf("uri attribute %v of generator not correctly formatted", g.URI)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
5
go.mod
5
go.mod
@ -2,4 +2,7 @@ module streifling.com/jason/atom-feed
|
|||||||
|
|
||||||
go 1.23.2
|
go 1.23.2
|
||||||
|
|
||||||
require github.com/google/uuid v1.6.0
|
require (
|
||||||
|
github.com/google/uuid v1.6.0
|
||||||
|
golang.org/x/text v0.19.0
|
||||||
|
)
|
||||||
|
2
go.sum
2
go.sum
@ -1,2 +1,4 @@
|
|||||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
|
golang.org/x/text v0.19.0 h1:kTxAhCbGbxhK0IwgSKiMO5awPoDQ0RpfiVYBfK860YM=
|
||||||
|
golang.org/x/text v0.19.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY=
|
||||||
|
6
icon.go
6
icon.go
@ -7,18 +7,18 @@ import (
|
|||||||
|
|
||||||
type Icon struct {
|
type Icon struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
URI URI `xml:"uri"`
|
URI IRI `xml:"uri"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIcon(uri string) *Icon {
|
func NewIcon(uri string) *Icon {
|
||||||
return &Icon{URI: URI(uri)}
|
return &Icon{URI: IRI(uri)}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Icon) Check() error {
|
func (i *Icon) Check() error {
|
||||||
if i.URI == "" {
|
if i.URI == "" {
|
||||||
return errors.New("uri element of icon empty")
|
return errors.New("uri element of icon empty")
|
||||||
} else {
|
} else {
|
||||||
if !isValidURI(i.URI) {
|
if !isValidIRI(i.URI) {
|
||||||
return fmt.Errorf("uri attribute %v of icon not correctly formatted", i.URI)
|
return fmt.Errorf("uri attribute %v of icon not correctly formatted", i.URI)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
6
id.go
6
id.go
@ -9,18 +9,18 @@ import (
|
|||||||
|
|
||||||
type ID struct {
|
type ID struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
URI URI `xml:"uri"`
|
URI IRI `xml:"uri"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewID() *ID {
|
func NewID() *ID {
|
||||||
return &ID{URI: URI(fmt.Sprint("urn:uuid:", uuid.New()))}
|
return &ID{URI: IRI(fmt.Sprint("urn:uuid:", uuid.New()))}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *ID) Check() error {
|
func (i *ID) Check() error {
|
||||||
if i.URI == "" {
|
if i.URI == "" {
|
||||||
return errors.New("uri element of id empty")
|
return errors.New("uri element of id empty")
|
||||||
} else {
|
} else {
|
||||||
if !isValidURI(i.URI) {
|
if !isValidIRI(i.URI) {
|
||||||
return fmt.Errorf("uri element %v of id not correctly formatted", i.URI)
|
return fmt.Errorf("uri element %v of id not correctly formatted", i.URI)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
23
link.go
23
link.go
@ -3,13 +3,14 @@ package atomfeed
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Link struct {
|
type Link struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
Title Text `xml:"title,attr,omitempty"`
|
Title string `xml:"title,attr,omitempty"`
|
||||||
Content Content `xml:"content"` // undefinedContent in RFC4287
|
Content Content `xml:"content"` // undefinedContent in RFC4287
|
||||||
Href URI `xml:"href,attr"`
|
Href IRI `xml:"href,attr"`
|
||||||
Rel string `xml:"rel,attr,omitempty"`
|
Rel string `xml:"rel,attr,omitempty"`
|
||||||
Type MediaType `xml:"type,attr,omitempty"`
|
Type MediaType `xml:"type,attr,omitempty"`
|
||||||
HrefLang LanguageTag `xml:"hreflang,attr,omitempty"`
|
HrefLang LanguageTag `xml:"hreflang,attr,omitempty"`
|
||||||
@ -22,18 +23,28 @@ func NewLink(href string) (*Link, error) {
|
|||||||
return nil, fmt.Errorf("error creating content element: %v", err)
|
return nil, fmt.Errorf("error creating content element: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return &Link{Href: URI(href), Content: content}, nil
|
return &Link{Href: IRI(href), Content: content}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *Link) Check() error {
|
func (l *Link) Check() error {
|
||||||
if l.Href == "" {
|
if l.Href == "" {
|
||||||
return errors.New("href attribute of link empty")
|
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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if l.Title != nil {
|
if strings.Contains(l.Rel, ":") || !isValidIRI(IRI(l.Rel)) {
|
||||||
if err := l.Title.Check(); err != nil {
|
return fmt.Errorf("rel attribute %v of link %v not correctly formatted", l.Rel, l.Href)
|
||||||
return fmt.Errorf("title attribute of link %v: %v", l.Href, err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if !isValidMediaType(string(l.Type)) {
|
||||||
|
return fmt.Errorf("type attribute %v of link %v invalid media type", l.Type, l.Href)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isValidLanguageTag(l.HrefLang) {
|
||||||
|
return fmt.Errorf("hreflang attribute %v of link %v invalid language tag", l.Type, l.HrefLang)
|
||||||
}
|
}
|
||||||
|
|
||||||
if l.Content == nil {
|
if l.Content == nil {
|
||||||
|
2
logo.go
2
logo.go
@ -4,7 +4,7 @@ import "errors"
|
|||||||
|
|
||||||
type Logo struct {
|
type Logo struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
URI URI `xml:"uri"`
|
URI IRI `xml:"uri"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *Logo) Check() error {
|
func (l *Logo) Check() error {
|
||||||
|
@ -10,7 +10,7 @@ import (
|
|||||||
type OutOfLineContent struct {
|
type OutOfLineContent struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
Type MediaType `xml:"type,attr,omitempty"`
|
Type MediaType `xml:"type,attr,omitempty"`
|
||||||
SRC URI `xml:"src,attr"`
|
SRC IRI `xml:"src,attr"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func newOutOfLineContent(mediaType string, content any) (*OutOfLineContent, error) {
|
func newOutOfLineContent(mediaType string, content any) (*OutOfLineContent, error) {
|
||||||
@ -22,11 +22,11 @@ func newOutOfLineContent(mediaType string, content any) (*OutOfLineContent, erro
|
|||||||
return nil, fmt.Errorf("content type %T incompatible with out of line content", content)
|
return nil, fmt.Errorf("content type %T incompatible with out of line content", content)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !isValidURI(content.(URI)) {
|
if !isValidIRI(content.(IRI)) {
|
||||||
return nil, errors.New("content not a valid uri")
|
return nil, errors.New("content not a valid uri")
|
||||||
}
|
}
|
||||||
|
|
||||||
return &OutOfLineContent{Type: MediaType(mediaType), SRC: content.(URI)}, nil
|
return &OutOfLineContent{Type: MediaType(mediaType), SRC: content.(IRI)}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (o *OutOfLineContent) isContent() bool { return true }
|
func (o *OutOfLineContent) isContent() bool { return true }
|
||||||
|
15
person.go
15
person.go
@ -1,7 +1,6 @@
|
|||||||
package atomfeed
|
package atomfeed
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/xml"
|
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/mail"
|
"net/mail"
|
||||||
@ -10,7 +9,7 @@ import (
|
|||||||
type Person struct {
|
type Person struct {
|
||||||
*CommonAttributes
|
*CommonAttributes
|
||||||
Name string `xml:"name"`
|
Name string `xml:"name"`
|
||||||
URI URI `xml:"uri,omitempty"`
|
URI IRI `xml:"uri,omitempty"`
|
||||||
Email EmailAddress `xml:"email,omitempty"`
|
Email EmailAddress `xml:"email,omitempty"`
|
||||||
Extensions []*ExtensionElement `xml:",any,omitempty"`
|
Extensions []*ExtensionElement `xml:",any,omitempty"`
|
||||||
}
|
}
|
||||||
@ -19,8 +18,14 @@ func NewPerson(name string) *Person {
|
|||||||
return &Person{Name: name}
|
return &Person{Name: name}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Person) AddExtension(name string, value any) {
|
// AddExtension adds the extension to the person
|
||||||
p.Extensions = append(p.Extensions, &ExtensionElement{XMLName: xml.Name{Local: name}, Value: value})
|
func (p *Person) AddExtension(e *ExtensionElement) {
|
||||||
|
if p.Extensions == nil {
|
||||||
|
p.Extensions = make([]*ExtensionElement, 1)
|
||||||
|
p.Extensions[0] = e
|
||||||
|
} else {
|
||||||
|
p.Extensions = append(p.Extensions, e)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Person) Check() error {
|
func (p *Person) Check() error {
|
||||||
@ -29,7 +34,7 @@ func (p *Person) Check() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if p.URI != "" {
|
if p.URI != "" {
|
||||||
if !isValidURI(p.URI) {
|
if !isValidIRI(p.URI) {
|
||||||
return fmt.Errorf("uri element of person %v not correctly formatted", p.Name)
|
return fmt.Errorf("uri element of person %v not correctly formatted", p.Name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ type PlainText struct {
|
|||||||
Text string `xml:"text"`
|
Text string `xml:"text"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PlainText) IsText() bool { return true }
|
func (p *PlainText) isText() bool { return true }
|
||||||
|
|
||||||
func (p *PlainText) Check() error {
|
func (p *PlainText) Check() error {
|
||||||
if p.Type != "" && p.Type != "text" && p.Type != "html" {
|
if p.Type != "" && p.Type != "text" && p.Type != "html" {
|
||||||
|
2
text.go
2
text.go
@ -6,8 +6,8 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Text interface {
|
type Text interface {
|
||||||
|
isText() bool
|
||||||
Check() error
|
Check() error
|
||||||
IsText() bool
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewText(textType, content string) (Text, error) {
|
func NewText(textType, content string) (Text, error) {
|
||||||
|
@ -17,7 +17,7 @@ type XHTMLText struct {
|
|||||||
XHTMLDiv XHTMLDiv
|
XHTMLDiv XHTMLDiv
|
||||||
}
|
}
|
||||||
|
|
||||||
func (x *XHTMLText) IsText() bool { return true }
|
func (x *XHTMLText) isText() bool { return true }
|
||||||
|
|
||||||
func (x *XHTMLText) Check() error {
|
func (x *XHTMLText) Check() error {
|
||||||
if x.Type != "xhtml" {
|
if x.Type != "xhtml" {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user