feedscout 1.8.1 → 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.
- package/README.md +1 -1
- package/dist/common/discover/utils.d.ts +2 -3
- package/dist/common/locales.js +214 -10
- package/dist/common/types.d.ts +2 -1
- package/dist/common/uris/feed/types.d.ts +2 -2
- package/dist/common/uris/html/types.d.ts +4 -4
- package/dist/common/utils.d.ts +6 -4
- package/dist/common/utils.js +12 -3
- package/dist/feeds/defaults.d.ts +2 -1
- package/dist/feeds/defaults.js +116 -9
- package/dist/feeds/extractors.js +2 -2
- package/dist/feeds/platform/handlers/acast.js +26 -0
- package/dist/feeds/platform/handlers/ameblo.js +36 -0
- package/dist/feeds/platform/handlers/applePodcasts.js +1 -1
- package/dist/feeds/platform/handlers/arena.js +42 -0
- package/dist/feeds/platform/handlers/artstation.js +52 -0
- package/dist/feeds/platform/handlers/audioboom.js +23 -0
- package/dist/feeds/platform/handlers/bearblog.js +45 -0
- package/dist/feeds/platform/handlers/behance.js +7 -0
- package/dist/feeds/platform/handlers/blogspot.js +38 -2
- package/dist/feeds/platform/handlers/bookwyrm.js +48 -0
- package/dist/feeds/platform/handlers/buttondown.js +43 -0
- package/dist/feeds/platform/handlers/buzzsprout.js +22 -0
- package/dist/feeds/platform/handlers/codeberg.js +5 -5
- package/dist/feeds/platform/handlers/dailymotion.js +16 -0
- package/dist/feeds/platform/handlers/deviantart.js +25 -6
- package/dist/feeds/platform/handlers/devto.js +8 -0
- package/dist/feeds/platform/handlers/discourse.js +70 -0
- package/dist/feeds/platform/handlers/dreamwidth.js +48 -0
- package/dist/feeds/platform/handlers/exblog.js +35 -0
- package/dist/feeds/platform/handlers/fireside.js +24 -0
- package/dist/feeds/platform/handlers/friendica.js +44 -0
- package/dist/feeds/platform/handlers/ghost.js +30 -0
- package/dist/feeds/platform/handlers/github.js +6 -0
- package/dist/feeds/platform/handlers/githubGist.js +16 -5
- package/dist/feeds/platform/handlers/gitlab.js +31 -13
- package/dist/feeds/platform/handlers/goodreads.js +18 -8
- package/dist/feeds/platform/handlers/hackernews.js +21 -0
- package/dist/feeds/platform/handlers/hashnode.js +1 -1
- package/dist/feeds/platform/handlers/hatenablog.js +4 -1
- package/dist/feeds/platform/handlers/hearthis.js +32 -0
- package/dist/feeds/platform/handlers/heyWorld.js +18 -0
- package/dist/feeds/platform/handlers/insanejournal.js +69 -0
- package/dist/feeds/platform/handlers/itchio.js +24 -1
- package/dist/feeds/platform/handlers/lemmy.js +46 -4
- package/dist/feeds/platform/handlers/letterboxd.js +4 -0
- package/dist/feeds/platform/handlers/libsyn.js +25 -0
- package/dist/feeds/platform/handlers/listed.js +20 -0
- package/dist/feeds/platform/handlers/livejournal.js +68 -0
- package/dist/feeds/platform/handlers/mastodon.js +32 -0
- package/dist/feeds/platform/handlers/mataroa.js +16 -0
- package/dist/feeds/platform/handlers/medium.js +2 -2
- package/dist/feeds/platform/handlers/microblog.js +55 -0
- package/dist/feeds/platform/handlers/misskey.js +40 -0
- package/dist/feeds/platform/handlers/myanimelist.js +43 -0
- package/dist/feeds/platform/handlers/naverBlog.js +21 -0
- package/dist/feeds/platform/handlers/nebula.js +68 -0
- package/dist/feeds/platform/handlers/note.js +53 -0
- package/dist/feeds/platform/handlers/observable.js +34 -0
- package/dist/feeds/platform/handlers/odysee.js +20 -0
- package/dist/feeds/platform/handlers/pagecord.js +16 -0
- package/dist/feeds/platform/handlers/paragraph.js +1 -2
- package/dist/feeds/platform/handlers/pika.js +35 -0
- package/dist/feeds/platform/handlers/pinterest.js +13 -0
- package/dist/feeds/platform/handlers/pixelfed.js +46 -0
- package/dist/feeds/platform/handlers/pleroma.js +34 -0
- package/dist/feeds/platform/handlers/podbean.js +29 -0
- package/dist/feeds/platform/handlers/podigee.js +29 -0
- package/dist/feeds/platform/handlers/posthaven.js +24 -0
- package/dist/feeds/platform/handlers/prose.js +26 -0
- package/dist/feeds/platform/handlers/qiita.js +58 -0
- package/dist/feeds/platform/handlers/reddit.js +83 -9
- package/dist/feeds/platform/handlers/rssCom.js +20 -0
- package/dist/feeds/platform/handlers/seesaa.js +22 -0
- package/dist/feeds/platform/handlers/sourceforge.js +37 -4
- package/dist/feeds/platform/handlers/spreaker.js +21 -0
- package/dist/feeds/platform/handlers/stackExchange.js +26 -2
- package/dist/feeds/platform/handlers/steam.js +7 -0
- package/dist/feeds/platform/handlers/tildes.js +41 -0
- package/dist/feeds/platform/handlers/tistory.js +16 -0
- package/dist/feeds/platform/handlers/transistor.js +29 -0
- package/dist/feeds/platform/handlers/velog.js +24 -0
- package/dist/feeds/platform/handlers/vimeo.js +6 -0
- package/dist/feeds/platform/handlers/weblogLol.js +26 -0
- package/dist/feeds/platform/handlers/weebly.js +25 -0
- package/dist/feeds/platform/handlers/wordpress.js +173 -28
- package/dist/feeds/platform/handlers/writeas.js +51 -0
- package/dist/feeds/platform/handlers/ximalaya.js +1 -1
- package/dist/feeds/platform/handlers/youtube.js +4 -1
- package/dist/feeds/platform/handlers/zenn.js +54 -0
- package/dist/feeds.d.ts +2 -2
- package/dist/feeds.js +2 -2
- package/dist/utils.d.ts +2 -1
- package/package.json +24 -64
- package/dist/blogrolls/defaults.cjs +0 -51
- package/dist/blogrolls/defaults.d.cts +0 -17
- package/dist/blogrolls/extractors.cjs +0 -21
- package/dist/blogrolls/extractors.d.cts +0 -7
- package/dist/blogrolls/index.cjs +0 -25
- package/dist/blogrolls/index.d.cts +0 -7
- package/dist/blogrolls/types.d.cts +0 -6
- package/dist/blogrolls.cjs +0 -13
- package/dist/blogrolls.d.cts +0 -4
- package/dist/common/discover/defaults.cjs +0 -46
- package/dist/common/discover/defaults.d.cts +0 -6
- package/dist/common/discover/index.cjs +0 -105
- package/dist/common/discover/utils.cjs +0 -101
- package/dist/common/discover/utils.d.cts +0 -6
- package/dist/common/locales.cjs +0 -144
- package/dist/common/types.cjs +0 -10
- package/dist/common/types.d.cts +0 -89
- package/dist/common/uris/feed/index.cjs +0 -12
- package/dist/common/uris/feed/index.d.cts +0 -6
- package/dist/common/uris/feed/types.d.cts +0 -9
- package/dist/common/uris/guess/index.cjs +0 -8
- package/dist/common/uris/guess/index.d.cts +0 -7
- package/dist/common/uris/guess/types.d.cts +0 -10
- package/dist/common/uris/guess/utils.cjs +0 -42
- package/dist/common/uris/guess/utils.d.cts +0 -8
- package/dist/common/uris/headers/index.cjs +0 -26
- package/dist/common/uris/headers/index.d.cts +0 -6
- package/dist/common/uris/headers/types.d.cts +0 -9
- package/dist/common/uris/html/handlers.cjs +0 -45
- package/dist/common/uris/html/index.cjs +0 -19
- package/dist/common/uris/html/index.d.cts +0 -6
- package/dist/common/uris/html/types.d.cts +0 -12
- package/dist/common/uris/index.cjs +0 -32
- package/dist/common/uris/platform/index.cjs +0 -10
- package/dist/common/uris/platform/index.d.cts +0 -7
- package/dist/common/uris/platform/types.d.cts +0 -13
- package/dist/common/utils.cjs +0 -89
- package/dist/common/utils.d.cts +0 -10
- package/dist/favicons/defaults.cjs +0 -65
- package/dist/favicons/defaults.d.cts +0 -18
- package/dist/favicons/extractors.cjs +0 -25
- package/dist/favicons/extractors.d.cts +0 -7
- package/dist/favicons/index.cjs +0 -29
- package/dist/favicons/index.d.cts +0 -7
- package/dist/favicons/platform/handlers/bluesky.cjs +0 -27
- package/dist/favicons/platform/handlers/codeberg.cjs +0 -19
- package/dist/favicons/platform/handlers/deviantart.cjs +0 -29
- package/dist/favicons/platform/handlers/devto.cjs +0 -31
- package/dist/favicons/platform/handlers/github.cjs +0 -19
- package/dist/favicons/platform/handlers/githubGist.cjs +0 -19
- package/dist/favicons/platform/handlers/gitlab.cjs +0 -38
- package/dist/favicons/platform/handlers/lobsters.cjs +0 -21
- package/dist/favicons/platform/handlers/mastodon.cjs +0 -40
- package/dist/favicons/platform/handlers/reddit.cjs +0 -42
- package/dist/favicons/platform/handlers/sourceforge.cjs +0 -21
- package/dist/favicons/platform/handlers/tumblr.cjs +0 -16
- package/dist/favicons/types.d.cts +0 -4
- package/dist/favicons/utils.cjs +0 -10
- package/dist/favicons.cjs +0 -12
- package/dist/favicons.d.cts +0 -4
- package/dist/feeds/defaults.cjs +0 -178
- package/dist/feeds/defaults.d.cts +0 -20
- package/dist/feeds/extractors.cjs +0 -46
- package/dist/feeds/extractors.d.cts +0 -7
- package/dist/feeds/index.cjs +0 -26
- package/dist/feeds/index.d.cts +0 -7
- package/dist/feeds/platform/handlers/applePodcasts.cjs +0 -26
- package/dist/feeds/platform/handlers/behance.cjs +0 -49
- package/dist/feeds/platform/handlers/blogspot.cjs +0 -36
- package/dist/feeds/platform/handlers/bluesky.cjs +0 -20
- package/dist/feeds/platform/handlers/codeberg.cjs +0 -69
- package/dist/feeds/platform/handlers/csdn.cjs +0 -20
- package/dist/feeds/platform/handlers/dailymotion.cjs +0 -70
- package/dist/feeds/platform/handlers/deviantart.cjs +0 -66
- package/dist/feeds/platform/handlers/devto.cjs +0 -50
- package/dist/feeds/platform/handlers/douban.cjs +0 -56
- package/dist/feeds/platform/handlers/github.cjs +0 -116
- package/dist/feeds/platform/handlers/githubGist.cjs +0 -45
- package/dist/feeds/platform/handlers/gitlab.cjs +0 -80
- package/dist/feeds/platform/handlers/goodreads.cjs +0 -39
- package/dist/feeds/platform/handlers/hashnode.cjs +0 -16
- package/dist/feeds/platform/handlers/hatenablog.cjs +0 -53
- package/dist/feeds/platform/handlers/itchio.cjs +0 -88
- package/dist/feeds/platform/handlers/kickstarter.cjs +0 -22
- package/dist/feeds/platform/handlers/lemmy.cjs +0 -46
- package/dist/feeds/platform/handlers/letterboxd.cjs +0 -42
- package/dist/feeds/platform/handlers/lobsters.cjs +0 -57
- package/dist/feeds/platform/handlers/mastodon.cjs +0 -42
- package/dist/feeds/platform/handlers/medium.cjs +0 -68
- package/dist/feeds/platform/handlers/paragraph.cjs +0 -21
- package/dist/feeds/platform/handlers/pinterest.cjs +0 -44
- package/dist/feeds/platform/handlers/producthunt.cjs +0 -29
- package/dist/feeds/platform/handlers/reddit.cjs +0 -75
- package/dist/feeds/platform/handlers/soundcloud.cjs +0 -39
- package/dist/feeds/platform/handlers/sourceforge.cjs +0 -20
- package/dist/feeds/platform/handlers/stackExchange.cjs +0 -40
- package/dist/feeds/platform/handlers/steam.cjs +0 -28
- package/dist/feeds/platform/handlers/substack.cjs +0 -23
- package/dist/feeds/platform/handlers/tumblr.cjs +0 -24
- package/dist/feeds/platform/handlers/v2ex.cjs +0 -35
- package/dist/feeds/platform/handlers/vimeo.cjs +0 -69
- package/dist/feeds/platform/handlers/wordpress.cjs +0 -64
- package/dist/feeds/platform/handlers/wpengine.cjs +0 -10
- package/dist/feeds/platform/handlers/ximalaya.cjs +0 -20
- package/dist/feeds/platform/handlers/youtube.cjs +0 -94
- package/dist/feeds/types.d.cts +0 -9
- package/dist/feeds.cjs +0 -15
- package/dist/feeds.d.cts +0 -4
- package/dist/hubs/discover/index.cjs +0 -30
- package/dist/hubs/discover/index.d.cts +0 -7
- package/dist/hubs/discover/types.d.cts +0 -15
- package/dist/hubs/feed/index.cjs +0 -32
- package/dist/hubs/headers/index.cjs +0 -17
- package/dist/hubs/html/index.cjs +0 -28
- package/dist/hubs.cjs +0 -0
- package/dist/hubs.d.cts +0 -2
- package/dist/index.cjs +0 -13
- package/dist/index.d.cts +0 -8
- package/dist/methods.cjs +0 -15
- package/dist/methods.d.cts +0 -7
- package/dist/utils.cjs +0 -9
- package/dist/utils.d.cts +0 -2
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { composeHint, isHostOf, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/libsyn.ts
|
|
3
|
+
const numericRegex = /^\d+$/;
|
|
4
|
+
const libsynHandler = {
|
|
5
|
+
match: (url) => {
|
|
6
|
+
return isSubdomainOf(url, "libsyn.com");
|
|
7
|
+
},
|
|
8
|
+
resolve: (url) => {
|
|
9
|
+
const { origin, pathname } = new URL(url);
|
|
10
|
+
if (isHostOf(url, "feeds.libsyn.com")) {
|
|
11
|
+
const showId = pathname.split("/").filter(Boolean)[0];
|
|
12
|
+
if (showId && numericRegex.test(showId)) return [{
|
|
13
|
+
uri: `https://feeds.libsyn.com/${showId}/rss`,
|
|
14
|
+
hint: composeHint("libsyn:podcast")
|
|
15
|
+
}];
|
|
16
|
+
return [];
|
|
17
|
+
}
|
|
18
|
+
return [{
|
|
19
|
+
uri: `${origin}/rss`,
|
|
20
|
+
hint: composeHint("libsyn:podcast")
|
|
21
|
+
}];
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
//#endregion
|
|
25
|
+
export { libsynHandler };
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { composeHint, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/listed.ts
|
|
3
|
+
const hosts = ["listed.to", "www.listed.to"];
|
|
4
|
+
const userRegex = /^\/@([^/]+)/;
|
|
5
|
+
const listedHandler = {
|
|
6
|
+
match: (url) => {
|
|
7
|
+
return isHostOf(url, hosts);
|
|
8
|
+
},
|
|
9
|
+
resolve: (url) => {
|
|
10
|
+
const { pathname } = new URL(url);
|
|
11
|
+
const userMatch = pathname.match(userRegex);
|
|
12
|
+
if (!userMatch?.[1]) return [];
|
|
13
|
+
return [{
|
|
14
|
+
uri: `https://listed.to/@${userMatch[1]}/feed.rss`,
|
|
15
|
+
hint: composeHint("listed:blog")
|
|
16
|
+
}];
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
//#endregion
|
|
20
|
+
export { listedHandler };
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { composeHint, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/livejournal.ts
|
|
3
|
+
const wwwUsersPathRegex = /^\/(?:users\/|~)([^/]+)/;
|
|
4
|
+
const legacyUserPathRegex = /^\/([^/]+)/;
|
|
5
|
+
const tagRegex = /^\/tag\/([^/]+)/;
|
|
6
|
+
const reservedHosts = new Set([
|
|
7
|
+
"livejournal.com",
|
|
8
|
+
"www.livejournal.com",
|
|
9
|
+
"users.livejournal.com",
|
|
10
|
+
"community.livejournal.com",
|
|
11
|
+
"syndicated.livejournal.com"
|
|
12
|
+
]);
|
|
13
|
+
const livejournalHandler = {
|
|
14
|
+
match: (url) => {
|
|
15
|
+
if (!isSubdomainOf(url, "livejournal.com")) return false;
|
|
16
|
+
const { hostname, pathname } = new URL(url);
|
|
17
|
+
const lower = hostname.toLowerCase();
|
|
18
|
+
if (reservedHosts.has(lower)) {
|
|
19
|
+
if (lower === "www.livejournal.com") return wwwUsersPathRegex.test(pathname);
|
|
20
|
+
if (lower === "users.livejournal.com" || lower === "community.livejournal.com") return legacyUserPathRegex.test(pathname);
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
return true;
|
|
24
|
+
},
|
|
25
|
+
resolve: (url) => {
|
|
26
|
+
const { origin, hostname, pathname } = new URL(url);
|
|
27
|
+
const lowerHostname = hostname.toLowerCase();
|
|
28
|
+
const uris = [];
|
|
29
|
+
let userOrigin = origin;
|
|
30
|
+
if (lowerHostname === "www.livejournal.com") {
|
|
31
|
+
const userMatch = pathname.match(wwwUsersPathRegex);
|
|
32
|
+
if (userMatch?.[1]) userOrigin = `https://${userMatch[1]}.livejournal.com`;
|
|
33
|
+
else return uris;
|
|
34
|
+
}
|
|
35
|
+
if (lowerHostname === "users.livejournal.com" || lowerHostname === "community.livejournal.com") {
|
|
36
|
+
const userMatch = pathname.match(legacyUserPathRegex);
|
|
37
|
+
if (userMatch?.[1]) userOrigin = `https://${userMatch[1]}.livejournal.com`;
|
|
38
|
+
else return uris;
|
|
39
|
+
}
|
|
40
|
+
const tagMatch = pathname.match(tagRegex);
|
|
41
|
+
if (tagMatch?.[1]) {
|
|
42
|
+
const tag = encodeURIComponent(tagMatch[1]);
|
|
43
|
+
uris.push({
|
|
44
|
+
uri: `${userOrigin}/data/rss?tag=${tag}`,
|
|
45
|
+
hint: composeHint("livejournal:posts-tag-rss")
|
|
46
|
+
});
|
|
47
|
+
uris.push({
|
|
48
|
+
uri: `${userOrigin}/data/atom?tag=${tag}`,
|
|
49
|
+
hint: composeHint("livejournal:posts-tag-atom")
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
uris.push({
|
|
53
|
+
uri: `${userOrigin}/data/rss`,
|
|
54
|
+
hint: composeHint("livejournal:posts-rss")
|
|
55
|
+
});
|
|
56
|
+
uris.push({
|
|
57
|
+
uri: `${userOrigin}/data/atom`,
|
|
58
|
+
hint: composeHint("livejournal:posts-atom")
|
|
59
|
+
});
|
|
60
|
+
uris.push({
|
|
61
|
+
uri: `${userOrigin}/data/userpics`,
|
|
62
|
+
hint: composeHint("livejournal:userpics-atom")
|
|
63
|
+
});
|
|
64
|
+
return uris;
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
//#endregion
|
|
68
|
+
export { livejournalHandler };
|
|
@@ -2,6 +2,9 @@ import { composeHint } from "../../../common/utils.js";
|
|
|
2
2
|
import { isMastodonHeaders, isMastodonHtml } from "../../../favicons/platform/handlers/mastodon.js";
|
|
3
3
|
//#region src/feeds/platform/handlers/mastodon.ts
|
|
4
4
|
const profileRegex = /^\/@([^/]+)/;
|
|
5
|
+
const taggedProfileRegex = /^\/@([^/]+)\/tagged\/([^/]+)/;
|
|
6
|
+
const repliesProfileRegex = /^\/@([^/]+)\/with_replies/;
|
|
7
|
+
const mediaProfileRegex = /^\/@([^/]+)\/media/;
|
|
5
8
|
const tagRegex = /^\/tags\/([^/]+)/;
|
|
6
9
|
const isProfilePath = (pathname) => {
|
|
7
10
|
const segments = pathname.split("/").filter(Boolean);
|
|
@@ -24,6 +27,35 @@ const mastodonHandler = {
|
|
|
24
27
|
resolve: (url) => {
|
|
25
28
|
try {
|
|
26
29
|
const { origin, pathname } = new URL(url);
|
|
30
|
+
const repliesMatch = pathname.match(repliesProfileRegex);
|
|
31
|
+
if (repliesMatch?.[1]) return [{
|
|
32
|
+
uri: `${origin}/@${repliesMatch[1]}/with_replies.rss`,
|
|
33
|
+
hint: composeHint("mastodon:replies")
|
|
34
|
+
}, {
|
|
35
|
+
uri: `${origin}/@${repliesMatch[1]}.rss`,
|
|
36
|
+
hint: composeHint("mastodon:posts")
|
|
37
|
+
}];
|
|
38
|
+
const mediaMatch = pathname.match(mediaProfileRegex);
|
|
39
|
+
if (mediaMatch?.[1]) return [{
|
|
40
|
+
uri: `${origin}/@${mediaMatch[1]}/media.rss`,
|
|
41
|
+
hint: composeHint("mastodon:media")
|
|
42
|
+
}, {
|
|
43
|
+
uri: `${origin}/@${mediaMatch[1]}.rss`,
|
|
44
|
+
hint: composeHint("mastodon:posts")
|
|
45
|
+
}];
|
|
46
|
+
const taggedMatch = pathname.match(taggedProfileRegex);
|
|
47
|
+
if (taggedMatch?.[1] && taggedMatch?.[2]) {
|
|
48
|
+
const uris = [];
|
|
49
|
+
uris.push({
|
|
50
|
+
uri: `${origin}/@${taggedMatch[1]}/tagged/${taggedMatch[2]}.rss`,
|
|
51
|
+
hint: composeHint("mastodon:tagged")
|
|
52
|
+
});
|
|
53
|
+
uris.push({
|
|
54
|
+
uri: `${origin}/@${taggedMatch[1]}.rss`,
|
|
55
|
+
hint: composeHint("mastodon:posts")
|
|
56
|
+
});
|
|
57
|
+
return uris;
|
|
58
|
+
}
|
|
27
59
|
const userMatch = pathname.match(profileRegex);
|
|
28
60
|
if (userMatch?.[1]) return [{
|
|
29
61
|
uri: `${origin}/@${userMatch[1]}.rss`,
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { composeHint, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/mataroa.ts
|
|
3
|
+
const mataroaHandler = {
|
|
4
|
+
match: (url) => {
|
|
5
|
+
return isSubdomainOf(url, "mataroa.blog");
|
|
6
|
+
},
|
|
7
|
+
resolve: (url) => {
|
|
8
|
+
const { origin } = new URL(url);
|
|
9
|
+
return [{
|
|
10
|
+
uri: `${origin}/rss/`,
|
|
11
|
+
hint: composeHint("mataroa:blog")
|
|
12
|
+
}];
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
//#endregion
|
|
16
|
+
export { mataroaHandler };
|
|
@@ -53,11 +53,11 @@ const mediumHandler = {
|
|
|
53
53
|
const subdomain = lowerHostname.replace(".medium.com", "");
|
|
54
54
|
const tagMatch = pathname.match(subdomainTagRegex);
|
|
55
55
|
if (tagMatch?.[1]) return [{
|
|
56
|
-
uri: `https
|
|
56
|
+
uri: `https://${subdomain}.medium.com/feed/tagged/${tagMatch[1]}`,
|
|
57
57
|
hint: composeHint("medium:tagged")
|
|
58
58
|
}];
|
|
59
59
|
return [{
|
|
60
|
-
uri: `https
|
|
60
|
+
uri: `https://${subdomain}.medium.com/feed`,
|
|
61
61
|
hint: composeHint("medium:publication")
|
|
62
62
|
}];
|
|
63
63
|
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import { composeHint, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/microblog.ts
|
|
3
|
+
const categoryRegex = /^\/categories\/([^/]+)/;
|
|
4
|
+
const microblogHandler = {
|
|
5
|
+
match: (url) => {
|
|
6
|
+
return isSubdomainOf(url, "micro.blog");
|
|
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 category = categoryMatch[1];
|
|
14
|
+
uris.push({
|
|
15
|
+
uri: `${origin}/categories/${category}/feed.xml`,
|
|
16
|
+
hint: composeHint("microblog:category-rss")
|
|
17
|
+
});
|
|
18
|
+
uris.push({
|
|
19
|
+
uri: `${origin}/categories/${category}/feed.json`,
|
|
20
|
+
hint: composeHint("microblog:category-json")
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
if (pathname.startsWith("/archive")) uris.push({
|
|
24
|
+
uri: `${origin}/archive/index.json`,
|
|
25
|
+
hint: composeHint("microblog:archive")
|
|
26
|
+
});
|
|
27
|
+
if (pathname.startsWith("/photos")) uris.push({
|
|
28
|
+
uri: `${origin}/photos/index.json`,
|
|
29
|
+
hint: composeHint("microblog:photos")
|
|
30
|
+
});
|
|
31
|
+
if (pathname.startsWith("/replies")) uris.push({
|
|
32
|
+
uri: `${origin}/replies.xml`,
|
|
33
|
+
hint: composeHint("microblog:replies")
|
|
34
|
+
});
|
|
35
|
+
uris.push({
|
|
36
|
+
uri: `${origin}/feed.xml`,
|
|
37
|
+
hint: composeHint("microblog:posts-rss")
|
|
38
|
+
});
|
|
39
|
+
uris.push({
|
|
40
|
+
uri: `${origin}/feed.json`,
|
|
41
|
+
hint: composeHint("microblog:posts-json")
|
|
42
|
+
});
|
|
43
|
+
uris.push({
|
|
44
|
+
uri: `${origin}/podcast.xml`,
|
|
45
|
+
hint: composeHint("microblog:podcast")
|
|
46
|
+
});
|
|
47
|
+
uris.push({
|
|
48
|
+
uri: `${origin}/podcast.json`,
|
|
49
|
+
hint: composeHint("microblog:podcast-json")
|
|
50
|
+
});
|
|
51
|
+
return uris;
|
|
52
|
+
}
|
|
53
|
+
};
|
|
54
|
+
//#endregion
|
|
55
|
+
export { microblogHandler };
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { composeHint, hasMetaContent } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/misskey.ts
|
|
3
|
+
const profileRegex = /^\/@([^/.]+)/;
|
|
4
|
+
const isMisskeyHtml = (content) => {
|
|
5
|
+
return hasMetaContent(content, "application-name", "Misskey");
|
|
6
|
+
};
|
|
7
|
+
const misskeyHandler = {
|
|
8
|
+
match: (url, content) => {
|
|
9
|
+
try {
|
|
10
|
+
if (!content || !isMisskeyHtml(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}/@${match[1]}.atom`,
|
|
24
|
+
hint: composeHint("misskey:posts-atom")
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
uri: `${origin}/@${match[1]}.rss`,
|
|
28
|
+
hint: composeHint("misskey:posts-rss")
|
|
29
|
+
},
|
|
30
|
+
{
|
|
31
|
+
uri: `${origin}/@${match[1]}.json`,
|
|
32
|
+
hint: composeHint("misskey:posts-json")
|
|
33
|
+
}
|
|
34
|
+
];
|
|
35
|
+
} catch {}
|
|
36
|
+
return [];
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
//#endregion
|
|
40
|
+
export { misskeyHandler };
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { composeHint, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/myanimelist.ts
|
|
3
|
+
const hosts = ["myanimelist.net", "www.myanimelist.net"];
|
|
4
|
+
const userRegex = /^\/(?:profile|animelist|mangalist|history)\/([^/]+)/;
|
|
5
|
+
const myanimelistHandler = {
|
|
6
|
+
match: (url) => {
|
|
7
|
+
return isHostOf(url, hosts);
|
|
8
|
+
},
|
|
9
|
+
resolve: (url) => {
|
|
10
|
+
const { pathname } = new URL(url);
|
|
11
|
+
if (pathname === "/news" || pathname.startsWith("/news/")) return [{
|
|
12
|
+
uri: "https://myanimelist.net/rss/news.xml",
|
|
13
|
+
hint: composeHint("myanimelist:news")
|
|
14
|
+
}];
|
|
15
|
+
if (pathname === "/featured" || pathname.startsWith("/featured/")) return [{
|
|
16
|
+
uri: "https://myanimelist.net/rss/featured.xml",
|
|
17
|
+
hint: composeHint("myanimelist:featured")
|
|
18
|
+
}];
|
|
19
|
+
const match = pathname.match(userRegex);
|
|
20
|
+
if (!match?.[1]) return [];
|
|
21
|
+
const user = match[1];
|
|
22
|
+
const uris = [];
|
|
23
|
+
uris.push({
|
|
24
|
+
uri: `https://myanimelist.net/rss.php?type=rw&u=${user}`,
|
|
25
|
+
hint: composeHint("myanimelist:anime")
|
|
26
|
+
});
|
|
27
|
+
uris.push({
|
|
28
|
+
uri: `https://myanimelist.net/rss.php?type=rm&u=${user}`,
|
|
29
|
+
hint: composeHint("myanimelist:manga")
|
|
30
|
+
});
|
|
31
|
+
uris.push({
|
|
32
|
+
uri: `https://myanimelist.net/rss.php?type=rrw&u=${user}`,
|
|
33
|
+
hint: composeHint("myanimelist:recently-watched")
|
|
34
|
+
});
|
|
35
|
+
uris.push({
|
|
36
|
+
uri: `https://myanimelist.net/rss.php?type=rrm&u=${user}`,
|
|
37
|
+
hint: composeHint("myanimelist:recently-read")
|
|
38
|
+
});
|
|
39
|
+
return uris;
|
|
40
|
+
}
|
|
41
|
+
};
|
|
42
|
+
//#endregion
|
|
43
|
+
export { myanimelistHandler };
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { composeHint, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/naverBlog.ts
|
|
3
|
+
const hosts = ["blog.naver.com", "m.blog.naver.com"];
|
|
4
|
+
const naverBlogHandler = {
|
|
5
|
+
match: (url) => {
|
|
6
|
+
return isHostOf(url, hosts);
|
|
7
|
+
},
|
|
8
|
+
resolve: (url) => {
|
|
9
|
+
const { pathname } = new URL(url);
|
|
10
|
+
const pathSegments = pathname.split("/").filter(Boolean);
|
|
11
|
+
if (pathSegments.length === 0) return [];
|
|
12
|
+
const blogId = pathSegments[0];
|
|
13
|
+
if (blogId.includes(".")) return [];
|
|
14
|
+
return [{
|
|
15
|
+
uri: `https://rss.blog.naver.com/${blogId}.xml`,
|
|
16
|
+
hint: composeHint("naver-blog:blog")
|
|
17
|
+
}];
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
//#endregion
|
|
21
|
+
export { naverBlogHandler };
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/nebula.ts
|
|
3
|
+
const hosts = ["nebula.tv", "www.nebula.tv"];
|
|
4
|
+
const excludedPaths = [
|
|
5
|
+
"about",
|
|
6
|
+
"classes",
|
|
7
|
+
"library",
|
|
8
|
+
"login",
|
|
9
|
+
"originals",
|
|
10
|
+
"pricing",
|
|
11
|
+
"privacy",
|
|
12
|
+
"search",
|
|
13
|
+
"settings",
|
|
14
|
+
"signup",
|
|
15
|
+
"terms"
|
|
16
|
+
];
|
|
17
|
+
const globalPaths = new Set(["videos", "explore"]);
|
|
18
|
+
const nebulaHandler = {
|
|
19
|
+
match: (url) => {
|
|
20
|
+
return isHostOf(url, hosts);
|
|
21
|
+
},
|
|
22
|
+
resolve: (url) => {
|
|
23
|
+
const { pathname, searchParams } = new URL(url);
|
|
24
|
+
const pathSegments = pathname.split("/").filter(Boolean);
|
|
25
|
+
if (pathSegments.length === 0 || globalPaths.has(pathSegments[0])) {
|
|
26
|
+
const rawCategory = searchParams.get("category");
|
|
27
|
+
const category = rawCategory ? rawCategory.toLowerCase() : null;
|
|
28
|
+
const uris = [];
|
|
29
|
+
if (category) {
|
|
30
|
+
uris.push({
|
|
31
|
+
uri: `https://rss.nebula.app/video/categories/${category}.rss`,
|
|
32
|
+
hint: composeHint("nebula:category")
|
|
33
|
+
});
|
|
34
|
+
uris.push({
|
|
35
|
+
uri: `https://rss.nebula.app/video/categories/${category}.rss?plus=true`,
|
|
36
|
+
hint: composeHint("nebula:category-plus")
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
uris.push({
|
|
40
|
+
uri: "https://rss.nebula.app/video.rss",
|
|
41
|
+
hint: composeHint("nebula:videos-all")
|
|
42
|
+
});
|
|
43
|
+
uris.push({
|
|
44
|
+
uri: "https://rss.nebula.app/video.rss?plus=true",
|
|
45
|
+
hint: composeHint("nebula:videos-all-plus")
|
|
46
|
+
});
|
|
47
|
+
uris.push({
|
|
48
|
+
uri: "https://rss.nebula.app/video/channels.rss",
|
|
49
|
+
hint: composeHint("nebula:channels")
|
|
50
|
+
});
|
|
51
|
+
return uris;
|
|
52
|
+
}
|
|
53
|
+
const slug = pathSegments[0];
|
|
54
|
+
if (isAnyOf(slug, excludedPaths)) return [];
|
|
55
|
+
const uris = [];
|
|
56
|
+
uris.push({
|
|
57
|
+
uri: `https://rss.nebula.app/video/channels/${slug}.rss`,
|
|
58
|
+
hint: composeHint("nebula:videos")
|
|
59
|
+
});
|
|
60
|
+
uris.push({
|
|
61
|
+
uri: `https://rss.nebula.app/video/channels/${slug}.rss?plus=true`,
|
|
62
|
+
hint: composeHint("nebula:videos-plus")
|
|
63
|
+
});
|
|
64
|
+
return uris;
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
//#endregion
|
|
68
|
+
export { nebulaHandler };
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { composeHint, isAnyOf, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/note.ts
|
|
3
|
+
const hosts = ["note.com", "www.note.com"];
|
|
4
|
+
const excludedPaths = [
|
|
5
|
+
"about",
|
|
6
|
+
"api",
|
|
7
|
+
"explore",
|
|
8
|
+
"hashtag",
|
|
9
|
+
"help",
|
|
10
|
+
"login",
|
|
11
|
+
"m",
|
|
12
|
+
"n",
|
|
13
|
+
"premium",
|
|
14
|
+
"privacy",
|
|
15
|
+
"ranking",
|
|
16
|
+
"search",
|
|
17
|
+
"settings",
|
|
18
|
+
"signup",
|
|
19
|
+
"terms"
|
|
20
|
+
];
|
|
21
|
+
const hashtagRegex = /^\/hashtag\/([^/]+)/;
|
|
22
|
+
const magazineRegex = /^\/([^/]+)\/m\/([^/]+)/;
|
|
23
|
+
const noteHandler = {
|
|
24
|
+
match: (url) => {
|
|
25
|
+
return isHostOf(url, hosts);
|
|
26
|
+
},
|
|
27
|
+
resolve: (url) => {
|
|
28
|
+
const { pathname } = new URL(url);
|
|
29
|
+
const hashtagMatch = pathname.match(hashtagRegex);
|
|
30
|
+
if (hashtagMatch?.[1]) return [{
|
|
31
|
+
uri: `https://note.com/hashtag/${hashtagMatch[1]}/rss`,
|
|
32
|
+
hint: composeHint("note:hashtag")
|
|
33
|
+
}];
|
|
34
|
+
const magazineMatch = pathname.match(magazineRegex);
|
|
35
|
+
if (magazineMatch?.[1] && magazineMatch?.[2]) return [{
|
|
36
|
+
uri: `https://note.com/${magazineMatch[1]}/m/${magazineMatch[2]}/rss`,
|
|
37
|
+
hint: composeHint("note:magazine")
|
|
38
|
+
}];
|
|
39
|
+
const pathSegments = pathname.split("/").filter(Boolean);
|
|
40
|
+
if (pathSegments.length === 0) return [{
|
|
41
|
+
uri: "https://note.com/rss",
|
|
42
|
+
hint: composeHint("note:featured")
|
|
43
|
+
}];
|
|
44
|
+
const username = pathSegments[0];
|
|
45
|
+
if (isAnyOf(username, excludedPaths)) return [];
|
|
46
|
+
return [{
|
|
47
|
+
uri: `https://note.com/${username}/rss`,
|
|
48
|
+
hint: composeHint("note:blog")
|
|
49
|
+
}];
|
|
50
|
+
}
|
|
51
|
+
};
|
|
52
|
+
//#endregion
|
|
53
|
+
export { noteHandler };
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { composeHint, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/observable.ts
|
|
3
|
+
const hosts = ["observablehq.com", "www.observablehq.com"];
|
|
4
|
+
const collectionRegex = /^\/@([^/]+)\/collection\/([^/]+)/;
|
|
5
|
+
const userRegex = /^\/@([^/]+)/;
|
|
6
|
+
const observableHandler = {
|
|
7
|
+
match: (url) => {
|
|
8
|
+
return isHostOf(url, hosts);
|
|
9
|
+
},
|
|
10
|
+
resolve: (url) => {
|
|
11
|
+
const { pathname } = new URL(url);
|
|
12
|
+
if (pathname === "/recent" || pathname === "/recent/") return [{
|
|
13
|
+
uri: "https://api.observablehq.com/documents/public.rss",
|
|
14
|
+
hint: composeHint("observable:recent")
|
|
15
|
+
}];
|
|
16
|
+
if (pathname === "/trending" || pathname === "/trending/") return [{
|
|
17
|
+
uri: "https://api.observablehq.com/documents/trending.rss",
|
|
18
|
+
hint: composeHint("observable:trending")
|
|
19
|
+
}];
|
|
20
|
+
const collectionMatch = pathname.match(collectionRegex);
|
|
21
|
+
if (collectionMatch?.[1] && collectionMatch?.[2]) return [{
|
|
22
|
+
uri: `https://api.observablehq.com/collection/@${collectionMatch[1]}/${collectionMatch[2]}.rss`,
|
|
23
|
+
hint: composeHint("observable:collection")
|
|
24
|
+
}];
|
|
25
|
+
const userMatch = pathname.match(userRegex);
|
|
26
|
+
if (!userMatch?.[1]) return [];
|
|
27
|
+
return [{
|
|
28
|
+
uri: `https://api.observablehq.com/documents/@${userMatch[1]}.rss`,
|
|
29
|
+
hint: composeHint("observable:notebooks")
|
|
30
|
+
}];
|
|
31
|
+
}
|
|
32
|
+
};
|
|
33
|
+
//#endregion
|
|
34
|
+
export { observableHandler };
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { composeHint, isHostOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/odysee.ts
|
|
3
|
+
const hosts = ["odysee.com", "www.odysee.com"];
|
|
4
|
+
const channelRegex = /^\/(@[^/:]+:[a-f0-9]+)/i;
|
|
5
|
+
const odyseeHandler = {
|
|
6
|
+
match: (url) => {
|
|
7
|
+
return isHostOf(url, hosts);
|
|
8
|
+
},
|
|
9
|
+
resolve: (url) => {
|
|
10
|
+
const { pathname } = new URL(url);
|
|
11
|
+
const match = pathname.match(channelRegex);
|
|
12
|
+
if (!match?.[1]) return [];
|
|
13
|
+
return [{
|
|
14
|
+
uri: `https://odysee.com/$/rss/${match[1]}`,
|
|
15
|
+
hint: composeHint("odysee:videos")
|
|
16
|
+
}];
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
//#endregion
|
|
20
|
+
export { odyseeHandler };
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { composeHint, isHostOf, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/pagecord.ts
|
|
3
|
+
const pagecordHandler = {
|
|
4
|
+
match: (url) => {
|
|
5
|
+
return isSubdomainOf(url, "pagecord.com") && !isHostOf(url, "www.pagecord.com");
|
|
6
|
+
},
|
|
7
|
+
resolve: (url) => {
|
|
8
|
+
const { origin } = new URL(url);
|
|
9
|
+
return [{
|
|
10
|
+
uri: `${origin}/feed.xml`,
|
|
11
|
+
hint: composeHint("pagecord:blog")
|
|
12
|
+
}];
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
//#endregion
|
|
16
|
+
export { pagecordHandler };
|
|
@@ -10,9 +10,8 @@ const paragraphHandler = {
|
|
|
10
10
|
const { pathname } = new URL(url);
|
|
11
11
|
const userMatch = pathname.match(userRegex);
|
|
12
12
|
if (!userMatch?.[1]) return [];
|
|
13
|
-
const username = userMatch[1];
|
|
14
13
|
return [{
|
|
15
|
-
uri:
|
|
14
|
+
uri: `https://api.paragraph.com/blogs/rss/@${userMatch[1]}`,
|
|
16
15
|
hint: composeHint("paragraph:blog")
|
|
17
16
|
}];
|
|
18
17
|
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { composeHint, isSubdomainOf } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/pika.ts
|
|
3
|
+
const tagRegex = /^\/tag\/([^/]+)/;
|
|
4
|
+
const pikaHandler = {
|
|
5
|
+
match: (url) => {
|
|
6
|
+
return isSubdomainOf(url, "pika.page");
|
|
7
|
+
},
|
|
8
|
+
resolve: (url) => {
|
|
9
|
+
const { origin, pathname } = new URL(url);
|
|
10
|
+
const uris = [];
|
|
11
|
+
const tagMatch = pathname.match(tagRegex);
|
|
12
|
+
if (tagMatch?.[1]) {
|
|
13
|
+
const tag = tagMatch[1];
|
|
14
|
+
uris.push({
|
|
15
|
+
uri: `${origin}/tag/${tag}/feed`,
|
|
16
|
+
hint: composeHint("pika:tag-atom")
|
|
17
|
+
});
|
|
18
|
+
uris.push({
|
|
19
|
+
uri: `${origin}/tag/${tag}/feed.rss`,
|
|
20
|
+
hint: composeHint("pika:tag-rss")
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
uris.push({
|
|
24
|
+
uri: `${origin}/posts_feed`,
|
|
25
|
+
hint: composeHint("pika:posts-atom")
|
|
26
|
+
});
|
|
27
|
+
uris.push({
|
|
28
|
+
uri: `${origin}/posts_feed.rss`,
|
|
29
|
+
hint: composeHint("pika:posts-rss")
|
|
30
|
+
});
|
|
31
|
+
return uris;
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
//#endregion
|
|
35
|
+
export { pikaHandler };
|
|
@@ -34,6 +34,19 @@ const pinterestHandler = {
|
|
|
34
34
|
if (pathSegments.length === 0) return [];
|
|
35
35
|
const username = pathSegments[0];
|
|
36
36
|
if (isAnyOf(username, excludedPaths)) return [];
|
|
37
|
+
const reservedBoardSlugs = new Set([
|
|
38
|
+
"pins",
|
|
39
|
+
"boards",
|
|
40
|
+
"_saved",
|
|
41
|
+
"_created",
|
|
42
|
+
"followers",
|
|
43
|
+
"following"
|
|
44
|
+
]);
|
|
45
|
+
const board = pathSegments[1];
|
|
46
|
+
if (board && !reservedBoardSlugs.has(board)) return [{
|
|
47
|
+
uri: `https://www.pinterest.com/${username}/${board}.rss`,
|
|
48
|
+
hint: composeHint("pinterest:board")
|
|
49
|
+
}];
|
|
37
50
|
return [{
|
|
38
51
|
uri: `https://www.pinterest.com/${username}/feed.rss`,
|
|
39
52
|
hint: composeHint("pinterest:pins")
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { composeHint, hasMetaContent } from "../../../common/utils.js";
|
|
2
|
+
//#region src/feeds/platform/handlers/pixelfed.ts
|
|
3
|
+
const profileRegex = /^\/(?:users\/)?([a-zA-Z0-9_]+)\/?$/;
|
|
4
|
+
const excludedPaths = [
|
|
5
|
+
"admin",
|
|
6
|
+
"api",
|
|
7
|
+
"discover",
|
|
8
|
+
"i",
|
|
9
|
+
"login",
|
|
10
|
+
"notifications",
|
|
11
|
+
"p",
|
|
12
|
+
"register",
|
|
13
|
+
"settings",
|
|
14
|
+
"site",
|
|
15
|
+
"storage",
|
|
16
|
+
"timeline",
|
|
17
|
+
"users"
|
|
18
|
+
];
|
|
19
|
+
const isPixelfedHtml = (content) => {
|
|
20
|
+
return hasMetaContent(content, "generator", "pixelfed");
|
|
21
|
+
};
|
|
22
|
+
const pixelfedHandler = {
|
|
23
|
+
match: (url, content) => {
|
|
24
|
+
try {
|
|
25
|
+
if (!content || !isPixelfedHtml(content)) return false;
|
|
26
|
+
const { pathname } = new URL(url);
|
|
27
|
+
const match = pathname.match(profileRegex);
|
|
28
|
+
return Boolean(match?.[1] && !excludedPaths.includes(match[1]));
|
|
29
|
+
} catch {}
|
|
30
|
+
return false;
|
|
31
|
+
},
|
|
32
|
+
resolve: (url) => {
|
|
33
|
+
try {
|
|
34
|
+
const { origin, pathname } = new URL(url);
|
|
35
|
+
const match = pathname.match(profileRegex);
|
|
36
|
+
if (!match?.[1] || excludedPaths.includes(match[1])) return [];
|
|
37
|
+
return [{
|
|
38
|
+
uri: `${origin}/users/${match[1]}.atom`,
|
|
39
|
+
hint: composeHint("pixelfed:posts")
|
|
40
|
+
}];
|
|
41
|
+
} catch {}
|
|
42
|
+
return [];
|
|
43
|
+
}
|
|
44
|
+
};
|
|
45
|
+
//#endregion
|
|
46
|
+
export { pixelfedHandler };
|