@zenalexa/unicli 0.218.1 → 0.220.0
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 +42 -2
- package/README.md +169 -95
- package/README.zh-CN.md +169 -95
- package/bin/unicli-mcp +14 -0
- package/dist/adapters/1point3acres/forum.d.ts +58 -0
- package/dist/adapters/1point3acres/forum.d.ts.map +1 -0
- package/dist/adapters/1point3acres/forum.js +708 -0
- package/dist/adapters/1point3acres/forum.js.map +1 -0
- package/dist/adapters/aibase/news.d.ts +21 -0
- package/dist/adapters/aibase/news.d.ts.map +1 -0
- package/dist/adapters/aibase/news.js +96 -0
- package/dist/adapters/aibase/news.js.map +1 -0
- package/dist/adapters/arxiv/papers.d.ts +27 -0
- package/dist/adapters/arxiv/papers.d.ts.map +1 -0
- package/dist/adapters/arxiv/papers.js +193 -0
- package/dist/adapters/arxiv/papers.js.map +1 -0
- package/dist/adapters/bbc/topic.d.ts +24 -0
- package/dist/adapters/bbc/topic.d.ts.map +1 -0
- package/dist/adapters/bbc/topic.js +122 -0
- package/dist/adapters/bbc/topic.js.map +1 -0
- package/dist/adapters/chatgpt/web.d.ts +24 -0
- package/dist/adapters/chatgpt/web.d.ts.map +1 -0
- package/dist/adapters/chatgpt/web.js +242 -0
- package/dist/adapters/chatgpt/web.js.map +1 -0
- package/dist/adapters/claude/web.d.ts +24 -0
- package/dist/adapters/claude/web.d.ts.map +1 -0
- package/dist/adapters/claude/web.js +575 -0
- package/dist/adapters/claude/web.js.map +1 -0
- package/dist/adapters/codex/projects.d.ts +27 -0
- package/dist/adapters/codex/projects.d.ts.map +1 -0
- package/dist/adapters/codex/projects.js +147 -0
- package/dist/adapters/codex/projects.js.map +1 -0
- package/dist/adapters/coingecko/markets.d.ts +19 -0
- package/dist/adapters/coingecko/markets.d.ts.map +1 -0
- package/dist/adapters/coingecko/markets.js +474 -0
- package/dist/adapters/coingecko/markets.js.map +1 -0
- package/dist/adapters/coupang/product.d.ts +27 -0
- package/dist/adapters/coupang/product.d.ts.map +1 -0
- package/dist/adapters/coupang/product.js +211 -0
- package/dist/adapters/coupang/product.js.map +1 -0
- package/dist/adapters/crates/registry.d.ts +44 -0
- package/dist/adapters/crates/registry.d.ts.map +1 -0
- package/dist/adapters/crates/registry.js +186 -0
- package/dist/adapters/crates/registry.js.map +1 -0
- package/dist/adapters/ctrip/travel.d.ts +83 -0
- package/dist/adapters/ctrip/travel.d.ts.map +1 -0
- package/dist/adapters/ctrip/travel.js +630 -0
- package/dist/adapters/ctrip/travel.js.map +1 -0
- package/dist/adapters/dblp/publications.d.ts +41 -0
- package/dist/adapters/dblp/publications.d.ts.map +1 -0
- package/dist/adapters/dblp/publications.js +409 -0
- package/dist/adapters/dblp/publications.js.map +1 -0
- package/dist/adapters/deepseek/web.d.ts +1 -1
- package/dist/adapters/deepseek/web.d.ts.map +1 -1
- package/dist/adapters/deepseek/web.js +66 -1
- package/dist/adapters/deepseek/web.js.map +1 -1
- package/dist/adapters/defillama/protocols.d.ts +13 -0
- package/dist/adapters/defillama/protocols.d.ts.map +1 -0
- package/dist/adapters/defillama/protocols.js +218 -0
- package/dist/adapters/defillama/protocols.js.map +1 -0
- package/dist/adapters/devto/read.d.ts +26 -0
- package/dist/adapters/devto/read.d.ts.map +1 -0
- package/dist/adapters/devto/read.js +110 -0
- package/dist/adapters/devto/read.js.map +1 -0
- package/dist/adapters/dianping/shop.d.ts +38 -0
- package/dist/adapters/dianping/shop.d.ts.map +1 -0
- package/dist/adapters/dianping/shop.js +194 -0
- package/dist/adapters/dianping/shop.js.map +1 -0
- package/dist/adapters/dockerhub/registry.d.ts +36 -0
- package/dist/adapters/dockerhub/registry.d.ts.map +1 -0
- package/dist/adapters/dockerhub/registry.js +172 -0
- package/dist/adapters/dockerhub/registry.js.map +1 -0
- package/dist/adapters/endoflife/product.d.ts +11 -0
- package/dist/adapters/endoflife/product.d.ts.map +1 -0
- package/dist/adapters/endoflife/product.js +113 -0
- package/dist/adapters/endoflife/product.js.map +1 -0
- package/dist/adapters/facebook/marketplace-extra.d.ts +9 -0
- package/dist/adapters/facebook/marketplace-extra.d.ts.map +1 -0
- package/dist/adapters/facebook/marketplace-extra.js +170 -0
- package/dist/adapters/facebook/marketplace-extra.js.map +1 -0
- package/dist/adapters/flathub/apps.d.ts +17 -0
- package/dist/adapters/flathub/apps.d.ts.map +1 -0
- package/dist/adapters/flathub/apps.js +220 -0
- package/dist/adapters/flathub/apps.js.map +1 -0
- package/dist/adapters/goproxy/module.d.ts +24 -0
- package/dist/adapters/goproxy/module.d.ts.map +1 -0
- package/dist/adapters/goproxy/module.js +221 -0
- package/dist/adapters/goproxy/module.js.map +1 -0
- package/dist/adapters/grok/web.d.ts +29 -0
- package/dist/adapters/grok/web.d.ts.map +1 -0
- package/dist/adapters/grok/web.js +553 -0
- package/dist/adapters/grok/web.js.map +1 -0
- package/dist/adapters/hackernews/read.d.ts +31 -0
- package/dist/adapters/hackernews/read.d.ts.map +1 -0
- package/dist/adapters/hackernews/read.js +201 -0
- package/dist/adapters/hackernews/read.js.map +1 -0
- package/dist/adapters/hf/paper.d.ts +22 -0
- package/dist/adapters/hf/paper.d.ts.map +1 -0
- package/dist/adapters/hf/paper.js +112 -0
- package/dist/adapters/hf/paper.js.map +1 -0
- package/dist/adapters/homebrew/packages.d.ts +52 -0
- package/dist/adapters/homebrew/packages.d.ts.map +1 -0
- package/dist/adapters/homebrew/packages.js +240 -0
- package/dist/adapters/homebrew/packages.js.map +1 -0
- package/dist/adapters/indeed/jobs.d.ts +38 -0
- package/dist/adapters/indeed/jobs.d.ts.map +1 -0
- package/dist/adapters/indeed/jobs.js +300 -0
- package/dist/adapters/indeed/jobs.js.map +1 -0
- package/dist/adapters/instagram/collections.d.ts +9 -0
- package/dist/adapters/instagram/collections.d.ts.map +1 -0
- package/dist/adapters/instagram/collections.js +174 -0
- package/dist/adapters/instagram/collections.js.map +1 -0
- package/dist/adapters/lichess/players.d.ts +46 -0
- package/dist/adapters/lichess/players.d.ts.map +1 -0
- package/dist/adapters/lichess/players.js +221 -0
- package/dist/adapters/lichess/players.js.map +1 -0
- package/dist/adapters/lobsters/read-domain.d.ts +35 -0
- package/dist/adapters/lobsters/read-domain.d.ts.map +1 -0
- package/dist/adapters/lobsters/read-domain.js +306 -0
- package/dist/adapters/lobsters/read-domain.js.map +1 -0
- package/dist/adapters/maven/artifact.d.ts +30 -0
- package/dist/adapters/maven/artifact.d.ts.map +1 -0
- package/dist/adapters/maven/artifact.js +121 -0
- package/dist/adapters/maven/artifact.js.map +1 -0
- package/dist/adapters/mdn/search.d.ts +11 -0
- package/dist/adapters/mdn/search.d.ts.map +1 -0
- package/dist/adapters/mdn/search.js +115 -0
- package/dist/adapters/mdn/search.js.map +1 -0
- package/dist/adapters/medium/tag.d.ts +15 -0
- package/dist/adapters/medium/tag.d.ts.map +1 -0
- package/dist/adapters/medium/tag.js +148 -0
- package/dist/adapters/medium/tag.js.map +1 -0
- package/dist/adapters/npm/package.d.ts +32 -0
- package/dist/adapters/npm/package.d.ts.map +1 -0
- package/dist/adapters/npm/package.js +141 -0
- package/dist/adapters/npm/package.js.map +1 -0
- package/dist/adapters/nuget/package.d.ts +34 -0
- package/dist/adapters/nuget/package.d.ts.map +1 -0
- package/dist/adapters/nuget/package.js +135 -0
- package/dist/adapters/nuget/package.js.map +1 -0
- package/dist/adapters/nvd/cve.d.ts +42 -0
- package/dist/adapters/nvd/cve.d.ts.map +1 -0
- package/dist/adapters/nvd/cve.js +132 -0
- package/dist/adapters/nvd/cve.js.map +1 -0
- package/dist/adapters/oeis/sequences.d.ts +14 -0
- package/dist/adapters/oeis/sequences.d.ts.map +1 -0
- package/dist/adapters/oeis/sequences.js +219 -0
- package/dist/adapters/oeis/sequences.js.map +1 -0
- package/dist/adapters/openalex/works.d.ts +43 -0
- package/dist/adapters/openalex/works.d.ts.map +1 -0
- package/dist/adapters/openalex/works.js +267 -0
- package/dist/adapters/openalex/works.js.map +1 -0
- package/dist/adapters/openfda/records.d.ts +18 -0
- package/dist/adapters/openfda/records.d.ts.map +1 -0
- package/dist/adapters/openfda/records.js +209 -0
- package/dist/adapters/openfda/records.js.map +1 -0
- package/dist/adapters/openreview/papers.d.ts +34 -0
- package/dist/adapters/openreview/papers.d.ts.map +1 -0
- package/dist/adapters/openreview/papers.js +463 -0
- package/dist/adapters/openreview/papers.js.map +1 -0
- package/dist/adapters/osv/security.d.ts +36 -0
- package/dist/adapters/osv/security.d.ts.map +1 -0
- package/dist/adapters/osv/security.js +247 -0
- package/dist/adapters/osv/security.js.map +1 -0
- package/dist/adapters/packagist/package.d.ts +31 -0
- package/dist/adapters/packagist/package.d.ts.map +1 -0
- package/dist/adapters/packagist/package.js +108 -0
- package/dist/adapters/packagist/package.js.map +1 -0
- package/dist/adapters/pubmed/articles.d.ts +31 -0
- package/dist/adapters/pubmed/articles.d.ts.map +1 -0
- package/dist/adapters/pubmed/articles.js +385 -0
- package/dist/adapters/pubmed/articles.js.map +1 -0
- package/dist/adapters/pypi/package.d.ts +38 -0
- package/dist/adapters/pypi/package.d.ts.map +1 -0
- package/dist/adapters/pypi/package.js +235 -0
- package/dist/adapters/pypi/package.js.map +1 -0
- package/dist/adapters/qwen/web.d.ts +26 -0
- package/dist/adapters/qwen/web.d.ts.map +1 -0
- package/dist/adapters/qwen/web.js +672 -0
- package/dist/adapters/qwen/web.js.map +1 -0
- package/dist/adapters/reddit/account.d.ts +12 -0
- package/dist/adapters/reddit/account.d.ts.map +1 -0
- package/dist/adapters/reddit/account.js +409 -0
- package/dist/adapters/reddit/account.js.map +1 -0
- package/dist/adapters/rednote/web.d.ts +30 -0
- package/dist/adapters/rednote/web.d.ts.map +1 -0
- package/dist/adapters/rednote/web.js +858 -0
- package/dist/adapters/rednote/web.js.map +1 -0
- package/dist/adapters/rest-countries/countries.d.ts +14 -0
- package/dist/adapters/rest-countries/countries.d.ts.map +1 -0
- package/dist/adapters/rest-countries/countries.js +231 -0
- package/dist/adapters/rest-countries/countries.js.map +1 -0
- package/dist/adapters/reuters/article-detail.d.ts +37 -0
- package/dist/adapters/reuters/article-detail.d.ts.map +1 -0
- package/dist/adapters/reuters/article-detail.js +139 -0
- package/dist/adapters/reuters/article-detail.js.map +1 -0
- package/dist/adapters/rfc/rfc.d.ts +11 -0
- package/dist/adapters/rfc/rfc.d.ts.map +1 -0
- package/dist/adapters/rfc/rfc.js +121 -0
- package/dist/adapters/rfc/rfc.js.map +1 -0
- package/dist/adapters/rubygems/gem.d.ts +26 -0
- package/dist/adapters/rubygems/gem.d.ts.map +1 -0
- package/dist/adapters/rubygems/gem.js +96 -0
- package/dist/adapters/rubygems/gem.js.map +1 -0
- package/dist/adapters/stackoverflow/questions.d.ts +79 -0
- package/dist/adapters/stackoverflow/questions.d.ts.map +1 -0
- package/dist/adapters/stackoverflow/questions.js +504 -0
- package/dist/adapters/stackoverflow/questions.js.map +1 -0
- package/dist/adapters/steam/app.d.ts +39 -0
- package/dist/adapters/steam/app.d.ts.map +1 -0
- package/dist/adapters/steam/app.js +165 -0
- package/dist/adapters/steam/app.js.map +1 -0
- package/dist/adapters/tiktok/creator-videos.d.ts +52 -0
- package/dist/adapters/tiktok/creator-videos.d.ts.map +1 -0
- package/dist/adapters/tiktok/creator-videos.js +267 -0
- package/dist/adapters/tiktok/creator-videos.js.map +1 -0
- package/dist/adapters/tvmaze/shows.d.ts +13 -0
- package/dist/adapters/tvmaze/shows.d.ts.map +1 -0
- package/dist/adapters/tvmaze/shows.js +240 -0
- package/dist/adapters/tvmaze/shows.js.map +1 -0
- package/dist/adapters/twitter/bookmark-folders.d.ts +33 -0
- package/dist/adapters/twitter/bookmark-folders.d.ts.map +1 -0
- package/dist/adapters/twitter/bookmark-folders.js +290 -0
- package/dist/adapters/twitter/bookmark-folders.js.map +1 -0
- package/dist/adapters/twitter/quote.d.ts +18 -0
- package/dist/adapters/twitter/quote.d.ts.map +1 -0
- package/dist/adapters/twitter/quote.js +285 -0
- package/dist/adapters/twitter/quote.js.map +1 -0
- package/dist/adapters/twitter/tweet-actions.d.ts +12 -0
- package/dist/adapters/twitter/tweet-actions.d.ts.map +1 -0
- package/dist/adapters/twitter/tweet-actions.js +145 -0
- package/dist/adapters/twitter/tweet-actions.js.map +1 -0
- package/dist/adapters/twitter/tweet-url.d.ts +15 -0
- package/dist/adapters/twitter/tweet-url.d.ts.map +1 -0
- package/dist/adapters/twitter/tweet-url.js +57 -0
- package/dist/adapters/twitter/tweet-url.js.map +1 -0
- package/dist/adapters/uisdc/news.d.ts +22 -0
- package/dist/adapters/uisdc/news.d.ts.map +1 -0
- package/dist/adapters/uisdc/news.js +91 -0
- package/dist/adapters/uisdc/news.js.map +1 -0
- package/dist/adapters/weibo/favorites-publish.d.ts +28 -0
- package/dist/adapters/weibo/favorites-publish.d.ts.map +1 -0
- package/dist/adapters/weibo/favorites-publish.js +356 -0
- package/dist/adapters/weibo/favorites-publish.js.map +1 -0
- package/dist/adapters/wikidata/entities.d.ts +15 -0
- package/dist/adapters/wikidata/entities.d.ts.map +1 -0
- package/dist/adapters/wikidata/entities.js +219 -0
- package/dist/adapters/wikidata/entities.js.map +1 -0
- package/dist/adapters/wikipedia/page.d.ts +21 -0
- package/dist/adapters/wikipedia/page.d.ts.map +1 -0
- package/dist/adapters/wikipedia/page.js +116 -0
- package/dist/adapters/wikipedia/page.js.map +1 -0
- package/dist/adapters/wttr/weather.d.ts +12 -0
- package/dist/adapters/wttr/weather.d.ts.map +1 -0
- package/dist/adapters/wttr/weather.js +207 -0
- package/dist/adapters/wttr/weather.js.map +1 -0
- package/dist/adapters/xianyu/publish.d.ts +31 -0
- package/dist/adapters/xianyu/publish.d.ts.map +1 -0
- package/dist/adapters/xianyu/publish.js +349 -0
- package/dist/adapters/xianyu/publish.js.map +1 -0
- package/dist/adapters/xiaohongshu/user-helpers.d.ts +2 -2
- package/dist/adapters/xiaohongshu/user-helpers.d.ts.map +1 -1
- package/dist/adapters/xiaohongshu/user-helpers.js +5 -4
- package/dist/adapters/xiaohongshu/user-helpers.js.map +1 -1
- package/dist/adapters/yuanbao/web.d.ts +27 -0
- package/dist/adapters/yuanbao/web.d.ts.map +1 -0
- package/dist/adapters/yuanbao/web.js +365 -0
- package/dist/adapters/yuanbao/web.js.map +1 -0
- package/dist/adapters/zhihu/collection.d.ts +33 -0
- package/dist/adapters/zhihu/collection.d.ts.map +1 -0
- package/dist/adapters/zhihu/collection.js +185 -0
- package/dist/adapters/zhihu/collection.js.map +1 -0
- package/dist/adapters/zlibrary/web.d.ts +19 -0
- package/dist/adapters/zlibrary/web.d.ts.map +1 -0
- package/dist/adapters/zlibrary/web.js +153 -0
- package/dist/adapters/zlibrary/web.js.map +1 -0
- package/dist/browser/daemon-client.js +2 -2
- package/dist/browser/daemon-client.js.map +1 -1
- package/dist/cli.js +1 -1
- package/dist/cli.js.map +1 -1
- package/dist/commands/adapter-authoring.d.ts +28 -0
- package/dist/commands/adapter-authoring.d.ts.map +1 -0
- package/dist/commands/adapter-authoring.js +231 -0
- package/dist/commands/adapter-authoring.js.map +1 -0
- package/dist/commands/browser/actions.d.ts +13 -0
- package/dist/commands/browser/actions.d.ts.map +1 -0
- package/dist/commands/{browser-operator.js → browser/actions.js} +19 -12
- package/dist/commands/browser/actions.js.map +1 -0
- package/dist/commands/browser/adapter.d.ts +10 -0
- package/dist/commands/browser/adapter.d.ts.map +1 -0
- package/dist/commands/{browser-adapter-authoring.js → browser/adapter.js} +14 -7
- package/dist/commands/browser/adapter.js.map +1 -0
- package/dist/commands/browser/authoring.d.ts +10 -0
- package/dist/commands/browser/authoring.d.ts.map +1 -0
- package/dist/commands/{browser-authoring-operator.js → browser/authoring.js} +12 -5
- package/dist/commands/browser/authoring.js.map +1 -0
- package/dist/commands/browser/index.d.ts +10 -0
- package/dist/commands/browser/index.d.ts.map +1 -0
- package/dist/commands/{browser.js → browser/index.js} +13 -14
- package/dist/commands/browser/index.js.map +1 -0
- package/dist/commands/{browser-operator-runtime.d.ts → browser/runtime.d.ts} +9 -2
- package/dist/commands/browser/runtime.d.ts.map +1 -0
- package/dist/commands/{browser-operator-runtime.js → browser/runtime.js} +16 -9
- package/dist/commands/browser/runtime.js.map +1 -0
- package/dist/commands/explore.d.ts +5 -5
- package/dist/commands/explore.d.ts.map +1 -1
- package/dist/commands/explore.js +7 -97
- package/dist/commands/explore.js.map +1 -1
- package/dist/commands/generate.d.ts +5 -5
- package/dist/commands/generate.d.ts.map +1 -1
- package/dist/commands/generate.js +8 -179
- package/dist/commands/generate.js.map +1 -1
- package/dist/commands/operate.js +1 -1
- package/dist/commands/operate.js.map +1 -1
- package/dist/commands/synthesize.d.ts +5 -5
- package/dist/commands/synthesize.d.ts.map +1 -1
- package/dist/commands/synthesize.js +8 -161
- package/dist/commands/synthesize.js.map +1 -1
- package/dist/discovery/aliases.d.ts +5 -10
- package/dist/discovery/aliases.d.ts.map +1 -1
- package/dist/discovery/aliases.js +10 -10
- package/dist/discovery/aliases.js.map +1 -1
- package/dist/discovery/macos-dynamic.d.ts.map +1 -1
- package/dist/discovery/macos-dynamic.js +17 -3
- package/dist/discovery/macos-dynamic.js.map +1 -1
- package/dist/engine/browser/action-evidence.js +1 -1
- package/dist/engine/browser/action-evidence.js.map +1 -1
- package/dist/engine/steps/map.d.ts +7 -0
- package/dist/engine/steps/map.d.ts.map +1 -1
- package/dist/engine/steps/map.js +16 -2
- package/dist/engine/steps/map.js.map +1 -1
- package/dist/manifest-compact.txt +10 -10
- package/dist/manifest-search.json +1 -1
- package/dist/manifest.json +5437 -247
- package/dist/transport/refs.d.ts +5 -3
- package/dist/transport/refs.d.ts.map +1 -1
- package/dist/transport/refs.js +8 -1
- package/dist/transport/refs.js.map +1 -1
- package/dist/transport/sidecar-binary.d.ts +7 -0
- package/dist/transport/sidecar-binary.d.ts.map +1 -1
- package/dist/transport/sidecar-binary.js +28 -8
- package/dist/transport/sidecar-binary.js.map +1 -1
- package/package.json +8 -6
- package/server.json +3 -3
- package/skills/unicli/SKILL.md +1 -1
- package/skills/unicli-claude-code/SKILL.md +1 -1
- package/skills/unicli-hermes/SKILL.md +1 -1
- package/src/adapters/1point3acres/forum.test.ts +300 -0
- package/src/adapters/1point3acres/forum.ts +852 -0
- package/src/adapters/36kr/hot.yaml +1 -0
- package/src/adapters/36kr/search.yaml +1 -0
- package/src/adapters/_archived/README.md +12 -0
- package/src/adapters/{apple-music → _archived/apple-music}/rate-album.yaml +5 -12
- package/src/adapters/_archived/archive.json +44 -0
- package/src/adapters/{az → _archived/az}/account.yaml +5 -1
- package/src/adapters/{ctrip → _archived/ctrip}/hot.yaml +5 -1
- package/src/adapters/{ctrip → _archived/ctrip}/search.yaml +6 -2
- package/src/adapters/{gcloud → _archived/gcloud}/projects.yaml +5 -1
- package/src/adapters/aibase/news.test.ts +42 -0
- package/src/adapters/aibase/news.ts +118 -0
- package/src/adapters/amazon/new-releases.yaml +1 -0
- package/src/adapters/amazon/search.yaml +1 -0
- package/src/adapters/apple-podcasts/search.yaml +1 -0
- package/src/adapters/apple-podcasts/top.yaml +1 -0
- package/src/adapters/arxiv/papers.test.ts +59 -0
- package/src/adapters/arxiv/papers.ts +226 -0
- package/src/adapters/arxiv/search.yaml +1 -0
- package/src/adapters/arxiv/trending.yaml +1 -0
- package/src/adapters/baidu/hot.yaml +1 -0
- package/src/adapters/baidu/search.yaml +1 -0
- package/src/adapters/bbc/top.yaml +1 -0
- package/src/adapters/bbc/topic.test.ts +52 -0
- package/src/adapters/bbc/topic.ts +149 -0
- package/src/adapters/bilibili/feed.yaml +1 -0
- package/src/adapters/bilibili/hot.yaml +1 -0
- package/src/adapters/bilibili/trending.yaml +1 -0
- package/src/adapters/binance/hot.yaml +1 -0
- package/src/adapters/binance/top.yaml +1 -0
- package/src/adapters/bluesky/search.yaml +1 -0
- package/src/adapters/bluesky/trending.yaml +1 -0
- package/src/adapters/chatgpt/web.test.ts +121 -0
- package/src/adapters/chatgpt/web.ts +286 -0
- package/src/adapters/claude/web.test.ts +206 -0
- package/src/adapters/claude/web.ts +684 -0
- package/src/adapters/cnki/search.yaml +1 -0
- package/src/adapters/cnn/top.yaml +1 -0
- package/src/adapters/codex/projects.test.ts +77 -0
- package/src/adapters/codex/projects.ts +178 -0
- package/src/adapters/coingecko/markets.test.ts +156 -0
- package/src/adapters/coingecko/markets.ts +574 -0
- package/src/adapters/coupang/hot.yaml +1 -0
- package/src/adapters/coupang/product.test.ts +111 -0
- package/src/adapters/coupang/product.ts +256 -0
- package/src/adapters/coupang/search.yaml +1 -0
- package/src/adapters/crates/registry.test.ts +89 -0
- package/src/adapters/crates/registry.ts +247 -0
- package/src/adapters/ctrip/travel.test.ts +359 -0
- package/src/adapters/ctrip/travel.ts +792 -0
- package/src/adapters/dangdang/hot.yaml +1 -0
- package/src/adapters/dangdang/search.yaml +1 -0
- package/src/adapters/dblp/publications.test.ts +123 -0
- package/src/adapters/dblp/publications.ts +494 -0
- package/src/adapters/deepseek/web.test.ts +69 -0
- package/src/adapters/deepseek/web.ts +78 -1
- package/src/adapters/defillama/protocols.test.ts +75 -0
- package/src/adapters/defillama/protocols.ts +253 -0
- package/src/adapters/devto/read.test.ts +49 -0
- package/src/adapters/devto/read.ts +145 -0
- package/src/adapters/devto/search.yaml +1 -0
- package/src/adapters/devto/top.yaml +1 -0
- package/src/adapters/dianping/hot.yaml +1 -0
- package/src/adapters/dianping/search.yaml +1 -0
- package/src/adapters/dianping/shop.test.ts +134 -0
- package/src/adapters/dianping/shop.ts +261 -0
- package/src/adapters/dictionary/search.yaml +1 -0
- package/src/adapters/dockerhub/registry.test.ts +97 -0
- package/src/adapters/dockerhub/registry.ts +223 -0
- package/src/adapters/douban/new-movies.yaml +1 -0
- package/src/adapters/douban/search.yaml +1 -0
- package/src/adapters/doubao/new.yaml +1 -0
- package/src/adapters/douyu/hot.yaml +1 -0
- package/src/adapters/douyu/search.yaml +1 -0
- package/src/adapters/eastmoney/hot.yaml +1 -0
- package/src/adapters/eastmoney/search.yaml +1 -0
- package/src/adapters/ele/hot.yaml +1 -0
- package/src/adapters/ele/search.yaml +1 -0
- package/src/adapters/endoflife/product.test.ts +50 -0
- package/src/adapters/endoflife/product.ts +128 -0
- package/src/adapters/exchangerate/list.yaml +1 -0
- package/src/adapters/facebook/feed.yaml +1 -0
- package/src/adapters/facebook/marketplace-extra.test.ts +132 -0
- package/src/adapters/facebook/marketplace-extra.ts +213 -0
- package/src/adapters/facebook/search.yaml +1 -0
- package/src/adapters/flathub/apps.test.ts +85 -0
- package/src/adapters/flathub/apps.ts +254 -0
- package/src/adapters/futu/hot.yaml +1 -0
- package/src/adapters/gemini/deep-research-result.yaml +1 -0
- package/src/adapters/gemini/new.yaml +1 -0
- package/src/adapters/gh/search-repos.yaml +1 -0
- package/src/adapters/gitee/search.yaml +1 -0
- package/src/adapters/gitee/trending.yaml +1 -0
- package/src/adapters/gitlab/search.yaml +1 -0
- package/src/adapters/gitlab/trending.yaml +1 -0
- package/src/adapters/google/search.yaml +1 -0
- package/src/adapters/goproxy/module.test.ts +72 -0
- package/src/adapters/goproxy/module.ts +258 -0
- package/src/adapters/grok/web.test.ts +181 -0
- package/src/adapters/grok/web.ts +640 -0
- package/src/adapters/hackernews/read.test.ts +68 -0
- package/src/adapters/hackernews/read.ts +265 -0
- package/src/adapters/hf/paper.test.ts +48 -0
- package/src/adapters/hf/paper.ts +138 -0
- package/src/adapters/hf/top.yaml +1 -0
- package/src/adapters/homebrew/packages.test.ts +109 -0
- package/src/adapters/homebrew/packages.ts +304 -0
- package/src/adapters/huggingface-papers/search.yaml +1 -0
- package/src/adapters/imdb/search.yaml +1 -0
- package/src/adapters/imdb/top.yaml +1 -0
- package/src/adapters/imdb/trending.yaml +1 -0
- package/src/adapters/imessage/recent.yaml +1 -0
- package/src/adapters/imessage/search.yaml +1 -0
- package/src/adapters/indeed/jobs.test.ts +230 -0
- package/src/adapters/indeed/jobs.ts +375 -0
- package/src/adapters/instagram/collections.test.ts +94 -0
- package/src/adapters/instagram/collections.ts +206 -0
- package/src/adapters/instagram/search.yaml +1 -0
- package/src/adapters/itch-io/popular.yaml +1 -0
- package/src/adapters/itch-io/search.yaml +1 -0
- package/src/adapters/itch-io/top.yaml +1 -0
- package/src/adapters/ithome/hot.yaml +1 -0
- package/src/adapters/jianyu/search.yaml +1 -0
- package/src/adapters/jike/feed.yaml +1 -0
- package/src/adapters/jike/search.yaml +1 -0
- package/src/adapters/juejin/hot.yaml +1 -0
- package/src/adapters/juejin/search.yaml +1 -0
- package/src/adapters/kuaishou/hot.yaml +1 -0
- package/src/adapters/kuaishou/search.yaml +1 -0
- package/src/adapters/leetcode/discuss-search.yaml +1 -0
- package/src/adapters/lichess/players.test.ts +99 -0
- package/src/adapters/lichess/players.ts +277 -0
- package/src/adapters/linear/issue-list.yaml +1 -0
- package/src/adapters/linkedin/search.yaml +1 -0
- package/src/adapters/linkedin/timeline.yaml +1 -0
- package/src/adapters/linux-do/feed.yaml +1 -0
- package/src/adapters/linux-do/hot.yaml +1 -0
- package/src/adapters/lobsters/hot.yaml +1 -0
- package/src/adapters/lobsters/read-domain.test.ts +121 -0
- package/src/adapters/lobsters/read-domain.ts +400 -0
- package/src/adapters/lobsters/search.yaml +1 -0
- package/src/adapters/macos/apps-list.yaml +1 -0
- package/src/adapters/macos/calendar-list.yaml +1 -0
- package/src/adapters/macos/contacts-search.yaml +1 -0
- package/src/adapters/macos/notes-list.yaml +1 -0
- package/src/adapters/macos/notes-search.yaml +1 -0
- package/src/adapters/macos/photos-search.yaml +1 -0
- package/src/adapters/macos/reminders-list.yaml +1 -0
- package/src/adapters/macos/shortcuts-list.yaml +1 -0
- package/src/adapters/maimai/search.yaml +1 -0
- package/src/adapters/maoyan/hot.yaml +1 -0
- package/src/adapters/maoyan/search.yaml +1 -0
- package/src/adapters/mastodon/search.yaml +1 -0
- package/src/adapters/mastodon/timeline.yaml +1 -0
- package/src/adapters/mastodon/trending.yaml +1 -0
- package/src/adapters/maven/artifact.test.ts +67 -0
- package/src/adapters/maven/artifact.ts +155 -0
- package/src/adapters/maven/info.yaml +46 -0
- package/src/adapters/maven/search.yaml +44 -0
- package/src/adapters/mdn/search.test.ts +39 -0
- package/src/adapters/mdn/search.ts +133 -0
- package/src/adapters/medium/feed.yaml +1 -0
- package/src/adapters/medium/search.yaml +1 -0
- package/src/adapters/medium/tag.test.ts +64 -0
- package/src/adapters/medium/tag.ts +164 -0
- package/src/adapters/medium/trending.yaml +1 -0
- package/src/adapters/meituan/search.yaml +1 -0
- package/src/adapters/mubu/list.yaml +1 -0
- package/src/adapters/mubu/search.yaml +1 -0
- package/src/adapters/netease-music/hot.yaml +1 -0
- package/src/adapters/netease-music/search.yaml +1 -0
- package/src/adapters/netease-music/top.yaml +1 -0
- package/src/adapters/notion/search.yaml +1 -0
- package/src/adapters/npm/package.test.ts +53 -0
- package/src/adapters/npm/package.ts +177 -0
- package/src/adapters/npm-trends/trending.yaml +1 -0
- package/src/adapters/nuget/info.yaml +41 -0
- package/src/adapters/nuget/package.test.ts +102 -0
- package/src/adapters/nuget/package.ts +193 -0
- package/src/adapters/nuget/search.yaml +43 -0
- package/src/adapters/nvd/cve.test.ts +66 -0
- package/src/adapters/nvd/cve.ts +182 -0
- package/src/adapters/nytimes/search.yaml +1 -0
- package/src/adapters/nytimes/top.yaml +1 -0
- package/src/adapters/obs/stream-start.yaml +1 -0
- package/src/adapters/obs/stream-stop.yaml +1 -0
- package/src/adapters/obsidian/search.yaml +1 -0
- package/src/adapters/oeis/sequences.test.ts +71 -0
- package/src/adapters/oeis/sequences.ts +234 -0
- package/src/adapters/ollama/list.yaml +1 -0
- package/src/adapters/openalex/works.test.ts +99 -0
- package/src/adapters/openalex/works.ts +319 -0
- package/src/adapters/openfda/records.test.ts +90 -0
- package/src/adapters/openfda/records.ts +239 -0
- package/src/adapters/openreview/papers.test.ts +139 -0
- package/src/adapters/openreview/papers.ts +560 -0
- package/src/adapters/openrouter/search.yaml +1 -0
- package/src/adapters/osv/security.test.ts +91 -0
- package/src/adapters/osv/security.ts +298 -0
- package/src/adapters/packagist/info.yaml +34 -0
- package/src/adapters/packagist/package.test.ts +62 -0
- package/src/adapters/packagist/package.ts +146 -0
- package/src/adapters/packagist/search.yaml +41 -0
- package/src/adapters/pexels/search.yaml +1 -0
- package/src/adapters/pinduoduo/hot.yaml +1 -0
- package/src/adapters/pinduoduo/search.yaml +1 -0
- package/src/adapters/powerpoint/list.yaml +1 -0
- package/src/adapters/producthunt/browse.yaml +1 -0
- package/src/adapters/producthunt/hot.yaml +1 -0
- package/src/adapters/producthunt/search.yaml +1 -0
- package/src/adapters/pub-dev/info.yaml +34 -0
- package/src/adapters/pub-dev/search.yaml +43 -0
- package/src/adapters/pubmed/articles.test.ts +96 -0
- package/src/adapters/pubmed/articles.ts +497 -0
- package/src/adapters/pypi/package.test.ts +131 -0
- package/src/adapters/pypi/package.ts +297 -0
- package/src/adapters/quark/search.yaml +1 -0
- package/src/adapters/qwen/web.test.ts +176 -0
- package/src/adapters/qwen/web.ts +758 -0
- package/src/adapters/reddit/account.test.ts +56 -0
- package/src/adapters/reddit/account.ts +493 -0
- package/src/adapters/rednote/web.test.ts +354 -0
- package/src/adapters/rednote/web.ts +968 -0
- package/src/adapters/replicate/search.yaml +1 -0
- package/src/adapters/replicate/trending.yaml +1 -0
- package/src/adapters/rest-countries/countries.test.ts +80 -0
- package/src/adapters/rest-countries/countries.ts +271 -0
- package/src/adapters/reuters/article-detail.test.ts +65 -0
- package/src/adapters/reuters/article-detail.ts +186 -0
- package/src/adapters/reuters/search.yaml +1 -0
- package/src/adapters/reuters/top.yaml +1 -0
- package/src/adapters/rfc/rfc.test.ts +37 -0
- package/src/adapters/rfc/rfc.ts +133 -0
- package/src/adapters/rubygems/gem.test.ts +43 -0
- package/src/adapters/rubygems/gem.ts +126 -0
- package/src/adapters/rubygems/info.yaml +27 -0
- package/src/adapters/rubygems/search.yaml +39 -0
- package/src/adapters/sinablog/hot.yaml +1 -0
- package/src/adapters/sinablog/search.yaml +1 -0
- package/src/adapters/slack/search.yaml +1 -0
- package/src/adapters/smzdm/hot.yaml +1 -0
- package/src/adapters/smzdm/search.yaml +1 -0
- package/src/adapters/spotify/search.yaml +1 -0
- package/src/adapters/spotify/top-tracks.yaml +1 -0
- package/src/adapters/sspai/hot.yaml +1 -0
- package/src/adapters/stackoverflow/hot.yaml +1 -0
- package/src/adapters/stackoverflow/questions.test.ts +207 -0
- package/src/adapters/stackoverflow/questions.ts +765 -0
- package/src/adapters/stackoverflow/search.yaml +1 -0
- package/src/adapters/steam/app.test.ts +68 -0
- package/src/adapters/steam/app.ts +218 -0
- package/src/adapters/steam/new-releases.yaml +1 -0
- package/src/adapters/steam/search.yaml +1 -0
- package/src/adapters/steam/top-sellers.yaml +1 -0
- package/src/adapters/substack/feed.yaml +1 -0
- package/src/adapters/substack/search.yaml +1 -0
- package/src/adapters/substack/trending.yaml +1 -0
- package/src/adapters/taobao/hot.yaml +1 -0
- package/src/adapters/taobao/search.yaml +1 -0
- package/src/adapters/techcrunch/search.yaml +1 -0
- package/src/adapters/theverge/search.yaml +1 -0
- package/src/adapters/threads/hot.yaml +1 -0
- package/src/adapters/threads/search.yaml +1 -0
- package/src/adapters/tiktok/creator-videos.test.ts +158 -0
- package/src/adapters/tiktok/creator-videos.ts +370 -0
- package/src/adapters/tiktok/search.yaml +1 -0
- package/src/adapters/tiktok/trending.yaml +1 -0
- package/src/adapters/toutiao/hot.yaml +1 -0
- package/src/adapters/toutiao/search.yaml +1 -0
- package/src/adapters/tvmaze/shows.test.ts +93 -0
- package/src/adapters/tvmaze/shows.ts +271 -0
- package/src/adapters/twitch/search.yaml +1 -0
- package/src/adapters/twitch/top.yaml +1 -0
- package/src/adapters/twitter/bookmark-folders.test.ts +164 -0
- package/src/adapters/twitter/bookmark-folders.ts +366 -0
- package/src/adapters/twitter/quote.test.ts +157 -0
- package/src/adapters/twitter/quote.ts +332 -0
- package/src/adapters/twitter/tweet-actions.test.ts +51 -0
- package/src/adapters/twitter/tweet-actions.ts +187 -0
- package/src/adapters/twitter/tweet-url.ts +65 -0
- package/src/adapters/uisdc/news.test.ts +46 -0
- package/src/adapters/uisdc/news.ts +111 -0
- package/src/adapters/unsplash/search.yaml +1 -0
- package/src/adapters/v2ex/hot.yaml +1 -0
- package/src/adapters/v2ex/search.yaml +1 -0
- package/src/adapters/vercel/list.yaml +1 -0
- package/src/adapters/wechat-channels/hot.yaml +1 -0
- package/src/adapters/wechat-channels/search.yaml +1 -0
- package/src/adapters/weibo/favorites-publish.test.ts +177 -0
- package/src/adapters/weibo/favorites-publish.ts +426 -0
- package/src/adapters/weibo/feed.yaml +1 -0
- package/src/adapters/weibo/hot.yaml +1 -0
- package/src/adapters/weibo/search.yaml +1 -0
- package/src/adapters/weibo/timeline.yaml +1 -0
- package/src/adapters/weibo/trending.yaml +1 -0
- package/src/adapters/weixin/hot.yaml +1 -0
- package/src/adapters/weixin/search.yaml +1 -0
- package/src/adapters/weread/search.yaml +1 -0
- package/src/adapters/wikidata/entities.test.ts +103 -0
- package/src/adapters/wikidata/entities.ts +253 -0
- package/src/adapters/wikipedia/page.test.ts +49 -0
- package/src/adapters/wikipedia/page.ts +158 -0
- package/src/adapters/wikipedia/search.yaml +1 -0
- package/src/adapters/wikipedia/trending.yaml +1 -0
- package/src/adapters/wrangler/list.yaml +1 -0
- package/src/adapters/wttr/weather.test.ts +99 -0
- package/src/adapters/wttr/weather.ts +239 -0
- package/src/adapters/xianyu/publish.test.ts +210 -0
- package/src/adapters/xianyu/publish.ts +420 -0
- package/src/adapters/xiaohongshu/feed.yaml +1 -0
- package/src/adapters/xiaohongshu/hot.yaml +1 -0
- package/src/adapters/xiaohongshu/trending.yaml +1 -0
- package/src/adapters/xiaohongshu/user-helpers.ts +5 -2
- package/src/adapters/xueqiu/feed.yaml +1 -0
- package/src/adapters/xueqiu/hot-stock.yaml +1 -0
- package/src/adapters/xueqiu/hot.yaml +1 -0
- package/src/adapters/xueqiu/search.yaml +1 -0
- package/src/adapters/yahoo-finance/search.yaml +1 -0
- package/src/adapters/yahoo-finance/trending.yaml +1 -0
- package/src/adapters/youtube/trending.yaml +1 -0
- package/src/adapters/yuanbao/new.yaml +1 -0
- package/src/adapters/yuanbao/web.test.ts +144 -0
- package/src/adapters/yuanbao/web.ts +435 -0
- package/src/adapters/zhihu/collection.test.ts +96 -0
- package/src/adapters/zhihu/collection.ts +239 -0
- package/src/adapters/zhihu/feed.yaml +1 -0
- package/src/adapters/zhihu/hot.yaml +1 -0
- package/src/adapters/zhihu/search.yaml +1 -0
- package/src/adapters/zhihu/trending.yaml +1 -0
- package/src/adapters/zlibrary/web.test.ts +104 -0
- package/src/adapters/zlibrary/web.ts +192 -0
- package/src/adapters/zotero/search.yaml +1 -0
- package/src/adapters/zsxq/search.yaml +1 -0
- package/dist/commands/browser-adapter-authoring.d.ts +0 -3
- package/dist/commands/browser-adapter-authoring.d.ts.map +0 -1
- package/dist/commands/browser-adapter-authoring.js.map +0 -1
- package/dist/commands/browser-authoring-operator.d.ts +0 -3
- package/dist/commands/browser-authoring-operator.d.ts.map +0 -1
- package/dist/commands/browser-authoring-operator.js.map +0 -1
- package/dist/commands/browser-operator-runtime.d.ts.map +0 -1
- package/dist/commands/browser-operator-runtime.js.map +0 -1
- package/dist/commands/browser-operator.d.ts +0 -6
- package/dist/commands/browser-operator.d.ts.map +0 -1
- package/dist/commands/browser-operator.js.map +0 -1
- package/dist/commands/browser.d.ts +0 -11
- package/dist/commands/browser.d.ts.map +0 -1
- package/dist/commands/browser.js.map +0 -1
- package/dist/mcp/sse-transport.d.ts +0 -34
- package/dist/mcp/sse-transport.d.ts.map +0 -1
- package/dist/mcp/sse-transport.js +0 -182
- package/dist/mcp/sse-transport.js.map +0 -1
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @owner src/adapters/rfc/rfc.ts
|
|
3
|
+
* @does Register agent-facing IETF RFC metadata command.
|
|
4
|
+
* @needs IETF datatracker public API, strict RFC number validation, date normalization.
|
|
5
|
+
* @feeds surface coverage ledger, RFC metadata rows, standards cross-reference URLs.
|
|
6
|
+
* @breaks Datatracker API drift, weak number parsing, or silent empty rows hide RFC metadata failures.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { cli, Strategy } from "../../registry.js";
|
|
10
|
+
|
|
11
|
+
const API_BASE = "https://datatracker.ietf.org";
|
|
12
|
+
|
|
13
|
+
export function requireRfcNumber(value: unknown): number {
|
|
14
|
+
const text = String(value ?? "")
|
|
15
|
+
.trim()
|
|
16
|
+
.toLowerCase()
|
|
17
|
+
.replace(/^rfc/, "");
|
|
18
|
+
if (!text) throw new Error("rfc number cannot be empty.");
|
|
19
|
+
if (!/^\d+$/.test(text))
|
|
20
|
+
throw new Error(`rfc number "${String(value)}" is not valid.`);
|
|
21
|
+
const number = Number(text);
|
|
22
|
+
if (!Number.isInteger(number) || number < 1 || number > 999_999) {
|
|
23
|
+
throw new Error("rfc number must be an integer in [1, 999999].");
|
|
24
|
+
}
|
|
25
|
+
return number;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export function trimRfcDate(value: unknown): string | null {
|
|
29
|
+
const text = String(value ?? "").trim();
|
|
30
|
+
return /^\d{4}-\d{2}-\d{2}/.test(text) ? text.slice(0, 10) : null;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function stringField(value: unknown): string {
|
|
34
|
+
return typeof value === "string" ? value.trim() : "";
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function numberField(value: unknown): number | null {
|
|
38
|
+
const n =
|
|
39
|
+
typeof value === "number"
|
|
40
|
+
? value
|
|
41
|
+
: typeof value === "string" && value.trim()
|
|
42
|
+
? Number(value)
|
|
43
|
+
: NaN;
|
|
44
|
+
return Number.isFinite(n) ? n : null;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function objectField(value: unknown): Record<string, unknown> {
|
|
48
|
+
return value && typeof value === "object"
|
|
49
|
+
? (value as Record<string, unknown>)
|
|
50
|
+
: {};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export function mapRfcRow(
|
|
54
|
+
number: number,
|
|
55
|
+
doc: Record<string, unknown>,
|
|
56
|
+
): Record<string, unknown> {
|
|
57
|
+
const authors = Array.isArray(doc.authors)
|
|
58
|
+
? doc.authors
|
|
59
|
+
.map((author) => stringField(objectField(author).name))
|
|
60
|
+
.filter(Boolean)
|
|
61
|
+
.join(", ")
|
|
62
|
+
: "";
|
|
63
|
+
const group = objectField(doc.group);
|
|
64
|
+
const name = `rfc${number}`;
|
|
65
|
+
return {
|
|
66
|
+
rfc: number,
|
|
67
|
+
title: stringField(doc.title),
|
|
68
|
+
state: stringField(doc.state),
|
|
69
|
+
stdLevel: stringField(doc.std_level),
|
|
70
|
+
group: stringField(group.name),
|
|
71
|
+
groupType: stringField(group.type),
|
|
72
|
+
pages: numberField(doc.pages),
|
|
73
|
+
published: trimRfcDate(doc.time),
|
|
74
|
+
authors,
|
|
75
|
+
abstract: stringField(doc.abstract),
|
|
76
|
+
rfcEditorUrl: `https://www.rfc-editor.org/rfc/rfc${number}`,
|
|
77
|
+
url: `${API_BASE}/doc/${name}/`,
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
async function fetchJson(url: URL | string, label: string): Promise<unknown> {
|
|
82
|
+
const response = await fetch(url, {
|
|
83
|
+
headers: {
|
|
84
|
+
accept: "application/json",
|
|
85
|
+
"user-agent": "unicli-rfc (https://github.com/olo-dot-io/Uni-CLI)",
|
|
86
|
+
},
|
|
87
|
+
});
|
|
88
|
+
if (response.status === 404) throw new Error(`${label} returned no result.`);
|
|
89
|
+
if (response.status === 429) throw new Error(`${label} returned HTTP 429.`);
|
|
90
|
+
if (!response.ok)
|
|
91
|
+
throw new Error(`${label} returned HTTP ${response.status}.`);
|
|
92
|
+
return response.json();
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
cli({
|
|
96
|
+
site: "rfc",
|
|
97
|
+
name: "rfc",
|
|
98
|
+
description: "Fetch IETF RFC metadata by RFC number",
|
|
99
|
+
domain: "datatracker.ietf.org",
|
|
100
|
+
strategy: Strategy.PUBLIC,
|
|
101
|
+
args: [
|
|
102
|
+
{
|
|
103
|
+
name: "number",
|
|
104
|
+
type: "int",
|
|
105
|
+
required: true,
|
|
106
|
+
positional: true,
|
|
107
|
+
description: "RFC number",
|
|
108
|
+
},
|
|
109
|
+
],
|
|
110
|
+
columns: [
|
|
111
|
+
"rfc",
|
|
112
|
+
"title",
|
|
113
|
+
"state",
|
|
114
|
+
"stdLevel",
|
|
115
|
+
"group",
|
|
116
|
+
"groupType",
|
|
117
|
+
"pages",
|
|
118
|
+
"published",
|
|
119
|
+
"authors",
|
|
120
|
+
"abstract",
|
|
121
|
+
"rfcEditorUrl",
|
|
122
|
+
"url",
|
|
123
|
+
],
|
|
124
|
+
func: async (_page, kwargs) => {
|
|
125
|
+
const number = requireRfcNumber(kwargs.number);
|
|
126
|
+
const doc = objectField(
|
|
127
|
+
await fetchJson(`${API_BASE}/doc/rfc${number}/doc.json`, "rfc rfc"),
|
|
128
|
+
);
|
|
129
|
+
if (!doc.name)
|
|
130
|
+
throw new Error(`rfc rfc returned no metadata for RFC ${number}.`);
|
|
131
|
+
return [mapRfcRow(number, doc)];
|
|
132
|
+
},
|
|
133
|
+
});
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { mapRubyGemRow, requireGemName } from "./gem.js";
|
|
3
|
+
|
|
4
|
+
describe("rubygems agent-facing gem command", () => {
|
|
5
|
+
it("validates gem tokens", () => {
|
|
6
|
+
expect(requireGemName(" rails ")).toBe("rails");
|
|
7
|
+
expect(() => requireGemName("../rails")).toThrow("gem name must be");
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
it("maps gem metadata with source and bug tracker fallbacks", () => {
|
|
11
|
+
expect(
|
|
12
|
+
mapRubyGemRow(
|
|
13
|
+
{
|
|
14
|
+
name: "rails",
|
|
15
|
+
version: "8.0.0",
|
|
16
|
+
version_created_at: "2026-05-01T00:00:00.000Z",
|
|
17
|
+
downloads: 1000,
|
|
18
|
+
version_downloads: 100,
|
|
19
|
+
licenses: ["MIT"],
|
|
20
|
+
authors: "DHH",
|
|
21
|
+
homepage_uri: "https://rubyonrails.org",
|
|
22
|
+
metadata: {
|
|
23
|
+
source_code_uri: "https://github.com/rails/rails",
|
|
24
|
+
bug_tracker_uri: "https://github.com/rails/rails/issues",
|
|
25
|
+
},
|
|
26
|
+
info: "Rails is a framework",
|
|
27
|
+
project_uri: "https://rubygems.org/gems/rails",
|
|
28
|
+
},
|
|
29
|
+
"rails",
|
|
30
|
+
),
|
|
31
|
+
).toMatchObject({
|
|
32
|
+
gem: "rails",
|
|
33
|
+
version: "8.0.0",
|
|
34
|
+
releasedAt: "2026-05-01",
|
|
35
|
+
downloads: 1000,
|
|
36
|
+
versionDownloads: 100,
|
|
37
|
+
license: "MIT",
|
|
38
|
+
authors: "DHH",
|
|
39
|
+
source: "https://github.com/rails/rails",
|
|
40
|
+
bugs: "https://github.com/rails/rails/issues",
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
});
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @owner src/adapters/rubygems/gem.ts
|
|
3
|
+
* @does Register agent-facing RubyGems single-gem metadata command.
|
|
4
|
+
* @needs rubygems.org public gem JSON API.
|
|
5
|
+
* @feeds surface coverage ledger, Ruby dependency inspection, registry command surface.
|
|
6
|
+
* @breaks RubyGems envelope drift or weak gem-name validation hides gem metadata.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { cli, Strategy } from "../../registry.js";
|
|
10
|
+
|
|
11
|
+
const GEMS_BASE = "https://rubygems.org/api/v1";
|
|
12
|
+
|
|
13
|
+
interface RubyGemBody {
|
|
14
|
+
name?: unknown;
|
|
15
|
+
version?: unknown;
|
|
16
|
+
version_created_at?: unknown;
|
|
17
|
+
downloads?: unknown;
|
|
18
|
+
version_downloads?: unknown;
|
|
19
|
+
licenses?: unknown;
|
|
20
|
+
authors?: unknown;
|
|
21
|
+
homepage_uri?: unknown;
|
|
22
|
+
source_code_uri?: unknown;
|
|
23
|
+
bug_tracker_uri?: unknown;
|
|
24
|
+
metadata?: Record<string, unknown>;
|
|
25
|
+
info?: unknown;
|
|
26
|
+
project_uri?: unknown;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function stringField(value: unknown): string {
|
|
30
|
+
return typeof value === "string" ? value.trim() : "";
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function numberField(value: unknown): number | null {
|
|
34
|
+
return typeof value === "number" && Number.isFinite(value) ? value : null;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function requireGemName(value: unknown): string {
|
|
38
|
+
const name = String(value ?? "").trim();
|
|
39
|
+
if (!/^[A-Za-z0-9][A-Za-z0-9._-]*$/.test(name)) {
|
|
40
|
+
throw new Error("gem name must be a non-empty RubyGems token.");
|
|
41
|
+
}
|
|
42
|
+
return name;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export function mapRubyGemRow(
|
|
46
|
+
body: RubyGemBody,
|
|
47
|
+
requested: string,
|
|
48
|
+
): Record<string, unknown> {
|
|
49
|
+
const name = stringField(body.name) || requested;
|
|
50
|
+
if (!name) throw new Error(`RubyGems returned no metadata for ${requested}.`);
|
|
51
|
+
const licenses = Array.isArray(body.licenses)
|
|
52
|
+
? body.licenses.map(stringField).filter(Boolean).join(", ")
|
|
53
|
+
: "";
|
|
54
|
+
const metadata = body.metadata ?? {};
|
|
55
|
+
return {
|
|
56
|
+
gem: name,
|
|
57
|
+
version: stringField(body.version),
|
|
58
|
+
releasedAt: stringField(body.version_created_at).slice(0, 10),
|
|
59
|
+
downloads: numberField(body.downloads),
|
|
60
|
+
versionDownloads: numberField(body.version_downloads),
|
|
61
|
+
license: licenses,
|
|
62
|
+
authors: stringField(body.authors),
|
|
63
|
+
homepage: stringField(body.homepage_uri),
|
|
64
|
+
source:
|
|
65
|
+
stringField(body.source_code_uri) ||
|
|
66
|
+
stringField(metadata.source_code_uri),
|
|
67
|
+
bugs:
|
|
68
|
+
stringField(body.bug_tracker_uri) ||
|
|
69
|
+
stringField(metadata.bug_tracker_uri),
|
|
70
|
+
info: stringField(body.info),
|
|
71
|
+
url: stringField(body.project_uri) || `https://rubygems.org/gems/${name}`,
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
async function fetchGemJson(name: string): Promise<unknown> {
|
|
76
|
+
const response = await fetch(
|
|
77
|
+
`${GEMS_BASE}/gems/${encodeURIComponent(name)}.json`,
|
|
78
|
+
{
|
|
79
|
+
headers: {
|
|
80
|
+
"User-Agent": "unicli (https://github.com/olo-dot-io/Uni-CLI)",
|
|
81
|
+
Accept: "application/json",
|
|
82
|
+
},
|
|
83
|
+
},
|
|
84
|
+
);
|
|
85
|
+
if (response.status === 404)
|
|
86
|
+
throw new Error(`rubygems gem ${name} returned no result.`);
|
|
87
|
+
if (!response.ok)
|
|
88
|
+
throw new Error(`rubygems gem ${name} returned HTTP ${response.status}.`);
|
|
89
|
+
return response.json();
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
cli({
|
|
93
|
+
site: "rubygems",
|
|
94
|
+
name: "gem",
|
|
95
|
+
description: "Fetch RubyGems gem metadata",
|
|
96
|
+
domain: "rubygems.org",
|
|
97
|
+
strategy: Strategy.PUBLIC,
|
|
98
|
+
args: [
|
|
99
|
+
{
|
|
100
|
+
name: "name",
|
|
101
|
+
type: "str",
|
|
102
|
+
required: true,
|
|
103
|
+
positional: true,
|
|
104
|
+
description: "Gem name",
|
|
105
|
+
},
|
|
106
|
+
],
|
|
107
|
+
columns: [
|
|
108
|
+
"gem",
|
|
109
|
+
"version",
|
|
110
|
+
"releasedAt",
|
|
111
|
+
"downloads",
|
|
112
|
+
"versionDownloads",
|
|
113
|
+
"license",
|
|
114
|
+
"authors",
|
|
115
|
+
"homepage",
|
|
116
|
+
"source",
|
|
117
|
+
"bugs",
|
|
118
|
+
"info",
|
|
119
|
+
"url",
|
|
120
|
+
],
|
|
121
|
+
func: async (_page, kwargs) => {
|
|
122
|
+
const name = requireGemName(kwargs.name);
|
|
123
|
+
const body = (await fetchGemJson(name)) as RubyGemBody;
|
|
124
|
+
return [mapRubyGemRow(body, name)];
|
|
125
|
+
},
|
|
126
|
+
});
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
# @owner src/adapters/rubygems/info.yaml
|
|
2
|
+
# @does Fetch RubyGems package metadata by gem name.
|
|
3
|
+
# @needs rubygems.org gem API
|
|
4
|
+
# @feeds unicli rubygems info, package registry expansion
|
|
5
|
+
# @breaks Agents cannot inspect Ruby gem version, license, and source metadata.
|
|
6
|
+
site: rubygems
|
|
7
|
+
name: info
|
|
8
|
+
description: Get RubyGems package metadata
|
|
9
|
+
type: web-api
|
|
10
|
+
domain: rubygems.org
|
|
11
|
+
strategy: public
|
|
12
|
+
args:
|
|
13
|
+
gem:
|
|
14
|
+
type: str
|
|
15
|
+
required: true
|
|
16
|
+
positional: true
|
|
17
|
+
description: Gem name, for example rails
|
|
18
|
+
pipeline:
|
|
19
|
+
- fetch:
|
|
20
|
+
url: "https://rubygems.org/api/v1/gems/${{ args.gem | urlencode }}.json"
|
|
21
|
+
columns: [name, version, info, downloads, project_uri, homepage_uri]
|
|
22
|
+
capabilities: ["http.fetch"]
|
|
23
|
+
minimum_capability: http.fetch
|
|
24
|
+
trust: public
|
|
25
|
+
confidentiality: public
|
|
26
|
+
quarantine: false
|
|
27
|
+
schema_version: v2
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
# @owner src/adapters/rubygems/search.yaml
|
|
2
|
+
# @does Search RubyGems packages through the public JSON API.
|
|
3
|
+
# @needs rubygems.org search API
|
|
4
|
+
# @feeds unicli rubygems search, package registry expansion
|
|
5
|
+
# @breaks Ruby package discovery loses first-class CLI coverage.
|
|
6
|
+
site: rubygems
|
|
7
|
+
name: search
|
|
8
|
+
description: Search RubyGems packages
|
|
9
|
+
type: web-api
|
|
10
|
+
domain: rubygems.org
|
|
11
|
+
strategy: public
|
|
12
|
+
args:
|
|
13
|
+
query:
|
|
14
|
+
type: str
|
|
15
|
+
required: true
|
|
16
|
+
positional: true
|
|
17
|
+
description: Gem search query
|
|
18
|
+
limit:
|
|
19
|
+
type: int
|
|
20
|
+
default: 20
|
|
21
|
+
pipeline:
|
|
22
|
+
- fetch:
|
|
23
|
+
url: "https://rubygems.org/api/v1/search.json"
|
|
24
|
+
params:
|
|
25
|
+
query: "${{ args.query }}"
|
|
26
|
+
- map:
|
|
27
|
+
name: "${{ item.name }}"
|
|
28
|
+
version: "${{ item.version }}"
|
|
29
|
+
description: "${{ (item.info || '') | truncate(80) }}"
|
|
30
|
+
downloads: "${{ item.downloads }}"
|
|
31
|
+
project: "${{ item.project_uri }}"
|
|
32
|
+
- limit: ${{ args.limit }}
|
|
33
|
+
columns: [name, version, description, downloads, project]
|
|
34
|
+
capabilities: ["http.fetch"]
|
|
35
|
+
minimum_capability: http.fetch
|
|
36
|
+
trust: public
|
|
37
|
+
confidentiality: public
|
|
38
|
+
quarantine: false
|
|
39
|
+
schema_version: v2
|
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import {
|
|
3
|
+
buildStackReadRows,
|
|
4
|
+
decodeStackHtmlEntities,
|
|
5
|
+
mapStackQuestionRows,
|
|
6
|
+
mapStackUserRows,
|
|
7
|
+
requireStackLimit,
|
|
8
|
+
requireStackMinInt,
|
|
9
|
+
requireStackQuestionId,
|
|
10
|
+
requireStackString,
|
|
11
|
+
sortStackAnswers,
|
|
12
|
+
stackHtmlToText,
|
|
13
|
+
} from "./questions.js";
|
|
14
|
+
|
|
15
|
+
describe("stackoverflow agent-facing commands", () => {
|
|
16
|
+
it("validates ids, strings, limits, and converts Stack Exchange HTML", () => {
|
|
17
|
+
expect(requireStackQuestionId(" 79935770 ")).toBe("79935770");
|
|
18
|
+
expect(() => requireStackQuestionId("abc")).toThrow("numeric question id");
|
|
19
|
+
expect(requireStackString(" rust ", "tag")).toBe("rust");
|
|
20
|
+
expect(() => requireStackString("", "tag")).toThrow("cannot be empty");
|
|
21
|
+
expect(requireStackLimit(undefined, 20, 100, "limit")).toBe(20);
|
|
22
|
+
expect(() => requireStackLimit("101", 20, 100, "limit")).toThrow(
|
|
23
|
+
"[1, 100]",
|
|
24
|
+
);
|
|
25
|
+
expect(requireStackMinInt(undefined, 4000, 100, "max-length")).toBe(4000);
|
|
26
|
+
expect(() => requireStackMinInt("99", 4000, 100, "max-length")).toThrow(
|
|
27
|
+
">= 100",
|
|
28
|
+
);
|
|
29
|
+
expect(decodeStackHtmlEntities("Rust's <Trait> & API")).toBe(
|
|
30
|
+
"Rust's <Trait> & API",
|
|
31
|
+
);
|
|
32
|
+
expect(
|
|
33
|
+
stackHtmlToText(
|
|
34
|
+
'<p>Hello & <code>world</code></p><ul><li><a href="https://x.test">link</a></li></ul>',
|
|
35
|
+
),
|
|
36
|
+
).toBe("Hello & `world`\n- link (https://x.test)");
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it("maps question and user list rows with surface column names", () => {
|
|
40
|
+
expect(
|
|
41
|
+
mapStackQuestionRows([
|
|
42
|
+
{
|
|
43
|
+
question_id: 1,
|
|
44
|
+
title: "Use <T>",
|
|
45
|
+
score: 5,
|
|
46
|
+
answer_count: 2,
|
|
47
|
+
view_count: 99,
|
|
48
|
+
is_answered: true,
|
|
49
|
+
tags: ["typescript", "generics"],
|
|
50
|
+
owner: { display_name: "Jon's Bot" },
|
|
51
|
+
creation_date: 1_700_000_000,
|
|
52
|
+
last_activity_date: 1_700_100_000,
|
|
53
|
+
},
|
|
54
|
+
]),
|
|
55
|
+
).toEqual([
|
|
56
|
+
{
|
|
57
|
+
rank: 1,
|
|
58
|
+
id: 1,
|
|
59
|
+
title: "Use <T>",
|
|
60
|
+
score: 5,
|
|
61
|
+
answers: 2,
|
|
62
|
+
views: 99,
|
|
63
|
+
isAnswered: true,
|
|
64
|
+
tags: "typescript, generics",
|
|
65
|
+
author: "Jon's Bot",
|
|
66
|
+
createdAt: "2023-11-14",
|
|
67
|
+
lastActivityAt: "2023-11-16",
|
|
68
|
+
url: "https://stackoverflow.com/questions/1",
|
|
69
|
+
},
|
|
70
|
+
]);
|
|
71
|
+
|
|
72
|
+
expect(
|
|
73
|
+
mapStackUserRows([
|
|
74
|
+
{
|
|
75
|
+
user_id: 22656,
|
|
76
|
+
display_name: "Jon Skeet",
|
|
77
|
+
reputation: 1_500_000,
|
|
78
|
+
badge_counts: { gold: 900, silver: 9000, bronze: 90000 },
|
|
79
|
+
location: "Reading & London",
|
|
80
|
+
creation_date: 1_200_000_000,
|
|
81
|
+
last_access_date: 1_700_000_000,
|
|
82
|
+
},
|
|
83
|
+
]),
|
|
84
|
+
).toEqual([
|
|
85
|
+
{
|
|
86
|
+
userId: 22656,
|
|
87
|
+
displayName: "Jon Skeet",
|
|
88
|
+
reputation: 1_500_000,
|
|
89
|
+
goldBadges: 900,
|
|
90
|
+
silverBadges: 9000,
|
|
91
|
+
bronzeBadges: 90000,
|
|
92
|
+
location: "Reading & London",
|
|
93
|
+
createdAt: "2008-01-10",
|
|
94
|
+
lastAccessAt: "2023-11-14",
|
|
95
|
+
url: "https://stackoverflow.com/users/22656",
|
|
96
|
+
},
|
|
97
|
+
]);
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
it("sorts accepted answers first and builds bounded read rows", async () => {
|
|
101
|
+
expect(
|
|
102
|
+
sortStackAnswers({ accepted_answer_id: 3 }, [
|
|
103
|
+
{ answer_id: 2, score: 50 },
|
|
104
|
+
{ answer_id: 3, score: 1 },
|
|
105
|
+
{ answer_id: 4, score: 40 },
|
|
106
|
+
]).map((answer) => answer.answer_id),
|
|
107
|
+
).toEqual([3, 2, 4]);
|
|
108
|
+
|
|
109
|
+
const rows = await buildStackReadRows(
|
|
110
|
+
{
|
|
111
|
+
title: "Why?",
|
|
112
|
+
body: "<p>Question body</p>",
|
|
113
|
+
score: 10,
|
|
114
|
+
accepted_answer_id: 3,
|
|
115
|
+
owner: { display_name: "asker" },
|
|
116
|
+
link: "https://stackoverflow.com/q/1",
|
|
117
|
+
},
|
|
118
|
+
[{ body: "<p>Needs code</p>", score: 1, owner: { display_name: "c1" } }],
|
|
119
|
+
[
|
|
120
|
+
{
|
|
121
|
+
answer_id: 2,
|
|
122
|
+
body: "<p>High score</p>",
|
|
123
|
+
score: 20,
|
|
124
|
+
owner: { display_name: "a2" },
|
|
125
|
+
},
|
|
126
|
+
{
|
|
127
|
+
answer_id: 3,
|
|
128
|
+
is_accepted: true,
|
|
129
|
+
body: "<p>Accepted</p>",
|
|
130
|
+
score: 2,
|
|
131
|
+
owner: { display_name: "a3" },
|
|
132
|
+
},
|
|
133
|
+
],
|
|
134
|
+
async () => ({
|
|
135
|
+
items: [
|
|
136
|
+
{
|
|
137
|
+
post_id: 3,
|
|
138
|
+
body: "<p>Good point</p>",
|
|
139
|
+
score: 0,
|
|
140
|
+
owner: { display_name: "c3" },
|
|
141
|
+
},
|
|
142
|
+
],
|
|
143
|
+
}),
|
|
144
|
+
{
|
|
145
|
+
answersLimit: 2,
|
|
146
|
+
commentsLimit: 2,
|
|
147
|
+
maxLength: 100,
|
|
148
|
+
label: "stackoverflow/1",
|
|
149
|
+
},
|
|
150
|
+
);
|
|
151
|
+
|
|
152
|
+
expect(rows).toEqual([
|
|
153
|
+
{
|
|
154
|
+
type: "POST",
|
|
155
|
+
author: "asker",
|
|
156
|
+
score: 10,
|
|
157
|
+
accepted: "",
|
|
158
|
+
text: "Why?\n\nQuestion body\n\nhttps://stackoverflow.com/q/1",
|
|
159
|
+
},
|
|
160
|
+
{
|
|
161
|
+
type: "Q-COMMENT",
|
|
162
|
+
author: "c1",
|
|
163
|
+
score: 1,
|
|
164
|
+
accepted: "",
|
|
165
|
+
text: " > Needs code",
|
|
166
|
+
},
|
|
167
|
+
{
|
|
168
|
+
type: "ANSWER",
|
|
169
|
+
author: "a3",
|
|
170
|
+
score: 2,
|
|
171
|
+
accepted: "true",
|
|
172
|
+
text: "Accepted",
|
|
173
|
+
},
|
|
174
|
+
{
|
|
175
|
+
type: "A-COMMENT",
|
|
176
|
+
author: "c3",
|
|
177
|
+
score: 0,
|
|
178
|
+
accepted: "",
|
|
179
|
+
text: " > Good point",
|
|
180
|
+
},
|
|
181
|
+
{
|
|
182
|
+
type: "ANSWER",
|
|
183
|
+
author: "a2",
|
|
184
|
+
score: 20,
|
|
185
|
+
accepted: "",
|
|
186
|
+
text: "High score",
|
|
187
|
+
},
|
|
188
|
+
]);
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
it("refuses partial answer-comment pages", async () => {
|
|
192
|
+
await expect(
|
|
193
|
+
buildStackReadRows(
|
|
194
|
+
{ title: "Q", accepted_answer_id: 2 },
|
|
195
|
+
[],
|
|
196
|
+
[{ answer_id: 2 }],
|
|
197
|
+
async () => ({ items: [], has_more: true }),
|
|
198
|
+
{
|
|
199
|
+
answersLimit: 1,
|
|
200
|
+
commentsLimit: 2,
|
|
201
|
+
maxLength: 100,
|
|
202
|
+
label: "stackoverflow/2",
|
|
203
|
+
},
|
|
204
|
+
),
|
|
205
|
+
).rejects.toThrow("exceed one API page");
|
|
206
|
+
});
|
|
207
|
+
});
|