generic_rss.py 1.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849
  1. __package__ = 'archivebox.parsers'
  2. from typing import IO, Iterable
  3. from datetime import datetime
  4. from ..index.schema import Link
  5. from ..util import (
  6. htmldecode,
  7. enforce_types,
  8. str_between,
  9. )
  10. @enforce_types
  11. def parse_generic_rss_export(rss_file: IO[str], **_kwargs) -> Iterable[Link]:
  12. """Parse RSS XML-format files into links"""
  13. rss_file.seek(0)
  14. items = rss_file.read().split('<item>')
  15. items = items[1:] if items else []
  16. for item in items:
  17. # example item:
  18. # <item>
  19. # <title><![CDATA[How JavaScript works: inside the V8 engine]]></title>
  20. # <category>Unread</category>
  21. # <link>https://blog.sessionstack.com/how-javascript-works-inside</link>
  22. # <guid>https://blog.sessionstack.com/how-javascript-works-inside</guid>
  23. # <pubDate>Mon, 21 Aug 2017 14:21:58 -0500</pubDate>
  24. # </item>
  25. trailing_removed = item.split('</item>', 1)[0]
  26. leading_removed = trailing_removed.split('<item>', 1)[-1].strip()
  27. rows = leading_removed.split('\n')
  28. def get_row(key):
  29. return [r for r in rows if r.strip().startswith('<{}>'.format(key))][0]
  30. url = str_between(get_row('link'), '<link>', '</link>')
  31. ts_str = str_between(get_row('pubDate'), '<pubDate>', '</pubDate>')
  32. time = datetime.strptime(ts_str, "%a, %d %b %Y %H:%M:%S %z")
  33. title = str_between(get_row('title'), '<![CDATA[', ']]').strip()
  34. yield Link(
  35. url=htmldecode(url),
  36. timestamp=str(time.timestamp()),
  37. title=htmldecode(title) or None,
  38. tags=None,
  39. sources=[rss_file.name],
  40. )