@rmdes/indiekit-endpoint-microsub 1.0.28 → 1.0.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,246 @@
1
+ # @rmdes/indiekit-endpoint-microsub
2
+
3
+ A comprehensive Microsub social reader plugin for Indiekit. Subscribe to feeds (RSS, Atom, JSON Feed, h-feed), organize them into channels, and read posts in a unified timeline interface with a built-in web reader UI.
4
+
5
+ ## Features
6
+
7
+ - **Microsub Protocol**: Full implementation of the [Microsub spec](https://indieweb.org/Microsub)
8
+ - **Multi-Format Feeds**: RSS, Atom, JSON Feed, h-feed (microformats)
9
+ - **Smart Polling**: Adaptive tiered polling (2 minutes to 17+ hours) based on update frequency
10
+ - **Real-Time Updates**: WebSub (PubSubHubbub) support for instant notifications
11
+ - **Web Reader UI**: Built-in reader interface with channel navigation and timeline view
12
+ - **Feed Discovery**: Automatic discovery of feeds from website URLs
13
+ - **Read State**: Per-user read tracking with automatic cleanup
14
+ - **Compose Interface**: Post replies, likes, reposts, and bookmarks via Micropub
15
+ - **Webmention Support**: Receive webmentions in your notifications channel
16
+ - **Media Proxy**: Privacy-friendly image proxying
17
+ - **OPML Export**: Export your subscriptions as OPML
18
+
19
+ ## Installation
20
+
21
+ ```bash
22
+ npm install @rmdes/indiekit-endpoint-microsub
23
+ ```
24
+
25
+ ## Configuration
26
+
27
+ Add to your Indiekit config:
28
+
29
+ ```javascript
30
+ import MicrosubEndpoint from "@rmdes/indiekit-endpoint-microsub";
31
+
32
+ export default {
33
+ plugins: [
34
+ new MicrosubEndpoint({
35
+ mountPath: "/microsub", // Default mount path
36
+ }),
37
+ ],
38
+ };
39
+ ```
40
+
41
+ ## Usage
42
+
43
+ ### Web Reader UI
44
+
45
+ Navigate to `/microsub/reader` in your Indiekit installation to access the web interface.
46
+
47
+ **Channels**: Organize feeds into channels (Technology, News, Friends, etc.)
48
+ - Create new channels
49
+ - Configure content filters (exclude types, regex patterns)
50
+ - Reorder channels
51
+
52
+ **Feeds**: Manage subscriptions within each channel
53
+ - Subscribe to feeds by URL
54
+ - Search and discover feeds from websites
55
+ - Edit or rediscover feed URLs
56
+ - Force refresh feeds
57
+ - View feed health status
58
+
59
+ **Timeline**: Read posts from subscribed feeds
60
+ - Paginated timeline view
61
+ - Mark individual items or all items as read
62
+ - View read items separately
63
+ - Click through to original posts
64
+
65
+ **Compose**: Create posts via Micropub
66
+ - Reply to posts
67
+ - Like posts
68
+ - Repost posts
69
+ - Bookmark posts
70
+ - Include syndication targets
71
+
72
+ ### Microsub API
73
+
74
+ Compatible with Microsub clients like [Indigenous](https://indigenous.realize.be/) and [Monocle](https://monocle.p3k.io/).
75
+
76
+ **Endpoint:** Your Indiekit URL + `/microsub`
77
+
78
+ **Supported Actions:**
79
+ - `channels` - List, create, update, delete, reorder channels
80
+ - `timeline` - Get timeline items (paginated)
81
+ - `follow` - Subscribe to a feed
82
+ - `unfollow` - Unsubscribe from a feed
83
+ - `mute` - Mute URLs
84
+ - `unmute` - Unmute URLs
85
+ - `block` - Block authors
86
+ - `unblock` - Unblock authors
87
+ - `search` - Discover feeds from URL
88
+ - `preview` - Preview feed before subscribing
89
+
90
+ **Example:**
91
+
92
+ ```bash
93
+ # List channels
94
+ curl "https://your-site.example/microsub?action=channels" \
95
+ -H "Authorization: Bearer YOUR_TOKEN"
96
+
97
+ # Get timeline for channel
98
+ curl "https://your-site.example/microsub?action=timeline&channel=CHANNEL_UID" \
99
+ -H "Authorization: Bearer YOUR_TOKEN"
100
+
101
+ # Subscribe to feed
102
+ curl "https://your-site.example/microsub" \
103
+ -X POST \
104
+ -H "Authorization: Bearer YOUR_TOKEN" \
105
+ -d "action=follow&channel=CHANNEL_UID&url=https://example.com/feed"
106
+ ```
107
+
108
+ ## Feed Polling
109
+
110
+ Feeds are polled using an adaptive tiered system:
111
+
112
+ - **Tier 0**: 1 minute (very active feeds)
113
+ - **Tier 1**: 2 minutes (active feeds)
114
+ - **Tier 2**: 4 minutes
115
+ - **Tier 3**: 8 minutes
116
+ - ...
117
+ - **Tier 10**: ~17 hours (inactive feeds)
118
+
119
+ Tiers adjust automatically:
120
+ - Feed updates → decrease tier (faster polling)
121
+ - No changes for 2+ fetches → increase tier (slower polling)
122
+
123
+ WebSub-enabled feeds receive instant updates when available.
124
+
125
+ ## Read State Management
126
+
127
+ Read items are tracked per user. To prevent database bloat, only the last 30 read items per channel are kept. Unread items are never deleted.
128
+
129
+ Cleanup runs automatically:
130
+ - On server startup
131
+ - After marking items read
132
+
133
+ ## Integration with Other Plugins
134
+
135
+ ### Blogroll Plugin
136
+
137
+ If `@rmdes/indiekit-endpoint-blogroll` is installed, Microsub will automatically sync feed subscriptions:
138
+ - Subscribe to feed → adds to blogroll
139
+ - Unsubscribe → soft-deletes from blogroll
140
+
141
+ ### Micropub Plugin
142
+
143
+ The compose interface posts via Micropub. Ensure `@indiekit/endpoint-micropub` is configured.
144
+
145
+ ## OPML Export
146
+
147
+ Export your subscriptions:
148
+
149
+ ```
150
+ GET /microsub/reader/opml
151
+ ```
152
+
153
+ Returns OPML XML with all subscribed feeds organized by channel.
154
+
155
+ ## Webmentions
156
+
157
+ The plugin accepts webmentions at `/microsub/webmention`. Received webmentions appear in the special "Notifications" channel.
158
+
159
+ To advertise your webmention endpoint, add to your site's `<head>`:
160
+
161
+ ```html
162
+ <link rel="webmention" href="https://your-site.example/microsub/webmention" />
163
+ ```
164
+
165
+ ## Media Proxy
166
+
167
+ External images are proxied through `/microsub/media/:hash` for privacy and caching. This prevents your IP address from being sent to third-party image hosts.
168
+
169
+ ## API Response Format
170
+
171
+ All API responses follow the Microsub spec. Timeline items use the [jf2 format](https://jf2.spec.indieweb.org/).
172
+
173
+ **Example timeline response:**
174
+
175
+ ```json
176
+ {
177
+ "items": [
178
+ {
179
+ "type": "entry",
180
+ "uid": "https://example.com/post/123",
181
+ "url": "https://example.com/post/123",
182
+ "published": "2026-02-13T12:00:00.000Z",
183
+ "name": "Post Title",
184
+ "content": {
185
+ "text": "Plain text content",
186
+ "html": "<p>HTML content</p>"
187
+ },
188
+ "author": {
189
+ "name": "Author Name",
190
+ "url": "https://author.example/",
191
+ "photo": "https://author.example/photo.jpg"
192
+ },
193
+ "_id": "507f1f77bcf86cd799439011",
194
+ "_is_read": false
195
+ }
196
+ ],
197
+ "paging": {
198
+ "after": "cursor-string"
199
+ }
200
+ }
201
+ ```
202
+
203
+ ## Database Collections
204
+
205
+ The plugin creates these MongoDB collections:
206
+
207
+ - `microsub_channels` - User channels
208
+ - `microsub_feeds` - Feed subscriptions with polling metadata
209
+ - `microsub_items` - Timeline items (posts)
210
+ - `microsub_notifications` - Notifications channel items
211
+ - `microsub_muted` - Muted URLs
212
+ - `microsub_blocked` - Blocked authors
213
+
214
+ ## Troubleshooting
215
+
216
+ ### Feeds not updating
217
+
218
+ - Check the feed's `nextFetchAt` time in the admin UI
219
+ - Use "Force Refresh" button to poll immediately
220
+ - Try "Rediscover" to find the correct feed URL
221
+
222
+ ### "Unable to detect feed type" error
223
+
224
+ - The URL may not be a valid feed
225
+ - Try using the search feature to discover feeds from the homepage
226
+ - Check if the feed requires authentication
227
+
228
+ ### Items disappearing after marking read
229
+
230
+ This is normal behavior - only the last 30 read items per channel are kept to prevent database bloat. Unread items are never deleted.
231
+
232
+ ### Duplicate items
233
+
234
+ Deduplication is based on the feed's GUID/URL. If a feed doesn't provide stable GUIDs, duplicates may appear.
235
+
236
+ ## Contributing
237
+
238
+ Issues and pull requests welcome at [github.com/rmdes/indiekit-endpoint-microsub](https://github.com/rmdes/indiekit-endpoint-microsub)
239
+
240
+ ## License
241
+
242
+ MIT
243
+
244
+ ## Credits
245
+
246
+ Built by [Ricardo Mendes](https://rmendes.net) for the IndieWeb community.
package/index.js CHANGED
@@ -1,4 +1,5 @@
1
1
  import path from "node:path";
2
+ import { fileURLToPath } from "node:url";
2
3
 
3
4
  import express from "express";
4
5
 
@@ -29,6 +30,14 @@ export default class MicrosubEndpoint {
29
30
  this.mountPath = this.options.mountPath;
30
31
  }
31
32
 
33
+ /**
34
+ * Locales directory path
35
+ * @returns {string} Path to locales directory
36
+ */
37
+ get localesDirectory() {
38
+ return path.join(path.dirname(fileURLToPath(import.meta.url)), "locales");
39
+ }
40
+
32
41
  /**
33
42
  * Navigation items for Indiekit admin
34
43
  * @returns {object} Navigation item configuration
@@ -7,6 +7,60 @@ import crypto from "node:crypto";
7
7
 
8
8
  import { getCache, setCache } from "../cache/redis.js";
9
9
 
10
+ /**
11
+ * Private/internal IP ranges that should never be fetched (SSRF protection)
12
+ */
13
+ const BLOCKED_HOSTNAMES = new Set(["localhost", "0.0.0.0"]);
14
+ const BLOCKED_IP_PREFIXES = [
15
+ "127.", // Loopback
16
+ "10.", // Private Class A
17
+ "192.168.", // Private Class C
18
+ "169.254.", // Link-local
19
+ "0.", // Current network
20
+ ];
21
+
22
+ /**
23
+ * Check if a hostname resolves to a private/internal address
24
+ * @param {string} urlString - URL to check
25
+ * @returns {boolean} True if the URL targets a private/internal address
26
+ */
27
+ export function isPrivateUrl(urlString) {
28
+ try {
29
+ const parsed = new URL(urlString);
30
+ const hostname = parsed.hostname;
31
+
32
+ // Block known private hostnames
33
+ if (BLOCKED_HOSTNAMES.has(hostname)) {
34
+ return true;
35
+ }
36
+
37
+ // Block IPv6 loopback
38
+ if (hostname === "::1" || hostname === "[::1]") {
39
+ return true;
40
+ }
41
+
42
+ // Block private IPv4 ranges
43
+ for (const prefix of BLOCKED_IP_PREFIXES) {
44
+ if (hostname.startsWith(prefix)) {
45
+ return true;
46
+ }
47
+ }
48
+
49
+ // Block 172.16.0.0/12 (172.16.x.x - 172.31.x.x)
50
+ const match172 = hostname.match(/^172\.(\d+)\./);
51
+ if (match172) {
52
+ const second = Number.parseInt(match172[1], 10);
53
+ if (second >= 16 && second <= 31) {
54
+ return true;
55
+ }
56
+ }
57
+
58
+ return false;
59
+ } catch {
60
+ return true; // Invalid URLs are blocked
61
+ }
62
+ }
63
+
10
64
  const MAX_SIZE = 2 * 1024 * 1024; // 2MB max image size
11
65
  const CACHE_TTL = 4 * 60 * 60; // 4 hours
12
66
  const ALLOWED_TYPES = new Set([
@@ -99,6 +153,12 @@ export function proxyItemImages(item, baseUrl) {
99
153
  * @returns {Promise<object|null>} Cached image data or null
100
154
  */
101
155
  export async function fetchImage(redis, url) {
156
+ // Block private/internal URLs (defense-in-depth)
157
+ if (isPrivateUrl(url)) {
158
+ console.error(`[Microsub] Media proxy blocked private URL: ${url}`);
159
+ return;
160
+ }
161
+
102
162
  const cacheKey = `media:${hashUrl(url)}`;
103
163
 
104
164
  // Try cache first
@@ -194,6 +254,11 @@ export async function handleMediaProxy(request, response) {
194
254
  return response.status(400).send("Invalid URL");
195
255
  }
196
256
 
257
+ // Block requests to private/internal networks (SSRF protection)
258
+ if (isPrivateUrl(url)) {
259
+ return response.status(403).send("URL not allowed");
260
+ }
261
+
197
262
  // Get Redis client from application
198
263
  const { application } = request.app.locals;
199
264
  const redis = application.redis;
@@ -202,8 +267,7 @@ export async function handleMediaProxy(request, response) {
202
267
  const imageData = await fetchImage(redis, url);
203
268
 
204
269
  if (!imageData) {
205
- // Redirect to original URL as fallback
206
- return response.redirect(url);
270
+ return response.status(404).send("Image not available");
207
271
  }
208
272
 
209
273
  // Set cache headers
@@ -602,7 +602,11 @@ export async function searchItems(application, channelId, query, limit = 20) {
602
602
  typeof channelId === "string" ? new ObjectId(channelId) : channelId;
603
603
 
604
604
  // Use regex search (consider adding text index for better performance)
605
- const regex = new RegExp(query, "i");
605
+ const escapedQuery = query.replaceAll(
606
+ /[$()*+.?[\\\]^{|}]/g,
607
+ String.raw`\$&`,
608
+ );
609
+ const regex = new RegExp(escapedQuery, "i");
606
610
  const items = await collection
607
611
  .find({
608
612
  channelId: objectId,
@@ -4,6 +4,29 @@
4
4
  */
5
5
 
6
6
  import { mf2 } from "microformats-parser";
7
+ import sanitizeHtml from "sanitize-html";
8
+
9
+ /**
10
+ * Sanitize HTML options (matches normalizer.js)
11
+ */
12
+ const SANITIZE_OPTIONS = {
13
+ allowedTags: [
14
+ "a", "abbr", "b", "blockquote", "br", "code", "em", "figcaption",
15
+ "figure", "h1", "h2", "h3", "h4", "h5", "h6", "hr", "i", "img",
16
+ "li", "ol", "p", "pre", "s", "span", "strike", "strong", "sub",
17
+ "sup", "table", "tbody", "td", "th", "thead", "tr", "u", "ul",
18
+ "video", "audio", "source",
19
+ ],
20
+ allowedAttributes: {
21
+ a: ["href", "title", "rel"],
22
+ img: ["src", "alt", "title", "width", "height"],
23
+ video: ["src", "poster", "controls", "width", "height"],
24
+ audio: ["src", "controls"],
25
+ source: ["src", "type"],
26
+ "*": ["class"],
27
+ },
28
+ allowedSchemes: ["http", "https", "mailto"],
29
+ };
7
30
 
8
31
  /**
9
32
  * Verify a webmention
@@ -276,7 +299,7 @@ function extractContent(entry) {
276
299
 
277
300
  return {
278
301
  text: content.value,
279
- html: content.html,
302
+ html: content.html ? sanitizeHtml(content.html, SANITIZE_OPTIONS) : undefined,
280
303
  };
281
304
  }
282
305
 
@@ -0,0 +1,104 @@
1
+ {
2
+ "microsub": {
3
+ "reader": {
4
+ "title": "Leser",
5
+ "empty": "Keine Elemente zum Anzeigen",
6
+ "markAllRead": "Alle als gelesen markieren",
7
+ "showRead": "Gelesene anzeigen ({{count}})",
8
+ "hideRead": "Gelesene Elemente ausblenden",
9
+ "allRead": "Alles aufgeholt!",
10
+ "newer": "Neuer",
11
+ "older": "Älter"
12
+ },
13
+ "channels": {
14
+ "title": "Kanäle",
15
+ "name": "Kanalname",
16
+ "new": "Neuer Kanal",
17
+ "create": "Kanal erstellen",
18
+ "delete": "Kanal löschen",
19
+ "settings": "Kanaleinstellungen",
20
+ "empty": "Noch keine Kanäle. Erstellen Sie einen, um zu beginnen.",
21
+ "notifications": "Benachrichtigungen"
22
+ },
23
+ "timeline": {
24
+ "title": "Zeitleiste",
25
+ "empty": "Keine Elemente in diesem Kanal",
26
+ "markRead": "Als gelesen markieren",
27
+ "markUnread": "Als ungelesen markieren",
28
+ "remove": "Entfernen"
29
+ },
30
+ "feeds": {
31
+ "title": "Feeds",
32
+ "follow": "Folgen",
33
+ "subscribe": "Einen Feed abonnieren",
34
+ "unfollow": "Entfolgen",
35
+ "empty": "Keine Feeds in diesem Kanal abonniert",
36
+ "url": "Feed-URL",
37
+ "urlPlaceholder": "https://beispiel.de/feed.xml",
38
+ "edit": "Feed bearbeiten",
39
+ "rediscover": "Feed neu entdecken",
40
+ "refresh": "Jetzt aktualisieren",
41
+ "status": {
42
+ "active": "Aktiv",
43
+ "error": "Fehler",
44
+ "stale": "Veraltet"
45
+ }
46
+ },
47
+ "item": {
48
+ "reply": "Antworten",
49
+ "like": "Gefällt mir",
50
+ "repost": "Teilen",
51
+ "bookmark": "Lesezeichen",
52
+ "viewOriginal": "Original anzeigen"
53
+ },
54
+ "compose": {
55
+ "title": "Verfassen",
56
+ "content": "Was beschäftigt Sie?",
57
+ "comment": "Kommentar hinzufügen (optional)",
58
+ "commentHint": "Ihr Kommentar wird bei der Syndizierung mit einbezogen",
59
+ "syndicateTo": "Syndizieren an",
60
+ "syndicateHint": "Wählen Sie aus, wo dies quergepostet werden soll",
61
+ "submit": "Veröffentlichen",
62
+ "cancel": "Abbrechen",
63
+ "replyTo": "Antworten an",
64
+ "likeOf": "Gefällt mir",
65
+ "repostOf": "Teilen",
66
+ "bookmarkOf": "Lesezeichen setzen"
67
+ },
68
+ "settings": {
69
+ "title": "{{channel}}-Einstellungen",
70
+ "excludeTypes": "Interaktionstypen ausschließen",
71
+ "excludeTypesHelp": "Wählen Sie Beitragstypen aus, die in diesem Kanal ausgeblendet werden sollen",
72
+ "excludeRegex": "Ausschlussmuster",
73
+ "excludeRegexHelp": "Regulärer Ausdruck zum Herausfiltern übereinstimmender Inhalte",
74
+ "save": "Einstellungen speichern",
75
+ "dangerZone": "Gefahrenzone",
76
+ "deleteWarning": "Das Löschen dieses Kanals entfernt dauerhaft alle Feeds und Elemente. Diese Aktion kann nicht rückgängig gemacht werden.",
77
+ "deleteConfirm": "Sind Sie sicher, dass Sie diesen Kanal und alle seine Inhalte löschen möchten?",
78
+ "delete": "Kanal löschen",
79
+ "types": {
80
+ "like": "Gefällt mir",
81
+ "repost": "Geteilte Beiträge",
82
+ "bookmark": "Lesezeichen",
83
+ "reply": "Antworten",
84
+ "checkin": "Check-ins"
85
+ }
86
+ },
87
+ "search": {
88
+ "title": "Suchen",
89
+ "placeholder": "URL oder Suchbegriff eingeben",
90
+ "submit": "Suchen",
91
+ "noResults": "Keine Ergebnisse gefunden"
92
+ },
93
+ "preview": {
94
+ "title": "Vorschau",
95
+ "subscribe": "Diesen Feed abonnieren"
96
+ },
97
+ "error": {
98
+ "channelNotFound": "Kanal nicht gefunden",
99
+ "feedNotFound": "Feed nicht gefunden",
100
+ "invalidUrl": "Ungültige URL",
101
+ "invalidAction": "Ungültige Aktion"
102
+ }
103
+ }
104
+ }
@@ -0,0 +1,104 @@
1
+ {
2
+ "microsub": {
3
+ "reader": {
4
+ "title": "Lector",
5
+ "empty": "No hay elementos para mostrar",
6
+ "markAllRead": "Marcar todo como leído",
7
+ "showRead": "Mostrar leídos ({{count}})",
8
+ "hideRead": "Ocultar elementos leídos",
9
+ "allRead": "¡Todo al día!",
10
+ "newer": "Más reciente",
11
+ "older": "Más antiguo"
12
+ },
13
+ "channels": {
14
+ "title": "Canales",
15
+ "name": "Nombre del canal",
16
+ "new": "Nuevo canal",
17
+ "create": "Crear canal",
18
+ "delete": "Eliminar canal",
19
+ "settings": "Configuración del canal",
20
+ "empty": "Todavía no hay canales. Creá uno para empezar.",
21
+ "notifications": "Notificaciones"
22
+ },
23
+ "timeline": {
24
+ "title": "Línea de tiempo",
25
+ "empty": "No hay elementos en este canal",
26
+ "markRead": "Marcar como leído",
27
+ "markUnread": "Marcar como no leído",
28
+ "remove": "Eliminar"
29
+ },
30
+ "feeds": {
31
+ "title": "Feeds",
32
+ "follow": "Seguir",
33
+ "subscribe": "Suscribirse a un feed",
34
+ "unfollow": "Dejar de seguir",
35
+ "empty": "No se sigue ningún feed en este canal",
36
+ "url": "URL del feed",
37
+ "urlPlaceholder": "https://ejemplo.com/feed.xml",
38
+ "edit": "Editar feed",
39
+ "rediscover": "Redescubrir feed",
40
+ "refresh": "Actualizar ahora",
41
+ "status": {
42
+ "active": "Activo",
43
+ "error": "Error",
44
+ "stale": "Obsoleto"
45
+ }
46
+ },
47
+ "item": {
48
+ "reply": "Responder",
49
+ "like": "Me gusta",
50
+ "repost": "Repostear",
51
+ "bookmark": "Marcador",
52
+ "viewOriginal": "Ver original"
53
+ },
54
+ "compose": {
55
+ "title": "Redactar",
56
+ "content": "¿Qué estás pensando?",
57
+ "comment": "Agregar un comentario (opcional)",
58
+ "commentHint": "Tu comentario se incluirá cuando esto se sindique",
59
+ "syndicateTo": "Sindicar a",
60
+ "syndicateHint": "Seleccioná dónde publicar esto",
61
+ "submit": "Publicar",
62
+ "cancel": "Cancelar",
63
+ "replyTo": "Respondiendo a",
64
+ "likeOf": "Me gusta",
65
+ "repostOf": "Reposteando",
66
+ "bookmarkOf": "Marcando"
67
+ },
68
+ "settings": {
69
+ "title": "Configuración de {{channel}}",
70
+ "excludeTypes": "Excluir tipos de interacción",
71
+ "excludeTypesHelp": "Seleccioná los tipos de publicaciones que querés ocultar de este canal",
72
+ "excludeRegex": "Patrón de exclusión",
73
+ "excludeRegexHelp": "Expresión regular para filtrar contenido coincidente",
74
+ "save": "Guardar configuración",
75
+ "dangerZone": "Zona de peligro",
76
+ "deleteWarning": "Eliminar este canal eliminará permanentemente todos los feeds y elementos. Esta acción no se puede deshacer.",
77
+ "deleteConfirm": "¿Estás seguro de que querés eliminar este canal y todo su contenido?",
78
+ "delete": "Eliminar canal",
79
+ "types": {
80
+ "like": "Me gusta",
81
+ "repost": "Reposteos",
82
+ "bookmark": "Marcadores",
83
+ "reply": "Respuestas",
84
+ "checkin": "Registros"
85
+ }
86
+ },
87
+ "search": {
88
+ "title": "Buscar",
89
+ "placeholder": "Ingresá URL o término de búsqueda",
90
+ "submit": "Buscar",
91
+ "noResults": "No se encontraron resultados"
92
+ },
93
+ "preview": {
94
+ "title": "Vista previa",
95
+ "subscribe": "Suscribirse a este feed"
96
+ },
97
+ "error": {
98
+ "channelNotFound": "Canal no encontrado",
99
+ "feedNotFound": "Feed no encontrado",
100
+ "invalidUrl": "URL no válida",
101
+ "invalidAction": "Acción no válida"
102
+ }
103
+ }
104
+ }