quicknotes/specs/feeds.md

178 lines
No EOL
5.8 KiB
Markdown

# Feeds Specification
## Overview
The Feeds module allows users to subscribe to and read RSS/Atom feeds. It provides feed management, automatic fetching of new entries, and a clean reading experience for feed content.
## Data Model
### Feed
The `Feed` entity has the following attributes:
| Field | Type | Description |
|-------|------|-------------|
| ID | string | Unique identifier for the feed (UUID) |
| Title | string | Title of the feed |
| URL | string | URL of the feed (RSS/Atom) |
| Description | string | Description of the feed |
| SiteURL | string | URL of the website associated with the feed |
| ImageURL | string | URL of the feed's image or logo |
| LastFetched | timestamp | When the feed was last fetched |
| CreatedAt | timestamp | When the feed was added |
| UpdatedAt | timestamp | When the feed was last updated |
### Entry
The `Entry` entity represents a single item in a feed:
| Field | Type | Description |
|-------|------|-------------|
| ID | string | Unique identifier for the entry (UUID) |
| FeedID | string | ID of the parent feed |
| Title | string | Title of the entry |
| URL | string | URL of the entry |
| Content | string | HTML content of the entry |
| Summary | string | Summary or excerpt of the entry |
| Author | string | Author of the entry |
| Published | timestamp | When the entry was published |
| Updated | timestamp | When the entry was last updated |
| ReadAt | timestamp | When the entry was marked as read (null if unread) |
| FullContent | string | Full content of the entry (if fetched separately) |
| CreatedAt | timestamp | When the entry was added to the system |
| UpdatedAt | timestamp | When the entry was last updated in the system |
## UUID Generation
Both Feed and Entry entities have GORM BeforeCreate hooks which ensure that a UUID is automatically generated if the ID is not provided:
- For the Feed entity:
```go
func (f *Feed) BeforeCreate(tx *gorm.DB) error {
if f.ID == "" {
f.ID = uuid.New().String()
}
return nil
}
```
- For the Entry entity:
```go
func (e *Entry) BeforeCreate(tx *gorm.DB) error {
if e.ID == "" {
e.ID = uuid.New().String()
}
return nil
}
```
## Features
### Feed Management
1. **Subscribe to Feed**: Users can subscribe to new feeds by providing a URL
2. **Unsubscribe from Feed**: Users can unsubscribe from feeds
3. **List Feeds**: Users can view a list of all subscribed feeds
4. **View Feed Details**: Users can view details about a specific feed
### Entry Management
1. **List Entries**: Users can view a list of entries from all feeds or a specific feed
2. **View Entry**: Users can view the content of a specific entry
3. **Mark as Read**: Users can mark entries as read
4. **Mark as Unread**: Users can mark entries as unread
5. **Filter Entries**: Users can filter entries by read/unread status
### Feed Fetching
1. **Manual Refresh**: Users can manually refresh feeds to fetch new entries
2. **Automatic Refresh**: The system can automatically refresh feeds at regular intervals
3. **Content Extraction**: For feeds that only provide summaries, the system can fetch the full content
## API Endpoints
| Method | Endpoint | Description |
|--------|----------|-------------|
| GET | /api/feeds | List all feeds |
| POST | /api/feeds | Subscribe to a new feed |
| GET | /api/feeds/:id | Get a specific feed by ID |
| PUT | /api/feeds/:id | Update a specific feed |
| DELETE | /api/feeds/:id | Unsubscribe from a feed |
| GET | /api/feeds/:id/entries | List entries for a specific feed |
| GET | /api/feeds/entries | List entries from all feeds |
| GET | /api/feeds/entries/:id | Get a specific entry by ID |
| PUT | /api/feeds/entries/:id/read | Mark an entry as read |
| PUT | /api/feeds/entries/:id/unread | Mark an entry as unread |
| POST | /api/feeds/refresh | Refresh all feeds |
| POST | /api/feeds/:id/refresh | Refresh a specific feed |
## Frontend Routes
| Route | Description |
|-------|-------------|
| /feeds | List of subscribed feeds |
| /feeds/:id | View entries from a specific feed |
| /feeds/entries/:id | View a specific entry |
## Implementation Details
### Feed Parsing
The system uses the `gofeed` library to parse RSS and Atom feeds:
```go
func (s *Service) parseFeed(feedURL string) (*gofeed.Feed, error) {
fp := gofeed.NewParser()
feed, err := fp.ParseURL(feedURL)
if err != nil {
return nil, fmt.Errorf("failed to parse feed: %w", err)
}
return feed, nil
}
```
### Feed Refreshing
When refreshing a feed, the system:
1. Fetches the feed content from the URL
2. Parses the feed to extract entries
3. Compares entries with existing ones to identify new entries
4. Saves new entries to the database
### Content Extraction
For feeds that only provide summaries, the system can optionally fetch the full content of entries:
1. Extract the URL of the entry
2. Fetch the web page at that URL
3. Use a content extraction algorithm to extract the main content
4. Save the extracted content as the entry's full content
## User Interface
### Feed List
- Displays a list of subscribed feeds with titles, descriptions, and icons
- Shows the number of unread entries for each feed
- Provides buttons for refreshing, editing, and unsubscribing
### Entry List
- Displays a list of entries from all feeds or a specific feed
- Shows entry titles, summaries, publication dates, and read status
- Provides filters for read/unread status
- Includes buttons for marking as read/unread
### Entry Viewer
- Displays the entry content in a clean, reader-friendly format
- Shows the entry title, author, and publication date
- Provides buttons for marking as read/unread and returning to the list
- Includes a link to the original article
### Subscribe Form
- Input field for the feed URL
- Automatic detection of feed format (RSS/Atom)
- Preview of feed details before subscribing