miniflux-v2/reader/atom/atom.go

215 lines
4.4 KiB
Go
Raw Normal View History

2017-11-20 06:10:04 +01:00
// Copyright 2017 Frédéric Guillot. All rights reserved.
// Use of this source code is governed by the Apache 2.0
// license that can be found in the LICENSE file.
2018-08-25 06:51:50 +02:00
package atom // import "miniflux.app/reader/atom"
2017-11-20 06:10:04 +01:00
import (
"encoding/xml"
"strconv"
"strings"
"time"
2018-08-25 06:51:50 +02:00
"miniflux.app/crypto"
"miniflux.app/logger"
"miniflux.app/model"
"miniflux.app/reader/date"
"miniflux.app/reader/sanitizer"
"miniflux.app/url"
2017-11-20 06:10:04 +01:00
)
type atomFeed struct {
2017-11-20 06:10:04 +01:00
XMLName xml.Name `xml:"http://www.w3.org/2005/Atom feed"`
ID string `xml:"id"`
Title string `xml:"title"`
Author atomAuthor `xml:"author"`
Links []atomLink `xml:"link"`
Entries []atomEntry `xml:"entry"`
2017-11-20 06:10:04 +01:00
}
type atomEntry struct {
ID string `xml:"id"`
2018-02-17 21:21:58 +01:00
Title atomContent `xml:"title"`
Published string `xml:"published"`
Updated string `xml:"updated"`
Links []atomLink `xml:"link"`
Summary string `xml:"summary"`
Content atomContent `xml:"content"`
MediaGroup atomMediaGroup `xml:"http://search.yahoo.com/mrss/ group"`
Author atomAuthor `xml:"author"`
2017-11-20 06:10:04 +01:00
}
type atomAuthor struct {
2017-11-20 06:10:04 +01:00
Name string `xml:"name"`
Email string `xml:"email"`
}
type atomLink struct {
URL string `xml:"href,attr"`
2017-11-20 06:10:04 +01:00
Type string `xml:"type,attr"`
Rel string `xml:"rel,attr"`
Length string `xml:"length,attr"`
}
type atomContent struct {
2017-11-20 06:10:04 +01:00
Type string `xml:"type,attr"`
Data string `xml:",chardata"`
XML string `xml:",innerxml"`
2017-11-20 06:10:04 +01:00
}
type atomMediaGroup struct {
2017-11-20 06:10:04 +01:00
Description string `xml:"http://search.yahoo.com/mrss/ description"`
}
func (a *atomFeed) Transform() *model.Feed {
2017-11-20 06:10:04 +01:00
feed := new(model.Feed)
feed.FeedURL = getRelationURL(a.Links, "self")
feed.SiteURL = getURL(a.Links)
2017-11-22 23:52:31 +01:00
feed.Title = strings.TrimSpace(a.Title)
2017-11-20 06:10:04 +01:00
if feed.Title == "" {
feed.Title = feed.SiteURL
}
for _, entry := range a.Entries {
item := entry.Transform()
2017-12-14 05:16:15 +01:00
entryURL, err := url.AbsoluteURL(feed.SiteURL, item.URL)
if err == nil {
item.URL = entryURL
}
2017-11-20 06:10:04 +01:00
if item.Author == "" {
item.Author = getAuthor(a.Author)
2017-11-20 06:10:04 +01:00
}
2017-12-14 05:16:15 +01:00
if item.Title == "" {
item.Title = item.URL
}
2017-11-20 06:10:04 +01:00
feed.Entries = append(feed.Entries, item)
}
return feed
}
func (a *atomEntry) Transform() *model.Entry {
entry := new(model.Entry)
entry.URL = getURL(a.Links)
entry.Date = getDate(a)
2017-11-22 23:52:31 +01:00
entry.Author = getAuthor(a.Author)
entry.Hash = getHash(a)
entry.Content = getContent(a)
2018-02-17 21:21:58 +01:00
entry.Title = getTitle(a)
entry.Enclosures = getEnclosures(a)
return entry
2017-11-20 06:10:04 +01:00
}
func getURL(links []atomLink) string {
for _, link := range links {
2017-11-20 06:10:04 +01:00
if strings.ToLower(link.Rel) == "alternate" {
2017-11-22 23:52:31 +01:00
return strings.TrimSpace(link.URL)
2017-11-20 06:10:04 +01:00
}
if link.Rel == "" && link.Type == "" {
2017-11-22 23:52:31 +01:00
return strings.TrimSpace(link.URL)
2017-11-20 06:10:04 +01:00
}
}
return ""
}
func getRelationURL(links []atomLink, relation string) string {
for _, link := range links {
if strings.ToLower(link.Rel) == relation {
2017-11-22 23:52:31 +01:00
return strings.TrimSpace(link.URL)
}
}
return ""
2017-11-20 06:10:04 +01:00
}
func getDate(a *atomEntry) time.Time {
dateText := a.Updated
if dateText == "" {
dateText = a.Published
}
if dateText != "" {
result, err := date.Parse(dateText)
if err != nil {
2017-12-16 03:55:57 +01:00
logger.Error("atom: %v", err)
return time.Now()
2017-11-20 06:10:04 +01:00
}
return result
2017-11-20 06:10:04 +01:00
}
return time.Now()
2017-11-20 06:10:04 +01:00
}
func getContent(a *atomEntry) string {
if a.Content.Type == "html" || a.Content.Type == "text" {
return a.Content.Data
2017-11-20 06:10:04 +01:00
}
if a.Content.Type == "xhtml" {
return a.Content.XML
2017-11-20 06:10:04 +01:00
}
if a.Summary != "" {
return a.Summary
2017-11-20 06:10:04 +01:00
}
if a.MediaGroup.Description != "" {
return a.MediaGroup.Description
2017-11-20 06:10:04 +01:00
}
return ""
}
2018-02-17 21:21:58 +01:00
func getTitle(a *atomEntry) string {
title := ""
if a.Title.Type == "xhtml" {
title = a.Title.XML
} else {
title = a.Title.Data
}
return strings.TrimSpace(sanitizer.StripTags(title))
}
func getHash(a *atomEntry) string {
for _, value := range []string{a.ID, getURL(a.Links)} {
if value != "" {
2018-01-03 04:15:08 +01:00
return crypto.Hash(value)
2017-11-20 06:10:04 +01:00
}
}
return ""
2017-11-20 06:10:04 +01:00
}
func getEnclosures(a *atomEntry) model.EnclosureList {
enclosures := make(model.EnclosureList, 0)
2017-11-20 06:10:04 +01:00
for _, link := range a.Links {
if strings.ToLower(link.Rel) == "enclosure" {
2018-03-15 04:09:06 +01:00
length, _ := strconv.ParseInt(link.Length, 10, 0)
enclosures = append(enclosures, &model.Enclosure{URL: link.URL, MimeType: link.Type, Size: length})
}
2017-11-20 06:10:04 +01:00
}
return enclosures
2017-11-20 06:10:04 +01:00
}
func getAuthor(author atomAuthor) string {
2017-11-20 06:10:04 +01:00
if author.Name != "" {
2017-11-22 23:52:31 +01:00
return strings.TrimSpace(author.Name)
2017-11-20 06:10:04 +01:00
}
if author.Email != "" {
2017-11-22 23:52:31 +01:00
return strings.TrimSpace(author.Email)
2017-11-20 06:10:04 +01:00
}
return ""
}