@zenalexa/unicli 0.219.0 → 0.220.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +15 -24
- package/README.md +62 -250
- package/README.zh-CN.md +62 -250
- 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/anilist/web.d.ts +11 -0
- package/dist/adapters/anilist/web.d.ts.map +1 -0
- package/dist/adapters/anilist/web.js +284 -0
- package/dist/adapters/anilist/web.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/bangumi/web.d.ts +14 -0
- package/dist/adapters/bangumi/web.d.ts.map +1 -0
- package/dist/adapters/bangumi/web.js +257 -0
- package/dist/adapters/bangumi/web.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/dlsite/web.d.ts +31 -0
- package/dist/adapters/dlsite/web.d.ts.map +1 -0
- package/dist/adapters/dlsite/web.js +455 -0
- package/dist/adapters/dlsite/web.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/ehentai/web.d.ts +66 -0
- package/dist/adapters/ehentai/web.d.ts.map +1 -0
- package/dist/adapters/ehentai/web.js +608 -0
- package/dist/adapters/ehentai/web.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/jikan/web.d.ts +9 -0
- package/dist/adapters/jikan/web.d.ts.map +1 -0
- package/dist/adapters/jikan/web.js +154 -0
- package/dist/adapters/jikan/web.js.map +1 -0
- package/dist/adapters/kitsu/web.d.ts +9 -0
- package/dist/adapters/kitsu/web.d.ts.map +1 -0
- package/dist/adapters/kitsu/web.js +97 -0
- package/dist/adapters/kitsu/web.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/mangadex/web.d.ts +10 -0
- package/dist/adapters/mangadex/web.d.ts.map +1 -0
- package/dist/adapters/mangadex/web.js +188 -0
- package/dist/adapters/mangadex/web.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/moegirl/web.d.ts +23 -0
- package/dist/adapters/moegirl/web.d.ts.map +1 -0
- package/dist/adapters/moegirl/web.js +269 -0
- package/dist/adapters/moegirl/web.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/safebooru/web.d.ts +10 -0
- package/dist/adapters/safebooru/web.d.ts.map +1 -0
- package/dist/adapters/safebooru/web.js +120 -0
- package/dist/adapters/safebooru/web.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/vndb/web.d.ts +10 -0
- package/dist/adapters/vndb/web.d.ts.map +1 -0
- package/dist/adapters/vndb/web.js +321 -0
- package/dist/adapters/vndb/web.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/agents/codex-pack.d.ts +62 -0
- package/dist/agents/codex-pack.d.ts.map +1 -0
- package/dist/agents/codex-pack.js +163 -0
- package/dist/agents/codex-pack.js.map +1 -0
- package/dist/browser/daemon-client.js +2 -2
- package/dist/browser/daemon-client.js.map +1 -1
- package/dist/commands/agents.d.ts.map +1 -1
- package/dist/commands/agents.js +6 -43
- package/dist/commands/agents.js.map +1 -1
- package/dist/commands/browser/adapter.d.ts.map +1 -1
- package/dist/commands/browser/adapter.js +17 -3
- package/dist/commands/browser/adapter.js.map +1 -1
- package/dist/commands/describe.d.ts.map +1 -1
- package/dist/commands/describe.js +6 -7
- package/dist/commands/describe.js.map +1 -1
- package/dist/commands/dispatch.d.ts +1 -1
- package/dist/commands/dispatch.d.ts.map +1 -1
- package/dist/commands/dispatch.js +4 -2
- package/dist/commands/dispatch.js.map +1 -1
- package/dist/commands/mcp.d.ts +1 -1
- package/dist/commands/mcp.d.ts.map +1 -1
- package/dist/commands/mcp.js +10 -5
- package/dist/commands/mcp.js.map +1 -1
- package/dist/core/command-contract-lint.d.ts +10 -0
- package/dist/core/command-contract-lint.d.ts.map +1 -0
- package/dist/core/command-contract-lint.js +41 -0
- package/dist/core/command-contract-lint.js.map +1 -0
- package/dist/core/command-contract.d.ts +100 -0
- package/dist/core/command-contract.d.ts.map +1 -0
- package/dist/core/command-contract.js +174 -0
- package/dist/core/command-contract.js.map +1 -0
- package/dist/core/index.d.ts +2 -0
- package/dist/core/index.d.ts.map +1 -1
- package/dist/core/index.js +2 -0
- package/dist/core/index.js.map +1 -1
- package/dist/discovery/aliases.d.ts +2 -2
- package/dist/discovery/aliases.d.ts.map +1 -1
- package/dist/discovery/aliases.js +464 -6
- 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/discovery/search.d.ts.map +1 -1
- package/dist/discovery/search.js +147 -2
- package/dist/discovery/search.js.map +1 -1
- package/dist/engine/args.d.ts.map +1 -1
- package/dist/engine/args.js +18 -1
- package/dist/engine/args.js.map +1 -1
- package/dist/engine/artifact-validation.d.ts +29 -0
- package/dist/engine/artifact-validation.d.ts.map +1 -0
- package/dist/engine/artifact-validation.js +211 -0
- package/dist/engine/artifact-validation.js.map +1 -0
- package/dist/engine/browser/diagnostics.d.ts +38 -0
- package/dist/engine/browser/diagnostics.d.ts.map +1 -0
- package/dist/engine/browser/diagnostics.js +40 -0
- package/dist/engine/browser/diagnostics.js.map +1 -0
- package/dist/engine/invoke.d.ts +1 -0
- package/dist/engine/invoke.d.ts.map +1 -1
- package/dist/engine/invoke.js +1 -0
- package/dist/engine/invoke.js.map +1 -1
- package/dist/engine/kernel/errors.d.ts +11 -0
- package/dist/engine/kernel/errors.d.ts.map +1 -0
- package/dist/engine/kernel/errors.js +15 -0
- package/dist/engine/kernel/errors.js.map +1 -0
- package/dist/engine/kernel/execute.d.ts +7 -18
- package/dist/engine/kernel/execute.d.ts.map +1 -1
- package/dist/engine/kernel/execute.js +25 -410
- package/dist/engine/kernel/execute.js.map +1 -1
- package/dist/engine/kernel/stages.d.ts +44 -0
- package/dist/engine/kernel/stages.d.ts.map +1 -0
- package/dist/engine/kernel/stages.js +428 -0
- package/dist/engine/kernel/stages.js.map +1 -0
- package/dist/engine/kernel/types.d.ts +21 -1
- package/dist/engine/kernel/types.d.ts.map +1 -1
- package/dist/engine/steps/download.d.ts +1 -0
- package/dist/engine/steps/download.d.ts.map +1 -1
- package/dist/engine/steps/download.js +10 -6
- package/dist/engine/steps/download.js.map +1 -1
- package/dist/fast-path/render.js +1 -1
- package/dist/fast-path/render.js.map +1 -1
- package/dist/manifest-compact.txt +10 -10
- package/dist/manifest-search.json +1 -1
- package/dist/manifest.json +8421 -332
- package/dist/mcp/handler.d.ts.map +1 -1
- package/dist/mcp/handler.js +11 -1
- package/dist/mcp/handler.js.map +1 -1
- package/dist/mcp/server.d.ts +1 -1
- package/dist/mcp/server.js +1 -1
- package/dist/mcp/tools.d.ts.map +1 -1
- package/dist/mcp/tools.js +18 -10
- package/dist/mcp/tools.js.map +1 -1
- package/dist/output/error-map.d.ts.map +1 -1
- package/dist/output/error-map.js +1 -1
- package/dist/output/error-map.js.map +1 -1
- package/dist/registry.d.ts.map +1 -1
- package/dist/registry.js +2 -1
- package/dist/registry.js.map +1 -1
- 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 +4 -3
- 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/aibase/news.test.ts +42 -0
- package/src/adapters/aibase/news.ts +118 -0
- package/src/adapters/anilist/web.test.ts +93 -0
- package/src/adapters/anilist/web.ts +341 -0
- package/src/adapters/arxiv/download.yaml +53 -0
- package/src/adapters/arxiv/papers.test.ts +59 -0
- package/src/adapters/arxiv/papers.ts +226 -0
- package/src/adapters/bangumi/web.test.ts +109 -0
- package/src/adapters/bangumi/web.ts +295 -0
- package/src/adapters/bbc/topic.test.ts +52 -0
- package/src/adapters/bbc/topic.ts +149 -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/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/product.test.ts +111 -0
- package/src/adapters/coupang/product.ts +256 -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/danbooru/artists.yaml +44 -0
- package/src/adapters/danbooru/comments.yaml +45 -0
- package/src/adapters/danbooru/detail.yaml +78 -0
- package/src/adapters/danbooru/download.yaml +51 -0
- package/src/adapters/danbooru/pools.yaml +56 -0
- package/src/adapters/danbooru/search.yaml +69 -0
- package/src/adapters/danbooru/tags.yaml +42 -0
- package/src/adapters/danbooru/wiki.yaml +44 -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/dianping/shop.test.ts +134 -0
- package/src/adapters/dianping/shop.ts +261 -0
- package/src/adapters/dlsite/web.test.ts +132 -0
- package/src/adapters/dlsite/web.ts +557 -0
- package/src/adapters/dockerhub/registry.test.ts +97 -0
- package/src/adapters/dockerhub/registry.ts +223 -0
- package/src/adapters/ehentai/web.test.ts +157 -0
- package/src/adapters/ehentai/web.ts +750 -0
- package/src/adapters/endoflife/product.test.ts +50 -0
- package/src/adapters/endoflife/product.ts +128 -0
- package/src/adapters/facebook/marketplace-extra.test.ts +132 -0
- package/src/adapters/facebook/marketplace-extra.ts +213 -0
- package/src/adapters/flathub/apps.test.ts +85 -0
- package/src/adapters/flathub/apps.ts +254 -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/homebrew/packages.test.ts +109 -0
- package/src/adapters/homebrew/packages.ts +304 -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/jikan/web.test.ts +50 -0
- package/src/adapters/jikan/web.ts +177 -0
- package/src/adapters/kitsu/web.test.ts +29 -0
- package/src/adapters/kitsu/web.ts +109 -0
- package/src/adapters/konachan/detail.yaml +62 -0
- package/src/adapters/konachan/download.yaml +55 -0
- package/src/adapters/konachan/search.yaml +65 -0
- package/src/adapters/konachan/tags.yaml +40 -0
- package/src/adapters/lichess/players.test.ts +99 -0
- package/src/adapters/lichess/players.ts +277 -0
- package/src/adapters/lobsters/read-domain.test.ts +121 -0
- package/src/adapters/lobsters/read-domain.ts +400 -0
- package/src/adapters/mangadex/web.test.ts +46 -0
- package/src/adapters/mangadex/web.ts +210 -0
- package/src/adapters/maven/artifact.test.ts +67 -0
- package/src/adapters/maven/artifact.ts +155 -0
- package/src/adapters/mdn/search.test.ts +39 -0
- package/src/adapters/mdn/search.ts +133 -0
- package/src/adapters/medium/tag.test.ts +64 -0
- package/src/adapters/medium/tag.ts +164 -0
- package/src/adapters/moegirl/web.test.ts +87 -0
- package/src/adapters/moegirl/web.ts +343 -0
- package/src/adapters/npm/package.test.ts +53 -0
- package/src/adapters/npm/package.ts +177 -0
- package/src/adapters/nuget/package.test.ts +102 -0
- package/src/adapters/nuget/package.ts +193 -0
- package/src/adapters/nvd/cve.test.ts +66 -0
- package/src/adapters/nvd/cve.ts +182 -0
- package/src/adapters/oeis/sequences.test.ts +71 -0
- package/src/adapters/oeis/sequences.ts +234 -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/osv/security.test.ts +91 -0
- package/src/adapters/osv/security.ts +298 -0
- package/src/adapters/packagist/package.test.ts +62 -0
- package/src/adapters/packagist/package.ts +146 -0
- package/src/adapters/pdf/read.yaml +49 -0
- package/src/adapters/pixiv/download.yaml +15 -2
- 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/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/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/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/safebooru/detail.yaml +63 -0
- package/src/adapters/safebooru/download.yaml +58 -0
- package/src/adapters/safebooru/search.yaml +69 -0
- package/src/adapters/safebooru/web.test.ts +60 -0
- package/src/adapters/safebooru/web.ts +130 -0
- package/src/adapters/stackoverflow/questions.test.ts +207 -0
- package/src/adapters/stackoverflow/questions.ts +765 -0
- package/src/adapters/steam/app.test.ts +68 -0
- package/src/adapters/steam/app.ts +218 -0
- package/src/adapters/tiktok/creator-videos.test.ts +158 -0
- package/src/adapters/tiktok/creator-videos.ts +370 -0
- package/src/adapters/tvmaze/shows.test.ts +93 -0
- package/src/adapters/tvmaze/shows.ts +271 -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/vndb/web.test.ts +86 -0
- package/src/adapters/vndb/web.ts +393 -0
- package/src/adapters/weibo/favorites-publish.test.ts +177 -0
- package/src/adapters/weibo/favorites-publish.ts +426 -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/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/user-helpers.ts +5 -2
- package/src/adapters/yandere/detail.yaml +61 -0
- package/src/adapters/yandere/download.yaml +56 -0
- package/src/adapters/yandere/search.yaml +67 -0
- package/src/adapters/yandere/tags.yaml +41 -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/zlibrary/web.test.ts +104 -0
- package/src/adapters/zlibrary/web.ts +192 -0
|
@@ -0,0 +1,560 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @owner src/adapters/openreview/papers.ts
|
|
3
|
+
* @does Register agent-facing OpenReview search, paper, author, venue, and reviews commands.
|
|
4
|
+
* @needs Public api2.openreview.net notes API, forum/profile id validation, note content normalization.
|
|
5
|
+
* @feeds surface coverage ledger, scholarly review workflow, agent-readable paper/review rows.
|
|
6
|
+
* @breaks OpenReview API envelope drift, content.value parsing, or silent empty threads hide paper review state.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { cli, Strategy } from "../../registry.js";
|
|
10
|
+
|
|
11
|
+
const OPENREVIEW_API = "https://api2.openreview.net";
|
|
12
|
+
const OPENREVIEW_BASE = "https://openreview.net";
|
|
13
|
+
const FORUM_ID_RE = /^[A-Za-z0-9_-]{6,20}$/;
|
|
14
|
+
const PROFILE_ID_RE = /^~(?=.*\p{L})[\p{L}\p{M}0-9._-]+\d+$/u;
|
|
15
|
+
const REVIEW_SECTION_FIELDS = [
|
|
16
|
+
["summary", "Summary"],
|
|
17
|
+
["strengths", "Strengths"],
|
|
18
|
+
["weaknesses", "Weaknesses"],
|
|
19
|
+
["questions", "Questions"],
|
|
20
|
+
["comment", "Comment"],
|
|
21
|
+
["rebuttal", "Rebuttal"],
|
|
22
|
+
["decision", "Decision"],
|
|
23
|
+
["recommendation", "Recommendation"],
|
|
24
|
+
["title", "Title"],
|
|
25
|
+
["abstract", "Abstract"],
|
|
26
|
+
["withdrawal_confirmation", "Withdrawal confirmation"],
|
|
27
|
+
] as const;
|
|
28
|
+
|
|
29
|
+
interface OpenReviewContentValue {
|
|
30
|
+
value?: unknown;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
type OpenReviewContent = Record<string, OpenReviewContentValue | undefined>;
|
|
34
|
+
|
|
35
|
+
interface OpenReviewNote {
|
|
36
|
+
id?: unknown;
|
|
37
|
+
forum?: unknown;
|
|
38
|
+
cdate?: unknown;
|
|
39
|
+
pdate?: unknown;
|
|
40
|
+
invitations?: unknown;
|
|
41
|
+
signatures?: unknown;
|
|
42
|
+
content?: OpenReviewContent;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
interface NotesEnvelope {
|
|
46
|
+
notes?: OpenReviewNote[];
|
|
47
|
+
error?: unknown;
|
|
48
|
+
errors?: unknown;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function stringField(value: unknown): string {
|
|
52
|
+
return typeof value === "string" ? value.trim() : "";
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function numberField(value: unknown): number | null {
|
|
56
|
+
return typeof value === "number" && Number.isFinite(value) ? value : null;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export function coerceOpenReviewInt(value: unknown): number {
|
|
60
|
+
if (value === undefined || value === null || value === "") return Number.NaN;
|
|
61
|
+
const n = typeof value === "number" ? value : Number(value);
|
|
62
|
+
return Number.isFinite(n) && Number.isInteger(n) ? n : Number.NaN;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
export function requireOpenReviewLimit(
|
|
66
|
+
value: unknown,
|
|
67
|
+
fallback: number,
|
|
68
|
+
max: number,
|
|
69
|
+
): number {
|
|
70
|
+
const n =
|
|
71
|
+
value === undefined || value === null || value === ""
|
|
72
|
+
? fallback
|
|
73
|
+
: coerceOpenReviewInt(value);
|
|
74
|
+
if (!Number.isInteger(n) || n < 1 || n > max) {
|
|
75
|
+
throw new Error(`openreview limit must be an integer in [1, ${max}].`);
|
|
76
|
+
}
|
|
77
|
+
return n;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
export function requireOpenReviewOffset(value: unknown, fallback = 0): number {
|
|
81
|
+
const n =
|
|
82
|
+
value === undefined || value === null || value === ""
|
|
83
|
+
? fallback
|
|
84
|
+
: coerceOpenReviewInt(value);
|
|
85
|
+
if (!Number.isInteger(n) || n < 0) {
|
|
86
|
+
throw new Error("openreview offset must be a non-negative integer.");
|
|
87
|
+
}
|
|
88
|
+
return n;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
export function requireForumId(value: unknown, label = "id"): string {
|
|
92
|
+
const id = String(value ?? "").trim();
|
|
93
|
+
if (!id) throw new Error(`openreview ${label} is required.`);
|
|
94
|
+
if (!FORUM_ID_RE.test(id)) {
|
|
95
|
+
throw new Error(
|
|
96
|
+
`openreview ${label} "${String(value)}" is not a valid forum id.`,
|
|
97
|
+
);
|
|
98
|
+
}
|
|
99
|
+
return id;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function requireProfileId(value: unknown): string {
|
|
103
|
+
const id = String(value ?? "").trim();
|
|
104
|
+
if (!id) throw new Error("openreview profile is required.");
|
|
105
|
+
if (!PROFILE_ID_RE.test(id)) {
|
|
106
|
+
throw new Error(`openreview profile "${String(value)}" is not valid.`);
|
|
107
|
+
}
|
|
108
|
+
return id;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export function readContent(
|
|
112
|
+
content: OpenReviewContent | undefined,
|
|
113
|
+
key: string,
|
|
114
|
+
): unknown {
|
|
115
|
+
return content?.[key]?.value;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
export function formatOpenReviewDate(value: unknown): string {
|
|
119
|
+
const n = numberField(value);
|
|
120
|
+
return n && n > 0 ? new Date(n).toISOString().slice(0, 10) : "";
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
export function absoluteOpenReviewPdf(value: unknown): string {
|
|
124
|
+
const pdf = stringField(value);
|
|
125
|
+
if (!pdf) return "";
|
|
126
|
+
if (/^https?:\/\//i.test(pdf)) return pdf;
|
|
127
|
+
return pdf.startsWith("/")
|
|
128
|
+
? `${OPENREVIEW_BASE}${pdf}`
|
|
129
|
+
: `${OPENREVIEW_BASE}/${pdf}`;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
function authorIdToName(value: unknown): string {
|
|
133
|
+
return String(value ?? "")
|
|
134
|
+
.replace(/^~/, "")
|
|
135
|
+
.replace(/\d+$/, "")
|
|
136
|
+
.replace(/_/g, " ")
|
|
137
|
+
.trim();
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
export function mapOpenReviewNoteRow(
|
|
141
|
+
note: OpenReviewNote,
|
|
142
|
+
): Record<string, unknown> {
|
|
143
|
+
const content = note.content ?? {};
|
|
144
|
+
const id = stringField(note.id);
|
|
145
|
+
const authors = readContent(content, "authors");
|
|
146
|
+
const authorIds = readContent(content, "authorids");
|
|
147
|
+
const authorList =
|
|
148
|
+
Array.isArray(authors) && authors.length > 0
|
|
149
|
+
? authors.map(stringField).filter(Boolean).join(", ")
|
|
150
|
+
: Array.isArray(authorIds)
|
|
151
|
+
? authorIds.map(authorIdToName).filter(Boolean).join(", ")
|
|
152
|
+
: "";
|
|
153
|
+
const keywords = readContent(content, "keywords");
|
|
154
|
+
const keywordList = Array.isArray(keywords)
|
|
155
|
+
? keywords.map(stringField).filter(Boolean).join(", ")
|
|
156
|
+
: stringField(keywords);
|
|
157
|
+
return {
|
|
158
|
+
id,
|
|
159
|
+
title: stringField(readContent(content, "title")).replace(/\s+/g, " "),
|
|
160
|
+
authors: authorList,
|
|
161
|
+
keywords: keywordList,
|
|
162
|
+
venue: stringField(readContent(content, "venue")),
|
|
163
|
+
venueid: stringField(readContent(content, "venueid")),
|
|
164
|
+
primary_area: stringField(readContent(content, "primary_area")),
|
|
165
|
+
abstract: stringField(readContent(content, "abstract")).replace(
|
|
166
|
+
/\s+/g,
|
|
167
|
+
" ",
|
|
168
|
+
),
|
|
169
|
+
pdate: formatOpenReviewDate(note.pdate ?? note.cdate),
|
|
170
|
+
pdf: absoluteOpenReviewPdf(readContent(content, "pdf")),
|
|
171
|
+
url: id ? `${OPENREVIEW_BASE}/forum?id=${id}` : "",
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
function invitationTail(note: OpenReviewNote): string {
|
|
176
|
+
const invitations = Array.isArray(note.invitations) ? note.invitations : [];
|
|
177
|
+
for (const invitation of invitations) {
|
|
178
|
+
const match = String(invitation).match(/\/-\/([^/]+)$/);
|
|
179
|
+
if (match) return match[1];
|
|
180
|
+
}
|
|
181
|
+
return "";
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
export function classifyReviewNote(
|
|
185
|
+
note: OpenReviewNote,
|
|
186
|
+
isRoot: boolean,
|
|
187
|
+
): string {
|
|
188
|
+
if (isRoot) return "PAPER";
|
|
189
|
+
const tail = invitationTail(note).toLowerCase();
|
|
190
|
+
if (tail.includes("decision")) return "DECISION";
|
|
191
|
+
if (tail.includes("withdrawal")) return "WITHDRAWAL";
|
|
192
|
+
if (tail.includes("rebuttal")) return "REBUTTAL";
|
|
193
|
+
if (tail.includes("meta")) return "META_REVIEW";
|
|
194
|
+
if (tail.includes("review")) return "REVIEW";
|
|
195
|
+
if (tail.includes("comment")) return "COMMENT";
|
|
196
|
+
return tail ? tail.toUpperCase() : "NOTE";
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
export function authorFromSignatures(signatures: unknown): string {
|
|
200
|
+
if (!Array.isArray(signatures) || signatures.length === 0) return "";
|
|
201
|
+
const signature = String(signatures[0]);
|
|
202
|
+
if (signature.startsWith("~")) return authorIdToName(signature);
|
|
203
|
+
const parts = signature.split("/");
|
|
204
|
+
return parts.at(-1) ?? signature;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
function joinReviewSections(content: OpenReviewContent | undefined): string {
|
|
208
|
+
const parts: string[] = [];
|
|
209
|
+
for (const [key, label] of REVIEW_SECTION_FIELDS) {
|
|
210
|
+
const value = readContent(content, key);
|
|
211
|
+
if (value === undefined || value === null) continue;
|
|
212
|
+
const text = Array.isArray(value) ? value.join(", ") : String(value);
|
|
213
|
+
const trimmed = text.replace(/\r\n/g, "\n").trim();
|
|
214
|
+
if (trimmed) parts.push(`${label}: ${trimmed}`);
|
|
215
|
+
}
|
|
216
|
+
return parts.join("\n\n");
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
function truncate(text: string, maxLength: number): string {
|
|
220
|
+
return text.length <= maxLength ? text : `${text.slice(0, maxLength - 3)}...`;
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
export function mapReviewThreadRows(
|
|
224
|
+
root: OpenReviewNote,
|
|
225
|
+
replies: OpenReviewNote[],
|
|
226
|
+
forum: string,
|
|
227
|
+
maxLength: number,
|
|
228
|
+
): Array<Record<string, unknown>> {
|
|
229
|
+
const sorted = [...replies]
|
|
230
|
+
.filter((note) => note.id !== forum)
|
|
231
|
+
.sort((a, b) => (numberField(a.cdate) ?? 0) - (numberField(b.cdate) ?? 0));
|
|
232
|
+
return [root, ...sorted].map((note) => {
|
|
233
|
+
const isRoot = note.id === forum;
|
|
234
|
+
const rating = readContent(note.content, "rating");
|
|
235
|
+
const confidence = readContent(note.content, "confidence");
|
|
236
|
+
return {
|
|
237
|
+
type: classifyReviewNote(note, isRoot),
|
|
238
|
+
author: authorFromSignatures(note.signatures),
|
|
239
|
+
rating: rating === undefined || rating === null ? "" : String(rating),
|
|
240
|
+
confidence:
|
|
241
|
+
confidence === undefined || confidence === null
|
|
242
|
+
? ""
|
|
243
|
+
: String(confidence),
|
|
244
|
+
text: truncate(joinReviewSections(note.content), maxLength),
|
|
245
|
+
};
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
async function fetchOpenReview(
|
|
250
|
+
path: string,
|
|
251
|
+
label: string,
|
|
252
|
+
): Promise<NotesEnvelope> {
|
|
253
|
+
const response = await fetch(`${OPENREVIEW_API}${path}`, {
|
|
254
|
+
headers: {
|
|
255
|
+
"User-Agent":
|
|
256
|
+
"unicli-openreview/1.0 (https://github.com/olo-dot-io/Uni-CLI)",
|
|
257
|
+
Accept: "application/json",
|
|
258
|
+
},
|
|
259
|
+
});
|
|
260
|
+
if (response.status === 404) return {};
|
|
261
|
+
if (!response.ok) {
|
|
262
|
+
const body = await response.text().catch(() => "");
|
|
263
|
+
throw new Error(
|
|
264
|
+
`OpenReview API HTTP ${response.status} for ${label}${body ? ` (${body.slice(0, 200)})` : ""}.`,
|
|
265
|
+
);
|
|
266
|
+
}
|
|
267
|
+
const json = (await response.json()) as NotesEnvelope;
|
|
268
|
+
const errors = Array.isArray(json.errors) ? json.errors : [];
|
|
269
|
+
const error = stringField(json.error);
|
|
270
|
+
if (errors.length > 0 || error) {
|
|
271
|
+
const detail =
|
|
272
|
+
error ||
|
|
273
|
+
errors
|
|
274
|
+
.map((entry) =>
|
|
275
|
+
typeof entry === "string"
|
|
276
|
+
? entry
|
|
277
|
+
: JSON.stringify(entry).slice(0, 200),
|
|
278
|
+
)
|
|
279
|
+
.join("; ");
|
|
280
|
+
throw new Error(`OpenReview API error for ${label}: ${detail}.`);
|
|
281
|
+
}
|
|
282
|
+
return json;
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
function notesFromEnvelope(json: NotesEnvelope): OpenReviewNote[] {
|
|
286
|
+
return Array.isArray(json.notes) ? json.notes : [];
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
cli({
|
|
290
|
+
site: "openreview",
|
|
291
|
+
name: "search",
|
|
292
|
+
description: "Search OpenReview papers by free-text query",
|
|
293
|
+
domain: "openreview.net",
|
|
294
|
+
strategy: Strategy.PUBLIC,
|
|
295
|
+
args: [
|
|
296
|
+
{
|
|
297
|
+
name: "query",
|
|
298
|
+
type: "str",
|
|
299
|
+
required: true,
|
|
300
|
+
positional: true,
|
|
301
|
+
description: "Search keyword",
|
|
302
|
+
},
|
|
303
|
+
{ name: "limit", type: "int", default: 25, description: "Max results" },
|
|
304
|
+
],
|
|
305
|
+
columns: ["rank", "id", "title", "authors", "venue", "pdate", "url"],
|
|
306
|
+
func: async (_page, kwargs) => {
|
|
307
|
+
const query = String(kwargs.query ?? "").trim();
|
|
308
|
+
if (!query) throw new Error("openreview search query cannot be empty.");
|
|
309
|
+
const limit = requireOpenReviewLimit(kwargs.limit, 25, 50);
|
|
310
|
+
const params = new URLSearchParams({
|
|
311
|
+
term: query,
|
|
312
|
+
type: "terms",
|
|
313
|
+
limit: String(limit),
|
|
314
|
+
});
|
|
315
|
+
const notes = notesFromEnvelope(
|
|
316
|
+
await fetchOpenReview(
|
|
317
|
+
`/notes/search?${params.toString()}`,
|
|
318
|
+
"openreview search",
|
|
319
|
+
),
|
|
320
|
+
);
|
|
321
|
+
if (notes.length === 0)
|
|
322
|
+
throw new Error(`No OpenReview papers found for "${query}".`);
|
|
323
|
+
return notes.slice(0, limit).map((note, index) => {
|
|
324
|
+
const row = mapOpenReviewNoteRow(note);
|
|
325
|
+
return {
|
|
326
|
+
rank: index + 1,
|
|
327
|
+
id: row.id,
|
|
328
|
+
title: row.title,
|
|
329
|
+
authors: row.authors,
|
|
330
|
+
venue: row.venue,
|
|
331
|
+
pdate: row.pdate,
|
|
332
|
+
url: row.url,
|
|
333
|
+
};
|
|
334
|
+
});
|
|
335
|
+
},
|
|
336
|
+
});
|
|
337
|
+
|
|
338
|
+
cli({
|
|
339
|
+
site: "openreview",
|
|
340
|
+
name: "paper",
|
|
341
|
+
description: "Show full metadata for a single OpenReview paper",
|
|
342
|
+
domain: "openreview.net",
|
|
343
|
+
strategy: Strategy.PUBLIC,
|
|
344
|
+
args: [
|
|
345
|
+
{
|
|
346
|
+
name: "id",
|
|
347
|
+
type: "str",
|
|
348
|
+
required: true,
|
|
349
|
+
positional: true,
|
|
350
|
+
description: "OpenReview note id",
|
|
351
|
+
},
|
|
352
|
+
],
|
|
353
|
+
columns: [
|
|
354
|
+
"id",
|
|
355
|
+
"title",
|
|
356
|
+
"authors",
|
|
357
|
+
"keywords",
|
|
358
|
+
"venue",
|
|
359
|
+
"venueid",
|
|
360
|
+
"primary_area",
|
|
361
|
+
"abstract",
|
|
362
|
+
"pdate",
|
|
363
|
+
"pdf",
|
|
364
|
+
"url",
|
|
365
|
+
],
|
|
366
|
+
func: async (_page, kwargs) => {
|
|
367
|
+
const id = requireForumId(kwargs.id);
|
|
368
|
+
const notes = notesFromEnvelope(
|
|
369
|
+
await fetchOpenReview(
|
|
370
|
+
`/notes?id=${encodeURIComponent(id)}`,
|
|
371
|
+
`openreview paper ${id}`,
|
|
372
|
+
),
|
|
373
|
+
);
|
|
374
|
+
if (notes.length === 0)
|
|
375
|
+
throw new Error(`No OpenReview paper found with id "${id}".`);
|
|
376
|
+
const row = mapOpenReviewNoteRow(notes[0]);
|
|
377
|
+
return [
|
|
378
|
+
{
|
|
379
|
+
id: row.id,
|
|
380
|
+
title: row.title,
|
|
381
|
+
authors: row.authors,
|
|
382
|
+
keywords: row.keywords,
|
|
383
|
+
venue: row.venue,
|
|
384
|
+
venueid: row.venueid,
|
|
385
|
+
primary_area: row.primary_area,
|
|
386
|
+
abstract: row.abstract,
|
|
387
|
+
pdate: row.pdate,
|
|
388
|
+
pdf: row.pdf,
|
|
389
|
+
url: row.url,
|
|
390
|
+
},
|
|
391
|
+
];
|
|
392
|
+
},
|
|
393
|
+
});
|
|
394
|
+
|
|
395
|
+
cli({
|
|
396
|
+
site: "openreview",
|
|
397
|
+
name: "author",
|
|
398
|
+
description: "List OpenReview submissions by author profile id",
|
|
399
|
+
domain: "openreview.net",
|
|
400
|
+
strategy: Strategy.PUBLIC,
|
|
401
|
+
args: [
|
|
402
|
+
{
|
|
403
|
+
name: "profile",
|
|
404
|
+
type: "str",
|
|
405
|
+
required: true,
|
|
406
|
+
positional: true,
|
|
407
|
+
description: "OpenReview profile id",
|
|
408
|
+
},
|
|
409
|
+
{ name: "limit", type: "int", default: 50, description: "Max submissions" },
|
|
410
|
+
],
|
|
411
|
+
columns: ["rank", "id", "title", "authors", "venue", "pdate", "url"],
|
|
412
|
+
func: async (_page, kwargs) => {
|
|
413
|
+
const profile = requireProfileId(kwargs.profile);
|
|
414
|
+
const limit = requireOpenReviewLimit(kwargs.limit, 50, 1000);
|
|
415
|
+
const params = new URLSearchParams({
|
|
416
|
+
"content.authorids": profile,
|
|
417
|
+
limit: String(limit),
|
|
418
|
+
sort: "cdate:desc",
|
|
419
|
+
});
|
|
420
|
+
const notes = notesFromEnvelope(
|
|
421
|
+
await fetchOpenReview(
|
|
422
|
+
`/notes?${params.toString()}`,
|
|
423
|
+
`openreview author ${profile}`,
|
|
424
|
+
),
|
|
425
|
+
);
|
|
426
|
+
if (notes.length === 0)
|
|
427
|
+
throw new Error(`No OpenReview submissions found for "${profile}".`);
|
|
428
|
+
return notes.slice(0, limit).map((note, index) => {
|
|
429
|
+
const row = mapOpenReviewNoteRow(note);
|
|
430
|
+
return {
|
|
431
|
+
rank: index + 1,
|
|
432
|
+
id: row.id,
|
|
433
|
+
title: row.title,
|
|
434
|
+
authors: row.authors,
|
|
435
|
+
venue: row.venue,
|
|
436
|
+
pdate: row.pdate,
|
|
437
|
+
url: row.url,
|
|
438
|
+
};
|
|
439
|
+
});
|
|
440
|
+
},
|
|
441
|
+
});
|
|
442
|
+
|
|
443
|
+
cli({
|
|
444
|
+
site: "openreview",
|
|
445
|
+
name: "venue",
|
|
446
|
+
description: "List papers at an OpenReview venue or invitation",
|
|
447
|
+
domain: "openreview.net",
|
|
448
|
+
strategy: Strategy.PUBLIC,
|
|
449
|
+
args: [
|
|
450
|
+
{
|
|
451
|
+
name: "venue",
|
|
452
|
+
type: "str",
|
|
453
|
+
required: true,
|
|
454
|
+
positional: true,
|
|
455
|
+
description: "Venue text or invitation id",
|
|
456
|
+
},
|
|
457
|
+
{ name: "limit", type: "int", default: 25, description: "Max results" },
|
|
458
|
+
{
|
|
459
|
+
name: "offset",
|
|
460
|
+
type: "int",
|
|
461
|
+
default: 0,
|
|
462
|
+
description: "Pagination offset",
|
|
463
|
+
},
|
|
464
|
+
],
|
|
465
|
+
columns: [
|
|
466
|
+
"rank",
|
|
467
|
+
"id",
|
|
468
|
+
"title",
|
|
469
|
+
"authors",
|
|
470
|
+
"keywords",
|
|
471
|
+
"primary_area",
|
|
472
|
+
"pdate",
|
|
473
|
+
"pdf",
|
|
474
|
+
"url",
|
|
475
|
+
],
|
|
476
|
+
func: async (_page, kwargs) => {
|
|
477
|
+
const venue = String(kwargs.venue ?? "").trim();
|
|
478
|
+
if (!venue) throw new Error("openreview venue cannot be empty.");
|
|
479
|
+
const limit = requireOpenReviewLimit(kwargs.limit, 25, 200);
|
|
480
|
+
const offset = requireOpenReviewOffset(kwargs.offset);
|
|
481
|
+
const params = new URLSearchParams({
|
|
482
|
+
[venue.includes("/-/") ? "invitation" : "content.venue"]: venue,
|
|
483
|
+
limit: String(limit),
|
|
484
|
+
offset: String(offset),
|
|
485
|
+
});
|
|
486
|
+
const notes = notesFromEnvelope(
|
|
487
|
+
await fetchOpenReview(
|
|
488
|
+
`/notes?${params.toString()}`,
|
|
489
|
+
`openreview venue ${venue}`,
|
|
490
|
+
),
|
|
491
|
+
);
|
|
492
|
+
if (notes.length === 0)
|
|
493
|
+
throw new Error(`No OpenReview papers found at venue "${venue}".`);
|
|
494
|
+
return notes.slice(0, limit).map((note, index) => {
|
|
495
|
+
const row = mapOpenReviewNoteRow(note);
|
|
496
|
+
return {
|
|
497
|
+
rank: offset + index + 1,
|
|
498
|
+
id: row.id,
|
|
499
|
+
title: row.title,
|
|
500
|
+
authors: row.authors,
|
|
501
|
+
keywords: row.keywords,
|
|
502
|
+
primary_area: row.primary_area,
|
|
503
|
+
pdate: row.pdate,
|
|
504
|
+
pdf: row.pdf,
|
|
505
|
+
url: row.url,
|
|
506
|
+
};
|
|
507
|
+
});
|
|
508
|
+
},
|
|
509
|
+
});
|
|
510
|
+
|
|
511
|
+
cli({
|
|
512
|
+
site: "openreview",
|
|
513
|
+
name: "reviews",
|
|
514
|
+
description:
|
|
515
|
+
"Show paper, reviews, decisions, and comments for an OpenReview forum",
|
|
516
|
+
domain: "openreview.net",
|
|
517
|
+
strategy: Strategy.PUBLIC,
|
|
518
|
+
args: [
|
|
519
|
+
{
|
|
520
|
+
name: "forum",
|
|
521
|
+
type: "str",
|
|
522
|
+
required: true,
|
|
523
|
+
positional: true,
|
|
524
|
+
description: "OpenReview forum id",
|
|
525
|
+
},
|
|
526
|
+
{
|
|
527
|
+
name: "max-length",
|
|
528
|
+
type: "int",
|
|
529
|
+
default: 4000,
|
|
530
|
+
description: "Per-row text truncation length",
|
|
531
|
+
},
|
|
532
|
+
],
|
|
533
|
+
columns: ["type", "author", "rating", "confidence", "text"],
|
|
534
|
+
func: async (_page, kwargs) => {
|
|
535
|
+
const forum = requireForumId(kwargs.forum, "forum");
|
|
536
|
+
const maxLength = coerceOpenReviewInt(
|
|
537
|
+
kwargs["max-length"] ?? kwargs.maxLength ?? 4000,
|
|
538
|
+
);
|
|
539
|
+
if (!Number.isInteger(maxLength) || maxLength < 200) {
|
|
540
|
+
throw new Error(
|
|
541
|
+
"openreview reviews max-length must be an integer >= 200.",
|
|
542
|
+
);
|
|
543
|
+
}
|
|
544
|
+
const rootNotes = notesFromEnvelope(
|
|
545
|
+
await fetchOpenReview(
|
|
546
|
+
`/notes?id=${encodeURIComponent(forum)}`,
|
|
547
|
+
`openreview paper ${forum}`,
|
|
548
|
+
),
|
|
549
|
+
);
|
|
550
|
+
const root = rootNotes[0];
|
|
551
|
+
if (!root) throw new Error(`No OpenReview forum found with id "${forum}".`);
|
|
552
|
+
const replies = notesFromEnvelope(
|
|
553
|
+
await fetchOpenReview(
|
|
554
|
+
`/notes?forum=${encodeURIComponent(forum)}&details=replies&limit=1000`,
|
|
555
|
+
`openreview reviews ${forum}`,
|
|
556
|
+
),
|
|
557
|
+
);
|
|
558
|
+
return mapReviewThreadRows(root, replies, forum, maxLength);
|
|
559
|
+
},
|
|
560
|
+
});
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import {
|
|
3
|
+
mapOsvQueryRows,
|
|
4
|
+
mapOsvVulnerabilityRow,
|
|
5
|
+
requireOsvEcosystem,
|
|
6
|
+
requireOsvLimit,
|
|
7
|
+
requireOsvString,
|
|
8
|
+
requireOsvVulnerabilityId,
|
|
9
|
+
} from "./security.js";
|
|
10
|
+
|
|
11
|
+
describe("osv agent-facing security commands", () => {
|
|
12
|
+
it("validates OSV query inputs", () => {
|
|
13
|
+
expect(requireOsvString(" lodash ", "package")).toBe("lodash");
|
|
14
|
+
expect(() => requireOsvString("", "package")).toThrow("cannot be empty");
|
|
15
|
+
expect(requireOsvEcosystem("PyPI")).toBe("PyPI");
|
|
16
|
+
expect(() => requireOsvEcosystem("pypi")).toThrow("not recognised");
|
|
17
|
+
expect(requireOsvLimit(undefined)).toBe(30);
|
|
18
|
+
expect(requireOsvLimit("200")).toBe(200);
|
|
19
|
+
expect(() => requireOsvLimit("0")).toThrow("osv limit must be");
|
|
20
|
+
expect(requireOsvVulnerabilityId("GHSA-29mw-wpgm-hmr9")).toBe(
|
|
21
|
+
"GHSA-29mw-wpgm-hmr9",
|
|
22
|
+
);
|
|
23
|
+
expect(() => requireOsvVulnerabilityId("has spaces")).toThrow("not valid");
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it("sorts OSV query rows by published date descending", () => {
|
|
27
|
+
expect(
|
|
28
|
+
mapOsvQueryRows(
|
|
29
|
+
[
|
|
30
|
+
{
|
|
31
|
+
id: "GHSA-old",
|
|
32
|
+
summary: "old",
|
|
33
|
+
published: "2020-01-01T00:00:00Z",
|
|
34
|
+
affected: [{ package: { ecosystem: "PyPI", name: "django" } }],
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
id: "GHSA-new",
|
|
38
|
+
summary: "new",
|
|
39
|
+
published: "2026-01-01T00:00:00Z",
|
|
40
|
+
affected: [{ package: { ecosystem: "PyPI", name: "django" } }],
|
|
41
|
+
},
|
|
42
|
+
],
|
|
43
|
+
10,
|
|
44
|
+
),
|
|
45
|
+
).toMatchObject([
|
|
46
|
+
{
|
|
47
|
+
rank: 1,
|
|
48
|
+
id: "GHSA-new",
|
|
49
|
+
published: "2026-01-01T00:00:00Z",
|
|
50
|
+
affectedPackages: "PyPI:django",
|
|
51
|
+
},
|
|
52
|
+
{
|
|
53
|
+
rank: 2,
|
|
54
|
+
id: "GHSA-old",
|
|
55
|
+
},
|
|
56
|
+
]);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it("maps OSV vulnerability details without sentinel rows", () => {
|
|
60
|
+
expect(
|
|
61
|
+
mapOsvVulnerabilityRow({
|
|
62
|
+
id: "GHSA-29mw-wpgm-hmr9",
|
|
63
|
+
summary: "ReDoS in lodash",
|
|
64
|
+
aliases: ["CVE-2020-28500"],
|
|
65
|
+
published: "2022-01-06T20:30:46Z",
|
|
66
|
+
modified: "2025-09-29T21:12:31.102523Z",
|
|
67
|
+
database_specific: {
|
|
68
|
+
severity: "MODERATE",
|
|
69
|
+
cwe_ids: ["CWE-1333", "CWE-400"],
|
|
70
|
+
},
|
|
71
|
+
affected: [
|
|
72
|
+
{ package: { name: "lodash", ecosystem: "npm" } },
|
|
73
|
+
{ package: { name: "lodash-rails", ecosystem: "RubyGems" } },
|
|
74
|
+
],
|
|
75
|
+
references: [{ url: "https://example.test" }],
|
|
76
|
+
}),
|
|
77
|
+
).toMatchObject({
|
|
78
|
+
id: "GHSA-29mw-wpgm-hmr9",
|
|
79
|
+
severity: "MODERATE",
|
|
80
|
+
aliases: "CVE-2020-28500",
|
|
81
|
+
modified: "2025-09-29T21:12:31Z",
|
|
82
|
+
affectedPackages: "npm:lodash, RubyGems:lodash-rails",
|
|
83
|
+
cwes: "CWE-1333, CWE-400",
|
|
84
|
+
referenceCount: 1,
|
|
85
|
+
url: "https://osv.dev/vulnerability/GHSA-29mw-wpgm-hmr9",
|
|
86
|
+
});
|
|
87
|
+
expect(() => mapOsvVulnerabilityRow({})).toThrow(
|
|
88
|
+
"OSV.dev returned no vulnerability record",
|
|
89
|
+
);
|
|
90
|
+
});
|
|
91
|
+
});
|