@zenalexa/unicli 0.215.1 → 0.216.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 +15 -15
- package/README.md +233 -389
- package/README.zh-CN.md +235 -362
- package/dist/adapters/1688/assets.d.ts +2 -0
- package/dist/adapters/1688/assets.d.ts.map +1 -0
- package/dist/adapters/1688/assets.js +102 -0
- package/dist/adapters/1688/assets.js.map +1 -0
- package/dist/adapters/51job/jobs.d.ts +2 -0
- package/dist/adapters/51job/jobs.d.ts.map +1 -0
- package/dist/adapters/51job/jobs.js +292 -0
- package/dist/adapters/51job/jobs.js.map +1 -0
- package/dist/adapters/_shared/browser-tools.d.ts +10 -0
- package/dist/adapters/_shared/browser-tools.d.ts.map +1 -0
- package/dist/adapters/_shared/browser-tools.js +42 -0
- package/dist/adapters/_shared/browser-tools.js.map +1 -0
- package/dist/adapters/antigravity/extra.d.ts +2 -0
- package/dist/adapters/antigravity/extra.d.ts.map +1 -0
- package/dist/adapters/antigravity/extra.js +52 -0
- package/dist/adapters/antigravity/extra.js.map +1 -0
- package/dist/adapters/baidu-scholar/search.d.ts +2 -0
- package/dist/adapters/baidu-scholar/search.d.ts.map +1 -0
- package/dist/adapters/baidu-scholar/search.js +34 -0
- package/dist/adapters/baidu-scholar/search.js.map +1 -0
- package/dist/adapters/bilibili/compat.d.ts +2 -0
- package/dist/adapters/bilibili/compat.d.ts.map +1 -0
- package/dist/adapters/bilibili/compat.js +114 -0
- package/dist/adapters/bilibili/compat.js.map +1 -0
- package/dist/adapters/chatgpt/image.d.ts +2 -0
- package/dist/adapters/chatgpt/image.d.ts.map +1 -0
- package/dist/adapters/chatgpt/image.js +81 -0
- package/dist/adapters/chatgpt/image.js.map +1 -0
- package/dist/adapters/chatgpt-app/chatgpt-app.d.ts +2 -0
- package/dist/adapters/chatgpt-app/chatgpt-app.d.ts.map +1 -0
- package/dist/adapters/chatgpt-app/chatgpt-app.js +9 -0
- package/dist/adapters/chatgpt-app/chatgpt-app.js.map +1 -0
- package/dist/adapters/chatwise/extra.d.ts +2 -0
- package/dist/adapters/chatwise/extra.d.ts.map +1 -0
- package/dist/adapters/chatwise/extra.js +35 -0
- package/dist/adapters/chatwise/extra.js.map +1 -0
- package/dist/adapters/codex/extra.d.ts +2 -0
- package/dist/adapters/codex/extra.d.ts.map +1 -0
- package/dist/adapters/codex/extra.js +35 -0
- package/dist/adapters/codex/extra.js.map +1 -0
- package/dist/adapters/deepseek/web.d.ts +2 -0
- package/dist/adapters/deepseek/web.d.ts.map +1 -0
- package/dist/adapters/deepseek/web.js +187 -0
- package/dist/adapters/deepseek/web.js.map +1 -0
- package/dist/adapters/doubao/web.d.ts +2 -0
- package/dist/adapters/doubao/web.d.ts.map +1 -0
- package/dist/adapters/doubao/web.js +138 -0
- package/dist/adapters/doubao/web.js.map +1 -0
- package/dist/adapters/eastmoney/market-data.d.ts +2 -0
- package/dist/adapters/eastmoney/market-data.d.ts.map +1 -0
- package/dist/adapters/eastmoney/market-data.js +753 -0
- package/dist/adapters/eastmoney/market-data.js.map +1 -0
- package/dist/adapters/gitee/user.d.ts +2 -0
- package/dist/adapters/gitee/user.d.ts.map +1 -0
- package/dist/adapters/gitee/user.js +54 -0
- package/dist/adapters/gitee/user.js.map +1 -0
- package/dist/adapters/google-scholar/cite.d.ts +2 -0
- package/dist/adapters/google-scholar/cite.d.ts.map +1 -0
- package/dist/adapters/google-scholar/cite.js +82 -0
- package/dist/adapters/google-scholar/cite.js.map +1 -0
- package/dist/adapters/google-scholar/profile.d.ts +2 -0
- package/dist/adapters/google-scholar/profile.d.ts.map +1 -0
- package/dist/adapters/google-scholar/profile.js +80 -0
- package/dist/adapters/google-scholar/profile.js.map +1 -0
- package/dist/adapters/google-scholar/search.d.ts +2 -0
- package/dist/adapters/google-scholar/search.d.ts.map +1 -0
- package/dist/adapters/google-scholar/search.js +54 -0
- package/dist/adapters/google-scholar/search.js.map +1 -0
- package/dist/adapters/gov-law/laws.d.ts +2 -0
- package/dist/adapters/gov-law/laws.d.ts.map +1 -0
- package/dist/adapters/gov-law/laws.js +58 -0
- package/dist/adapters/gov-law/laws.js.map +1 -0
- package/dist/adapters/gov-policy/policy.d.ts +2 -0
- package/dist/adapters/gov-policy/policy.d.ts.map +1 -0
- package/dist/adapters/gov-policy/policy.js +58 -0
- package/dist/adapters/gov-policy/policy.js.map +1 -0
- package/dist/adapters/jd/commerce.d.ts +2 -0
- package/dist/adapters/jd/commerce.d.ts.map +1 -0
- package/dist/adapters/jd/commerce.js +133 -0
- package/dist/adapters/jd/commerce.js.map +1 -0
- package/dist/adapters/jianyu/detail.d.ts +2 -0
- package/dist/adapters/jianyu/detail.d.ts.map +1 -0
- package/dist/adapters/jianyu/detail.js +27 -0
- package/dist/adapters/jianyu/detail.js.map +1 -0
- package/dist/adapters/jimeng/workspace.d.ts +2 -0
- package/dist/adapters/jimeng/workspace.d.ts.map +1 -0
- package/dist/adapters/jimeng/workspace.js +49 -0
- package/dist/adapters/jimeng/workspace.js.map +1 -0
- package/dist/adapters/ke/rent-transaction.d.ts +2 -0
- package/dist/adapters/ke/rent-transaction.d.ts.map +1 -0
- package/dist/adapters/ke/rent-transaction.js +63 -0
- package/dist/adapters/ke/rent-transaction.js.map +1 -0
- package/dist/adapters/linux-do/search.d.ts +2 -0
- package/dist/adapters/linux-do/search.d.ts.map +1 -0
- package/dist/adapters/linux-do/search.js +75 -0
- package/dist/adapters/linux-do/search.js.map +1 -0
- package/dist/adapters/linux-do/topic-content.d.ts +2 -0
- package/dist/adapters/linux-do/topic-content.d.ts.map +1 -0
- package/dist/adapters/linux-do/topic-content.js +32 -0
- package/dist/adapters/linux-do/topic-content.js.map +1 -0
- package/dist/adapters/maimai/talents.d.ts +2 -0
- package/dist/adapters/maimai/talents.d.ts.map +1 -0
- package/dist/adapters/maimai/talents.js +64 -0
- package/dist/adapters/maimai/talents.js.map +1 -0
- package/dist/adapters/mubu/docs.d.ts +2 -0
- package/dist/adapters/mubu/docs.d.ts.map +1 -0
- package/dist/adapters/mubu/docs.js +96 -0
- package/dist/adapters/mubu/docs.js.map +1 -0
- package/dist/adapters/nowcoder/nowcoder.d.ts +2 -0
- package/dist/adapters/nowcoder/nowcoder.d.ts.map +1 -0
- package/dist/adapters/nowcoder/nowcoder.js +480 -0
- package/dist/adapters/nowcoder/nowcoder.js.map +1 -0
- package/dist/adapters/powerchina/search.d.ts +2 -0
- package/dist/adapters/powerchina/search.d.ts.map +1 -0
- package/dist/adapters/powerchina/search.js +41 -0
- package/dist/adapters/powerchina/search.js.map +1 -0
- package/dist/adapters/quark/actions.d.ts +2 -0
- package/dist/adapters/quark/actions.d.ts.map +1 -0
- package/dist/adapters/quark/actions.js +151 -0
- package/dist/adapters/quark/actions.js.map +1 -0
- package/dist/adapters/reddit/browser-utils.d.ts +7 -0
- package/dist/adapters/reddit/browser-utils.d.ts.map +1 -0
- package/dist/adapters/reddit/browser-utils.js +68 -0
- package/dist/adapters/reddit/browser-utils.js.map +1 -0
- package/dist/adapters/reddit/listings.d.ts +2 -0
- package/dist/adapters/reddit/listings.d.ts.map +1 -0
- package/dist/adapters/reddit/listings.js +193 -0
- package/dist/adapters/reddit/listings.js.map +1 -0
- package/dist/adapters/reddit/search.d.ts +2 -0
- package/dist/adapters/reddit/search.d.ts.map +1 -0
- package/dist/adapters/reddit/search.js +66 -0
- package/dist/adapters/reddit/search.js.map +1 -0
- package/dist/adapters/spotify/api.d.ts +2 -0
- package/dist/adapters/spotify/api.d.ts.map +1 -0
- package/dist/adapters/spotify/api.js +252 -0
- package/dist/adapters/spotify/api.js.map +1 -0
- package/dist/adapters/taobao/commerce.d.ts +2 -0
- package/dist/adapters/taobao/commerce.d.ts.map +1 -0
- package/dist/adapters/taobao/commerce.js +130 -0
- package/dist/adapters/taobao/commerce.js.map +1 -0
- package/dist/adapters/tdx/hot-rank.d.ts +2 -0
- package/dist/adapters/tdx/hot-rank.d.ts.map +1 -0
- package/dist/adapters/tdx/hot-rank.js +34 -0
- package/dist/adapters/tdx/hot-rank.js.map +1 -0
- package/dist/adapters/ths/hot-rank.d.ts +2 -0
- package/dist/adapters/ths/hot-rank.d.ts.map +1 -0
- package/dist/adapters/ths/hot-rank.js +34 -0
- package/dist/adapters/ths/hot-rank.js.map +1 -0
- package/dist/adapters/toutiao/articles.d.ts +2 -0
- package/dist/adapters/toutiao/articles.d.ts.map +1 -0
- package/dist/adapters/toutiao/articles.js +41 -0
- package/dist/adapters/toutiao/articles.js.map +1 -0
- package/dist/adapters/twitter/lists-extra.d.ts +2 -0
- package/dist/adapters/twitter/lists-extra.d.ts.map +1 -0
- package/dist/adapters/twitter/lists-extra.js +125 -0
- package/dist/adapters/twitter/lists-extra.js.map +1 -0
- package/dist/adapters/uiverse/components.d.ts +2 -0
- package/dist/adapters/uiverse/components.d.ts.map +1 -0
- package/dist/adapters/uiverse/components.js +138 -0
- package/dist/adapters/uiverse/components.js.map +1 -0
- package/dist/adapters/wanfang/search.d.ts +2 -0
- package/dist/adapters/wanfang/search.d.ts.map +1 -0
- package/dist/adapters/wanfang/search.js +34 -0
- package/dist/adapters/wanfang/search.js.map +1 -0
- package/dist/adapters/weixin/drafts.d.ts +2 -0
- package/dist/adapters/weixin/drafts.d.ts.map +1 -0
- package/dist/adapters/weixin/drafts.js +69 -0
- package/dist/adapters/weixin/drafts.js.map +1 -0
- package/dist/adapters/weread/ai-outline.d.ts +2 -0
- package/dist/adapters/weread/ai-outline.d.ts.map +1 -0
- package/dist/adapters/weread/ai-outline.js +28 -0
- package/dist/adapters/weread/ai-outline.js.map +1 -0
- package/dist/adapters/xiaoyuzhou/media.d.ts +2 -0
- package/dist/adapters/xiaoyuzhou/media.d.ts.map +1 -0
- package/dist/adapters/xiaoyuzhou/media.js +105 -0
- package/dist/adapters/xiaoyuzhou/media.js.map +1 -0
- package/dist/adapters/xueqiu/extra.d.ts +2 -0
- package/dist/adapters/xueqiu/extra.d.ts.map +1 -0
- package/dist/adapters/xueqiu/extra.js +74 -0
- package/dist/adapters/xueqiu/extra.js.map +1 -0
- package/dist/adapters/youtube/personal.d.ts +2 -0
- package/dist/adapters/youtube/personal.d.ts.map +1 -0
- package/dist/adapters/youtube/personal.js +183 -0
- package/dist/adapters/youtube/personal.js.map +1 -0
- package/dist/adapters/zhihu/actions.d.ts +2 -0
- package/dist/adapters/zhihu/actions.d.ts.map +1 -0
- package/dist/adapters/zhihu/actions.js +110 -0
- package/dist/adapters/zhihu/actions.js.map +1 -0
- package/dist/browser/adapter-authoring.d.ts.map +1 -1
- package/dist/browser/adapter-authoring.js +3 -3
- package/dist/browser/adapter-authoring.js.map +1 -1
- package/dist/browser/bridge.d.ts.map +1 -1
- package/dist/browser/bridge.js +2 -3
- package/dist/browser/bridge.js.map +1 -1
- package/dist/browser/daemon-client.d.ts.map +1 -1
- package/dist/browser/daemon-client.js +17 -5
- package/dist/browser/daemon-client.js.map +1 -1
- package/dist/browser/network-cache.js +2 -2
- package/dist/browser/network-cache.js.map +1 -1
- package/dist/browser/site-memory.d.ts.map +1 -1
- package/dist/browser/site-memory.js +11 -8
- package/dist/browser/site-memory.js.map +1 -1
- package/dist/browser/target-errors.d.ts +2 -2
- package/dist/browser/target-errors.js +2 -2
- package/dist/browser/verify-fixture.d.ts.map +1 -1
- package/dist/browser/verify-fixture.js +2 -2
- package/dist/browser/verify-fixture.js.map +1 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +9 -4
- package/dist/cli.js.map +1 -1
- package/dist/commands/agents.d.ts.map +1 -1
- package/dist/commands/agents.js +5 -2
- package/dist/commands/agents.js.map +1 -1
- package/dist/commands/auth.d.ts.map +1 -1
- package/dist/commands/auth.js +8 -2
- package/dist/commands/auth.js.map +1 -1
- package/dist/commands/browser-operator-runtime.d.ts.map +1 -1
- package/dist/commands/browser-operator-runtime.js +2 -2
- package/dist/commands/browser-operator-runtime.js.map +1 -1
- package/dist/commands/browser-operator.d.ts.map +1 -1
- package/dist/commands/browser-operator.js +2 -2
- package/dist/commands/browser-operator.js.map +1 -1
- package/dist/commands/describe.d.ts +2 -2
- package/dist/commands/describe.d.ts.map +1 -1
- package/dist/commands/describe.js +14 -3
- package/dist/commands/describe.js.map +1 -1
- package/dist/commands/dispatch.d.ts.map +1 -1
- package/dist/commands/dispatch.js +7 -6
- package/dist/commands/dispatch.js.map +1 -1
- package/dist/commands/explore.js +2 -2
- package/dist/commands/explore.js.map +1 -1
- package/dist/commands/generate.js +3 -3
- package/dist/commands/generate.js.map +1 -1
- package/dist/commands/health.d.ts.map +1 -1
- package/dist/commands/health.js +4 -10
- package/dist/commands/health.js.map +1 -1
- package/dist/commands/migrate.d.ts +9 -9
- package/dist/commands/migrate.d.ts.map +1 -1
- package/dist/commands/migrate.js +22 -22
- package/dist/commands/migrate.js.map +1 -1
- package/dist/commands/repair.d.ts.map +1 -1
- package/dist/commands/repair.js +9 -4
- package/dist/commands/repair.js.map +1 -1
- package/dist/commands/search.d.ts +3 -3
- package/dist/commands/search.js +4 -4
- package/dist/commands/skills.d.ts +3 -0
- package/dist/commands/skills.d.ts.map +1 -1
- package/dist/commands/skills.js +8 -4
- package/dist/commands/skills.js.map +1 -1
- package/dist/commands/synthesize.js +2 -2
- package/dist/commands/synthesize.js.map +1 -1
- package/dist/discovery/aliases.d.ts.map +1 -1
- package/dist/discovery/aliases.js +69 -0
- package/dist/discovery/aliases.js.map +1 -1
- package/dist/discovery/loader.d.ts.map +1 -1
- package/dist/discovery/loader.js +130 -0
- package/dist/discovery/loader.js.map +1 -1
- package/dist/engine/kernel/execute.d.ts.map +1 -1
- package/dist/engine/kernel/execute.js +27 -4
- package/dist/engine/kernel/execute.js.map +1 -1
- package/dist/engine/user-home.d.ts +9 -0
- package/dist/engine/user-home.d.ts.map +1 -0
- package/dist/engine/user-home.js +12 -0
- package/dist/engine/user-home.js.map +1 -0
- package/dist/fast-path.d.ts +14 -0
- package/dist/fast-path.d.ts.map +1 -0
- package/dist/fast-path.js +565 -0
- package/dist/fast-path.js.map +1 -0
- package/dist/hub/index.d.ts +1 -1
- package/dist/hub/index.d.ts.map +1 -1
- package/dist/hub/index.js +28 -17
- package/dist/hub/index.js.map +1 -1
- package/dist/main.js +6 -3
- package/dist/main.js.map +1 -1
- package/dist/manifest-compact.txt +11 -11
- package/dist/manifest-search.json +1 -1
- package/dist/manifest.json +27600 -1016
- package/dist/mcp/tools.js +1 -1
- package/dist/mcp/tools.js.map +1 -1
- package/dist/output/error-map.d.ts.map +1 -1
- package/dist/output/error-map.js +16 -5
- package/dist/output/error-map.js.map +1 -1
- package/dist/protocol/acp.js +1 -1
- package/dist/protocol/acp.js.map +1 -1
- package/dist/registry.d.ts +3 -0
- package/dist/registry.d.ts.map +1 -1
- package/dist/registry.js +34 -1
- package/dist/registry.js.map +1 -1
- package/dist/runtime/usage-ledger.d.ts +6 -7
- package/dist/runtime/usage-ledger.d.ts.map +1 -1
- package/dist/runtime/usage-ledger.js +6 -7
- package/dist/runtime/usage-ledger.js.map +1 -1
- package/dist/types.d.ts +2 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js.map +1 -1
- package/package.json +12 -6
- package/src/adapters/1688/assets.ts +114 -0
- package/src/adapters/51job/jobs.ts +327 -0
- package/src/adapters/_shared/browser-tools.ts +59 -0
- package/src/adapters/antigravity/extra.ts +53 -0
- package/src/adapters/baidu-scholar/search.ts +38 -0
- package/src/adapters/bilibili/compat.ts +132 -0
- package/src/adapters/binance/asks.yaml +41 -0
- package/src/adapters/binance/depth.yaml +41 -0
- package/src/adapters/binance/gainers.yaml +42 -0
- package/src/adapters/binance/hot.yaml +1 -1
- package/src/adapters/binance/kline.yaml +1 -1
- package/src/adapters/binance/klines.yaml +46 -0
- package/src/adapters/binance/losers.yaml +42 -0
- package/src/adapters/binance/pairs.yaml +38 -0
- package/src/adapters/binance/price.yaml +32 -0
- package/src/adapters/binance/prices.yaml +30 -0
- package/src/adapters/binance/ticker.yaml +2 -2
- package/src/adapters/binance/top.yaml +46 -0
- package/src/adapters/binance/trades.yaml +42 -0
- package/src/adapters/chatgpt/image.ts +84 -0
- package/src/adapters/chatgpt-app/chatgpt-app.ts +11 -0
- package/src/adapters/chatwise/extra.ts +36 -0
- package/src/adapters/codex/extra.ts +36 -0
- package/src/adapters/deepseek/web.ts +203 -0
- package/src/adapters/doubao/web.ts +154 -0
- package/src/adapters/eastmoney/market-data.ts +829 -0
- package/src/adapters/excel/insert-image.yaml +80 -0
- package/src/adapters/excel/insert-link.yaml +90 -0
- package/src/adapters/excel/list.yaml +63 -0
- package/src/adapters/excel/read.yaml +74 -0
- package/src/adapters/excel/set-cell.yaml +77 -0
- package/src/adapters/excel/set-font.yaml +87 -0
- package/src/adapters/excel/status.yaml +29 -0
- package/src/adapters/gh/search-repos.yaml +62 -0
- package/src/adapters/gitee/user.ts +59 -0
- package/src/adapters/google-scholar/cite.ts +95 -0
- package/src/adapters/google-scholar/profile.ts +91 -0
- package/src/adapters/google-scholar/search.ts +58 -0
- package/src/adapters/gov-law/laws.ts +61 -0
- package/src/adapters/gov-policy/policy.ts +61 -0
- package/src/adapters/imessage/contact.yaml +7 -2
- package/src/adapters/imessage/recent.yaml +7 -2
- package/src/adapters/imessage/search.yaml +7 -2
- package/src/adapters/jd/commerce.ts +142 -0
- package/src/adapters/jianyu/detail.ts +28 -0
- package/src/adapters/jimeng/workspace.ts +53 -0
- package/src/adapters/ke/rent-transaction.ts +70 -0
- package/src/adapters/linux-do/search.ts +92 -0
- package/src/adapters/linux-do/topic-content.ts +45 -0
- package/src/adapters/maimai/talents.ts +65 -0
- package/src/adapters/mubu/docs.ts +107 -0
- package/src/adapters/nowcoder/nowcoder.ts +570 -0
- package/src/adapters/powerchina/search.ts +45 -0
- package/src/adapters/powerpoint/add-slide.yaml +93 -0
- package/src/adapters/powerpoint/insert-image.yaml +71 -0
- package/src/adapters/powerpoint/insert-link.yaml +94 -0
- package/src/adapters/powerpoint/list.yaml +58 -0
- package/src/adapters/powerpoint/set-font.yaml +101 -0
- package/src/adapters/powerpoint/slides.yaml +73 -0
- package/src/adapters/powerpoint/status.yaml +29 -0
- package/src/adapters/quark/actions.ts +169 -0
- package/src/adapters/reddit/browser-utils.ts +93 -0
- package/src/adapters/reddit/listings.ts +223 -0
- package/src/adapters/reddit/search.ts +74 -0
- package/src/adapters/spotify/api.ts +292 -0
- package/src/adapters/taobao/commerce.ts +134 -0
- package/src/adapters/tdx/hot-rank.ts +35 -0
- package/src/adapters/ths/hot-rank.ts +35 -0
- package/src/adapters/toutiao/articles.ts +45 -0
- package/src/adapters/twitter/lists-extra.ts +139 -0
- package/src/adapters/uiverse/components.ts +154 -0
- package/src/adapters/vercel/list.yaml +2 -1
- package/src/adapters/wanfang/search.ts +38 -0
- package/src/adapters/weixin/drafts.ts +74 -0
- package/src/adapters/weread/ai-outline.ts +32 -0
- package/src/adapters/word/insert-image.yaml +73 -0
- package/src/adapters/word/insert-link.yaml +82 -0
- package/src/adapters/word/insert-text.yaml +72 -0
- package/src/adapters/word/list.yaml +63 -0
- package/src/adapters/word/read.yaml +63 -0
- package/src/adapters/word/set-font.yaml +97 -0
- package/src/adapters/word/status.yaml +29 -0
- package/src/adapters/xiaoyuzhou/media.ts +139 -0
- package/src/adapters/xueqiu/extra.ts +78 -0
- package/src/adapters/youtube/personal.ts +215 -0
- package/src/adapters/zhihu/actions.ts +111 -0
- package/src/hub/external-clis-harness.yaml +211 -0
- package/src/hub/index.ts +29 -25
- package/src/adapters/linux-do/search.yaml +0 -44
- package/src/adapters/meituan/hot.yaml +0 -34
- package/src/adapters/reddit/frontpage.test.ts +0 -15
- package/src/adapters/reddit/frontpage.yaml +0 -41
- package/src/adapters/reddit/hot.test.ts +0 -15
- package/src/adapters/reddit/hot.yaml +0 -41
- package/src/adapters/reddit/new.test.ts +0 -15
- package/src/adapters/reddit/new.yaml +0 -42
- package/src/adapters/reddit/popular.test.ts +0 -15
- package/src/adapters/reddit/popular.yaml +0 -41
- package/src/adapters/reddit/rising.test.ts +0 -15
- package/src/adapters/reddit/rising.yaml +0 -41
- package/src/adapters/reddit/search.test.ts +0 -15
- package/src/adapters/reddit/search.yaml +0 -52
- package/src/adapters/reddit/subreddit.test.ts +0 -15
- package/src/adapters/reddit/subreddit.yaml +0 -50
- package/src/adapters/reddit/top.test.ts +0 -15
- package/src/adapters/reddit/top.yaml +0 -48
- package/src/adapters/reddit/trending.test.ts +0 -15
- package/src/adapters/reddit/trending.yaml +0 -33
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
import { cli, Strategy } from "../../registry.js";
|
|
2
|
+
import type { AdapterArg, IPage } from "../../types.js";
|
|
3
|
+
import {
|
|
4
|
+
clampLimit,
|
|
5
|
+
mapRedditPosts,
|
|
6
|
+
normalizeSubreddit,
|
|
7
|
+
redditChildren,
|
|
8
|
+
redditJson,
|
|
9
|
+
} from "./browser-utils.js";
|
|
10
|
+
|
|
11
|
+
const POST_COLUMNS = [
|
|
12
|
+
"rank",
|
|
13
|
+
"title",
|
|
14
|
+
"subreddit",
|
|
15
|
+
"author",
|
|
16
|
+
"score",
|
|
17
|
+
"comments",
|
|
18
|
+
"url",
|
|
19
|
+
];
|
|
20
|
+
|
|
21
|
+
const LIMIT_ARG: AdapterArg = {
|
|
22
|
+
name: "limit",
|
|
23
|
+
type: "int",
|
|
24
|
+
default: 20,
|
|
25
|
+
description: "Number of posts",
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
const SUBREDDIT_ARG: AdapterArg = {
|
|
29
|
+
name: "subreddit",
|
|
30
|
+
type: "str",
|
|
31
|
+
default: "",
|
|
32
|
+
description: "Subreddit name without r/",
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
const TIME_ARG: AdapterArg = {
|
|
36
|
+
name: "time",
|
|
37
|
+
type: "str",
|
|
38
|
+
default: "day",
|
|
39
|
+
choices: ["hour", "day", "week", "month", "year", "all"],
|
|
40
|
+
description: "Time window",
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
async function postsFromPath(
|
|
44
|
+
page: IPage,
|
|
45
|
+
path: string,
|
|
46
|
+
kwargs: Record<string, unknown>,
|
|
47
|
+
params: Record<string, string | number | boolean | undefined> = {},
|
|
48
|
+
): Promise<Array<Record<string, unknown>>> {
|
|
49
|
+
const limit = clampLimit(kwargs.limit);
|
|
50
|
+
const data = await redditJson(page, path, { limit, ...params });
|
|
51
|
+
return mapRedditPosts(redditChildren(data), limit);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
cli({
|
|
55
|
+
site: "reddit",
|
|
56
|
+
name: "hot",
|
|
57
|
+
description: "Reddit front page hot posts",
|
|
58
|
+
domain: "www.reddit.com",
|
|
59
|
+
strategy: Strategy.COOKIE,
|
|
60
|
+
browser: true,
|
|
61
|
+
args: [SUBREDDIT_ARG, LIMIT_ARG],
|
|
62
|
+
columns: POST_COLUMNS,
|
|
63
|
+
func: async (page, kwargs) => {
|
|
64
|
+
const subreddit = normalizeSubreddit(kwargs.subreddit);
|
|
65
|
+
const path = subreddit
|
|
66
|
+
? `/r/${encodeURIComponent(subreddit)}/hot.json`
|
|
67
|
+
: "/r/all/hot.json";
|
|
68
|
+
return postsFromPath(page as IPage, path, kwargs);
|
|
69
|
+
},
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
cli({
|
|
73
|
+
site: "reddit",
|
|
74
|
+
name: "frontpage",
|
|
75
|
+
description: "Reddit front page / r/all",
|
|
76
|
+
domain: "www.reddit.com",
|
|
77
|
+
strategy: Strategy.COOKIE,
|
|
78
|
+
browser: true,
|
|
79
|
+
args: [LIMIT_ARG],
|
|
80
|
+
columns: POST_COLUMNS,
|
|
81
|
+
func: async (page, kwargs) =>
|
|
82
|
+
postsFromPath(page as IPage, "/r/all.json", kwargs),
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
cli({
|
|
86
|
+
site: "reddit",
|
|
87
|
+
name: "popular",
|
|
88
|
+
description: "Reddit popular posts (/r/popular)",
|
|
89
|
+
domain: "www.reddit.com",
|
|
90
|
+
strategy: Strategy.COOKIE,
|
|
91
|
+
browser: true,
|
|
92
|
+
args: [LIMIT_ARG],
|
|
93
|
+
columns: POST_COLUMNS,
|
|
94
|
+
func: async (page, kwargs) =>
|
|
95
|
+
postsFromPath(page as IPage, "/r/popular.json", kwargs),
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
cli({
|
|
99
|
+
site: "reddit",
|
|
100
|
+
name: "new",
|
|
101
|
+
description: "Reddit newest posts",
|
|
102
|
+
domain: "www.reddit.com",
|
|
103
|
+
strategy: Strategy.COOKIE,
|
|
104
|
+
browser: true,
|
|
105
|
+
args: [SUBREDDIT_ARG, LIMIT_ARG],
|
|
106
|
+
columns: POST_COLUMNS,
|
|
107
|
+
func: async (page, kwargs) => {
|
|
108
|
+
const subreddit = normalizeSubreddit(kwargs.subreddit);
|
|
109
|
+
const path = subreddit
|
|
110
|
+
? `/r/${encodeURIComponent(subreddit)}/new.json`
|
|
111
|
+
: "/new.json";
|
|
112
|
+
return postsFromPath(page as IPage, path, kwargs);
|
|
113
|
+
},
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
cli({
|
|
117
|
+
site: "reddit",
|
|
118
|
+
name: "top",
|
|
119
|
+
description: "Reddit top posts",
|
|
120
|
+
domain: "www.reddit.com",
|
|
121
|
+
strategy: Strategy.COOKIE,
|
|
122
|
+
browser: true,
|
|
123
|
+
args: [SUBREDDIT_ARG, TIME_ARG, LIMIT_ARG],
|
|
124
|
+
columns: POST_COLUMNS,
|
|
125
|
+
func: async (page, kwargs) => {
|
|
126
|
+
const subreddit = normalizeSubreddit(kwargs.subreddit);
|
|
127
|
+
const path = subreddit
|
|
128
|
+
? `/r/${encodeURIComponent(subreddit)}/top.json`
|
|
129
|
+
: "/top.json";
|
|
130
|
+
return postsFromPath(page as IPage, path, kwargs, {
|
|
131
|
+
t: String(kwargs.time ?? "day"),
|
|
132
|
+
});
|
|
133
|
+
},
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
cli({
|
|
137
|
+
site: "reddit",
|
|
138
|
+
name: "rising",
|
|
139
|
+
description: "Reddit rising posts",
|
|
140
|
+
domain: "www.reddit.com",
|
|
141
|
+
strategy: Strategy.COOKIE,
|
|
142
|
+
browser: true,
|
|
143
|
+
args: [SUBREDDIT_ARG, LIMIT_ARG],
|
|
144
|
+
columns: POST_COLUMNS,
|
|
145
|
+
func: async (page, kwargs) => {
|
|
146
|
+
const subreddit = normalizeSubreddit(kwargs.subreddit);
|
|
147
|
+
const path = subreddit
|
|
148
|
+
? `/r/${encodeURIComponent(subreddit)}/rising.json`
|
|
149
|
+
: "/rising.json";
|
|
150
|
+
return postsFromPath(page as IPage, path, kwargs);
|
|
151
|
+
},
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
cli({
|
|
155
|
+
site: "reddit",
|
|
156
|
+
name: "subreddit",
|
|
157
|
+
description: "Get posts from a specific subreddit",
|
|
158
|
+
domain: "www.reddit.com",
|
|
159
|
+
strategy: Strategy.COOKIE,
|
|
160
|
+
browser: true,
|
|
161
|
+
args: [
|
|
162
|
+
{
|
|
163
|
+
name: "name",
|
|
164
|
+
type: "str",
|
|
165
|
+
required: true,
|
|
166
|
+
positional: true,
|
|
167
|
+
description: "Subreddit name",
|
|
168
|
+
},
|
|
169
|
+
{
|
|
170
|
+
name: "sort",
|
|
171
|
+
type: "str",
|
|
172
|
+
default: "hot",
|
|
173
|
+
choices: ["hot", "new", "top", "rising"],
|
|
174
|
+
description: "Sort order",
|
|
175
|
+
},
|
|
176
|
+
TIME_ARG,
|
|
177
|
+
LIMIT_ARG,
|
|
178
|
+
],
|
|
179
|
+
columns: POST_COLUMNS,
|
|
180
|
+
func: async (page, kwargs) => {
|
|
181
|
+
const name = normalizeSubreddit(kwargs.name);
|
|
182
|
+
const sort = String(kwargs.sort ?? "hot");
|
|
183
|
+
const limit = clampLimit(kwargs.limit);
|
|
184
|
+
const data = await redditJson(
|
|
185
|
+
page as IPage,
|
|
186
|
+
`/r/${encodeURIComponent(name)}/${sort}.json`,
|
|
187
|
+
{
|
|
188
|
+
limit,
|
|
189
|
+
t: sort === "top" ? String(kwargs.time ?? "day") : undefined,
|
|
190
|
+
},
|
|
191
|
+
);
|
|
192
|
+
return mapRedditPosts(redditChildren(data), limit);
|
|
193
|
+
},
|
|
194
|
+
});
|
|
195
|
+
|
|
196
|
+
cli({
|
|
197
|
+
site: "reddit",
|
|
198
|
+
name: "trending",
|
|
199
|
+
description: "Reddit trending subreddits",
|
|
200
|
+
domain: "www.reddit.com",
|
|
201
|
+
strategy: Strategy.COOKIE,
|
|
202
|
+
browser: true,
|
|
203
|
+
args: [LIMIT_ARG],
|
|
204
|
+
columns: ["name", "subscribers", "title", "description", "url"],
|
|
205
|
+
func: async (page, kwargs) => {
|
|
206
|
+
const limit = clampLimit(kwargs.limit, 25);
|
|
207
|
+
const data = await redditJson(page as IPage, "/subreddits/popular.json", {
|
|
208
|
+
limit,
|
|
209
|
+
});
|
|
210
|
+
return redditChildren(data)
|
|
211
|
+
.slice(0, limit)
|
|
212
|
+
.map((child) => {
|
|
213
|
+
const item = (child.data ?? {}) as Record<string, unknown>;
|
|
214
|
+
return {
|
|
215
|
+
name: String(item.display_name ?? ""),
|
|
216
|
+
subscribers: Number(item.subscribers ?? 0),
|
|
217
|
+
title: String(item.title ?? ""),
|
|
218
|
+
description: String(item.public_description ?? "").slice(0, 120),
|
|
219
|
+
url: item.url ? `https://www.reddit.com${String(item.url)}` : "",
|
|
220
|
+
};
|
|
221
|
+
});
|
|
222
|
+
},
|
|
223
|
+
});
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { cli, Strategy } from "../../registry.js";
|
|
2
|
+
import type { IPage } from "../../types.js";
|
|
3
|
+
import {
|
|
4
|
+
clampLimit,
|
|
5
|
+
mapRedditPosts,
|
|
6
|
+
normalizeSubreddit,
|
|
7
|
+
redditChildren,
|
|
8
|
+
redditJson,
|
|
9
|
+
} from "./browser-utils.js";
|
|
10
|
+
|
|
11
|
+
cli({
|
|
12
|
+
site: "reddit",
|
|
13
|
+
name: "search",
|
|
14
|
+
description: "Search Reddit posts",
|
|
15
|
+
domain: "www.reddit.com",
|
|
16
|
+
strategy: Strategy.COOKIE,
|
|
17
|
+
browser: true,
|
|
18
|
+
args: [
|
|
19
|
+
{
|
|
20
|
+
name: "query",
|
|
21
|
+
type: "str",
|
|
22
|
+
required: true,
|
|
23
|
+
positional: true,
|
|
24
|
+
description: "Search query",
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
name: "subreddit",
|
|
28
|
+
type: "str",
|
|
29
|
+
default: "",
|
|
30
|
+
description: "Restrict search to a subreddit",
|
|
31
|
+
},
|
|
32
|
+
{
|
|
33
|
+
name: "sort",
|
|
34
|
+
type: "str",
|
|
35
|
+
default: "relevance",
|
|
36
|
+
choices: ["relevance", "hot", "top", "new", "comments"],
|
|
37
|
+
description: "Sort order",
|
|
38
|
+
},
|
|
39
|
+
{
|
|
40
|
+
name: "time",
|
|
41
|
+
type: "str",
|
|
42
|
+
default: "all",
|
|
43
|
+
choices: ["hour", "day", "week", "month", "year", "all"],
|
|
44
|
+
description: "Time window",
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
name: "limit",
|
|
48
|
+
type: "int",
|
|
49
|
+
default: 20,
|
|
50
|
+
description: "Number of results",
|
|
51
|
+
},
|
|
52
|
+
],
|
|
53
|
+
columns: ["title", "subreddit", "author", "score", "comments", "url"],
|
|
54
|
+
func: async (page, kwargs) => {
|
|
55
|
+
const p = page as IPage;
|
|
56
|
+
const query = String(kwargs.query ?? "");
|
|
57
|
+
const subreddit = normalizeSubreddit(kwargs.subreddit);
|
|
58
|
+
const sort = String(kwargs.sort ?? "relevance");
|
|
59
|
+
const time = String(kwargs.time ?? "all");
|
|
60
|
+
const limit = clampLimit(kwargs.limit);
|
|
61
|
+
const path = subreddit
|
|
62
|
+
? `/r/${encodeURIComponent(subreddit)}/search.json`
|
|
63
|
+
: "/search.json";
|
|
64
|
+
const data = await redditJson(p, path, {
|
|
65
|
+
q: query,
|
|
66
|
+
sort,
|
|
67
|
+
t: time,
|
|
68
|
+
limit,
|
|
69
|
+
restrict_sr: subreddit ? "on" : "off",
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
return mapRedditPosts(redditChildren(data), limit);
|
|
73
|
+
},
|
|
74
|
+
});
|
|
@@ -0,0 +1,292 @@
|
|
|
1
|
+
import { mkdir, readFile, writeFile } from "node:fs/promises";
|
|
2
|
+
import { homedir } from "node:os";
|
|
3
|
+
import { dirname, join } from "node:path";
|
|
4
|
+
import { cli, Strategy } from "../../registry.js";
|
|
5
|
+
import { intArg, str } from "../_shared/browser-tools.js";
|
|
6
|
+
|
|
7
|
+
interface SpotifyTokens {
|
|
8
|
+
access_token?: string;
|
|
9
|
+
refresh_token?: string;
|
|
10
|
+
expires_at?: number;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
interface SpotifyConfig {
|
|
14
|
+
SPOTIFY_CLIENT_ID?: string;
|
|
15
|
+
SPOTIFY_CLIENT_SECRET?: string;
|
|
16
|
+
SPOTIFY_REDIRECT_URI?: string;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
const TOKEN_PATH = join(homedir(), ".unicli", "spotify-tokens.json");
|
|
20
|
+
const ENV_PATH = join(homedir(), ".unicli", "spotify.env");
|
|
21
|
+
|
|
22
|
+
async function readConfig(): Promise<SpotifyConfig> {
|
|
23
|
+
const config: SpotifyConfig = {
|
|
24
|
+
SPOTIFY_CLIENT_ID: process.env.SPOTIFY_CLIENT_ID,
|
|
25
|
+
SPOTIFY_CLIENT_SECRET: process.env.SPOTIFY_CLIENT_SECRET,
|
|
26
|
+
SPOTIFY_REDIRECT_URI: process.env.SPOTIFY_REDIRECT_URI,
|
|
27
|
+
};
|
|
28
|
+
try {
|
|
29
|
+
const text = await readFile(ENV_PATH, "utf8");
|
|
30
|
+
for (const line of text.split(/\r?\n/)) {
|
|
31
|
+
const match = /^\s*([A-Z0-9_]+)\s*=\s*(.*?)\s*$/.exec(line);
|
|
32
|
+
if (match) config[match[1] as keyof SpotifyConfig] = match[2];
|
|
33
|
+
}
|
|
34
|
+
} catch {
|
|
35
|
+
// Environment variables are enough for non-interactive use.
|
|
36
|
+
}
|
|
37
|
+
return config;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
async function readTokens(): Promise<SpotifyTokens> {
|
|
41
|
+
try {
|
|
42
|
+
return JSON.parse(await readFile(TOKEN_PATH, "utf8")) as SpotifyTokens;
|
|
43
|
+
} catch {
|
|
44
|
+
return {};
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
async function writeTokens(tokens: SpotifyTokens): Promise<void> {
|
|
49
|
+
await mkdir(dirname(TOKEN_PATH), { recursive: true });
|
|
50
|
+
await writeFile(TOKEN_PATH, JSON.stringify(tokens, null, 2));
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
async function refreshAccessToken(): Promise<string> {
|
|
54
|
+
const config = await readConfig();
|
|
55
|
+
const tokens = await readTokens();
|
|
56
|
+
if (tokens.access_token && (tokens.expires_at ?? 0) > Date.now() + 60_000) {
|
|
57
|
+
return tokens.access_token;
|
|
58
|
+
}
|
|
59
|
+
if (!tokens.refresh_token) {
|
|
60
|
+
throw new Error("Spotify refresh token missing. Run spotify auth first.");
|
|
61
|
+
}
|
|
62
|
+
if (!config.SPOTIFY_CLIENT_ID || !config.SPOTIFY_CLIENT_SECRET) {
|
|
63
|
+
throw new Error(`Spotify app config missing in ${ENV_PATH}`);
|
|
64
|
+
}
|
|
65
|
+
const response = await fetch("https://accounts.spotify.com/api/token", {
|
|
66
|
+
method: "POST",
|
|
67
|
+
headers: {
|
|
68
|
+
authorization: `Basic ${Buffer.from(
|
|
69
|
+
`${config.SPOTIFY_CLIENT_ID}:${config.SPOTIFY_CLIENT_SECRET}`,
|
|
70
|
+
).toString("base64")}`,
|
|
71
|
+
"content-type": "application/x-www-form-urlencoded",
|
|
72
|
+
},
|
|
73
|
+
body: new URLSearchParams({
|
|
74
|
+
grant_type: "refresh_token",
|
|
75
|
+
refresh_token: tokens.refresh_token,
|
|
76
|
+
}),
|
|
77
|
+
});
|
|
78
|
+
if (!response.ok) {
|
|
79
|
+
throw new Error(`Spotify token refresh failed: HTTP ${response.status}`);
|
|
80
|
+
}
|
|
81
|
+
const data = (await response.json()) as SpotifyTokens & {
|
|
82
|
+
expires_in?: number;
|
|
83
|
+
};
|
|
84
|
+
const next = {
|
|
85
|
+
...tokens,
|
|
86
|
+
...data,
|
|
87
|
+
expires_at: Date.now() + Number(data.expires_in ?? 3600) * 1000,
|
|
88
|
+
};
|
|
89
|
+
await writeTokens(next);
|
|
90
|
+
return str(next.access_token);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
async function spotifyApi(
|
|
94
|
+
path: string,
|
|
95
|
+
init: RequestInit = {},
|
|
96
|
+
): Promise<unknown> {
|
|
97
|
+
const token = await refreshAccessToken();
|
|
98
|
+
const response = await fetch(`https://api.spotify.com/v1${path}`, {
|
|
99
|
+
...init,
|
|
100
|
+
headers: {
|
|
101
|
+
authorization: `Bearer ${token}`,
|
|
102
|
+
"content-type": "application/json",
|
|
103
|
+
...init.headers,
|
|
104
|
+
},
|
|
105
|
+
});
|
|
106
|
+
if (response.status === 204) return { ok: true };
|
|
107
|
+
if (!response.ok) {
|
|
108
|
+
const preview = await response.text().catch(() => "");
|
|
109
|
+
throw new Error(
|
|
110
|
+
`Spotify API failed: HTTP ${response.status} ${preview.slice(0, 160)}`,
|
|
111
|
+
);
|
|
112
|
+
}
|
|
113
|
+
return response.json();
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
async function searchTrack(query: string): Promise<string> {
|
|
117
|
+
const data = (await spotifyApi(
|
|
118
|
+
`/search?type=track&limit=1&q=${encodeURIComponent(query)}`,
|
|
119
|
+
)) as { tracks?: { items?: Array<{ uri?: string }> } };
|
|
120
|
+
const uri = data.tracks?.items?.[0]?.uri;
|
|
121
|
+
if (!uri) throw new Error(`No Spotify track found for query: ${query}`);
|
|
122
|
+
return uri;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
cli({
|
|
126
|
+
site: "spotify",
|
|
127
|
+
name: "auth",
|
|
128
|
+
description: "Create or complete Spotify OAuth setup",
|
|
129
|
+
domain: "api.spotify.com",
|
|
130
|
+
strategy: Strategy.PUBLIC,
|
|
131
|
+
args: [{ name: "code", type: "str", required: false }],
|
|
132
|
+
columns: ["status", "url"],
|
|
133
|
+
func: async (_page, kwargs) => {
|
|
134
|
+
const config = await readConfig();
|
|
135
|
+
const redirect =
|
|
136
|
+
config.SPOTIFY_REDIRECT_URI ?? "http://127.0.0.1:8888/callback";
|
|
137
|
+
if (!config.SPOTIFY_CLIENT_ID) {
|
|
138
|
+
throw new Error(`Spotify client id missing in ${ENV_PATH}`);
|
|
139
|
+
}
|
|
140
|
+
if (!kwargs.code) {
|
|
141
|
+
const scopes = [
|
|
142
|
+
"user-read-playback-state",
|
|
143
|
+
"user-modify-playback-state",
|
|
144
|
+
"user-read-currently-playing",
|
|
145
|
+
"streaming",
|
|
146
|
+
].join(" ");
|
|
147
|
+
const url = new URL("https://accounts.spotify.com/authorize");
|
|
148
|
+
url.searchParams.set("client_id", config.SPOTIFY_CLIENT_ID);
|
|
149
|
+
url.searchParams.set("response_type", "code");
|
|
150
|
+
url.searchParams.set("redirect_uri", redirect);
|
|
151
|
+
url.searchParams.set("scope", scopes);
|
|
152
|
+
return [{ status: "open_authorize_url", url: url.href }];
|
|
153
|
+
}
|
|
154
|
+
if (!config.SPOTIFY_CLIENT_SECRET) {
|
|
155
|
+
throw new Error(`Spotify client secret missing in ${ENV_PATH}`);
|
|
156
|
+
}
|
|
157
|
+
const response = await fetch("https://accounts.spotify.com/api/token", {
|
|
158
|
+
method: "POST",
|
|
159
|
+
headers: {
|
|
160
|
+
authorization: `Basic ${Buffer.from(
|
|
161
|
+
`${config.SPOTIFY_CLIENT_ID}:${config.SPOTIFY_CLIENT_SECRET}`,
|
|
162
|
+
).toString("base64")}`,
|
|
163
|
+
"content-type": "application/x-www-form-urlencoded",
|
|
164
|
+
},
|
|
165
|
+
body: new URLSearchParams({
|
|
166
|
+
grant_type: "authorization_code",
|
|
167
|
+
code: str(kwargs.code),
|
|
168
|
+
redirect_uri: redirect,
|
|
169
|
+
}),
|
|
170
|
+
});
|
|
171
|
+
if (!response.ok) {
|
|
172
|
+
throw new Error(`Spotify auth failed: HTTP ${response.status}`);
|
|
173
|
+
}
|
|
174
|
+
const data = (await response.json()) as SpotifyTokens & {
|
|
175
|
+
expires_in?: number;
|
|
176
|
+
};
|
|
177
|
+
await writeTokens({
|
|
178
|
+
...data,
|
|
179
|
+
expires_at: Date.now() + Number(data.expires_in ?? 3600) * 1000,
|
|
180
|
+
});
|
|
181
|
+
return [{ status: "saved", url: TOKEN_PATH }];
|
|
182
|
+
},
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
cli({
|
|
186
|
+
site: "spotify",
|
|
187
|
+
name: "status",
|
|
188
|
+
description: "Show current Spotify playback status",
|
|
189
|
+
domain: "api.spotify.com",
|
|
190
|
+
strategy: Strategy.COOKIE,
|
|
191
|
+
columns: ["track", "artist", "is_playing", "progress_ms"],
|
|
192
|
+
func: async () => {
|
|
193
|
+
const data = (await spotifyApi("/me/player")) as {
|
|
194
|
+
is_playing?: boolean;
|
|
195
|
+
progress_ms?: number;
|
|
196
|
+
item?: { name?: string; artists?: Array<{ name?: string }> };
|
|
197
|
+
device?: { name?: string; volume_percent?: number };
|
|
198
|
+
};
|
|
199
|
+
return [
|
|
200
|
+
{
|
|
201
|
+
track: data.item?.name ?? "",
|
|
202
|
+
artist: data.item?.artists?.map((a) => a.name).join(", ") ?? "",
|
|
203
|
+
is_playing: data.is_playing ?? false,
|
|
204
|
+
progress_ms: data.progress_ms ?? 0,
|
|
205
|
+
device: data.device?.name ?? "",
|
|
206
|
+
volume: data.device?.volume_percent ?? "",
|
|
207
|
+
},
|
|
208
|
+
];
|
|
209
|
+
},
|
|
210
|
+
});
|
|
211
|
+
|
|
212
|
+
cli({
|
|
213
|
+
site: "spotify",
|
|
214
|
+
name: "volume",
|
|
215
|
+
description: "Set Spotify playback volume",
|
|
216
|
+
domain: "api.spotify.com",
|
|
217
|
+
strategy: Strategy.COOKIE,
|
|
218
|
+
args: [{ name: "percent", type: "int", required: true, positional: true }],
|
|
219
|
+
columns: ["ok", "volume"],
|
|
220
|
+
func: async (_page, kwargs) => {
|
|
221
|
+
const volume = intArg(kwargs.percent, 50, 100);
|
|
222
|
+
await spotifyApi(`/me/player/volume?volume_percent=${volume}`, {
|
|
223
|
+
method: "PUT",
|
|
224
|
+
});
|
|
225
|
+
return [{ ok: true, volume }];
|
|
226
|
+
},
|
|
227
|
+
});
|
|
228
|
+
|
|
229
|
+
cli({
|
|
230
|
+
site: "spotify",
|
|
231
|
+
name: "queue",
|
|
232
|
+
description: "Add a Spotify track to the playback queue",
|
|
233
|
+
domain: "api.spotify.com",
|
|
234
|
+
strategy: Strategy.COOKIE,
|
|
235
|
+
args: [{ name: "query", type: "str", required: true, positional: true }],
|
|
236
|
+
columns: ["ok", "uri"],
|
|
237
|
+
func: async (_page, kwargs) => {
|
|
238
|
+
const uri = await searchTrack(str(kwargs.query));
|
|
239
|
+
await spotifyApi(`/me/player/queue?uri=${encodeURIComponent(uri)}`, {
|
|
240
|
+
method: "POST",
|
|
241
|
+
});
|
|
242
|
+
return [{ ok: true, uri }];
|
|
243
|
+
},
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
cli({
|
|
247
|
+
site: "spotify",
|
|
248
|
+
name: "shuffle",
|
|
249
|
+
description: "Toggle Spotify shuffle mode",
|
|
250
|
+
domain: "api.spotify.com",
|
|
251
|
+
strategy: Strategy.COOKIE,
|
|
252
|
+
args: [
|
|
253
|
+
{
|
|
254
|
+
name: "state",
|
|
255
|
+
type: "str",
|
|
256
|
+
required: true,
|
|
257
|
+
positional: true,
|
|
258
|
+
choices: ["on", "off"],
|
|
259
|
+
},
|
|
260
|
+
],
|
|
261
|
+
columns: ["ok", "state"],
|
|
262
|
+
func: async (_page, kwargs) => {
|
|
263
|
+
const state = str(kwargs.state).toLowerCase() === "on";
|
|
264
|
+
await spotifyApi(`/me/player/shuffle?state=${state}`, { method: "PUT" });
|
|
265
|
+
return [{ ok: true, state }];
|
|
266
|
+
},
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
cli({
|
|
270
|
+
site: "spotify",
|
|
271
|
+
name: "repeat",
|
|
272
|
+
description: "Set Spotify repeat mode",
|
|
273
|
+
domain: "api.spotify.com",
|
|
274
|
+
strategy: Strategy.COOKIE,
|
|
275
|
+
args: [
|
|
276
|
+
{
|
|
277
|
+
name: "state",
|
|
278
|
+
type: "str",
|
|
279
|
+
required: true,
|
|
280
|
+
positional: true,
|
|
281
|
+
choices: ["off", "track", "context"],
|
|
282
|
+
},
|
|
283
|
+
],
|
|
284
|
+
columns: ["ok", "state"],
|
|
285
|
+
func: async (_page, kwargs) => {
|
|
286
|
+
const state = str(kwargs.state, "off");
|
|
287
|
+
await spotifyApi(`/me/player/repeat?state=${encodeURIComponent(state)}`, {
|
|
288
|
+
method: "PUT",
|
|
289
|
+
});
|
|
290
|
+
return [{ ok: true, state }];
|
|
291
|
+
},
|
|
292
|
+
});
|