feedscout 1.8.0 → 2.0.0-beta.1

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.
Files changed (236) hide show
  1. package/README.md +1 -1
  2. package/dist/blogrolls/extractors.d.ts +2 -2
  3. package/dist/blogrolls/extractors.js +2 -2
  4. package/dist/blogrolls/index.js +4 -5
  5. package/dist/blogrolls.d.ts +2 -2
  6. package/dist/blogrolls.js +2 -2
  7. package/dist/common/discover/defaults.d.ts +6 -0
  8. package/dist/common/discover/defaults.js +44 -0
  9. package/dist/common/discover/index.js +2 -2
  10. package/dist/common/discover/utils.d.ts +3 -6
  11. package/dist/common/discover/utils.js +1 -28
  12. package/dist/common/locales.js +214 -10
  13. package/dist/common/types.d.ts +3 -2
  14. package/dist/common/uris/feed/types.d.ts +2 -2
  15. package/dist/common/uris/html/types.d.ts +4 -4
  16. package/dist/common/utils.d.ts +6 -4
  17. package/dist/common/utils.js +13 -11
  18. package/dist/favicons/extractors.d.ts +2 -2
  19. package/dist/favicons/extractors.js +2 -2
  20. package/dist/favicons/index.js +4 -5
  21. package/dist/favicons.d.ts +2 -2
  22. package/dist/favicons.js +2 -2
  23. package/dist/feeds/defaults.d.ts +2 -1
  24. package/dist/feeds/defaults.js +116 -9
  25. package/dist/feeds/extractors.d.ts +2 -2
  26. package/dist/feeds/extractors.js +6 -6
  27. package/dist/feeds/index.js +4 -5
  28. package/dist/feeds/platform/handlers/acast.js +26 -0
  29. package/dist/feeds/platform/handlers/ameblo.js +36 -0
  30. package/dist/feeds/platform/handlers/applePodcasts.js +1 -1
  31. package/dist/feeds/platform/handlers/arena.js +42 -0
  32. package/dist/feeds/platform/handlers/artstation.js +52 -0
  33. package/dist/feeds/platform/handlers/audioboom.js +23 -0
  34. package/dist/feeds/platform/handlers/bearblog.js +45 -0
  35. package/dist/feeds/platform/handlers/behance.js +7 -0
  36. package/dist/feeds/platform/handlers/blogspot.js +38 -2
  37. package/dist/feeds/platform/handlers/bookwyrm.js +48 -0
  38. package/dist/feeds/platform/handlers/buttondown.js +43 -0
  39. package/dist/feeds/platform/handlers/buzzsprout.js +22 -0
  40. package/dist/feeds/platform/handlers/codeberg.js +5 -5
  41. package/dist/feeds/platform/handlers/dailymotion.js +16 -0
  42. package/dist/feeds/platform/handlers/deviantart.js +25 -6
  43. package/dist/feeds/platform/handlers/devto.js +8 -0
  44. package/dist/feeds/platform/handlers/discourse.js +70 -0
  45. package/dist/feeds/platform/handlers/dreamwidth.js +48 -0
  46. package/dist/feeds/platform/handlers/exblog.js +35 -0
  47. package/dist/feeds/platform/handlers/fireside.js +24 -0
  48. package/dist/feeds/platform/handlers/friendica.js +44 -0
  49. package/dist/feeds/platform/handlers/ghost.js +30 -0
  50. package/dist/feeds/platform/handlers/github.js +6 -0
  51. package/dist/feeds/platform/handlers/githubGist.js +16 -5
  52. package/dist/feeds/platform/handlers/gitlab.js +31 -13
  53. package/dist/feeds/platform/handlers/goodreads.js +18 -8
  54. package/dist/feeds/platform/handlers/hackernews.js +21 -0
  55. package/dist/feeds/platform/handlers/hashnode.js +1 -1
  56. package/dist/feeds/platform/handlers/hatenablog.js +4 -1
  57. package/dist/feeds/platform/handlers/hearthis.js +32 -0
  58. package/dist/feeds/platform/handlers/heyWorld.js +18 -0
  59. package/dist/feeds/platform/handlers/insanejournal.js +69 -0
  60. package/dist/feeds/platform/handlers/itchio.js +24 -1
  61. package/dist/feeds/platform/handlers/lemmy.js +46 -4
  62. package/dist/feeds/platform/handlers/letterboxd.js +4 -0
  63. package/dist/feeds/platform/handlers/libsyn.js +25 -0
  64. package/dist/feeds/platform/handlers/listed.js +20 -0
  65. package/dist/feeds/platform/handlers/livejournal.js +68 -0
  66. package/dist/feeds/platform/handlers/mastodon.js +32 -0
  67. package/dist/feeds/platform/handlers/mataroa.js +16 -0
  68. package/dist/feeds/platform/handlers/medium.js +2 -2
  69. package/dist/feeds/platform/handlers/microblog.js +55 -0
  70. package/dist/feeds/platform/handlers/misskey.js +40 -0
  71. package/dist/feeds/platform/handlers/myanimelist.js +43 -0
  72. package/dist/feeds/platform/handlers/naverBlog.js +21 -0
  73. package/dist/feeds/platform/handlers/nebula.js +68 -0
  74. package/dist/feeds/platform/handlers/note.js +53 -0
  75. package/dist/feeds/platform/handlers/observable.js +34 -0
  76. package/dist/feeds/platform/handlers/odysee.js +20 -0
  77. package/dist/feeds/platform/handlers/pagecord.js +16 -0
  78. package/dist/feeds/platform/handlers/paragraph.js +1 -2
  79. package/dist/feeds/platform/handlers/pika.js +35 -0
  80. package/dist/feeds/platform/handlers/pinterest.js +13 -0
  81. package/dist/feeds/platform/handlers/pixelfed.js +46 -0
  82. package/dist/feeds/platform/handlers/pleroma.js +34 -0
  83. package/dist/feeds/platform/handlers/podbean.js +29 -0
  84. package/dist/feeds/platform/handlers/podigee.js +29 -0
  85. package/dist/feeds/platform/handlers/posthaven.js +24 -0
  86. package/dist/feeds/platform/handlers/prose.js +26 -0
  87. package/dist/feeds/platform/handlers/qiita.js +58 -0
  88. package/dist/feeds/platform/handlers/reddit.js +83 -9
  89. package/dist/feeds/platform/handlers/rssCom.js +20 -0
  90. package/dist/feeds/platform/handlers/seesaa.js +22 -0
  91. package/dist/feeds/platform/handlers/sourceforge.js +37 -4
  92. package/dist/feeds/platform/handlers/spreaker.js +21 -0
  93. package/dist/feeds/platform/handlers/stackExchange.js +26 -2
  94. package/dist/feeds/platform/handlers/steam.js +7 -0
  95. package/dist/feeds/platform/handlers/tildes.js +41 -0
  96. package/dist/feeds/platform/handlers/tistory.js +16 -0
  97. package/dist/feeds/platform/handlers/transistor.js +29 -0
  98. package/dist/feeds/platform/handlers/velog.js +24 -0
  99. package/dist/feeds/platform/handlers/vimeo.js +6 -0
  100. package/dist/feeds/platform/handlers/weblogLol.js +26 -0
  101. package/dist/feeds/platform/handlers/weebly.js +25 -0
  102. package/dist/feeds/platform/handlers/wordpress.js +173 -28
  103. package/dist/feeds/platform/handlers/writeas.js +51 -0
  104. package/dist/feeds/platform/handlers/ximalaya.js +1 -1
  105. package/dist/feeds/platform/handlers/youtube.js +4 -1
  106. package/dist/feeds/platform/handlers/zenn.js +54 -0
  107. package/dist/feeds.d.ts +3 -3
  108. package/dist/feeds.js +3 -3
  109. package/dist/hubs/discover/index.js +3 -3
  110. package/dist/hubs/feed/index.js +2 -2
  111. package/dist/hubs/headers/index.js +2 -2
  112. package/dist/hubs/html/index.js +2 -2
  113. package/dist/index.d.ts +2 -1
  114. package/dist/index.js +2 -1
  115. package/dist/utils.d.ts +2 -1
  116. package/package.json +24 -64
  117. package/dist/blogrolls/defaults.cjs +0 -51
  118. package/dist/blogrolls/defaults.d.cts +0 -17
  119. package/dist/blogrolls/extractors.cjs +0 -21
  120. package/dist/blogrolls/extractors.d.cts +0 -7
  121. package/dist/blogrolls/index.cjs +0 -26
  122. package/dist/blogrolls/index.d.cts +0 -7
  123. package/dist/blogrolls/types.d.cts +0 -6
  124. package/dist/blogrolls.cjs +0 -13
  125. package/dist/blogrolls.d.cts +0 -4
  126. package/dist/common/discover/index.cjs +0 -105
  127. package/dist/common/discover/utils.cjs +0 -130
  128. package/dist/common/discover/utils.d.cts +0 -8
  129. package/dist/common/locales.cjs +0 -144
  130. package/dist/common/types.cjs +0 -10
  131. package/dist/common/types.d.cts +0 -89
  132. package/dist/common/uris/feed/index.cjs +0 -12
  133. package/dist/common/uris/feed/index.d.cts +0 -6
  134. package/dist/common/uris/feed/types.d.cts +0 -9
  135. package/dist/common/uris/guess/index.cjs +0 -8
  136. package/dist/common/uris/guess/index.d.cts +0 -7
  137. package/dist/common/uris/guess/types.d.cts +0 -10
  138. package/dist/common/uris/guess/utils.cjs +0 -42
  139. package/dist/common/uris/guess/utils.d.cts +0 -8
  140. package/dist/common/uris/headers/index.cjs +0 -26
  141. package/dist/common/uris/headers/index.d.cts +0 -6
  142. package/dist/common/uris/headers/types.d.cts +0 -9
  143. package/dist/common/uris/html/handlers.cjs +0 -45
  144. package/dist/common/uris/html/index.cjs +0 -19
  145. package/dist/common/uris/html/index.d.cts +0 -6
  146. package/dist/common/uris/html/types.d.cts +0 -12
  147. package/dist/common/uris/index.cjs +0 -32
  148. package/dist/common/uris/platform/index.cjs +0 -10
  149. package/dist/common/uris/platform/index.d.cts +0 -7
  150. package/dist/common/uris/platform/types.d.cts +0 -13
  151. package/dist/common/utils.cjs +0 -97
  152. package/dist/common/utils.d.cts +0 -10
  153. package/dist/favicons/defaults.cjs +0 -65
  154. package/dist/favicons/defaults.d.cts +0 -18
  155. package/dist/favicons/extractors.cjs +0 -25
  156. package/dist/favicons/extractors.d.cts +0 -7
  157. package/dist/favicons/index.cjs +0 -30
  158. package/dist/favicons/index.d.cts +0 -7
  159. package/dist/favicons/platform/handlers/bluesky.cjs +0 -27
  160. package/dist/favicons/platform/handlers/codeberg.cjs +0 -19
  161. package/dist/favicons/platform/handlers/deviantart.cjs +0 -29
  162. package/dist/favicons/platform/handlers/devto.cjs +0 -31
  163. package/dist/favicons/platform/handlers/github.cjs +0 -19
  164. package/dist/favicons/platform/handlers/githubGist.cjs +0 -19
  165. package/dist/favicons/platform/handlers/gitlab.cjs +0 -38
  166. package/dist/favicons/platform/handlers/lobsters.cjs +0 -21
  167. package/dist/favicons/platform/handlers/mastodon.cjs +0 -40
  168. package/dist/favicons/platform/handlers/reddit.cjs +0 -42
  169. package/dist/favicons/platform/handlers/sourceforge.cjs +0 -21
  170. package/dist/favicons/platform/handlers/tumblr.cjs +0 -16
  171. package/dist/favicons/types.d.cts +0 -4
  172. package/dist/favicons/utils.cjs +0 -10
  173. package/dist/favicons.cjs +0 -12
  174. package/dist/favicons.d.cts +0 -4
  175. package/dist/feeds/defaults.cjs +0 -178
  176. package/dist/feeds/defaults.d.cts +0 -20
  177. package/dist/feeds/extractors.cjs +0 -46
  178. package/dist/feeds/extractors.d.cts +0 -7
  179. package/dist/feeds/index.cjs +0 -27
  180. package/dist/feeds/index.d.cts +0 -7
  181. package/dist/feeds/platform/handlers/applePodcasts.cjs +0 -26
  182. package/dist/feeds/platform/handlers/behance.cjs +0 -49
  183. package/dist/feeds/platform/handlers/blogspot.cjs +0 -36
  184. package/dist/feeds/platform/handlers/bluesky.cjs +0 -20
  185. package/dist/feeds/platform/handlers/codeberg.cjs +0 -69
  186. package/dist/feeds/platform/handlers/csdn.cjs +0 -20
  187. package/dist/feeds/platform/handlers/dailymotion.cjs +0 -70
  188. package/dist/feeds/platform/handlers/deviantart.cjs +0 -66
  189. package/dist/feeds/platform/handlers/devto.cjs +0 -50
  190. package/dist/feeds/platform/handlers/douban.cjs +0 -56
  191. package/dist/feeds/platform/handlers/github.cjs +0 -116
  192. package/dist/feeds/platform/handlers/githubGist.cjs +0 -45
  193. package/dist/feeds/platform/handlers/gitlab.cjs +0 -80
  194. package/dist/feeds/platform/handlers/goodreads.cjs +0 -39
  195. package/dist/feeds/platform/handlers/hashnode.cjs +0 -16
  196. package/dist/feeds/platform/handlers/hatenablog.cjs +0 -53
  197. package/dist/feeds/platform/handlers/itchio.cjs +0 -88
  198. package/dist/feeds/platform/handlers/kickstarter.cjs +0 -22
  199. package/dist/feeds/platform/handlers/lemmy.cjs +0 -46
  200. package/dist/feeds/platform/handlers/letterboxd.cjs +0 -42
  201. package/dist/feeds/platform/handlers/lobsters.cjs +0 -57
  202. package/dist/feeds/platform/handlers/mastodon.cjs +0 -42
  203. package/dist/feeds/platform/handlers/medium.cjs +0 -68
  204. package/dist/feeds/platform/handlers/paragraph.cjs +0 -21
  205. package/dist/feeds/platform/handlers/pinterest.cjs +0 -44
  206. package/dist/feeds/platform/handlers/producthunt.cjs +0 -29
  207. package/dist/feeds/platform/handlers/reddit.cjs +0 -75
  208. package/dist/feeds/platform/handlers/soundcloud.cjs +0 -39
  209. package/dist/feeds/platform/handlers/sourceforge.cjs +0 -20
  210. package/dist/feeds/platform/handlers/stackExchange.cjs +0 -40
  211. package/dist/feeds/platform/handlers/steam.cjs +0 -28
  212. package/dist/feeds/platform/handlers/substack.cjs +0 -23
  213. package/dist/feeds/platform/handlers/tumblr.cjs +0 -24
  214. package/dist/feeds/platform/handlers/v2ex.cjs +0 -35
  215. package/dist/feeds/platform/handlers/vimeo.cjs +0 -69
  216. package/dist/feeds/platform/handlers/wordpress.cjs +0 -64
  217. package/dist/feeds/platform/handlers/wpengine.cjs +0 -10
  218. package/dist/feeds/platform/handlers/ximalaya.cjs +0 -20
  219. package/dist/feeds/platform/handlers/youtube.cjs +0 -94
  220. package/dist/feeds/types.d.cts +0 -9
  221. package/dist/feeds.cjs +0 -15
  222. package/dist/feeds.d.cts +0 -4
  223. package/dist/hubs/discover/index.cjs +0 -30
  224. package/dist/hubs/discover/index.d.cts +0 -7
  225. package/dist/hubs/discover/types.d.cts +0 -15
  226. package/dist/hubs/feed/index.cjs +0 -32
  227. package/dist/hubs/headers/index.cjs +0 -17
  228. package/dist/hubs/html/index.cjs +0 -28
  229. package/dist/hubs.cjs +0 -0
  230. package/dist/hubs.d.cts +0 -2
  231. package/dist/index.cjs +0 -12
  232. package/dist/index.d.cts +0 -7
  233. package/dist/methods.cjs +0 -15
  234. package/dist/methods.d.cts +0 -7
  235. package/dist/utils.cjs +0 -9
  236. package/dist/utils.d.cts +0 -2
