@zenalexa/unicli 0.209.0 → 0.211.2
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/AGENTS.md +75 -20
- package/README.md +832 -285
- package/dist/adapters/cursor/cursor.d.ts +2 -1
- package/dist/adapters/cursor/cursor.d.ts.map +1 -1
- package/dist/adapters/cursor/cursor.js +87 -1
- package/dist/adapters/cursor/cursor.js.map +1 -1
- package/dist/adapters/discord-app/discord-app.d.ts +1 -1
- package/dist/adapters/discord-app/discord-app.js +94 -1
- package/dist/adapters/discord-app/discord-app.js.map +1 -1
- package/dist/browser/bridge.d.ts +45 -0
- package/dist/browser/bridge.d.ts.map +1 -1
- package/dist/browser/bridge.js +94 -1
- package/dist/browser/bridge.js.map +1 -1
- package/dist/browser/cdp-client.d.ts +24 -1
- package/dist/browser/cdp-client.d.ts.map +1 -1
- package/dist/browser/cdp-client.js +56 -2
- package/dist/browser/cdp-client.js.map +1 -1
- package/dist/browser/launcher.d.ts +5 -0
- package/dist/browser/launcher.d.ts.map +1 -1
- package/dist/browser/launcher.js +7 -0
- package/dist/browser/launcher.js.map +1 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +72 -2
- package/dist/cli.js.map +1 -1
- package/dist/commands/agents.d.ts +14 -3
- package/dist/commands/agents.d.ts.map +1 -1
- package/dist/commands/agents.js +369 -140
- package/dist/commands/agents.js.map +1 -1
- package/dist/commands/browser.d.ts.map +1 -1
- package/dist/commands/browser.js +79 -1
- package/dist/commands/browser.js.map +1 -1
- package/dist/commands/ext.d.ts +11 -0
- package/dist/commands/ext.d.ts.map +1 -0
- package/dist/commands/ext.js +122 -0
- package/dist/commands/ext.js.map +1 -0
- package/dist/commands/mcp.d.ts +3 -4
- package/dist/commands/mcp.d.ts.map +1 -1
- package/dist/commands/mcp.js +47 -62
- package/dist/commands/mcp.js.map +1 -1
- package/dist/commands/schema.d.ts +12 -0
- package/dist/commands/schema.d.ts.map +1 -0
- package/dist/commands/schema.js +72 -0
- package/dist/commands/schema.js.map +1 -0
- package/dist/commands/search.d.ts +12 -0
- package/dist/commands/search.d.ts.map +1 -0
- package/dist/commands/search.js +47 -0
- package/dist/commands/search.js.map +1 -0
- package/dist/commands/skills.d.ts +2 -3
- package/dist/commands/skills.d.ts.map +1 -1
- package/dist/commands/skills.js +2 -3
- package/dist/commands/skills.js.map +1 -1
- package/dist/commands/status.d.ts +12 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +137 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/discovery/aliases.d.ts +31 -0
- package/dist/discovery/aliases.d.ts.map +1 -0
- package/dist/discovery/aliases.js +477 -0
- package/dist/discovery/aliases.js.map +1 -0
- package/dist/discovery/loader.d.ts.map +1 -1
- package/dist/discovery/loader.js +25 -0
- package/dist/discovery/loader.js.map +1 -1
- package/dist/discovery/search.d.ts +73 -0
- package/dist/discovery/search.d.ts.map +1 -0
- package/dist/discovery/search.js +355 -0
- package/dist/discovery/search.js.map +1 -0
- package/dist/engine/yaml-runner.d.ts +10 -0
- package/dist/engine/yaml-runner.d.ts.map +1 -1
- package/dist/engine/yaml-runner.js +51 -3
- package/dist/engine/yaml-runner.js.map +1 -1
- package/dist/hub/index.d.ts +37 -0
- package/dist/hub/index.d.ts.map +1 -0
- package/dist/hub/index.js +98 -0
- package/dist/hub/index.js.map +1 -0
- package/dist/hub/passthrough.d.ts +21 -0
- package/dist/hub/passthrough.d.ts.map +1 -0
- package/dist/hub/passthrough.js +71 -0
- package/dist/hub/passthrough.js.map +1 -0
- package/dist/manifest-compact.txt +15 -0
- package/dist/manifest-search.json +1 -0
- package/dist/manifest.json +2163 -268
- package/dist/mcp/oauth.d.ts +33 -0
- package/dist/mcp/oauth.d.ts.map +1 -0
- package/dist/mcp/oauth.js +220 -0
- package/dist/mcp/oauth.js.map +1 -0
- package/dist/mcp/schema.d.ts +65 -0
- package/dist/mcp/schema.d.ts.map +1 -0
- package/dist/mcp/schema.js +136 -0
- package/dist/mcp/schema.js.map +1 -0
- package/dist/mcp/server.d.ts +23 -10
- package/dist/mcp/server.d.ts.map +1 -1
- package/dist/mcp/server.js +350 -182
- package/dist/mcp/server.js.map +1 -1
- package/dist/mcp/sse-transport.d.ts +34 -0
- package/dist/mcp/sse-transport.d.ts.map +1 -0
- package/dist/mcp/sse-transport.js +182 -0
- package/dist/mcp/sse-transport.js.map +1 -0
- package/dist/mcp/streamable-http.d.ts +64 -0
- package/dist/mcp/streamable-http.d.ts.map +1 -0
- package/dist/mcp/streamable-http.js +312 -0
- package/dist/mcp/streamable-http.js.map +1 -0
- package/dist/permissions/sensitive-paths.js +2 -2
- package/dist/permissions/sensitive-paths.js.map +1 -1
- package/dist/types.d.ts +15 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js.map +1 -1
- package/package.json +8 -7
- package/src/adapters/1688/_site.json +9 -0
- package/src/adapters/36kr/latest.yaml +36 -0
- package/src/adapters/arxiv/trending.yaml +37 -0
- package/src/adapters/baidu/hot.yaml +37 -0
- package/src/adapters/baidu/search.yaml +41 -0
- package/src/adapters/barchart/_site.json +10 -0
- package/src/adapters/bbc/technology.yaml +26 -0
- package/src/adapters/bbc/top.yaml +26 -0
- package/src/adapters/bbc/world.yaml +26 -0
- package/src/adapters/bilibili/coin.yaml +33 -0
- package/src/adapters/bilibili/later.yaml +31 -0
- package/src/adapters/bilibili/live.yaml +41 -0
- package/src/adapters/bilibili/trending.yaml +29 -0
- package/src/adapters/binance/hot.yaml +39 -0
- package/src/adapters/binance/kline.yaml +35 -0
- package/src/adapters/binance/ticker.yaml +33 -0
- package/src/adapters/bluesky/likes.yaml +35 -0
- package/src/adapters/bluesky/notifications.yaml +30 -0
- package/src/adapters/bluesky/post.yaml +25 -0
- package/src/adapters/chrome/bookmarks.yaml +26 -0
- package/src/adapters/chrome/tabs.yaml +25 -0
- package/src/adapters/cnki/search.yaml +40 -0
- package/src/adapters/cnn/technology.yaml +26 -0
- package/src/adapters/cocoapods/info.yaml +33 -0
- package/src/adapters/coinbase/prices.yaml +27 -0
- package/src/adapters/coinbase/rates.yaml +24 -0
- package/src/adapters/coupang/hot.yaml +29 -0
- package/src/adapters/crates-io/info.yaml +29 -0
- package/src/adapters/crates-io/versions.yaml +33 -0
- package/src/adapters/ctrip/hot.yaml +28 -0
- package/src/adapters/cursor/cursor.ts +91 -1
- package/src/adapters/dangdang/hot.yaml +38 -0
- package/src/adapters/dangdang/search.yaml +43 -0
- package/src/adapters/deepseek/chat.yaml +33 -0
- package/src/adapters/deepseek/models.yaml +18 -0
- package/src/adapters/devto/latest.yaml +29 -0
- package/src/adapters/devto/search.yaml +34 -0
- package/src/adapters/dianping/hot.yaml +43 -0
- package/src/adapters/dianping/search.yaml +48 -0
- package/src/adapters/discord-app/discord-app.ts +99 -1
- package/src/adapters/docker/networks.yaml +20 -0
- package/src/adapters/docker/volumes.yaml +19 -0
- package/src/adapters/docker-hub/info.yaml +26 -0
- package/src/adapters/docker-hub/tags.yaml +34 -0
- package/src/adapters/douban/group-hot.yaml +29 -0
- package/src/adapters/douban/new-movies.yaml +32 -0
- package/src/adapters/douban/tv-hot.yaml +32 -0
- package/src/adapters/doubao/ask.yaml +64 -0
- package/src/adapters/doubao/new.yaml +18 -0
- package/src/adapters/doubao/status.yaml +21 -0
- package/src/adapters/douyu/hot.yaml +34 -0
- package/src/adapters/douyu/search.yaml +42 -0
- package/src/adapters/eastmoney/fund.yaml +36 -0
- package/src/adapters/eastmoney/market.yaml +26 -0
- package/src/adapters/ele/hot.yaml +40 -0
- package/src/adapters/ele/search.yaml +47 -0
- package/src/adapters/exchangerate/list.yaml +25 -0
- package/src/adapters/facebook/marketplace.yaml +43 -0
- package/src/adapters/facebook/post.yaml +29 -0
- package/src/adapters/futu/hot.yaml +49 -0
- package/src/adapters/futu/quote.yaml +46 -0
- package/src/adapters/gitee/repos.yaml +34 -0
- package/src/adapters/github-trending/developers.yaml +33 -0
- package/src/adapters/github-trending/weekly.yaml +34 -0
- package/src/adapters/gitlab/projects.yaml +34 -0
- package/src/adapters/godot/scene-export.yaml +1 -1
- package/src/adapters/hackernews/comments.yaml +36 -0
- package/src/adapters/hackernews/item.yaml +28 -0
- package/src/adapters/hf/datasets.yaml +34 -0
- package/src/adapters/hf/spaces.yaml +34 -0
- package/src/adapters/homebrew/search.yaml +27 -0
- package/src/adapters/huggingface-papers/search.yaml +33 -0
- package/src/adapters/imdb/box-office.yaml +28 -0
- package/src/adapters/infoq/latest.yaml +26 -0
- package/src/adapters/instagram/activity.yaml +45 -0
- package/src/adapters/instagram/highlights.yaml +48 -0
- package/src/adapters/instagram/reels-trending.yaml +51 -0
- package/src/adapters/instagram/reels.yaml +34 -0
- package/src/adapters/instagram/stories.yaml +29 -0
- package/src/adapters/instagram/suggested.yaml +40 -0
- package/src/adapters/instagram/tags.yaml +52 -0
- package/src/adapters/itch-io/search.yaml +35 -0
- package/src/adapters/itch-io/top.yaml +31 -0
- package/src/adapters/ithome/hot.yaml +27 -0
- package/src/adapters/ithome/latest.yaml +26 -0
- package/src/adapters/jd/_site.json +9 -0
- package/src/adapters/jianyu/search.yaml +42 -0
- package/src/adapters/ke/ershoufang.yaml +75 -0
- package/src/adapters/ke/xiaoqu.yaml +73 -0
- package/src/adapters/kuaishou/hot.yaml +39 -0
- package/src/adapters/kuaishou/search.yaml +44 -0
- package/src/adapters/linkedin/_site.json +10 -0
- package/src/adapters/linkedin/jobs.yaml +47 -0
- package/src/adapters/lobsters/search.yaml +35 -0
- package/src/adapters/macos/active-app.yaml +16 -0
- package/src/adapters/macos/apps-list.yaml +14 -0
- package/src/adapters/macos/apps.yaml +31 -0
- package/src/adapters/macos/battery.yaml +26 -0
- package/src/adapters/macos/bluetooth.yaml +14 -0
- package/src/adapters/macos/brightness.yaml +22 -0
- package/src/adapters/macos/caffeinate.yaml +25 -0
- package/src/adapters/macos/calendar-create.yaml +81 -0
- package/src/adapters/macos/calendar-list.yaml +53 -0
- package/src/adapters/macos/calendar-today.yaml +30 -0
- package/src/adapters/macos/clipboard.yaml +23 -0
- package/src/adapters/macos/contacts-search.yaml +59 -0
- package/src/adapters/macos/dark-mode.yaml +19 -0
- package/src/adapters/macos/disk-info.yaml +31 -0
- package/src/adapters/macos/disk-usage.yaml +14 -0
- package/src/adapters/macos/do-not-disturb.yaml +21 -0
- package/src/adapters/macos/empty-trash.yaml +16 -0
- package/src/adapters/macos/finder-copy.yaml +40 -0
- package/src/adapters/macos/finder-move.yaml +40 -0
- package/src/adapters/macos/finder-new-folder.yaml +36 -0
- package/src/adapters/macos/finder-recent.yaml +24 -0
- package/src/adapters/macos/finder-selection.yaml +22 -0
- package/src/adapters/macos/finder-tags.yaml +40 -0
- package/src/adapters/macos/lock-screen.yaml +16 -0
- package/src/adapters/macos/mail-send.yaml +54 -0
- package/src/adapters/macos/mail-status.yaml +38 -0
- package/src/adapters/macos/messages-send.yaml +42 -0
- package/src/adapters/macos/music-control.yaml +43 -0
- package/src/adapters/macos/music-now.yaml +34 -0
- package/src/adapters/macos/notes-list.yaml +30 -0
- package/src/adapters/macos/notes-search.yaml +45 -0
- package/src/adapters/macos/notification.yaml +27 -0
- package/src/adapters/macos/notify.yaml +46 -0
- package/src/adapters/macos/open-app.yaml +21 -0
- package/src/adapters/macos/open.yaml +24 -0
- package/src/adapters/macos/photos-search.yaml +44 -0
- package/src/adapters/macos/processes.yaml +14 -0
- package/src/adapters/macos/reminder-create.yaml +75 -0
- package/src/adapters/macos/reminders-complete.yaml +33 -0
- package/src/adapters/macos/reminders-list.yaml +28 -0
- package/src/adapters/macos/safari-history.yaml +23 -0
- package/src/adapters/macos/safari-tabs.yaml +26 -0
- package/src/adapters/macos/safari-url.yaml +22 -0
- package/src/adapters/macos/say.yaml +36 -0
- package/src/adapters/macos/screen-lock.yaml +16 -0
- package/src/adapters/macos/screen-recording.yaml +32 -0
- package/src/adapters/macos/screenshot.yaml +34 -0
- package/src/adapters/macos/shortcuts-list.yaml +14 -0
- package/src/adapters/macos/shortcuts-run.yaml +28 -0
- package/src/adapters/macos/sleep.yaml +14 -0
- package/src/adapters/macos/spotlight.yaml +43 -0
- package/src/adapters/macos/system-info.yaml +32 -0
- package/src/adapters/macos/trash.yaml +32 -0
- package/src/adapters/macos/uptime.yaml +14 -0
- package/src/adapters/macos/volume.yaml +19 -0
- package/src/adapters/macos/wallpaper.yaml +33 -0
- package/src/adapters/macos/wifi-info.yaml +14 -0
- package/src/adapters/macos/wifi.yaml +63 -0
- package/src/adapters/maimai/search.yaml +62 -0
- package/src/adapters/maoyan/hot.yaml +33 -0
- package/src/adapters/maoyan/search.yaml +40 -0
- package/src/adapters/mastodon/timeline.yaml +33 -0
- package/src/adapters/mastodon/user.yaml +32 -0
- package/src/adapters/medium/article.yaml +22 -0
- package/src/adapters/medium/trending.yaml +30 -0
- package/src/adapters/meituan/hot.yaml +26 -0
- package/src/adapters/mubu/list.yaml +36 -0
- package/src/adapters/mubu/search.yaml +54 -0
- package/src/adapters/netease-music/playlist.yaml +34 -0
- package/src/adapters/netease-music/top.yaml +33 -0
- package/src/adapters/notion/databases.yaml +41 -0
- package/src/adapters/notion/pages.yaml +44 -0
- package/src/adapters/notion/search.yaml +41 -0
- package/src/adapters/npm/downloads.yaml +28 -0
- package/src/adapters/npm/info.yaml +25 -0
- package/src/adapters/npm/search.yaml +35 -0
- package/src/adapters/npm/versions.yaml +33 -0
- package/src/adapters/npm-trends/trending.yaml +17 -0
- package/src/adapters/nytimes/search.yaml +35 -0
- package/src/adapters/obsidian/daily.yaml +23 -0
- package/src/adapters/obsidian/open.yaml +26 -0
- package/src/adapters/obsidian/search.yaml +28 -0
- package/src/adapters/ollama/generate.yaml +33 -0
- package/src/adapters/ollama/models.yaml +29 -0
- package/src/adapters/ollama/ps.yaml +20 -0
- package/src/adapters/openrouter/search.yaml +31 -0
- package/src/adapters/perplexity/ask.yaml +33 -0
- package/src/adapters/pexels/curated.yaml +31 -0
- package/src/adapters/pinduoduo/search.yaml +36 -0
- package/src/adapters/producthunt/search.yaml +41 -0
- package/src/adapters/pypi/search.yaml +37 -0
- package/src/adapters/pypi/versions.yaml +31 -0
- package/src/adapters/quark/ls.yaml +39 -0
- package/src/adapters/quark/search.yaml +39 -0
- package/src/adapters/qweather/forecast.yaml +37 -0
- package/src/adapters/reddit/new.yaml +34 -0
- package/src/adapters/reddit/rising.yaml +33 -0
- package/src/adapters/reddit/top.yaml +40 -0
- package/src/adapters/reddit/trending.yaml +25 -0
- package/src/adapters/replicate/run.yaml +39 -0
- package/src/adapters/reuters/_site.json +9 -0
- package/src/adapters/reuters/latest.yaml +26 -0
- package/src/adapters/sinablog/_site.json +9 -0
- package/src/adapters/sinafinance/market.yaml +21 -0
- package/src/adapters/slack/messages.yaml +41 -0
- package/src/adapters/slack/post.yaml +32 -0
- package/src/adapters/slack/search.yaml +28 -0
- package/src/adapters/slack/send.yaml +35 -0
- package/src/adapters/slack/status.yaml +19 -0
- package/src/adapters/slack/users.yaml +37 -0
- package/src/adapters/slock/servers.yaml +32 -0
- package/src/adapters/smzdm/_site.json +9 -0
- package/src/adapters/spotify/now-playing.yaml +28 -0
- package/src/adapters/spotify/playlists.yaml +37 -0
- package/src/adapters/spotify/top-tracks.yaml +42 -0
- package/src/adapters/stackoverflow/question.yaml +34 -0
- package/src/adapters/stackoverflow/tags.yaml +30 -0
- package/src/adapters/steam/app-details.yaml +32 -0
- package/src/adapters/steam/new-releases.yaml +28 -0
- package/src/adapters/steam/search.yaml +36 -0
- package/src/adapters/steam/specials.yaml +29 -0
- package/src/adapters/steam/wishlist.yaml +36 -0
- package/src/adapters/substack/trending.yaml +28 -0
- package/src/adapters/taobao/hot.yaml +31 -0
- package/src/adapters/taobao/search.yaml +44 -0
- package/src/adapters/techcrunch/search.yaml +33 -0
- package/src/adapters/theverge/search.yaml +33 -0
- package/src/adapters/threads/hot.yaml +44 -0
- package/src/adapters/threads/search.yaml +47 -0
- package/src/adapters/tiktok/trending.yaml +28 -0
- package/src/adapters/toutiao/hot.yaml +31 -0
- package/src/adapters/toutiao/search.yaml +47 -0
- package/src/adapters/twitch/games.yaml +31 -0
- package/src/adapters/twitch/search.yaml +39 -0
- package/src/adapters/twitch/streams.yaml +38 -0
- package/src/adapters/twitter/lists.yaml +49 -0
- package/src/adapters/twitter/media.yaml +48 -0
- package/src/adapters/twitter/mentions.yaml +38 -0
- package/src/adapters/twitter/mute.yaml +42 -0
- package/src/adapters/twitter/pin.yaml +43 -0
- package/src/adapters/twitter/quotes.yaml +43 -0
- package/src/adapters/twitter/retweets.yaml +44 -0
- package/src/adapters/twitter/spaces.yaml +43 -0
- package/src/adapters/twitter/unmute.yaml +42 -0
- package/src/adapters/unsplash/random.yaml +27 -0
- package/src/adapters/v2ex/search.yaml +36 -0
- package/src/adapters/vscode/extensions.yaml +23 -0
- package/src/adapters/vscode/install-ext.yaml +26 -0
- package/src/adapters/vscode/open.yaml +25 -0
- package/src/adapters/wechat-channels/hot.yaml +39 -0
- package/src/adapters/wechat-channels/search.yaml +57 -0
- package/src/adapters/weibo/trending.yaml +28 -0
- package/src/adapters/weixin/_site.json +9 -0
- package/src/adapters/weixin/hot.yaml +22 -0
- package/src/adapters/wikipedia/today.yaml +25 -0
- package/src/adapters/xiaohongshu/follow.yaml +35 -0
- package/src/adapters/xiaohongshu/hashtag.yaml +44 -0
- package/src/adapters/xiaohongshu/hot.yaml +34 -0
- package/src/adapters/xiaohongshu/like.yaml +27 -0
- package/src/adapters/xiaohongshu/profile.yaml +39 -0
- package/src/adapters/xiaohongshu/save.yaml +27 -0
- package/src/adapters/xiaohongshu/suggest.yaml +34 -0
- package/src/adapters/xiaohongshu/trending.yaml +35 -0
- package/src/adapters/xiaohongshu/unfollow.yaml +40 -0
- package/src/adapters/xueqiu/market.yaml +25 -0
- package/src/adapters/yahoo-finance/quote.yaml +35 -0
- package/src/adapters/yahoo-finance/search.yaml +42 -0
- package/src/adapters/yahoo-finance/trending.yaml +30 -0
- package/src/adapters/youtube/playlist.yaml +43 -0
- package/src/adapters/youtube/shorts.yaml +42 -0
- package/src/adapters/youtube/trending.yaml +41 -0
- package/src/adapters/zhihu/answer.yaml +29 -0
- package/src/adapters/zhihu/answers.yaml +38 -0
- package/src/adapters/zhihu/article.yaml +27 -0
- package/src/adapters/zhihu/articles.yaml +38 -0
- package/src/adapters/zhihu/collections.yaml +37 -0
- package/src/adapters/zhihu/columns.yaml +38 -0
- package/src/adapters/zhihu/comment.yaml +42 -0
- package/src/adapters/zhihu/followers.yaml +37 -0
- package/src/adapters/zhihu/following.yaml +37 -0
- package/src/adapters/zhihu/pins.yaml +37 -0
- package/src/adapters/zhihu/topic.yaml +35 -0
- package/src/adapters/zhihu/topics.yaml +37 -0
- package/src/adapters/zhihu/trending.yaml +36 -0
- package/src/adapters/zhihu/user.yaml +30 -0
- package/src/adapters/zoom/join.yaml +24 -0
- package/src/adapters/zoom/start.yaml +17 -0
- package/src/hub/external-clis.yaml +343 -0
- package/src/hub/index.ts +128 -0
- package/src/hub/passthrough.ts +85 -0
- package/src/adapters/1688/manifest.yaml +0 -7
- package/src/adapters/barchart/manifest.yaml +0 -8
- package/src/adapters/jd/manifest.yaml +0 -7
- package/src/adapters/linkedin/manifest.yaml +0 -8
- package/src/adapters/reuters/manifest.yaml +0 -7
- package/src/adapters/sinablog/manifest.yaml +0 -7
- package/src/adapters/smzdm/manifest.yaml +0 -7
- package/src/adapters/weixin/manifest.yaml +0 -7
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
site: tiktok
|
|
2
|
+
name: trending
|
|
3
|
+
description: TikTok trending hashtags and sounds
|
|
4
|
+
domain: www.tiktok.com
|
|
5
|
+
type: web-api
|
|
6
|
+
strategy: cookie
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
limit:
|
|
10
|
+
type: int
|
|
11
|
+
default: 20
|
|
12
|
+
|
|
13
|
+
pipeline:
|
|
14
|
+
- fetch:
|
|
15
|
+
url: https://www.tiktok.com/api/discover/
|
|
16
|
+
params:
|
|
17
|
+
count: "${{ args.limit }}"
|
|
18
|
+
|
|
19
|
+
- select: data
|
|
20
|
+
|
|
21
|
+
- map:
|
|
22
|
+
rank: ${{ index + 1 }}
|
|
23
|
+
name: "${{ item.cardItem?.title || item.title || '' }}"
|
|
24
|
+
description: "${{ (item.cardItem?.subTitle || item.description || '') | truncate(80) }}"
|
|
25
|
+
|
|
26
|
+
- limit: ${{ args.limit }}
|
|
27
|
+
|
|
28
|
+
columns: [rank, name, description]
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
site: toutiao
|
|
2
|
+
name: hot
|
|
3
|
+
description: Get Toutiao (Jinri Toutiao) hot topics
|
|
4
|
+
type: web-api
|
|
5
|
+
domain: www.toutiao.com
|
|
6
|
+
strategy: public
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
limit:
|
|
10
|
+
type: int
|
|
11
|
+
default: 30
|
|
12
|
+
description: Number of hot topics
|
|
13
|
+
|
|
14
|
+
pipeline:
|
|
15
|
+
- fetch:
|
|
16
|
+
url: "https://www.toutiao.com/hot-event/hot-board/?origin=toutiao_pc"
|
|
17
|
+
headers:
|
|
18
|
+
User-Agent: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36"
|
|
19
|
+
|
|
20
|
+
- select: data
|
|
21
|
+
|
|
22
|
+
- map:
|
|
23
|
+
rank: "${{ index + 1 }}"
|
|
24
|
+
title: "${{ item.Title }}"
|
|
25
|
+
hot_value: "${{ item.HotValue }}"
|
|
26
|
+
url: "${{ item.Url }}"
|
|
27
|
+
label: "${{ item.LabelDesc || '' }}"
|
|
28
|
+
|
|
29
|
+
- limit: ${{ args.limit }}
|
|
30
|
+
|
|
31
|
+
columns: [rank, title, hot_value, label, url]
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
site: toutiao
|
|
2
|
+
name: search
|
|
3
|
+
description: Search Toutiao for articles
|
|
4
|
+
domain: www.toutiao.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
query:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Search query
|
|
14
|
+
limit:
|
|
15
|
+
type: int
|
|
16
|
+
default: 10
|
|
17
|
+
description: Number of results
|
|
18
|
+
|
|
19
|
+
columns: [rank, title, source, time, url]
|
|
20
|
+
|
|
21
|
+
pipeline:
|
|
22
|
+
- navigate:
|
|
23
|
+
url: https://so.toutiao.com/search?keyword=${{ args.query | urlencode }}
|
|
24
|
+
settleMs: 4000
|
|
25
|
+
|
|
26
|
+
- scroll:
|
|
27
|
+
times: 2
|
|
28
|
+
|
|
29
|
+
- evaluate: |
|
|
30
|
+
(async () => {
|
|
31
|
+
const limit = ${{ args.limit }};
|
|
32
|
+
const clean = v => (v || '').replace(/\s+/g, ' ').trim();
|
|
33
|
+
const items = document.querySelectorAll('[class*="result-content"], [class*="search-card"], .cs-view a[href*="toutiao"]');
|
|
34
|
+
return Array.from(items).slice(0, limit).map((item, i) => {
|
|
35
|
+
const titleEl = item.querySelector('h3, [class*="title"], a[class*="link"]');
|
|
36
|
+
const sourceEl = item.querySelector('[class*="source"], [class*="author"], [class*="media"]');
|
|
37
|
+
const timeEl = item.querySelector('[class*="time"], [class*="date"], time');
|
|
38
|
+
const linkEl = item.querySelector('a[href]') || item.closest('a');
|
|
39
|
+
return {
|
|
40
|
+
rank: i + 1,
|
|
41
|
+
title: clean(titleEl?.textContent),
|
|
42
|
+
source: clean(sourceEl?.textContent),
|
|
43
|
+
time: clean(timeEl?.textContent),
|
|
44
|
+
url: linkEl?.href || ''
|
|
45
|
+
};
|
|
46
|
+
}).filter(r => r.title);
|
|
47
|
+
})()
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
site: twitch
|
|
2
|
+
name: games
|
|
3
|
+
description: Get top games on Twitch by viewer count
|
|
4
|
+
type: web-api
|
|
5
|
+
domain: api.twitch.tv
|
|
6
|
+
strategy: cookie
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
limit:
|
|
10
|
+
type: int
|
|
11
|
+
default: 20
|
|
12
|
+
|
|
13
|
+
pipeline:
|
|
14
|
+
- fetch:
|
|
15
|
+
url: https://api.twitch.tv/helix/games/top
|
|
16
|
+
params:
|
|
17
|
+
first: "${{ args.limit }}"
|
|
18
|
+
headers:
|
|
19
|
+
Client-Id: "${{ env.TWITCH_CLIENT_ID || '' }}"
|
|
20
|
+
Authorization: "Bearer ${{ env.TWITCH_TOKEN || '' }}"
|
|
21
|
+
|
|
22
|
+
- select: data
|
|
23
|
+
|
|
24
|
+
- map:
|
|
25
|
+
rank: ${{ index + 1 }}
|
|
26
|
+
name: ${{ item.name }}
|
|
27
|
+
id: ${{ item.id }}
|
|
28
|
+
|
|
29
|
+
- limit: ${{ args.limit }}
|
|
30
|
+
|
|
31
|
+
columns: [rank, name, id]
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
site: twitch
|
|
2
|
+
name: search
|
|
3
|
+
description: Search Twitch channels
|
|
4
|
+
type: web-api
|
|
5
|
+
domain: api.twitch.tv
|
|
6
|
+
strategy: cookie
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
query:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Search query
|
|
14
|
+
limit:
|
|
15
|
+
type: int
|
|
16
|
+
default: 20
|
|
17
|
+
|
|
18
|
+
pipeline:
|
|
19
|
+
- fetch:
|
|
20
|
+
url: https://api.twitch.tv/helix/search/channels
|
|
21
|
+
params:
|
|
22
|
+
query: "${{ args.query }}"
|
|
23
|
+
first: "${{ args.limit }}"
|
|
24
|
+
headers:
|
|
25
|
+
Client-Id: "${{ env.TWITCH_CLIENT_ID || '' }}"
|
|
26
|
+
Authorization: "Bearer ${{ env.TWITCH_TOKEN || '' }}"
|
|
27
|
+
|
|
28
|
+
- select: data
|
|
29
|
+
|
|
30
|
+
- map:
|
|
31
|
+
name: ${{ item.display_name }}
|
|
32
|
+
game: ${{ item.game_name }}
|
|
33
|
+
is_live: ${{ item.is_live }}
|
|
34
|
+
title: "${{ (item.title || '') | truncate(60) }}"
|
|
35
|
+
id: ${{ item.id }}
|
|
36
|
+
|
|
37
|
+
- limit: ${{ args.limit }}
|
|
38
|
+
|
|
39
|
+
columns: [name, game, is_live, title]
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
site: twitch
|
|
2
|
+
name: streams
|
|
3
|
+
description: Get top live streams on Twitch
|
|
4
|
+
type: web-api
|
|
5
|
+
domain: api.twitch.tv
|
|
6
|
+
strategy: cookie
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
game_id:
|
|
10
|
+
type: str
|
|
11
|
+
description: Filter by game ID
|
|
12
|
+
limit:
|
|
13
|
+
type: int
|
|
14
|
+
default: 20
|
|
15
|
+
|
|
16
|
+
pipeline:
|
|
17
|
+
- fetch:
|
|
18
|
+
url: https://api.twitch.tv/helix/streams
|
|
19
|
+
params:
|
|
20
|
+
first: "${{ args.limit }}"
|
|
21
|
+
game_id: "${{ args.game_id || '' }}"
|
|
22
|
+
headers:
|
|
23
|
+
Client-Id: "${{ env.TWITCH_CLIENT_ID || '' }}"
|
|
24
|
+
Authorization: "Bearer ${{ env.TWITCH_TOKEN || '' }}"
|
|
25
|
+
|
|
26
|
+
- select: data
|
|
27
|
+
|
|
28
|
+
- map:
|
|
29
|
+
rank: ${{ index + 1 }}
|
|
30
|
+
streamer: ${{ item.user_name }}
|
|
31
|
+
title: "${{ (item.title || '') | truncate(60) }}"
|
|
32
|
+
game: ${{ item.game_name }}
|
|
33
|
+
viewers: ${{ item.viewer_count }}
|
|
34
|
+
language: ${{ item.language }}
|
|
35
|
+
|
|
36
|
+
- limit: ${{ args.limit }}
|
|
37
|
+
|
|
38
|
+
columns: [rank, streamer, game, viewers, title, language]
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: lists
|
|
3
|
+
description: Get Twitter/X lists for a user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
user:
|
|
10
|
+
type: str
|
|
11
|
+
positional: true
|
|
12
|
+
description: Twitter username (omit for logged-in user)
|
|
13
|
+
limit:
|
|
14
|
+
type: int
|
|
15
|
+
default: 20
|
|
16
|
+
description: Number of lists
|
|
17
|
+
|
|
18
|
+
columns: [name, members, followers, mode, url]
|
|
19
|
+
|
|
20
|
+
pipeline:
|
|
21
|
+
- navigate: https://x.com/${{ args.user || 'i/lists' }}/lists
|
|
22
|
+
- wait: 4000
|
|
23
|
+
- evaluate: |
|
|
24
|
+
(async () => {
|
|
25
|
+
const limit = ${{ args.limit }};
|
|
26
|
+
await new Promise(r => setTimeout(r, 2000));
|
|
27
|
+
const cards = [];
|
|
28
|
+
const seen = new Set();
|
|
29
|
+
for (const anchor of document.querySelectorAll('a[href*="/i/lists/"]')) {
|
|
30
|
+
const href = anchor.getAttribute('href') || '';
|
|
31
|
+
if (!/\/i\/lists\/\d+/.test(href) || seen.has(href)) continue;
|
|
32
|
+
seen.add(href);
|
|
33
|
+
const container = anchor.closest('[data-testid="cellInnerDiv"]') || anchor;
|
|
34
|
+
const text = (container.innerText || anchor.innerText || '').trim();
|
|
35
|
+
if (!text) continue;
|
|
36
|
+
const lines = text.split('\n').map(l => l.trim()).filter(Boolean);
|
|
37
|
+
const name = lines[0] || '';
|
|
38
|
+
const membersMatch = text.match(/(\d+)\s*(?:members?|成员)/i);
|
|
39
|
+
const followersMatch = text.match(/(\d+)\s*(?:followers?|关注者)/i);
|
|
40
|
+
cards.push({
|
|
41
|
+
name,
|
|
42
|
+
members: membersMatch ? parseInt(membersMatch[1]) : 0,
|
|
43
|
+
followers: followersMatch ? parseInt(followersMatch[1]) : 0,
|
|
44
|
+
mode: /private|私密/.test(text) ? 'Private' : 'Public',
|
|
45
|
+
url: 'https://x.com' + href,
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
return cards.slice(0, limit);
|
|
49
|
+
})()
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: media
|
|
3
|
+
description: Get media tweets from a Twitter/X user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
user:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Twitter username
|
|
14
|
+
limit:
|
|
15
|
+
type: int
|
|
16
|
+
default: 20
|
|
17
|
+
description: Number of media tweets
|
|
18
|
+
|
|
19
|
+
columns: [rank, text, likes, retweets, type, date, url]
|
|
20
|
+
|
|
21
|
+
pipeline:
|
|
22
|
+
- navigate: https://x.com/${{ args.user }}/media
|
|
23
|
+
- wait: 4000
|
|
24
|
+
- evaluate: |
|
|
25
|
+
(async () => {
|
|
26
|
+
const limit = ${{ args.limit }};
|
|
27
|
+
const tweets = [];
|
|
28
|
+
const articles = document.querySelectorAll('article[data-testid="tweet"]');
|
|
29
|
+
for (const article of Array.from(articles).slice(0, limit)) {
|
|
30
|
+
const textEl = article.querySelector('[data-testid="tweetText"]');
|
|
31
|
+
const timeEl = article.querySelector('time');
|
|
32
|
+
const linkEl = article.querySelector('a[href*="/status/"]');
|
|
33
|
+
const metrics = article.querySelectorAll('[data-testid$="count"], [data-testid$="-group"] span');
|
|
34
|
+
const imgs = article.querySelectorAll('img[src*="pbs.twimg.com/media"]');
|
|
35
|
+
const vids = article.querySelectorAll('video');
|
|
36
|
+
const metricsArr = Array.from(metrics).map(m => (m.textContent || '').trim());
|
|
37
|
+
tweets.push({
|
|
38
|
+
rank: tweets.length + 1,
|
|
39
|
+
text: (textEl?.textContent || '').substring(0, 100),
|
|
40
|
+
likes: metricsArr[2] || '0',
|
|
41
|
+
retweets: metricsArr[1] || '0',
|
|
42
|
+
type: vids.length ? 'video' : imgs.length > 1 ? 'photos' : 'photo',
|
|
43
|
+
date: timeEl?.getAttribute('datetime')?.slice(0, 10) || '',
|
|
44
|
+
url: linkEl ? 'https://x.com' + linkEl.getAttribute('href') : '',
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
return tweets;
|
|
48
|
+
})()
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: mentions
|
|
3
|
+
description: Get recent mentions/replies for the logged-in Twitter/X user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
limit:
|
|
10
|
+
type: int
|
|
11
|
+
default: 20
|
|
12
|
+
description: Number of mentions
|
|
13
|
+
|
|
14
|
+
columns: [rank, author, text, date, url]
|
|
15
|
+
|
|
16
|
+
pipeline:
|
|
17
|
+
- navigate: https://x.com/notifications/mentions
|
|
18
|
+
- wait: 4000
|
|
19
|
+
- evaluate: |
|
|
20
|
+
(async () => {
|
|
21
|
+
const limit = ${{ args.limit }};
|
|
22
|
+
const articles = document.querySelectorAll('article[data-testid="tweet"]');
|
|
23
|
+
const results = [];
|
|
24
|
+
for (const article of Array.from(articles).slice(0, limit)) {
|
|
25
|
+
const userEl = article.querySelector('[data-testid="User-Name"]');
|
|
26
|
+
const textEl = article.querySelector('[data-testid="tweetText"]');
|
|
27
|
+
const timeEl = article.querySelector('time');
|
|
28
|
+
const linkEl = article.querySelector('a[href*="/status/"]');
|
|
29
|
+
results.push({
|
|
30
|
+
rank: results.length + 1,
|
|
31
|
+
author: (userEl?.textContent || '').split('@').pop()?.split('\n')[0]?.trim() || '',
|
|
32
|
+
text: (textEl?.textContent || '').substring(0, 140),
|
|
33
|
+
date: timeEl?.getAttribute('datetime')?.slice(0, 10) || '',
|
|
34
|
+
url: linkEl ? 'https://x.com' + linkEl.getAttribute('href') : '',
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
return results;
|
|
38
|
+
})()
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: mute
|
|
3
|
+
description: Mute a Twitter/X user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
user:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Twitter username to mute
|
|
14
|
+
|
|
15
|
+
columns: [status, username]
|
|
16
|
+
|
|
17
|
+
pipeline:
|
|
18
|
+
- navigate: https://x.com/${{ args.user }}
|
|
19
|
+
- wait: 3000
|
|
20
|
+
- evaluate: |
|
|
21
|
+
(async () => {
|
|
22
|
+
const username = ${{ args.user | json }};
|
|
23
|
+
const ct0 = document.cookie.match(/ct0=([^;]+)/)?.[1] || '';
|
|
24
|
+
if (!ct0) throw new Error('Not logged in (ct0 cookie missing)');
|
|
25
|
+
const resp = await fetch('https://x.com/' + username);
|
|
26
|
+
const html = await resp.text();
|
|
27
|
+
const idMatch = html.match(/"rest_id":"(\d+)"/);
|
|
28
|
+
if (!idMatch) throw new Error('Could not find user ID for ' + username);
|
|
29
|
+
const userId = idMatch[1];
|
|
30
|
+
const muteResp = await fetch('https://x.com/i/api/1.1/mutes/users/create.json', {
|
|
31
|
+
method: 'POST',
|
|
32
|
+
credentials: 'include',
|
|
33
|
+
headers: {
|
|
34
|
+
'Content-Type': 'application/x-www-form-urlencoded',
|
|
35
|
+
'X-Csrf-Token': ct0,
|
|
36
|
+
'Authorization': 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA',
|
|
37
|
+
},
|
|
38
|
+
body: 'user_id=' + userId,
|
|
39
|
+
});
|
|
40
|
+
if (!muteResp.ok) throw new Error('Failed to mute: HTTP ' + muteResp.status);
|
|
41
|
+
return [{ status: 'Muted', username }];
|
|
42
|
+
})()
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: pin
|
|
3
|
+
description: Get pinned tweet from a Twitter/X user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
user:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Twitter username
|
|
14
|
+
|
|
15
|
+
columns: [author, text, likes, retweets, replies, date, url]
|
|
16
|
+
|
|
17
|
+
pipeline:
|
|
18
|
+
- navigate: https://x.com/${{ args.user }}
|
|
19
|
+
- wait: 4000
|
|
20
|
+
- evaluate: |
|
|
21
|
+
(async () => {
|
|
22
|
+
const username = ${{ args.user | json }};
|
|
23
|
+
const pinLabel = document.querySelector('[data-testid="socialContext"]');
|
|
24
|
+
if (!pinLabel || !/pinned|置顶/i.test(pinLabel.textContent || '')) {
|
|
25
|
+
return [{ author: username, text: '(no pinned tweet)', likes: '', retweets: '', replies: '', date: '', url: '' }];
|
|
26
|
+
}
|
|
27
|
+
const article = pinLabel.closest('article[data-testid="tweet"]');
|
|
28
|
+
if (!article) throw new Error('Pinned tweet article not found');
|
|
29
|
+
const textEl = article.querySelector('[data-testid="tweetText"]');
|
|
30
|
+
const timeEl = article.querySelector('time');
|
|
31
|
+
const linkEl = article.querySelector('a[href*="/status/"]');
|
|
32
|
+
const groups = article.querySelectorAll('[role="group"] [data-testid$="count"], [role="group"] span');
|
|
33
|
+
const metrics = Array.from(groups).map(g => (g.textContent || '').trim()).filter(Boolean);
|
|
34
|
+
return [{
|
|
35
|
+
author: username,
|
|
36
|
+
text: (textEl?.textContent || '').substring(0, 200),
|
|
37
|
+
likes: metrics[2] || '0',
|
|
38
|
+
retweets: metrics[1] || '0',
|
|
39
|
+
replies: metrics[0] || '0',
|
|
40
|
+
date: timeEl?.getAttribute('datetime')?.slice(0, 10) || '',
|
|
41
|
+
url: linkEl ? 'https://x.com' + linkEl.getAttribute('href') : '',
|
|
42
|
+
}];
|
|
43
|
+
})()
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: quotes
|
|
3
|
+
description: Get quote tweets for a specific tweet
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
url:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Tweet URL or ID
|
|
14
|
+
limit:
|
|
15
|
+
type: int
|
|
16
|
+
default: 20
|
|
17
|
+
description: Number of quote tweets
|
|
18
|
+
|
|
19
|
+
columns: [rank, author, text, likes, date]
|
|
20
|
+
|
|
21
|
+
pipeline:
|
|
22
|
+
- navigate: ${{ args.url }}/quotes
|
|
23
|
+
- wait: 4000
|
|
24
|
+
- evaluate: |
|
|
25
|
+
(async () => {
|
|
26
|
+
const limit = ${{ args.limit }};
|
|
27
|
+
const articles = document.querySelectorAll('article[data-testid="tweet"]');
|
|
28
|
+
const results = [];
|
|
29
|
+
for (const article of Array.from(articles).slice(0, limit)) {
|
|
30
|
+
const userEl = article.querySelector('[data-testid="User-Name"]');
|
|
31
|
+
const textEl = article.querySelector('[data-testid="tweetText"]');
|
|
32
|
+
const timeEl = article.querySelector('time');
|
|
33
|
+
const likeEl = article.querySelector('[data-testid="like"] span, [data-testid="unlike"] span');
|
|
34
|
+
results.push({
|
|
35
|
+
rank: results.length + 1,
|
|
36
|
+
author: (userEl?.textContent || '').split('@').pop()?.split('\n')[0]?.trim() || '',
|
|
37
|
+
text: (textEl?.textContent || '').substring(0, 140),
|
|
38
|
+
likes: (likeEl?.textContent || '0').trim(),
|
|
39
|
+
date: timeEl?.getAttribute('datetime')?.slice(0, 10) || '',
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
return results;
|
|
43
|
+
})()
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: retweets
|
|
3
|
+
description: Get users who retweeted a specific tweet
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
url:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Tweet URL
|
|
14
|
+
limit:
|
|
15
|
+
type: int
|
|
16
|
+
default: 20
|
|
17
|
+
description: Number of users
|
|
18
|
+
|
|
19
|
+
columns: [rank, name, username, verified, followers]
|
|
20
|
+
|
|
21
|
+
pipeline:
|
|
22
|
+
- navigate: ${{ args.url }}/retweets
|
|
23
|
+
- wait: 4000
|
|
24
|
+
- evaluate: |
|
|
25
|
+
(async () => {
|
|
26
|
+
const limit = ${{ args.limit }};
|
|
27
|
+
const cells = document.querySelectorAll('[data-testid="cellInnerDiv"]');
|
|
28
|
+
const results = [];
|
|
29
|
+
for (const cell of Array.from(cells).slice(0, limit)) {
|
|
30
|
+
const nameEl = cell.querySelector('[data-testid="UserCell"] a div[dir="ltr"] span');
|
|
31
|
+
const handleEl = cell.querySelector('[data-testid="UserCell"] a[tabindex="-1"]');
|
|
32
|
+
const verifiedEl = cell.querySelector('svg[data-testid="icon-verified"]');
|
|
33
|
+
const bioEl = cell.querySelector('[data-testid="UserCell"] [dir="auto"]:last-child');
|
|
34
|
+
if (!nameEl) continue;
|
|
35
|
+
results.push({
|
|
36
|
+
rank: results.length + 1,
|
|
37
|
+
name: (nameEl?.textContent || '').trim(),
|
|
38
|
+
username: (handleEl?.textContent || '').replace('@', '').trim(),
|
|
39
|
+
verified: verifiedEl ? 'Yes' : 'No',
|
|
40
|
+
followers: '',
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
return results;
|
|
44
|
+
})()
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: spaces
|
|
3
|
+
description: Browse live and upcoming Twitter/X Spaces
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
limit:
|
|
10
|
+
type: int
|
|
11
|
+
default: 10
|
|
12
|
+
description: Number of spaces
|
|
13
|
+
|
|
14
|
+
columns: [rank, title, host, listeners, state, url]
|
|
15
|
+
|
|
16
|
+
pipeline:
|
|
17
|
+
- navigate: https://x.com/i/spaces
|
|
18
|
+
- wait: 5000
|
|
19
|
+
- evaluate: |
|
|
20
|
+
(async () => {
|
|
21
|
+
const limit = ${{ args.limit }};
|
|
22
|
+
await new Promise(r => setTimeout(r, 3000));
|
|
23
|
+
const cards = document.querySelectorAll('[data-testid="card.wrapper"], [data-testid="cellInnerDiv"]');
|
|
24
|
+
const results = [];
|
|
25
|
+
for (const card of Array.from(cards).slice(0, limit * 2)) {
|
|
26
|
+
const titleEl = card.querySelector('[data-testid="card.layoutLarge.detail"] span, [dir="ltr"] span');
|
|
27
|
+
const linkEl = card.querySelector('a[href*="/i/spaces/"]');
|
|
28
|
+
if (!titleEl || !linkEl) continue;
|
|
29
|
+
const text = card.textContent || '';
|
|
30
|
+
const listenersMatch = text.match(/(\d[\d,]*)\s*(?:listening|listeners?|人在听)/i);
|
|
31
|
+
const hostEl = card.querySelector('[data-testid="UserCell"] span, .r-1awozwy');
|
|
32
|
+
results.push({
|
|
33
|
+
rank: results.length + 1,
|
|
34
|
+
title: (titleEl.textContent || '').trim().substring(0, 80),
|
|
35
|
+
host: (hostEl?.textContent || '').trim(),
|
|
36
|
+
listeners: listenersMatch ? listenersMatch[1] : '0',
|
|
37
|
+
state: /live|直播/i.test(text) ? 'Live' : 'Scheduled',
|
|
38
|
+
url: 'https://x.com' + linkEl.getAttribute('href'),
|
|
39
|
+
});
|
|
40
|
+
if (results.length >= limit) break;
|
|
41
|
+
}
|
|
42
|
+
return results;
|
|
43
|
+
})()
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
site: twitter
|
|
2
|
+
name: unmute
|
|
3
|
+
description: Unmute a Twitter/X user
|
|
4
|
+
domain: x.com
|
|
5
|
+
strategy: intercept
|
|
6
|
+
browser: true
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
user:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
description: Twitter username to unmute
|
|
14
|
+
|
|
15
|
+
columns: [status, username]
|
|
16
|
+
|
|
17
|
+
pipeline:
|
|
18
|
+
- navigate: https://x.com/${{ args.user }}
|
|
19
|
+
- wait: 3000
|
|
20
|
+
- evaluate: |
|
|
21
|
+
(async () => {
|
|
22
|
+
const username = ${{ args.user | json }};
|
|
23
|
+
const ct0 = document.cookie.match(/ct0=([^;]+)/)?.[1] || '';
|
|
24
|
+
if (!ct0) throw new Error('Not logged in (ct0 cookie missing)');
|
|
25
|
+
const resp = await fetch('https://x.com/' + username);
|
|
26
|
+
const html = await resp.text();
|
|
27
|
+
const idMatch = html.match(/"rest_id":"(\d+)"/);
|
|
28
|
+
if (!idMatch) throw new Error('Could not find user ID for ' + username);
|
|
29
|
+
const userId = idMatch[1];
|
|
30
|
+
const unmuteResp = await fetch('https://x.com/i/api/1.1/mutes/users/destroy.json', {
|
|
31
|
+
method: 'POST',
|
|
32
|
+
credentials: 'include',
|
|
33
|
+
headers: {
|
|
34
|
+
'Content-Type': 'application/x-www-form-urlencoded',
|
|
35
|
+
'X-Csrf-Token': ct0,
|
|
36
|
+
'Authorization': 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA',
|
|
37
|
+
},
|
|
38
|
+
body: 'user_id=' + userId,
|
|
39
|
+
});
|
|
40
|
+
if (!unmuteResp.ok) throw new Error('Failed to unmute: HTTP ' + unmuteResp.status);
|
|
41
|
+
return [{ status: 'Unmuted', username }];
|
|
42
|
+
})()
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
site: unsplash
|
|
2
|
+
name: random
|
|
3
|
+
description: Get a random Unsplash photo
|
|
4
|
+
type: web-api
|
|
5
|
+
domain: api.unsplash.com
|
|
6
|
+
strategy: public
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
query:
|
|
10
|
+
type: str
|
|
11
|
+
description: Optional topic filter
|
|
12
|
+
|
|
13
|
+
pipeline:
|
|
14
|
+
- fetch:
|
|
15
|
+
url: https://unsplash.com/napi/photos/random
|
|
16
|
+
params:
|
|
17
|
+
query: "${{ args.query || '' }}"
|
|
18
|
+
|
|
19
|
+
- map:
|
|
20
|
+
id: ${{ item.id }}
|
|
21
|
+
description: "${{ (item.description || item.alt_description || '') | truncate(120) }}"
|
|
22
|
+
author: "${{ item.user ? item.user.name : '' }}"
|
|
23
|
+
likes: ${{ item.likes }}
|
|
24
|
+
url: "${{ item.urls ? item.urls.regular : '' }}"
|
|
25
|
+
download: "${{ item.links ? item.links.download : '' }}"
|
|
26
|
+
|
|
27
|
+
columns: [id, description, author, likes, url]
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
site: v2ex
|
|
2
|
+
name: search
|
|
3
|
+
description: Search V2EX topics via SOV2EX
|
|
4
|
+
domain: www.sov2ex.com
|
|
5
|
+
type: web-api
|
|
6
|
+
strategy: public
|
|
7
|
+
|
|
8
|
+
args:
|
|
9
|
+
query:
|
|
10
|
+
type: str
|
|
11
|
+
required: true
|
|
12
|
+
positional: true
|
|
13
|
+
limit:
|
|
14
|
+
type: int
|
|
15
|
+
default: 20
|
|
16
|
+
|
|
17
|
+
pipeline:
|
|
18
|
+
- fetch:
|
|
19
|
+
url: https://www.sov2ex.com/api/search
|
|
20
|
+
params:
|
|
21
|
+
q: "${{ args.query }}"
|
|
22
|
+
from: "0"
|
|
23
|
+
size: "${{ args.limit }}"
|
|
24
|
+
|
|
25
|
+
- select: hits
|
|
26
|
+
|
|
27
|
+
- map:
|
|
28
|
+
title: "${{ item._source ? item._source.title : '' }}"
|
|
29
|
+
node: "${{ item._source ? item._source.node : '' }}"
|
|
30
|
+
author: "${{ item._source ? item._source.member : '' }}"
|
|
31
|
+
replies: "${{ item._source ? item._source.replies : 0 }}"
|
|
32
|
+
created: "${{ item._source ? item._source.created : '' }}"
|
|
33
|
+
|
|
34
|
+
- limit: ${{ args.limit }}
|
|
35
|
+
|
|
36
|
+
columns: [title, node, author, replies, created]
|