miniflux/reader/atom/parser.go

63 lines
1.3 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 (
2019-12-23 07:18:21 +01:00
"bytes"
"encoding/xml"
2017-11-20 06:10:04 +01:00
"io"
2018-08-25 06:51:50 +02:00
"miniflux.app/errors"
"miniflux.app/model"
2019-12-23 07:18:21 +01:00
xml_decoder "miniflux.app/reader/xml"
2017-11-20 06:10:04 +01:00
)
2019-12-23 07:18:21 +01:00
type atomFeed interface {
Transform(baseURL string) *model.Feed
2019-12-23 07:18:21 +01:00
}
2017-11-21 03:34:11 +01:00
// Parse returns a normalized feed struct from a Atom feed.
func Parse(baseURL string, r io.Reader) (*model.Feed, *errors.LocalizedError) {
2019-12-23 07:18:21 +01:00
var buf bytes.Buffer
tee := io.TeeReader(r, &buf)
var rawFeed atomFeed
if getAtomFeedVersion(tee) == "0.3" {
rawFeed = new(atom03Feed)
} else {
rawFeed = new(atom10Feed)
}
decoder := xml_decoder.NewDecoder(&buf)
err := decoder.Decode(rawFeed)
2017-11-20 06:10:04 +01:00
if err != nil {
2018-02-28 06:19:59 +01:00
return nil, errors.NewLocalizedError("Unable to parse Atom feed: %q", err)
2017-11-20 06:10:04 +01:00
}
return rawFeed.Transform(baseURL), nil
2019-12-23 07:18:21 +01:00
}
func getAtomFeedVersion(data io.Reader) string {
decoder := xml_decoder.NewDecoder(data)
for {
token, _ := decoder.Token()
if token == nil {
break
}
if element, ok := token.(xml.StartElement); ok {
if element.Name.Local == "feed" {
for _, attr := range element.Attr {
if attr.Name.Local == "version" && attr.Value == "0.3" {
return "0.3"
}
}
return "1.0"
}
}
}
return "1.0"
2017-11-20 06:10:04 +01:00
}