Skip to content

Latest commit

 

History

History
772 lines (488 loc) · 24.4 KB

File metadata and controls

772 lines (488 loc) · 24.4 KB

Development

.. module:: reader
  :no-index:


Development should follow a problem-solution approach.

.. seealso:: :ref:`philosophy`

Roadmap

The plan is to continue evolving the library to support as many "feed reader application" use cases as possible, while still following the :ref:`philosophy`. Even if a specific feature is not a good fit for the library itself, it should be possible to find a more generic solution that makes it possible to build the feature on top.

Following is an unsorted, non-exhausive list of known areas for improvement. I am working on reader based on my current interests, in my spare time, but I will prioritize supporting :doc:`contributors <contributing>` (discussions, reviews and so on).

.. seealso::

    `Open issues`_ and :ref:`Design notes`.

Command-line interface

The :doc:`cli` is more or less stable, although it could use additional tests.

A terminal feed reader is not in scope, since I don't need one, but I'm not opposed to the idea; note that the web app should already work with text-based browsers such as Lynx.

Web application

The new :doc:`app` is under active development. I am working on adding features already implemented in the library and the :ref:`legacy web app <legacy web app>`, and possibly on a hosted version.

Like with the library, the goal is to keep it simple to use and sustainable to maintain long-term; the decision to use Flask, Bootstrap, and htmx is part of this (if they were around for ~15 years, they should be around for 15 more).

Backwards compatibility

reader uses semantic versioning.

Breaking compatibility is done by incrementing the major version, announcing it in the :doc:`changelog`, and raising deprecation warnings for at least one minor version before the new major version is released (if possible).

There may be minor exceptions to this, e.g. bug fixes and gross violation of specifications; they will be announced in the :doc:`changelog` with a This is a minor compatibility break warning.

Schema migrations for the default storage must happen automatically. Migrations can be removed in new major versions, with at least 3 months provided since the last migration.

What is the public API

reader follows the PEP 8 definition of public interface.

The following are part of the public API:

  • Every interface documented in the :doc:`API reference <api>`.
  • Any (documented) module, function, object, method, and attribute, defined in the reader package, that is accessible without passing through a name that starts with underscore.
  • The number and position of positional arguments.
  • The names of keyword arguments.
  • Argument types (argument types cannot become more strict).
  • Attribute types (attribute types cannot become less strict).

Undocumented type aliases (even if not private) are not part of the public API.

Other exceptions are possible; they will be marked aggresively as such.

.. seealso::

  The `Twisted Compatibility Policy <https://github.com/twisted/twisted/blob/twisted-16.2.0/docs/core/development/policy/compatibility-policy.rst>`_,
  which served as inspiration for this.


Internal API

The :doc:`internal` is not stable, but the long term goal is for it to become so.

In order to support / encourage potential users (e.g. plugins, alternate storage implementations), changes should at least be announced in the :doc:`changelog`.

Supported Python versions

The oldest Python version reader should support is:

  • the newest CPython available on the latest Ubuntu LTS (3 months after LTS release)
  • at least 1 stable PyPy version

This usually ends up being the last 3 stable CPython versions.

Dropping support for a Python version should be announced at least 1 release prior.

Releases

For convenience, reader only releases major and minor versions (bugfixes go in minor versions). Changes go only to the next release (no backports).

Making a release

Note

The :gh:`release.py <scripts/release.py>` script together with the :gh:`publish <.github/workflows/publish.yaml>` GitHub Actions workflow already do most of these.