@@ -0,0 +1,48 @@
1
+ import { composeHint, hasMetaContent } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/bookwyrm.ts
3
+ const profileRegex = /^\/user\/([^/]+)/;
4
+ const shelfRegex = /^\/user\/([^/]+)\/(?:shelf|books)\/([^/]+)\/?/;
5
+ const isBookwyrmHtml = (content) => {
6
+ return hasMetaContent(content, "generator", "BookWyrm");
7
+ };
8
+ const bookwyrmHandler = {
9
+ match: (url, content) => {
10
+ try {
11
+ if (!content || !isBookwyrmHtml(content)) return false;
12
+ const { pathname } = new URL(url);
13
+ return profileRegex.test(pathname);
14
+ } catch {}
15
+ return false;
16
+ },
17
+ resolve: (url) => {
18
+ try {
19
+ const { origin, pathname } = new URL(url);
20
+ const match = pathname.match(profileRegex);
21
+ if (!match?.[1]) return [];
22
+ const user = match[1];
23
+ const uris = [];
24
+ const shelfMatch = pathname.match(shelfRegex);
25
+ if (shelfMatch?.[2]) uris.push({
26
+ uri: `${origin}/user/${user}/${pathname.split("/")[3]}/${shelfMatch[2]}/rss`,
27
+ hint: composeHint("bookwyrm:shelf")
28
+ });
29
+ uris.push({
30
+ uri: `${origin}/user/${user}/rss`,
31
+ hint: composeHint("bookwyrm:activity")
32
+ }, {
33
+ uri: `${origin}/user/${user}/rss-reviews`,
34
+ hint: composeHint("bookwyrm:reviews")
35
+ }, {
36
+ uri: `${origin}/user/${user}/rss-quotes`,
37
+ hint: composeHint("bookwyrm:quotes")
38
+ }, {
39
+ uri: `${origin}/user/${user}/rss-comments`,
40
+ hint: composeHint("bookwyrm:comments")
41
+ });
42
+ return uris;
43
+ } catch {}
44
+ return [];
45
+ }
46
+ };
47
+ //#endregion
48
+ export { bookwyrmHandler };
@@ -0,0 +1,43 @@
1
+ import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/buttondown.ts
3
+ const hosts = [
4
+ "buttondown.com",
5
+ "www.buttondown.com",
6
+ "buttondown.email",
7
+ "www.buttondown.email"
8
+ ];
9
+ const excludedPaths = [
10
+ "about",
11
+ "api",
12
+ "blog",
13
+ "changelog",
14
+ "docs",
15
+ "features",
16
+ "help",
17
+ "legal",
18
+ "login",
19
+ "pricing",
20
+ "privacy",
21
+ "refer",
22
+ "register",
23
+ "settings",
24
+ "terms"
25
+ ];
26
+ const buttondownHandler = {
27
+ match: (url) => {
28
+ return isHostOf(url, hosts);
29
+ },
30
+ resolve: (url) => {
31
+ const { pathname } = new URL(url);
32
+ const pathSegments = pathname.split("/").filter(Boolean);
33
+ if (pathSegments.length === 0) return [];
34
+ const username = pathSegments[0];
35
+ if (isAnyOf(username, excludedPaths)) return [];
36
+ return [{
37
+ uri: `https://buttondown.com/${username}/rss`,
38
+ hint: composeHint("buttondown:newsletter")
39
+ }];
40
+ }
41
+ };
42
+ //#endregion
43
+ export { buttondownHandler };
@@ -0,0 +1,22 @@
1
+ import { composeHint, isHostOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/buzzsprout.ts
3
+ const hosts = ["buzzsprout.com", "www.buzzsprout.com"];
4
+ const numericRegex = /^\d+$/;
5
+ const buzzsproutHandler = {
6
+ match: (url) => {
7
+ return isHostOf(url, hosts);
8
+ },
9
+ resolve: (url) => {
10
+ const { pathname } = new URL(url);
11
+ const pathSegments = pathname.split("/").filter(Boolean);
12
+ if (pathSegments.length === 0) return [];
13
+ const podcastId = pathSegments[0];
14
+ if (!numericRegex.test(podcastId)) return [];
15
+ return [{
16
+ uri: `https://rss.buzzsprout.com/${podcastId}.rss`,
17
+ hint: composeHint("buzzsprout:podcast")
18
+ }];
19
+ }
20
+ };
21
+ //#endregion
22
+ export { buzzsproutHandler };
@@ -27,7 +27,7 @@ const codebergHandler = {
27
27
  if (pathSegments.length === 1) {
28
28
  const user = pathSegments[0];
29
29
  if (!isAnyOf(user, excludedPaths)) return [{
30
- uri: `${origin}/${user}.rss`,
30
+ uri: [`${origin}/${user}.atom`, `${origin}/${user}.rss`],
31
31
  hint: composeHint("codeberg:activity")
32
32
  }];
33
33
  }
@@ -37,19 +37,19 @@ const codebergHandler = {
37
37
  if (!isAnyOf(user, excludedPaths)) {
38
38
  const feeds = [
39
39
  {
40
- uri: `${origin}/${user}/${repo}/releases.rss`,
40
+ uri: [`${origin}/${user}/${repo}/releases.atom`, `${origin}/${user}/${repo}/releases.rss`],
41
41
  hint: composeHint("codeberg:releases")
42
42
  },
43
43
  {
44
- uri: `${origin}/${user}/${repo}/tags.rss`,
44
+ uri: [`${origin}/${user}/${repo}/tags.atom`, `${origin}/${user}/${repo}/tags.rss`],
45
45
  hint: composeHint("codeberg:tags")
46
46
  },
47
47
  {
48
- uri: `${origin}/${user}/${repo}.rss`,
48
+ uri: [`${origin}/${user}/${repo}.atom`, `${origin}/${user}/${repo}.rss`],
49
49
  hint: composeHint("codeberg:activity")
50
50
  }
51
51
  ];
52
- if (pathSegments[2] === "src" && pathSegments[3] === "branch" && pathSegments[4]) {
52
+ if (isHostOf(url, ["gitea.com", "www.gitea.com"]) && pathSegments[2] === "src" && pathSegments[3] === "branch" && pathSegments[4]) {
53
53
  const branch = pathSegments[4];
54
54
  const filePath = pathSegments.slice(5).join("/");
55
55
  feeds.unshift({
@@ -3,6 +3,8 @@ import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
3
3
  const hosts = ["dailymotion.com", "www.dailymotion.com"];
4
4
  const userRegex = /^\/([a-zA-Z0-9_-]+)$/;
5
5
  const playlistRegex = /^\/playlist\/([a-zA-Z0-9_-]+)/;
6
+ const channelRegex = /^\/channel\/([a-zA-Z0-9_-]+)/;
7
+ const searchRegex = /^\/search\/([^/]+)/;
6
8
  const excludedPaths = [
7
9
  "signin",
8
10
  "signout",
@@ -50,11 +52,25 @@ const dailymotionHandler = {
50
52
  },
51
53
  resolve: (url) => {
52
54
  const { pathname } = new URL(url);
55
+ if (pathname === "/" || pathname === "" || pathname === "/trending") return [{
56
+ uri: "https://www.dailymotion.com/rss/trending",
57
+ hint: composeHint("dailymotion:trending")
58
+ }];
53
59
  const playlistMatch = pathname.match(playlistRegex);
54
60
  if (playlistMatch?.[1]) return [{
55
61
  uri: `https://www.dailymotion.com/rss/playlist/${playlistMatch[1]}`,
56
62
  hint: composeHint("dailymotion:playlist")
57
63
  }];
64
+ const searchMatch = pathname.match(searchRegex);
65
+ if (searchMatch?.[1]) return [{
66
+ uri: `https://www.dailymotion.com/rss/search/${searchMatch[1]}`,
67
+ hint: composeHint("dailymotion:search")
68
+ }];
69
+ const channelMatch = pathname.match(channelRegex);
70
+ if (channelMatch?.[1]) return [{
71
+ uri: `https://www.dailymotion.com/rss/channel/${channelMatch[1]}`,
72
+ hint: composeHint("dailymotion:channel")
73
+ }];
58
74
  const userMatch = pathname.match(userRegex);
59
75
  if (userMatch?.[1]) {
60
76
  const username = userMatch[1];
@@ -3,22 +3,25 @@ import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
3
3
  const tagRegex = /^\/tag\/([^/]+)/;
4
4
  const favouritesRegex = /^\/([a-zA-Z0-9_-]+)\/favourites\/?$/;
5
5
  const folderRegex = /^\/([a-zA-Z0-9_-]+)\/gallery\/(\d+)(?:\/|$)/;
6
+ const journalRegex = /^\/([a-zA-Z0-9_-]+)\/journal(?:\/|$)/;
6
7
  const profileRegex = /^\/([a-zA-Z0-9_-]+)(?:\/gallery(?:\/all)?)?(?:\/|$)/;
7
8
  const hosts = ["deviantart.com", "www.deviantart.com"];
8
9
  const feedBaseUrl = "https://backend.deviantart.com/rss.xml";
9
10
  const excludedPaths = [
10
11
  "about",
12
+ "core-membership",
13
+ "daily-deviations",
14
+ "developers",
11
15
  "join",
12
- "search",
13
- "topic",
14
- "watch",
15
16
  "notifications",
17
+ "popular",
18
+ "search",
16
19
  "settings",
17
- "submit",
18
20
  "shop",
19
- "core-membership",
21
+ "submit",
20
22
  "team",
21
- "developers"
23
+ "topic",
24
+ "watch"
22
25
  ];
23
26
  const deviantartHandler = {
24
27
  match: (url) => {
@@ -26,6 +29,14 @@ const deviantartHandler = {
26
29
  },
27
30
  resolve: (url) => {
28
31
  const { pathname } = new URL(url);
32
+ if (pathname === "/daily-deviations" || pathname === "/daily-deviations/") return [{
33
+ uri: `${feedBaseUrl}?q=${encodeURIComponent("special:dd")}`,
34
+ hint: composeHint("deviantart:daily-deviations")
35
+ }];
36
+ if (pathname === "/popular" || pathname === "/popular/") return [{
37
+ uri: `${feedBaseUrl}?type=deviation&q=${encodeURIComponent("boost:popular")}`,
38
+ hint: composeHint("deviantart:popular")
39
+ }];
29
40
  const tagMatch = pathname.match(tagRegex);
30
41
  if (tagMatch?.[1]) {
31
42
  const tag = tagMatch[1];
@@ -51,6 +62,14 @@ const deviantartHandler = {
51
62
  hint: composeHint("deviantart:gallery")
52
63
  }];
53
64
  }
65
+ const journalMatch = pathname.match(journalRegex);
66
+ if (journalMatch?.[1]) {
67
+ const username = journalMatch[1];
68
+ if (!isAnyOf(username, excludedPaths)) return [{
69
+ uri: `${feedBaseUrl}?q=${encodeURIComponent(`journal:${username}`)}`,
70
+ hint: composeHint("deviantart:journal")
71
+ }];
72
+ }
54
73
  const username = pathname.match(profileRegex)?.[1];
55
74
  if (!username || isAnyOf(username, excludedPaths)) return [];
56
75
  const query = `by:${username} sort:time meta:all`;
@@ -28,6 +28,14 @@ const devtoHandler = {
28
28
  },
29
29
  resolve: (url) => {
30
30
  const { pathname } = new URL(url);
31
+ if (pathname === "/" || pathname === "") return [{
32
+ uri: "https://dev.to/feed",
33
+ hint: composeHint("devto:community")
34
+ }];
35
+ if (pathname === "/latest" || pathname === "/latest/") return [{
36
+ uri: "https://dev.to/feed/latest",
37
+ hint: composeHint("devto:latest")
38
+ }];
31
39
  const userMatch = pathname.match(userRegex);
32
40
  if (userMatch?.[1]) {
33
41
  const username = userMatch[1];
@@ -0,0 +1,70 @@
1
+ import { composeHint, hasMetaContent } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/discourse.ts
3
+ const userRegex = /^\/u\/([^/]+)/;
4
+ const categoryRegex = /^\/c\/(.+?)\/?$/;
5
+ const topicRegex = /^\/t\/([^/]+)\/(\d+)/;
6
+ const topRegex = /^\/top(?:\/([^/]+))?\/?$/;
7
+ const validTopPeriods = new Set([
8
+ "daily",
9
+ "weekly",
10
+ "monthly",
11
+ "quarterly",
12
+ "yearly",
13
+ "all"
14
+ ]);
15
+ const getTopPeriodSuffix = (pathPeriod, searchParams) => {
16
+ const period = pathPeriod ?? searchParams.get("period") ?? void 0;
17
+ if (period && validTopPeriods.has(period)) return `?period=${period}`;
18
+ return "";
19
+ };
20
+ const isDiscourseHtml = (content) => {
21
+ return hasMetaContent(content, "generator", "Discourse");
22
+ };
23
+ const discourseHandler = {
24
+ match: (url, content) => {
25
+ try {
26
+ if (!content || !isDiscourseHtml(content)) return false;
27
+ new URL(url);
28
+ return true;
29
+ } catch {}
30
+ return false;
31
+ },
32
+ resolve: (url) => {
33
+ try {
34
+ const { origin, pathname, searchParams } = new URL(url);
35
+ const topicMatch = pathname.match(topicRegex);
36
+ if (topicMatch?.[1] && topicMatch?.[2]) return [{
37
+ uri: `${origin}/t/${topicMatch[1]}/${topicMatch[2]}.rss`,
38
+ hint: composeHint("discourse:topic")
39
+ }];
40
+ const userMatch = pathname.match(userRegex);
41
+ if (userMatch?.[1]) return [{
42
+ uri: `${origin}/u/${userMatch[1]}/activity.rss`,
43
+ hint: composeHint("discourse:activity")
44
+ }];
45
+ const categoryMatch = pathname.match(categoryRegex);
46
+ if (categoryMatch?.[1]) return [{
47
+ uri: `${origin}/c/${categoryMatch[1]}.rss`,
48
+ hint: composeHint("discourse:category")
49
+ }];
50
+ const topMatch = pathname.match(topRegex);
51
+ if (topMatch) return [{
52
+ uri: `${origin}/top.rss${getTopPeriodSuffix(topMatch[1], searchParams)}`,
53
+ hint: composeHint("discourse:top")
54
+ }];
55
+ const uris = [];
56
+ uris.push({
57
+ uri: `${origin}/latest.rss`,
58
+ hint: composeHint("discourse:latest")
59
+ });
60
+ uris.push({
61
+ uri: `${origin}/posts.rss`,
62
+ hint: composeHint("discourse:posts")
63
+ });
64
+ return uris;
65
+ } catch {}
66
+ return [];
67
+ }
68
+ };
69
+ //#endregion
70
+ export { discourseHandler };
@@ -0,0 +1,48 @@
1
+ import { composeHint, isHostOf, isSubdomainOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/dreamwidth.ts
3
+ const usersPathRegex = /^\/(?:users\/|~)([^/]+)/;
4
+ const tagRegex = /^\/tag\/([^/]+)/;
5
+ const dreamwidthHandler = {
6
+ match: (url) => {
7
+ if (!isSubdomainOf(url, "dreamwidth.org")) return false;
8
+ if (isHostOf(url, ["www.dreamwidth.org", "dreamwidth.org"])) return usersPathRegex.test(new URL(url).pathname);
9
+ return true;
10
+ },
11
+ resolve: (url) => {
12
+ const { origin, pathname } = new URL(url);
13
+ const uris = [];
14
+ let userOrigin = origin;
15
+ if (isHostOf(url, ["www.dreamwidth.org", "dreamwidth.org"])) {
16
+ const userMatch = pathname.match(usersPathRegex);
17
+ if (userMatch?.[1]) userOrigin = `https://${userMatch[1]}.dreamwidth.org`;
18
+ else return uris;
19
+ }
20
+ const tagMatch = pathname.match(tagRegex);
21
+ if (tagMatch?.[1]) {
22
+ const tag = encodeURIComponent(tagMatch[1]);
23
+ uris.push({
24
+ uri: `${userOrigin}/data/rss?tag=${tag}`,
25
+ hint: composeHint("dreamwidth:posts-tag-rss")
26
+ });
27
+ uris.push({
28
+ uri: `${userOrigin}/data/atom?tag=${tag}`,
29
+ hint: composeHint("dreamwidth:posts-tag-atom")
30
+ });
31
+ }
32
+ uris.push({
33
+ uri: `${userOrigin}/data/rss`,
34
+ hint: composeHint("dreamwidth:posts-rss")
35
+ });
36
+ uris.push({
37
+ uri: `${userOrigin}/data/atom`,
38
+ hint: composeHint("dreamwidth:posts-atom")
39
+ });
40
+ uris.push({
41
+ uri: `${userOrigin}/data/userpics`,
42
+ hint: composeHint("dreamwidth:userpics-atom")
43
+ });
44
+ return uris;
45
+ }
46
+ };
47
+ //#endregion
48
+ export { dreamwidthHandler };
@@ -0,0 +1,35 @@
1
+ import { composeHint, isSubdomainOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/exblog.ts
3
+ const categoryRegex = /^\/i(\d+)/;
4
+ const exblogHandler = {
5
+ match: (url) => {
6
+ return isSubdomainOf(url, "exblog.jp");
7
+ },
8
+ resolve: (url) => {
9
+ const { origin, pathname } = new URL(url);
10
+ const uris = [];
11
+ const categoryMatch = pathname.match(categoryRegex);
12
+ if (categoryMatch?.[1]) {
13
+ const categoryId = categoryMatch[1];
14
+ uris.push({
15
+ uri: `${origin}/i${categoryId}/index.xml`,
16
+ hint: composeHint("exblog:category-rss")
17
+ });
18
+ uris.push({
19
+ uri: `${origin}/i${categoryId}/atom.xml`,
20
+ hint: composeHint("exblog:category-atom")
21
+ });
22
+ }
23
+ uris.push({
24
+ uri: `${origin}/index.xml`,
25
+ hint: composeHint("exblog:posts-rss")
26
+ });
27
+ uris.push({
28
+ uri: `${origin}/atom.xml`,
29
+ hint: composeHint("exblog:posts-atom")
30
+ });
31
+ return uris;
32
+ }
33
+ };
34
+ //#endregion
35
+ export { exblogHandler };
@@ -0,0 +1,24 @@
1
+ import { composeHint, isSubdomainOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/fireside.ts
3
+ const domainSuffix = /\.fireside\.fm$/i;
4
+ const firesideHandler = {
5
+ match: (url) => {
6
+ return isSubdomainOf(url, "fireside.fm");
7
+ },
8
+ resolve: (url) => {
9
+ const { hostname } = new URL(url);
10
+ const slug = hostname.replace(domainSuffix, "");
11
+ const uris = [];
12
+ uris.push({
13
+ uri: `https://feeds.fireside.fm/${slug}/rss`,
14
+ hint: composeHint("fireside:podcast-rss")
15
+ });
16
+ uris.push({
17
+ uri: `https://${slug}.fireside.fm/json`,
18
+ hint: composeHint("fireside:podcast-json")
19
+ });
20
+ return uris;
21
+ }
22
+ };
23
+ //#endregion
24
+ export { firesideHandler };
@@ -0,0 +1,44 @@
1
+ import { composeHint, hasMetaContent } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/friendica.ts
3
+ const profileRegex = /^\/profile\/([^/]+)/;
4
+ const isFriendicaHtml = (content) => {
5
+ return hasMetaContent(content, "generator", "Friendica");
6
+ };
7
+ const friendicaHandler = {
8
+ match: (url, content) => {
9
+ try {
10
+ if (!content || !isFriendicaHtml(content)) return false;
11
+ const { pathname } = new URL(url);
12
+ return profileRegex.test(pathname);
13
+ } catch {}
14
+ return false;
15
+ },
16
+ resolve: (url) => {
17
+ try {
18
+ const { origin, pathname } = new URL(url);
19
+ const match = pathname.match(profileRegex);
20
+ if (!match?.[1]) return [];
21
+ return [
22
+ {
23
+ uri: `${origin}/feed/${match[1]}`,
24
+ hint: composeHint("friendica:posts")
25
+ },
26
+ {
27
+ uri: `${origin}/feed/${match[1]}/comments`,
28
+ hint: composeHint("friendica:comments")
29
+ },
30
+ {
31
+ uri: `${origin}/feed/${match[1]}/replies`,
32
+ hint: composeHint("friendica:replies")
33
+ },
34
+ {
35
+ uri: `${origin}/feed/${match[1]}/activity`,
36
+ hint: composeHint("friendica:activity")
37
+ }
38
+ ];
39
+ } catch {}
40
+ return [];
41
+ }
42
+ };
43
+ //#endregion
44
+ export { friendicaHandler };
@@ -0,0 +1,30 @@
1
+ import { composeHint, isSubdomainOf } from "../../../common/utils.js";
2
+ //#region src/feeds/platform/handlers/ghost.ts
3
+ const tagRegex = /^\/tag\/([^/]+)/;
4
+ const authorRegex = /^\/author\/([^/]+)/;
5
+ const ghostHandler = {
6
+ match: (url) => {
7
+ return isSubdomainOf(url, "ghost.io");
8
+ },
9
+ resolve: (url) => {
10
+ const { origin, pathname } = new URL(url);
11
+ const uris = [];
12
+ const tagMatch = pathname.match(tagRegex);
13
+ if (tagMatch?.[1]) uris.push({
14
+ uri: `${origin}/tag/${tagMatch[1]}/rss/`,
15
+ hint: composeHint("ghost:tag")
16
+ });
17
+ const authorMatch = pathname.match(authorRegex);
18
+ if (authorMatch?.[1]) uris.push({
19
+ uri: `${origin}/author/${authorMatch[1]}/rss/`,
20
+ hint: composeHint("ghost:author")
21
+ });
22
+ uris.push({
23
+ uri: `${origin}/rss/`,
24
+ hint: composeHint("ghost:blog")
25
+ });
26
+ return uris;
27
+ }
28
+ };
29
+ //#endregion
30
+ export { ghostHandler };
@@ -4,6 +4,7 @@ const userRegex = /^\/([^/]+)\/?$/;
4
4
  const repoRegex = /^\/([^/]+)\/([^/]+)/;
5
5
  const wikiRegex = /\/wiki(\/|$)/;
6
6
  const discussionsRegex = /\/discussions(\/|$)/;
7
+ const discussionCategoryRegex = /\/discussions\/categories\/([^/]+)/;
7
8
  const branchRegex = /^\/[^/]+\/[^/]+\/tree\/([^/]+)\/?$/;
8
9
  const fileRegex = /^\/[^/]+\/[^/]+\/(?:blob|commits)\/([^/]+)\/(.+)/;
9
10
  const hosts = ["github.com", "www.github.com"];
@@ -90,6 +91,11 @@ const githubHandler = {
90
91
  uri: `https://github.com/${owner}/${repo}/discussions.atom`,
91
92
  hint: composeHint("github:discussions")
92
93
  });
94
+ const discussionCategoryMatch = pathname.match(discussionCategoryRegex);
95
+ if (discussionCategoryMatch?.[1]) uris.push({
96
+ uri: `https://github.com/${owner}/${repo}/discussions/categories/${discussionCategoryMatch[1]}.atom`,
97
+ hint: composeHint("github:discussion-category")
98
+ });
93
99
  const branchMatch = pathname.match(branchRegex);
94
100
  if (branchMatch?.[1]) {
95
101
  const branch = branchMatch[1];
@@ -2,7 +2,9 @@ import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
2
2
  //#region src/feeds/platform/handlers/githubGist.ts
3
3
  const gistRegex = /^\/([^/]+)\/([a-f0-9]+)/;
4
4
  const starredRegex = /^\/([^/]+)\/starred\/?$/;
5
+ const forksRegex = /^\/([^/]+)\/forks\/?$/;
5
6
  const userRegex = /^\/([^/]+)\/?$/;
7
+ const discoverRegex = /^\/discover\/?$/;
6
8
  const hosts = ["gist.github.com"];
7
9
  const excludedPaths = [
8
10
  "discover",
@@ -17,6 +19,20 @@ const githubGistHandler = {
17
19
  },
18
20
  resolve: (url) => {
19
21
  const { pathname } = new URL(url);
22
+ if (discoverRegex.test(pathname)) return [{
23
+ uri: "https://gist.github.com/discover.atom",
24
+ hint: composeHint("github-gist:discover")
25
+ }];
26
+ const starredMatch = pathname.match(starredRegex);
27
+ if (starredMatch?.[1] && !isAnyOf(starredMatch[1], excludedPaths)) return [{
28
+ uri: `https://gist.github.com/${starredMatch[1]}/starred.atom`,
29
+ hint: composeHint("github-gist:starred")
30
+ }];
31
+ const forksMatch = pathname.match(forksRegex);
32
+ if (forksMatch?.[1] && !isAnyOf(forksMatch[1], excludedPaths)) return [{
33
+ uri: `https://gist.github.com/${forksMatch[1]}/forks.atom`,
34
+ hint: composeHint("github-gist:forks")
35
+ }];
20
36
  const gistMatch = pathname.match(gistRegex);
21
37
  if (gistMatch?.[1] && gistMatch?.[2]) {
22
38
  const username = gistMatch[1];
@@ -26,11 +42,6 @@ const githubGistHandler = {
26
42
  }];
27
43
  return [];
28
44
  }
29
- const starredMatch = pathname.match(starredRegex);
30
- if (starredMatch?.[1] && !isAnyOf(starredMatch[1], excludedPaths)) return [{
31
- uri: `https://gist.github.com/${starredMatch[1]}/starred.atom`,
32
- hint: composeHint("github-gist:starred")
33
- }];
34
45
  const userMatch = pathname.match(userRegex);
35
46
  if (userMatch?.[1] && !isAnyOf(userMatch[1], excludedPaths)) return [{
36
47
  uri: `https://gist.github.com/${userMatch[1]}.atom`,
@@ -54,20 +54,38 @@ const gitlabHandler = {
54
54
  if (pathSegments.length >= 2) {
55
55
  const user = pathSegments[0];
56
56
  const repo = pathSegments[1];
57
- if (!isAnyOf(user, excludedPaths)) return [
58
- {
59
- uri: `${origin}/${user}/${repo}/-/releases.atom`,
60
- hint: composeHint("gitlab:releases")
61
- },
62
- {
63
- uri: `${origin}/${user}/${repo}/-/tags?format=atom`,
64
- hint: composeHint("gitlab:tags")
65
- },
66
- {
67
- uri: `${origin}/${user}/${repo}.atom`,
68
- hint: composeHint("gitlab:activity")
57
+ if (!isAnyOf(user, excludedPaths)) {
58
+ const repoFeeds = [
59
+ {
60
+ uri: `${origin}/${user}/${repo}/-/releases.atom`,
61
+ hint: composeHint("gitlab:releases")
62
+ },
63
+ {
64
+ uri: `${origin}/${user}/${repo}/-/tags?format=atom`,
65
+ hint: composeHint("gitlab:tags")
66
+ },
67
+ {
68
+ uri: `${origin}/${user}/${repo}/-/issues.atom`,
69
+ hint: composeHint("gitlab:issues")
70
+ },
71
+ {
72
+ uri: `${origin}/${user}/${repo}/-/merge_requests.atom`,
73
+ hint: composeHint("gitlab:merge-requests")
74
+ },
75
+ {
76
+ uri: `${origin}/${user}/${repo}.atom`,
77
+ hint: composeHint("gitlab:activity")
78
+ }
79
+ ];
80
+ if (pathSegments[2] === "-" && (pathSegments[3] === "commits" || pathSegments[3] === "tree") && pathSegments[4]) {
81
+ const branch = pathSegments[4];
82
+ repoFeeds.unshift({
83
+ uri: `${origin}/${user}/${repo}/-/commits/${branch}?format=atom`,
84
+ hint: composeHint("gitlab:branch-commits")
85
+ });
69
86
  }
70
- ];
87
+ return repoFeeds;
88
+ }
71
89
  }
72
90
  return [];
73
91
  }
@@ -10,8 +10,9 @@ const goodreadsHandler = {
10
10
  return isHostOf(url, hosts);
11
11
  },
12
12
  resolve: (url) => {
13
- const { origin, pathname } = new URL(url);
13
+ const { origin, pathname, searchParams } = new URL(url);
14
14
  const pathSegments = pathname.split("/").filter(Boolean);
15
+ const shelf = searchParams.get("shelf");
15
16
  if (pathSegments[0] === "user" && pathSegments[1] === "show" && pathSegments[2]) {
16
17
  const userId = parseUserId(pathSegments[2]);
17
18
  if (userId) return [{
@@ -24,13 +25,22 @@ const goodreadsHandler = {
24
25
  }
25
26
  if (pathSegments[0] === "review" && pathSegments[1] === "list" && pathSegments[2]) {
26
27
  const userId = parseUserId(pathSegments[2]);
27
- if (userId) return [{
28
- uri: `${origin}/review/list_rss/${userId}`,
29
- hint: composeHint("goodreads:reviews")
30
- }, {
31
- uri: `${origin}/user/updates_rss/${userId}`,
32
- hint: composeHint("goodreads:updates")
33
- }];
28
+ if (userId) {
29
+ const uris = [];
30
+ if (shelf) uris.push({
31
+ uri: `${origin}/review/list_rss/${userId}?shelf=${encodeURIComponent(shelf)}`,
32
+ hint: composeHint("goodreads:shelf")
33
+ });
34
+ uris.push({
35
+ uri: `${origin}/review/list_rss/${userId}`,
36
+ hint: composeHint("goodreads:reviews")
37
+ });
38
+ uris.push({
39
+ uri: `${origin}/user/updates_rss/${userId}`,
40
+ hint: composeHint("goodreads:updates")
41
+ });
42
+ return uris;
43
+ }
34
44
  }
35
45
  return [];
36
46
  }