| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839 |
- // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
- // SPDX-License-Identifier: Apache-2.0
- package atom // import "miniflux.app/v2/internal/reader/atom"
- import (
- "bytes"
- "testing"
- "time"
- )
- func TestParseAtomSample(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <updated>2003-12-13T18:30:02Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <id>urn:uuid:60a76c80-d399-11d9-b93C-0003939e0af6</id>
- <entry>
- <title>Atom-Powered Robots Run Amok</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("http://example.org/feed.xml", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "Example Feed" {
- t.Errorf("Incorrect title, got: %s", feed.Title)
- }
- if feed.FeedURL != "http://example.org/feed.xml" {
- t.Errorf("Incorrect feed URL, got: %s", feed.FeedURL)
- }
- if feed.SiteURL != "http://example.org/" {
- t.Errorf("Incorrect site URL, got: %s", feed.SiteURL)
- }
- if feed.IconURL != "" {
- t.Errorf("Incorrect icon URL, got: %s", feed.IconURL)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if !feed.Entries[0].Date.Equal(time.Date(2003, time.December, 13, 18, 30, 2, 0, time.UTC)) {
- t.Errorf("Incorrect entry date, got: %v", feed.Entries[0].Date)
- }
- if feed.Entries[0].Hash != "3841e5cf232f5111fc5841e9eba5f4b26d95e7d7124902e0f7272729d65601a6" {
- t.Errorf("Incorrect entry hash, got: %s", feed.Entries[0].Hash)
- }
- if feed.Entries[0].URL != "http://example.org/2003/12/13/atom03" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if feed.Entries[0].CommentsURL != "" {
- t.Errorf("Incorrect entry Comments URL, got: %s", feed.Entries[0].CommentsURL)
- }
- if feed.Entries[0].Title != "Atom-Powered Robots Run Amok" {
- t.Errorf("Incorrect entry title, got: %s", feed.Entries[0].Title)
- }
- if feed.Entries[0].Content != "Some text." {
- t.Errorf("Incorrect entry content, got: %s", feed.Entries[0].Content)
- }
- if feed.Entries[0].Author != "John Doe" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseFeedWithSubtitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <subtitle>This is a subtitle</subtitle>
- <link href="http://example.org/"/>
- <updated>2003-12-13T18:30:02Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <id>urn:uuid:60a76c80-d399-11d9-b93C-0003939e0af6</id>
- </feed>`
- feed, err := Parse("http://example.org/feed.xml", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Description != "This is a subtitle" {
- t.Errorf("Incorrect description, got: %s", feed.Description)
- }
- }
- func TestParseFeedWithoutTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <link rel="alternate" type="text/html" href="https://example.org/"/>
- <link rel="self" type="application/atom+xml" href="https://example.org/feed"/>
- <updated>2003-12-13T18:30:02Z</updated>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "https://example.org/" {
- t.Errorf("Incorrect feed title, got: %s", feed.Title)
- }
- }
- func TestParseEntryWithoutTitleButWithURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <updated>2003-12-13T18:30:02Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <id>urn:uuid:60a76c80-d399-11d9-b93C-0003939e0af6</id>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != "http://example.org/2003/12/13/atom03" {
- t.Errorf("Incorrect entry title, got: %s", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithoutTitleButWithSummary(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <updated>2003-12-13T18:30:02Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <id>urn:uuid:60a76c80-d399-11d9-b93C-0003939e0af6</id>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != "Some text." {
- t.Errorf("Incorrect entry title, got: %s", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithoutTitleButWithXHTMLContent(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <updated>2003-12-13T18:30:02Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <id>urn:uuid:60a76c80-d399-11d9-b93C-0003939e0af6</id>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="xhtml">
- <div xmlns="http://www.w3.org/1999/xhtml">AT&T bought <b>by SBC</b>!</div>
- </content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != "AT&T bought by SBC!" {
- t.Errorf("Incorrect entry title, got: %s", feed.Entries[0].Title)
- }
- }
- func TestParseFeedURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link rel="alternate" type="text/html" href="https://example.org/"/>
- <link rel="self" type="application/atom+xml" href="https://example.org/feed"/>
- <updated>2003-12-13T18:30:02Z</updated>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.SiteURL != "https://example.org/" {
- t.Errorf("Incorrect site URL, got: %s", feed.SiteURL)
- }
- if feed.FeedURL != "https://example.org/feed" {
- t.Errorf("Incorrect feed URL, got: %s", feed.FeedURL)
- }
- }
- func TestParseFeedWithRelativeFeedURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link rel="alternate" type="text/html" href="https://example.org/"/>
- <link rel="self" type="application/atom+xml" href="/feed"/>
- <updated>2003-12-13T18:30:02Z</updated>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.FeedURL != "https://example.org/feed" {
- t.Errorf("Incorrect feed URL, got: %s", feed.FeedURL)
- }
- }
- func TestParseFeedWithRelativeSiteURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="/blog/atom.xml" rel="self" type="application/atom+xml"/>
- <link href="/blog "/>
- <entry>
- <title>Test</title>
- <link href="/blog/article.html"/>
- <link href="/blog/article.html" rel="alternate" type="text/html"/>
- <id>/blog/article.html</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.FeedURL != "https://example.org/blog/atom.xml" {
- t.Errorf("Incorrect feed URL, got: %q", feed.FeedURL)
- }
- if feed.SiteURL != "https://example.org/blog" {
- t.Errorf("Incorrect site URL, got: %q", feed.SiteURL)
- }
- if feed.Entries[0].URL != "https://example.org/blog/article.html" {
- t.Errorf("Incorrect entry URL, got: %q", feed.Entries[0].URL)
- }
- }
- func TestParseFeedSiteURLWithTrailingSpace(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <link href="http://example.org "/>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.SiteURL != "http://example.org" {
- t.Errorf("Incorrect site URL, got: %q", feed.SiteURL)
- }
- }
- func TestParseFeedWithFeedURLWithTrailingSpace(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <link href="/blog/atom.xml " rel="self" type="application/atom+xml"/>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.FeedURL != "https://example.org/blog/atom.xml" {
- t.Errorf("Incorrect site URL, got: %q", feed.FeedURL)
- }
- }
- func TestParseEntryWithRelativeURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Test</title>
- <link href="something.html"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.net/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].URL != "http://example.org/something.html" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- }
- func TestParseEntryURLWithTextHTMLType(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Test</title>
- <link href="http://example.org/something.html" type="text/html"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.net/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].URL != "http://example.org/something.html" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- }
- func TestParseEntryURLWithNoRelAndNoType(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Test</title>
- <link href="http://example.org/something.html"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.net/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].URL != "http://example.org/something.html" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- }
- func TestParseEntryURLWithAlternateRel(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Test</title>
- <link href="http://example.org/something.html" rel="alternate"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.net/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].URL != "http://example.org/something.html" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- }
- func TestParseEntryTitleWithWhitespaces(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>
- Some Title
- </title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != "Some Title" {
- t.Errorf("Incorrect entry title, got: %s", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithPlainTextTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="text">AT&T bought by SBC!</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- <entry>
- <title>AT&T bought by SBC!</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- expected := `AT&T bought by SBC!`
- for i := range 2 {
- if feed.Entries[i].Title != expected {
- t.Errorf("Incorrect title for entry #%d, got: %q instead of %q", i, feed.Entries[i].Title, expected)
- }
- }
- }
- func TestParseEntryWithHTMLTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="html"><code>Code</code> Test</title>
- <link href="http://example.org/z"/>
- </entry>
- <entry>
- <title type="html"><![CDATA[Test with “unicode quote”]]></title>
- <link href="http://example.org/b"/>
- </entry>
- <entry>
- <title>
- <![CDATA[Entry title with space around CDATA]]>
- </title>
- <link href="http://example.org/c"/>
- </entry>
- <entry>
- <title type="html"><![CDATA[Test with self-closing <tag>]]></title>
- <link href="http://example.org/d"/>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 4 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].Title != "<code>Code</code> Test" {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- if feed.Entries[1].Title != "Test with “unicode quote”" {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[1].Title)
- }
- if feed.Entries[2].Title != "Entry title with space around CDATA" {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[2].Title)
- }
- if feed.Entries[3].Title != "Test with self-closing <tag>" {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[3].Title)
- }
- }
- func TestParseEntryWithXHTMLTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="xhtml">
- <div xmlns="http://www.w3.org/1999/xhtml">
- This is <b>XHTML</b> content.
- </div>
- </title>
- <link href="http://example.org/b"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != `This is <b>XHTML</b> content.` {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithEmptyXHTMLTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="xhtml">
- <div xmlns="http://www.w3.org/1999/xhtml"/>
- </title>
- <link href="http://example.org/entry"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != `http://example.org/entry` {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithXHTMLTitleWithoutDiv(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="xhtml">
- test
- </title>
- <link href="http://example.org/entry"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != `test` {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithNumericCharacterReferenceTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Σ ß</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != "Σ ß" {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithDoubleEncodedEntitiesTitle(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>&#39;AT&amp;T&#39;</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Title != `'AT&T'` {
- t.Errorf("Incorrect entry title, got: %q", feed.Entries[0].Title)
- }
- }
- func TestParseEntryWithXHTMLSummary(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="xhtml">Example</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary type="xhtml"><div xmlns="http://www.w3.org/1999/xhtml"><p>Test: <code>std::unique_ptr<S></code></p></div></summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Content != `<p>Test: <code>std::unique_ptr<S></code></p>` {
- t.Errorf("Incorrect entry content, got: %s", feed.Entries[1].Content)
- }
- }
- func TestParseEntryWithHTMLSummary(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="html">Example 1</title>
- <link href="http://example.org/1"/>
- <summary type="html"><code>std::unique_ptr&lt;S&gt; myvar;</code></summary>
- </entry>
- <entry>
- <title type="html">Example 2</title>
- <link href="http://example.org/2"/>
- <summary type="text/html"><code>std::unique_ptr&lt;S&gt; myvar;</code></summary>
- </entry>
- <entry>
- <title type="html">Example 3</title>
- <link href="http://example.org/3"/>
- <summary type="html"><![CDATA[<code>std::unique_ptr<S> myvar;</code>]]></summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 3 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- expected := `<code>std::unique_ptr<S> myvar;</code>`
- for i := range 3 {
- if feed.Entries[i].Content != expected {
- t.Errorf("Incorrect content for entry #%d, got: %q", i, feed.Entries[i].Content)
- }
- }
- }
- func TestParseEntryWithTextSummary(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/a"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>AT&T <S></summary>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/b"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary type="text">AT&T <S></summary>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/c"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary type="text/plain">AT&T <S></summary>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/d"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary type="text"><![CDATA[AT&T <S>]]></summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- expected := `AT&T <S>`
- for i := range 4 {
- if feed.Entries[i].Content != expected {
- t.Errorf("Incorrect content for entry #%d, got: %q", i, feed.Entries[i].Content)
- }
- }
- }
- func TestParseEntryWithTextContent(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/a"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content>AT&T <strong>Strong Element</strong></content>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/b"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="text">AT&T <strong>Strong Element</strong></content>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/c"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="text/plain">AT&T <strong>Strong Element</strong></content>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/d"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content><![CDATA[AT&T <strong>Strong Element</strong>]]></content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- expected := `AT&T <strong>Strong Element</strong>`
- for i := range 4 {
- if feed.Entries[i].Content != expected {
- t.Errorf("Incorrect content for entry #%d, got: %q instead of %q", i, feed.Entries[i].Content, expected)
- }
- }
- }
- func TestParseEntryWithHTMLContent(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/a"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="html">AT&amp;T bought <b>by SBC</b>!</content>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/b"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="text/html">AT&amp;T bought <b>by SBC</b>!</content>
- </entry>
- <entry>
- <title type="html">Example</title>
- <link href="http://example.org/c"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="html"><![CDATA[AT&T bought <b>by SBC</b>!]]></content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- expected := `AT&T bought <b>by SBC</b>!`
- for i := range 3 {
- if feed.Entries[i].Content != expected {
- t.Errorf("Incorrect content for entry #%d, got: %q", i, feed.Entries[i].Content)
- }
- }
- }
- func TestParseEntryWithXHTMLContent(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <title>Example</title>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <content type="xhtml">
- <div xmlns="http://www.w3.org/1999/xhtml">AT&T bought <b>by SBC</b>!</div>
- </content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Content != `AT&T bought <b>by SBC</b>!` {
- t.Errorf("Incorrect entry content, got: %q", feed.Entries[0].Content)
- }
- }
- func TestParseEntryWithAuthorName(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- <author>
- <name>Me</name>
- <email>me@localhost</email>
- </author>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "Me" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseEntryWithoutAuthorName(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- <author>
- <name/>
- <email>me@localhost</email>
- </author>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "me@localhost" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseEntryWithMultipleAuthors(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- <author>
- <name>Alice</name>
- </author>
- <author>
- <name>Bob</name>
- </author>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "Alice, Bob" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseFeedWithEntryWithoutAuthor(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <author>
- <name>John Doe</name>
- </author>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "John Doe" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseFeedWithMultipleAuthors(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <author>
- <name>Alice</name>
- </author>
- <author>
- <name>Bob</name>
- </author>
- <author>
- <name>Bob</name>
- </author>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "Alice, Bob" {
- t.Errorf("Incorrect entry author, got: %s", feed.Entries[0].Author)
- }
- }
- func TestParseFeedWithoutAuthor(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Entries[0].Author != "" {
- t.Errorf("Incorrect entry author, got: %q", feed.Entries[0].Author)
- }
- }
- func TestParseEntryWithEnclosures(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <id>http://www.example.org/myfeed</id>
- <title>My Podcast Feed</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <author>
- <name>John Doe</name>
- </author>
- <link href="http://example.org" />
- <link rel="self" href="http://example.org/myfeed" />
- <entry>
- <id>http://www.example.org/entries/1</id>
- <title>Atom 1.0</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="http://www.example.org/entries/1" />
- <summary>An overview of Atom 1.0</summary>
- <link rel="enclosure"
- type="audio/mpeg"
- title="MP3"
- href="http://www.example.org/myaudiofile.mp3"
- length="1234" />
- <link rel="enclosure"
- type="application/x-bittorrent"
- title="BitTorrent"
- href="http://www.example.org/myaudiofile.torrent"
- length="4567" />
- <content type="xhtml">
- <div xmlns="http://www.w3.org/1999/xhtml">
- <h1>Show Notes</h1>
- <ul>
- <li>00:01:00 -- Introduction</li>
- <li>00:15:00 -- Talking about Atom 1.0</li>
- <li>00:30:00 -- Wrapping up</li>
- </ul>
- </div>
- </content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if len(feed.Entries[0].Enclosures) != 2 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- expectedResults := []struct {
- url string
- mimeType string
- size int64
- }{
- {"http://www.example.org/myaudiofile.mp3", "audio/mpeg", 1234},
- {"http://www.example.org/myaudiofile.torrent", "application/x-bittorrent", 4567},
- }
- for index, enclosure := range feed.Entries[0].Enclosures {
- if expectedResults[index].url != enclosure.URL {
- t.Errorf(`Unexpected enclosure URL, got %q instead of %q`, enclosure.URL, expectedResults[index].url)
- }
- if expectedResults[index].mimeType != enclosure.MimeType {
- t.Errorf(`Unexpected enclosure type, got %q instead of %q`, enclosure.MimeType, expectedResults[index].mimeType)
- }
- if expectedResults[index].size != enclosure.Size {
- t.Errorf(`Unexpected enclosure size, got %d instead of %d`, enclosure.Size, expectedResults[index].size)
- }
- }
- }
- func TestParseEntryWithRelativeEnclosureURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <id>https://www.example.org/myfeed</id>
- <title>My Podcast Feed</title>
- <link href="https://example.org" />
- <link rel="self" href="https://example.org/myfeed" />
- <entry>
- <id>https://www.example.org/entries/1</id>
- <title>Atom 1.0</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="https://www.example.org/entries/1" />
- <link rel="enclosure"
- type="audio/mpeg"
- title="MP3"
- href=" /myaudiofile.mp3 "
- length="1234" />
- </content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if len(feed.Entries[0].Enclosures) != 1 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- if feed.Entries[0].Enclosures[0].URL != "https://example.org/myaudiofile.mp3" {
- t.Errorf("Incorrect enclosure URL, got: %q", feed.Entries[0].Enclosures[0].URL)
- }
- }
- func TestParseEntryWithDuplicateEnclosureURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <id>http://www.example.org/myfeed</id>
- <title>My Podcast Feed</title>
- <link href="http://example.org" />
- <link rel="self" href="http://example.org/myfeed" />
- <entry>
- <id>http://www.example.org/entries/1</id>
- <title>Atom 1.0</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="http://www.example.org/entries/1" />
- <link rel="enclosure"
- type="audio/mpeg"
- title="MP3"
- href="http://www.example.org/myaudiofile.mp3"
- length="1234" />
- <link rel="enclosure"
- type="audio/mpeg"
- title="MP3"
- href=" http://www.example.org/myaudiofile.mp3 "
- length="1234" />
- </content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if len(feed.Entries[0].Enclosures) != 1 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- if feed.Entries[0].Enclosures[0].URL != "http://www.example.org/myaudiofile.mp3" {
- t.Errorf("Incorrect enclosure URL, got: %q", feed.Entries[0].Enclosures[0].URL)
- }
- }
- func TestParseEntryWithoutEnclosureURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <id>http://www.example.org/myfeed</id>
- <title>My Podcast Feed</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="http://example.org" />
- <link rel="self" href="http://example.org/myfeed" />
- <entry>
- <id>http://www.example.org/entries/1</id>
- <title>Atom 1.0</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="http://www.example.org/entries/1" />
- <summary>An overview of Atom 1.0</summary>
- <link rel="enclosure" href="" length="0" />
- <content type="xhtml">Test</content>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if len(feed.Entries[0].Enclosures) != 0 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- }
- func TestParseEntryWithPublished(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <published>2003-12-13T18:30:02Z</published>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if !feed.Entries[0].Date.Equal(time.Date(2003, time.December, 13, 18, 30, 2, 0, time.UTC)) {
- t.Errorf("Incorrect entry date, got: %v", feed.Entries[0].Date)
- }
- }
- func TestParseEntryWithPublishedAndUpdated(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://example.org/2003/12/13/atom03"/>
- <id>urn:uuid:1225c695-cfb8-4ebb-aaaa-80da344efa6a</id>
- <published>2002-11-12T18:30:02Z</published>
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if !feed.Entries[0].Date.Equal(time.Date(2002, time.November, 12, 18, 30, 2, 0, time.UTC)) {
- t.Errorf("Incorrect entry date, got: %v", feed.Entries[0].Date)
- }
- }
- func TestParseInvalidXml(t *testing.T) {
- data := `garbage`
- _, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err == nil {
- t.Error("Parse should returns an error")
- }
- }
- func TestParseTitleWithSingleQuote(t *testing.T) {
- data := `
- <?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>' or ’</title>
- <link href="http://example.org/"/>
- </feed>
- `
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "' or ’" {
- t.Errorf(`Incorrect title, got: %q`, feed.Title)
- }
- }
- func TestParseTitleWithEncodedSingleQuote(t *testing.T) {
- data := `
- <?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title type="html">Test's Blog</title>
- <link href="http://example.org/"/>
- </feed>
- `
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "Test's Blog" {
- t.Errorf(`Incorrect title, got: %q`, feed.Title)
- }
- }
- func TestParseTitleWithSingleQuoteAndHTMLType(t *testing.T) {
- data := `
- <?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title type="html">O’Hara</title>
- <link href="http://example.org/"/>
- </feed>
- `
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "O’Hara" {
- t.Errorf(`Incorrect title, got: %q`, feed.Title)
- }
- }
- func TestParseWithHTMLEntity(t *testing.T) {
- data := `
- <?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- </feed>
- `
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.Title != "Example \u00a0 Feed" {
- t.Errorf(`Incorrect title, got: %q`, feed.Title)
- }
- }
- func TestParseWithInvalidCharacterEntity(t *testing.T) {
- data := `
- <?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/a&b"/>
- </feed>
- `
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.SiteURL != "http://example.org/a&b" {
- t.Errorf(`Incorrect URL, got: %q`, feed.SiteURL)
- }
- }
- func TestParseMediaGroup(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom" xmlns:media="http://search.yahoo.com/mrss/">
- <id>https://www.example.org/myfeed</id>
- <title>My Video Feed</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="https://example.org" />
- <link rel="self" href="https://example.org/myfeed" />
- <entry>
- <id>https://www.example.org/entries/1</id>
- <title>Some Video</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="https://www.example.org/entries/1" />
- <media:group>
- <media:title>Another title</media:title>
- <media:content url="https://www.youtube.com/v/abcd" type="application/x-shockwave-flash" width="640" height="390"/>
- <media:content url=" /v/efg " type="application/x-shockwave-flash" width="640" height="390"/>
- <media:content url=" " type="application/x-shockwave-flash" width="640" height="390"/>
- <media:thumbnail url="https://www.example.org/duplicate-thumbnail.jpg" width="480" height="360"/>
- <media:thumbnail url="https://www.example.org/duplicate-thumbnail.jpg" width="480" height="360"/>
- <media:thumbnail url=" /thumbnail2.jpg " width="480" height="360"/>
- <media:thumbnail url=" " width="480" height="360"/>
- <media:description>Some description
- A website: http://example.org/</media:description>
- </media:group>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if len(feed.Entries[0].Enclosures) != 4 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- expectedResults := []struct {
- url string
- mimeType string
- size int64
- }{
- {"https://www.example.org/duplicate-thumbnail.jpg", "image/*", 0},
- {"https://example.org/thumbnail2.jpg", "image/*", 0},
- {"https://www.youtube.com/v/abcd", "application/x-shockwave-flash", 0},
- {"https://example.org/v/efg", "application/x-shockwave-flash", 0},
- }
- for index, enclosure := range feed.Entries[0].Enclosures {
- if expectedResults[index].url != enclosure.URL {
- t.Errorf(`Unexpected enclosure URL, got %q instead of %q`, enclosure.URL, expectedResults[index].url)
- }
- if expectedResults[index].mimeType != enclosure.MimeType {
- t.Errorf(`Unexpected enclosure type, got %q instead of %q`, enclosure.MimeType, expectedResults[index].mimeType)
- }
- if expectedResults[index].size != enclosure.Size {
- t.Errorf(`Unexpected enclosure size, got %d instead of %d`, enclosure.Size, expectedResults[index].size)
- }
- }
- }
- func TestParseMediaElements(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom" xmlns:media="http://search.yahoo.com/mrss/">
- <id>https://www.example.org/myfeed</id>
- <title>My Video Feed</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="https://example.org" />
- <link rel="self" href="https://example.org/myfeed" />
- <entry>
- <id>https://www.example.org/entries/1</id>
- <title>Some Video</title>
- <updated>2005-07-15T12:00:00Z</updated>
- <link href="https://www.example.org/entries/1" />
- <media:title>Another title</media:title>
- <media:content url="https://www.youtube.com/v/abcd" type="application/x-shockwave-flash" width="640" height="390"/>
- <media:content url=" /relative/media.mp4 " type="application/x-shockwave-flash" width="640" height="390"/>
- <media:content url=" " type="application/x-shockwave-flash" width="640" height="390"/>
- <media:thumbnail url="https://example.org/duplicated-thumbnail.jpg" width="480" height="360"/>
- <media:thumbnail url=" https://example.org/duplicated-thumbnail.jpg " width="480" height="360"/>
- <media:thumbnail url=" " width="480" height="360"/>
- <media:peerLink type="application/x-bittorrent" href=" http://www.example.org/sampleFile.torrent " />
- <media:peerLink type="application/x-bittorrent" href=" /sampleFile2.torrent" />
- <media:peerLink type="application/x-bittorrent" href=" " />
- <media:description>Some description
- A website: http://example.org/</media:description>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Fatalf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if len(feed.Entries[0].Enclosures) != 5 {
- t.Fatalf("Incorrect number of enclosures, got: %d", len(feed.Entries[0].Enclosures))
- }
- expectedResults := []struct {
- url string
- mimeType string
- size int64
- }{
- {"https://example.org/duplicated-thumbnail.jpg", "image/*", 0},
- {"https://www.youtube.com/v/abcd", "application/x-shockwave-flash", 0},
- {"https://example.org/relative/media.mp4", "application/x-shockwave-flash", 0},
- {"http://www.example.org/sampleFile.torrent", "application/x-bittorrent", 0},
- {"https://example.org/sampleFile2.torrent", "application/x-bittorrent", 0},
- }
- for index, enclosure := range feed.Entries[0].Enclosures {
- if expectedResults[index].url != enclosure.URL {
- t.Errorf(`Unexpected enclosure URL, got %q instead of %q`, enclosure.URL, expectedResults[index].url)
- }
- if expectedResults[index].mimeType != enclosure.MimeType {
- t.Errorf(`Unexpected enclosure type, got %q instead of %q`, enclosure.MimeType, expectedResults[index].mimeType)
- }
- if expectedResults[index].size != enclosure.Size {
- t.Errorf(`Unexpected enclosure size, got %d instead of %d`, enclosure.Size, expectedResults[index].size)
- }
- }
- }
- func TestParseRepliesLinkRelationWithHTMLType(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom"
- xmlns:thr="http://purl.org/syndication/thread/1.0">
- <id>http://www.example.org/myfeed</id>
- <title>My Example Feed</title>
- <updated>2005-07-28T12:00:00Z</updated>
- <link href="http://www.example.org/myfeed" />
- <author><name>James</name></author>
- <entry>
- <id>tag:entries.com,2005:1</id>
- <title>My original entry</title>
- <updated>2006-03-01T12:12:12Z</updated>
- <link href="http://www.example.org/entries/1" />
- <link rel="replies"
- type="application/atom+xml"
- href="http://www.example.org/mycommentsfeed.xml"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <link rel="replies"
- type="text/html"
- href="http://www.example.org/comments.html"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <summary>This is my original entry</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if feed.Entries[0].CommentsURL != "http://www.example.org/comments.html" {
- t.Errorf("Incorrect entry comments URL, got: %s", feed.Entries[0].CommentsURL)
- }
- }
- func TestParseRepliesLinkRelationWithXHTMLType(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom"
- xmlns:thr="http://purl.org/syndication/thread/1.0">
- <id>http://www.example.org/myfeed</id>
- <title>My Example Feed</title>
- <updated>2005-07-28T12:00:00Z</updated>
- <link href="http://www.example.org/myfeed" />
- <author><name>James</name></author>
- <entry>
- <id>tag:entries.com,2005:1</id>
- <title>My original entry</title>
- <updated>2006-03-01T12:12:12Z</updated>
- <link href="http://www.example.org/entries/1" />
- <link rel="replies"
- type="application/atom+xml"
- href="http://www.example.org/mycommentsfeed.xml"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <link rel="replies"
- type="application/xhtml+xml"
- href="http://www.example.org/comments.xhtml"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <summary>This is my original entry</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if feed.Entries[0].CommentsURL != "http://www.example.org/comments.xhtml" {
- t.Errorf("Incorrect entry comments URL, got: %s", feed.Entries[0].CommentsURL)
- }
- }
- func TestParseRepliesLinkRelationWithNoType(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom"
- xmlns:thr="http://purl.org/syndication/thread/1.0">
- <id>http://www.example.org/myfeed</id>
- <title>My Example Feed</title>
- <updated>2005-07-28T12:00:00Z</updated>
- <link href="http://www.example.org/myfeed" />
- <author><name>James</name></author>
- <entry>
- <id>tag:entries.com,2005:1</id>
- <title>My original entry</title>
- <updated>2006-03-01T12:12:12Z</updated>
- <link href="http://www.example.org/entries/1" />
- <link rel="replies"
- href="http://www.example.org/mycommentsfeed.xml"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <summary>This is my original entry</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if feed.Entries[0].CommentsURL != "" {
- t.Errorf("Incorrect entry comments URL, got: %s", feed.Entries[0].CommentsURL)
- }
- }
- func TestAbsoluteCommentsURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom"
- xmlns:thr="http://purl.org/syndication/thread/1.0">
- <id>http://www.example.org/myfeed</id>
- <title>My Example Feed</title>
- <updated>2005-07-28T12:00:00Z</updated>
- <link href="http://www.example.org/myfeed" />
- <author><name>James</name></author>
- <entry>
- <id>tag:entries.com,2005:1</id>
- <title>My original entry</title>
- <updated>2006-03-01T12:12:12Z</updated>
- <link href="http://www.example.org/entries/1" />
- <link rel="replies"
- type="text/html"
- href="invalid url"
- thr:count="10" thr:updated="2005-07-28T12:10:00Z" />
- <summary>This is my original entry</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries) != 1 {
- t.Errorf("Incorrect number of entries, got: %d", len(feed.Entries))
- }
- if feed.Entries[0].URL != "http://www.example.org/entries/1" {
- t.Errorf("Incorrect entry URL, got: %s", feed.Entries[0].URL)
- }
- if feed.Entries[0].CommentsURL != "" {
- t.Errorf("Incorrect entry comments URL, got: %s", feed.Entries[0].CommentsURL)
- }
- }
- func TestParseItemWithCategories(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <entry>
- <link href="http://www.example.org/entries/1" />
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- <category term='ZZZZ' />
- <category term='ZZZZ' />
- <category term=" " />
- <category term='Technology' label='Science' />
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries[0].Tags) != 2 {
- t.Fatalf("Incorrect number of tags, got: %d", len(feed.Entries[0].Tags))
- }
- expected := []string{"Science", "ZZZZ"}
- result := feed.Entries[0].Tags
- for i, tag := range result {
- if tag != expected[i] {
- t.Errorf("Incorrect entry tag, got %q instead of %q", tag, expected[i])
- }
- }
- }
- func TestParseFeedWithCategories(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <category term='C term' label='C label' />
- <category term='B term' label='B label' />
- <category term='B term' label='B label' />
- <category term='A term' label='A label' />
- <entry>
- <link href="http://www.example.org/entries/1" />
- <updated>2003-12-13T18:30:02Z</updated>
- <summary>Some text.</summary>
- </entry>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if len(feed.Entries[0].Tags) != 3 {
- t.Fatalf("Incorrect number of tags, got: %d", len(feed.Entries[0].Tags))
- }
- expected := []string{"A label", "B label", "C label"}
- result := feed.Entries[0].Tags
- for i, tag := range result {
- if tag != expected[i] {
- t.Errorf("Incorrect entry tag, got %q instead of %q", tag, expected[i])
- }
- }
- }
- func TestParseFeedWithIconURL(t *testing.T) {
- data := `<?xml version="1.0" encoding="utf-8"?>
- <feed xmlns="http://www.w3.org/2005/Atom">
- <title>Example Feed</title>
- <link href="http://example.org/"/>
- <icon>http://example.org/icon.png</icon>
- </feed>`
- feed, err := Parse("https://example.org/", bytes.NewReader([]byte(data)), "10")
- if err != nil {
- t.Fatal(err)
- }
- if feed.IconURL != "http://example.org/icon.png" {
- t.Errorf("Incorrect icon URL, got: %s", feed.IconURL)
- }
- }
|