Making a release (from x to y == x + 1):

  • (release.py) bump version in src/reader/__init__.py to y
  • (release.py) update changelog with release version and date
  • (release.py) make sure tests pass / docs build
  • (release.py) push to GitHub
  • (release.py prompts) wait for GitHub Actions / Read the Docs builds to pass
  • (release.py) tag current commit with <major>.<minor> and <major>.x (e.g. when releasing 1.20: 1.20 and 1.x) and push tags to GitHub
  • (release.py prompts) approve publish workflow
  • (publish workflow) build and upload to PyPI
  • alternatively, to make a manual release (should not be needed)
    • clean up dist/: rm -rf dist/
    • build tarball and wheel: python -m build
    • upload to test PyPI and check: twine upload --repository-url https://test.pypi.org/legacy/ dist/*
    • upload to PyPI: twine upload dist/*
  • (publish workflow) create draft release in GitHub
  • (release.py prompts) publish release in GitHub
  • (release.py) bump versions from y to (y + 1).dev0, add (y + 1) changelog section

Documentation

Following are notes about what documentation should look like, especially for the stable high-level :doc:`API <api>`, since that's what most users will see.

We prefer type information in the method description, not in the signature, since the result is more readable. For the same reason, we prefer hand-written Sphinx-style field list types.

We still use autodoc-provided type hints as fallback for parameters that don't have hand-written types, for type documentation for dataclasses, and for the unstable :doc:`internal`, where it's too much effort to maintain hand-written types.

Known issues (October 2023, Sphinx version ~7):

  • Overloads are shown with full annotation regardless of autodoc_typehints (known, documented behavior). May get better with sphinx-doc/sphinx#10359.
  • Type aliases that do not come from hand-written types but from the autodoc typehints are expanded in-place; this also affects the overload type annotations. The documented work-around is to add the aliases to autodoc_type_aliases.
  • Type alias names that appear in parameter types do not link to the documentation in :ref:`type aliases`. May get better with sphinx-doc/sphinx#9705

Design notes

Folowing are various design notes that aren't captured somewhere else (either in the code, or in the issue where a feature was initially developed).

Why use SQLite and not SQLAlchemy?

tl;dr: For "historical reasons".

In the beginning:

  • I wanted to keep things as simple as possible, so I don't get demotivated and stop working on it. I also wanted to try out a "problem-solution" approach.
  • I think by that time I was already a great SQLite fan, and knew that because of the relatively single-user nature of the thing I won't have to change databases because of concurrency issues.
  • The fact that I didn't know exactly where and how I would deploy the web app (and that SQLite is in stdlib) kinda cemented that assumption.

Since then, I did come up with some of my own complexity: there's a SQL query builder, a schema migration system, and there were some concurrency issues. SQLAlchemy would have likely helped with the first two, but not with the last one (not without dropping SQLite).

Note that it is possible to use a different storage implementation; all storage stuff happens through a DAO-style interface, and SQLAlchemy was the main real alternative I had in mind. The API is private at the moment (1.10), but if anyone wants to use it I can make it public.

It is unlikely I'll write a SQLAlchemy storage myself, since I don't need it (yet), and I think testing it with multiple databases would take quite some time.

Multiple storage implementations

Detailed requirements and API discussion: :issue:`168#issuecomment-642002049`.

Minimal work needed to support alternate storages: :issue:`168#issuecomment-1383127564`.

Storage internal API documented in version 3.10 (November 2023) in :issue:`325`.

Database optimization

Optimization lessons learned while fixing "database is locked" errors: :issue:`175#issuecomment-657495233`.

Some general guidance on schema/index design: :issue:`327#issuecomment-1859147186`.

Speeding up get_entries(sort='recent'):

  • first attempt at adding indexes: :issue:`134`
  • using a computed column (recent_sort) didn't change things very much: :issue:`279`
  • an index on recent_sort alone is not enough for pagination, the index needs to match 1:1 the WHERE clause: :issue:`330`.

Speeding up get_entry_counts(feed=...):

Official guidance: https://www.sqlite.org/queryplanner-ng.html#howtofix

Parser

file:// handling, feed root, per-URL-prefix parsers (later retrievers, see below):

Requests session plugins:

Retriever / parser split:

Alternative feed parsers:

Lessons learned from the :ref:`Twitter` plugin:

  • It is useful for a retriever to pass an arbitrary resource to the parser.

    This is already codified in :meth:`~reader._parser.RetrieverType` and :meth:`~reader._parser.ParserType` being generic.

  • It is useful for a Retriever to store arbitrary caching data; the plugin (mis)used RetrieveResult.http_etag to store the (integer) id of the newest tweet in the thread.

    Update: This was formalized as :attr:`.RetrievedFeed.caching_info` in 3.15.

  • It is useful for a Retriever to pass arbitrary data to itself; the plugin (mis)used FeedForUpdate.http_etag to pass from :meth:`~reader._parser.FeedForUpdateRetrieverType.process_feed_for_update` to :meth:`~reader._parser.RetrieverType.__call__`:

    • the bearer token and the ids of recent entries (used to retrieve tweets)
    • the ids of entries to re-render, triggered by a one-off tag (passed along to the parser)

    This distinction was made so that process_feed_for_update() takes all the decisions upfront (possibly taking advantage of Storage.get_feeds_for_update() future optimisations to e.g. also get tags), and calling the retriever (in parallel) doesn't do any reader operations.

    It would be nice to formalize this as well.

  • A plugin can coordinate between a custom retriever and custom parser with an unregistered RetrieveResult MIME type (e.g. application/x.twitter).

  • A plugin can keep arbitrary data as a content with an unregistered type (e.g. application/x.twitter+json).

Metrics

Some thoughts on implementing metrics: :issue:`68#issuecomment-450025175`.

Per-call timings introduced in the :mod:`~reader._plugins.timer` experimental plugin.

Query builder

Survey of possible options: :issue:`123#issuecomment-582307504`.

In 2021, I've written an entire series about it: https://death.andgravity.com/query-builder

Pagination for methods that return iterators

Why do it for the private implementation: :issue:`167#issuecomment-626753299`;

Detailed requirements and API discussion for public pagination: :issue:`196#issuecomment-706038363`.

Search

From the initial issue:

Enabling search by default, and alternative search APIs: :issue:`252`.

Change tracking API: :issue:`323#issuecomment-1930756417`, model validated in this gist.

External resources:

reader types to Atom mapping

This whole issue: :issue:`153`.

Sort by random

Some thoughts in the initial issue: :issue:`105`.

Sort by tag values

It may be useful to be able to sort by tag values in order to allow sorting by cached entry counts: :issue:`306#issuecomment-1694655504`.

Entry/feed "primary key" attribute naming

This whole issue: :issue:`159#issuecomment-612914956`.

Change feed URL

From the initial issue:

Splitting a feed into two feeds is discussed in :issue:`221`. The main takeaway at the time (2021) was that not all entry data can be retrieved through the storage API, so it's not fully possible to round-trip an entry. Based on a quick 2024 survey, the only affected attributes are :attr:`~.EntryUpdateIntent.feed_order` (can't get; not critical) and :attr:`~Entry.original_feed_url` (can't set; will be required for archiving important entries of deleted feeds, :issue:`290`).

Resource tags / metadata

Feed tags

Detailed requirements and API discussion, and a case study of how to implement categories on top of tags: :issue:`184#issuecomment-689587006`.

Merging tags and metadata, and the addition of a new, generic (global, feed, entry) tag API: :issue:`266#issuecomment-1013739526`.

Entry tags

:issue:`228#issuecomment-810098748` discusses three different kinds of entry user data, how they would be implemented, and why I want more use-cases before implementing them (basically, YAGNI):

  • entry searchable text fields (for notes etc.)
  • entry tags (similar to feed tags, may be used as additional bool flags)
  • entry metadata (similar to feed metadata)
    • also discusses how to build an enclosure cache/preloader (doesn't need special reader features besides what's available in 1.16)

:issue:`253` discusses using entry tags to implement the current entry flags (read, important); tl;dr: it's not worth adding entry tags just for this. :issue:`327` discusses using entry tags for has_enclosures; tl;dr: it wouldn't save a lot of code, it would be only a bit slower, and it reconfirms that read and important are integral to the data model, so we still want them as regular columns.

After closing :issue:`228` with wontfix in late 2021, in early 2022 (following the :issue:`266` tag/metadata unification) I implemented entry and global tags in :issue:`272`; there's a list of known use cases in the issue description.

Resource tags

Optimistic locking for tags: :issue:`308`.

Filter tags by prefix: :issue:`309`.

User-added entries

Discussion about API/typing, and things we didn't do: :issue:`239`.

Feed updates

Some thoughts about adding a map argument: :issue:`152#issuecomment-606636200`.

How update_feeds() is like a pipeline: comment.

Data flow diagram for the update process, as of v1.13: :issue:`204#issuecomment-779709824`.

update_feeds_iter():

Disabling updates:

Updating entries based on a hash of their content (regardless of updated):

Decision to ignore feed.updated when updating feeds: :issue:`231`.

:ref:`scheduled` (initial design + postponed features): :issue:`322`.

Deleting entries

Requirements, open questions, and how it interacts with :mod:`~reader.plugins.entry_dedupe`: :issue:`96`.

A summary of why it isn't easy to do: :issue:`301#issuecomment-1442423151`.

Counts API

Detailed requirements and API discussion: :issue:`185#issuecomment-731743327`.

Tracking additional statistics (e.g. :attr:`~Entry.read_modified`): :issue:`254`; how to expose said statistics: :issue:`254#issuecomment-1807064610`.

Notebook with a successful attempt to determine a feed "usefulness" score based on how many entries I mark as read / important / don't care; highlights a number of gaps in the reader API: https://gist.github.com/lemon24/93222ef4bc4a775092b56546a6e6cd0f

Using None as a special argument value

This comment: :issue:`177#issuecomment-674786498`.

Batch methods

Some initial thoughts on batch get methods (including API/typing) in :issue:`191` (closed with wontfix, for now).

Why I want to postpone batch update/set methods: :issue:`187#issuecomment-700740251`.

tl:dr: Performance is likely a non-issue with SQLite, convenience can be added on top as a plugin.

(2025 update) Why it may be worth adding batch interfaces anyway (even if underneath the storage implementation doesn't actually batch) – it allows for future optimization: https://blog.glyph.im/2022/12/potato-programming.html

See the 2.12 reader._app.ResourceTags class for an idea of how to represent a bunch of tags in a reserved-name-scheme-agnostic way (useful e.g. for when get_entries() should return tags x, y, z of each entry).

Some web app measurements that show a few cases where batch methods may help: :issue:`306#issuecomment-1694655504`.

Using a single Reader objects from multiple threads

Some thoughts on why it's difficult to do: :issue:`206#issuecomment-751383418`.

Requirements and use cases: :issue:`206#issuecomment-1179739301`.

When/how to run pragma optimize: :issue:`206#issuecomment-1183660880`.

Full support added in version 2.16 (July 2022).

Plugins

List of potential hooks (from mid-2018): :issue:`80`.

Minimal plugin API (from 2021) – case study and built-in plugin naming scheme: :issue:`229#issuecomment-803870781`.

We'll add / document new (public) hooks as needed.

"Tag before, clear tag after" pattern for resilient plugins: :issue:`246#issuecomment-1596097300`.

Update hook error handling:

Considerations on using pluggy: :issue:`329`, how adapting reader use cases to the pluggy model may work. Conclusions:

even if using pluggy would require some work-arounds for calling, I think the collection / discovery functionality is still more than worth it

long-term and/or if reader becomes really popular, pluggy is definitely the way to go

Reserved names

Requirements, thoughts about the naming scheme and prefixes unlikely to collide with user names: :issue:`186` (multiple comments).

Wrapping underlying storage exceptions

Which exception to wrap, and which not: :issue:`21#issuecomment-365442439`.

Timezone handling

Aware vs. naive, and what's needed to go fully aware: :issue:`233#issuecomment-881618002`.

In version 3.10 (November 2023), all internal APIs were changed to use timezone-aware datetimes, with the timezone set to UTC, in preparation for support for any timezone.

OPML support

Thoughts on dynamic lists of feeds: :issue:`165#issuecomment-905893420`.

Duplicate entries

Duplicate entries are handled by :mod:`~reader.plugins.entry_dedupe` (extensive notes in the :gh:`docstring <src/reader/plugins/entry_dedupe.py>`).

However, it is also possible for a feed to have two entries with the same id – yes, even though in most (if not all) formats, the id is meant to be universally unique! As of 3.13, we do not support multiple entries with the same id (the first entry wins); see :issue:`335` for thoughts on this.

REST API

Some early thoughts: :issue:`192#issuecomment-700773138` (closed with wontfix, for now).