@jackwener/opencli 1.7.12 → 1.7.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +8 -7
- package/README.zh-CN.md +9 -8
- package/cli-manifest.json +12128 -6665
- package/clis/1point3acres/digest.js +35 -0
- package/clis/1point3acres/forum.js +51 -0
- package/clis/1point3acres/forums.js +44 -0
- package/clis/1point3acres/hot.js +35 -0
- package/clis/1point3acres/latest.js +35 -0
- package/clis/1point3acres/notifications.js +64 -0
- package/clis/1point3acres/search.js +71 -0
- package/clis/1point3acres/thread.js +117 -0
- package/clis/1point3acres/user.js +77 -0
- package/clis/1point3acres/utils.js +247 -0
- package/clis/_shared/desktop-commands.js +4 -0
- package/clis/aibase/news.js +110 -0
- package/clis/aibase/news.test.js +59 -0
- package/clis/amazon/discussion.test.js +1 -28
- package/clis/antigravity/watch.js +3 -2
- package/clis/arxiv/author.js +44 -0
- package/clis/baidu-scholar/search.js +0 -1
- package/clis/bbc/topic.js +57 -0
- package/clis/bbc/utils.js +79 -0
- package/clis/chaoxing/assignments.js +1 -1
- package/clis/chaoxing/exams.js +1 -1
- package/clis/chatgpt/ask.js +57 -0
- package/clis/chatgpt/commands.test.js +45 -0
- package/clis/chatgpt/detail.js +46 -0
- package/clis/chatgpt/history.js +39 -0
- package/clis/chatgpt/image.js +12 -11
- package/clis/chatgpt/image.test.js +23 -0
- package/clis/chatgpt/new.js +25 -0
- package/clis/chatgpt/read.js +43 -0
- package/clis/chatgpt/send.js +46 -0
- package/clis/chatgpt/status.js +29 -0
- package/clis/chatgpt/utils.js +294 -4
- package/clis/chatgpt/utils.test.js +13 -0
- package/clis/chatgpt-app/ask.js +6 -3
- package/clis/chatwise/ask.js +16 -43
- package/clis/chatwise/composer.test.js +186 -0
- package/clis/chatwise/send.js +2 -24
- package/clis/chatwise/utils.js +143 -0
- package/clis/claude/ask.js +1 -1
- package/clis/claude/detail.js +1 -0
- package/clis/claude/history.js +1 -0
- package/clis/claude/new.js +1 -0
- package/clis/claude/read.js +1 -0
- package/clis/claude/send.js +1 -0
- package/clis/claude/status.js +1 -0
- package/clis/codex/ask.js +15 -9
- package/clis/codex/history.js +16 -33
- package/clis/codex/projects.js +28 -0
- package/clis/codex/read.js +10 -4
- package/clis/codex/send.js +10 -3
- package/clis/codex/sidebar.js +356 -0
- package/clis/codex/sidebar.test.js +329 -0
- package/clis/coingecko/categories.js +75 -0
- package/clis/coingecko/coin.js +107 -0
- package/clis/coingecko/coingecko.test.js +109 -0
- package/clis/coingecko/derivatives.js +84 -0
- package/clis/coingecko/exchanges.js +74 -0
- package/clis/coingecko/global.js +71 -0
- package/clis/coingecko/top.js +64 -0
- package/clis/coingecko/trending.js +55 -0
- package/clis/coupang/add-to-cart.js +21 -13
- package/clis/coupang/coupang.test.js +159 -0
- package/clis/coupang/product.js +257 -0
- package/clis/coupang/search.js +38 -16
- package/clis/coupang/utils.js +55 -1
- package/clis/crates/crate.js +62 -0
- package/clis/crates/search.js +44 -0
- package/clis/crates/utils.js +72 -0
- package/clis/ctrip/ctrip.test.js +234 -0
- package/clis/ctrip/hotel-suggest.js +45 -0
- package/clis/ctrip/search.js +22 -68
- package/clis/ctrip/utils.js +175 -0
- package/clis/cursor/ask.js +6 -3
- package/clis/dblp/author.js +133 -0
- package/clis/dblp/venue.js +64 -0
- package/clis/deepseek/ask.js +12 -7
- package/clis/deepseek/ask.test.js +13 -13
- package/clis/deepseek/detail.js +38 -0
- package/clis/deepseek/detail.test.js +81 -0
- package/clis/deepseek/history.js +1 -0
- package/clis/deepseek/new.js +1 -0
- package/clis/deepseek/read.js +1 -0
- package/clis/deepseek/send.js +140 -0
- package/clis/deepseek/send.test.js +107 -0
- package/clis/deepseek/status.js +1 -0
- package/clis/deepseek/utils.js +66 -0
- package/clis/deepseek/utils.test.js +107 -1
- package/clis/defillama/defillama.test.js +99 -0
- package/clis/defillama/protocol.js +84 -0
- package/clis/defillama/protocols.js +55 -0
- package/clis/defillama/utils.js +99 -0
- package/clis/devto/latest.js +74 -0
- package/clis/dockerhub/image.js +52 -0
- package/clis/dockerhub/search.js +47 -0
- package/clis/dockerhub/utils.js +100 -0
- package/clis/doubao/ask.js +7 -3
- package/clis/doubao/detail.js +1 -0
- package/clis/doubao/history.js +1 -0
- package/clis/doubao/meeting-summary.js +1 -0
- package/clis/doubao/meeting-transcript.js +1 -0
- package/clis/doubao/new.js +1 -0
- package/clis/doubao/read.js +1 -0
- package/clis/doubao/send.js +1 -0
- package/clis/doubao/status.js +1 -0
- package/clis/douyin/draft.test.js +1 -30
- package/clis/endoflife/endoflife.test.js +51 -0
- package/clis/endoflife/product.js +55 -0
- package/clis/endoflife/utils.js +89 -0
- package/clis/facebook/__fixtures__/notifications-page.html +13 -0
- package/clis/facebook/notifications.js +326 -30
- package/clis/facebook/notifications.test.js +458 -0
- package/clis/flathub/app.js +71 -0
- package/clis/flathub/flathub.test.js +90 -0
- package/clis/flathub/search.js +80 -0
- package/clis/flathub/utils.js +114 -0
- package/clis/gemini/ask.js +7 -3
- package/clis/gemini/ask.test.js +2 -2
- package/clis/gemini/deep-research-result.js +6 -2
- package/clis/gemini/deep-research-result.test.js +15 -14
- package/clis/gemini/deep-research.js +8 -4
- package/clis/gemini/deep-research.test.js +15 -18
- package/clis/gemini/image.js +7 -2
- package/clis/gemini/new.js +1 -0
- package/clis/gemini/utils.js +0 -4
- package/clis/google-scholar/cite.js +0 -1
- package/clis/google-scholar/profile.js +0 -1
- package/clis/google-scholar/search.js +0 -1
- package/clis/goproxy/goproxy.test.js +103 -0
- package/clis/goproxy/module.js +47 -0
- package/clis/goproxy/utils.js +165 -0
- package/clis/goproxy/versions.js +59 -0
- package/clis/gov-law/recent.js +0 -1
- package/clis/gov-law/search.js +0 -1
- package/clis/gov-policy/__fixtures__/recent.html +16 -0
- package/clis/gov-policy/__fixtures__/search.html +41 -0
- package/clis/gov-policy/gov-policy.test.js +224 -0
- package/clis/gov-policy/recent.js +66 -24
- package/clis/gov-policy/search.js +65 -23
- package/clis/gov-policy/utils.js +54 -0
- package/clis/grok/ask.js +49 -265
- package/clis/grok/ask.test.js +21 -46
- package/clis/grok/detail.js +60 -0
- package/clis/grok/history.js +48 -0
- package/clis/grok/{image.ts → image.js} +56 -70
- package/clis/grok/image.test.ts +20 -0
- package/clis/grok/new.js +20 -0
- package/clis/grok/read.js +39 -0
- package/clis/grok/send.js +50 -0
- package/clis/grok/status.js +41 -0
- package/clis/grok/utils.js +326 -0
- package/clis/grok/utils.test.js +103 -0
- package/clis/hf/datasets.js +88 -0
- package/clis/hf/hf.test.js +16 -0
- package/clis/hf/models.js +91 -0
- package/clis/hf/paper.js +79 -0
- package/clis/hf/spaces.js +101 -0
- package/clis/hf/top.js +1 -0
- package/clis/homebrew/cask.js +39 -0
- package/clis/homebrew/formula.js +41 -0
- package/clis/homebrew/popular.js +54 -0
- package/clis/homebrew/utils.js +100 -0
- package/clis/hupu/__fixtures__/hot-home.html +64 -0
- package/clis/hupu/detail.js +0 -1
- package/clis/hupu/hot.js +156 -35
- package/clis/hupu/hot.test.js +224 -0
- package/clis/hupu/search.js +0 -1
- package/clis/instagram/note.js +1 -1
- package/clis/instagram/note.test.js +1 -29
- package/clis/instagram/post.js +1 -1
- package/clis/instagram/post.test.js +1 -1
- package/clis/instagram/reel.js +1 -1
- package/clis/instagram/story.js +1 -1
- package/clis/instagram/story.test.js +1 -34
- package/clis/jd/commands.test.js +1 -24
- package/clis/lichess/lichess.test.js +85 -0
- package/clis/lichess/top.js +46 -0
- package/clis/lichess/user.js +91 -0
- package/clis/lichess/utils.js +97 -0
- package/clis/linkedin/search.js +107 -10
- package/clis/linkedin/search.test.js +222 -0
- package/clis/linux-do/feed.js +2 -5
- package/clis/linux-do/feed.test.js +35 -0
- package/clis/lobsters/domain.js +92 -0
- package/clis/maven/artifact.js +49 -0
- package/clis/maven/search.js +51 -0
- package/clis/maven/utils.js +110 -0
- package/clis/mdn/search.js +97 -0
- package/clis/medium/tag.js +135 -0
- package/clis/npm/downloads.js +59 -0
- package/clis/npm/package.js +70 -0
- package/clis/npm/search.js +49 -0
- package/clis/npm/utils.js +76 -0
- package/clis/nuget/nuget.test.js +111 -0
- package/clis/nuget/package.js +101 -0
- package/clis/nuget/search.js +69 -0
- package/clis/nuget/utils.js +87 -0
- package/clis/nvd/cve.js +121 -0
- package/clis/oeis/oeis.test.js +88 -0
- package/clis/oeis/search.js +63 -0
- package/clis/oeis/sequence.js +71 -0
- package/clis/oeis/utils.js +88 -0
- package/clis/openalex/search.js +69 -0
- package/clis/openalex/utils.js +160 -0
- package/clis/openalex/work.js +65 -0
- package/clis/openfda/drug-label.js +74 -0
- package/clis/openfda/food-recall.js +65 -0
- package/clis/openfda/openfda.test.js +114 -0
- package/clis/openfda/utils.js +67 -0
- package/clis/osv/osv.test.js +97 -0
- package/clis/osv/query.js +72 -0
- package/clis/osv/utils.js +169 -0
- package/clis/osv/vulnerability.js +54 -0
- package/clis/packagist/package.js +49 -0
- package/clis/packagist/search.js +43 -0
- package/clis/packagist/utils.js +113 -0
- package/clis/paperreview/feedback.js +1 -1
- package/clis/paperreview/review.js +1 -1
- package/clis/paperreview/submit.js +1 -1
- package/clis/pixiv/download.test.js +1 -1
- package/clis/pixiv/illusts.test.js +1 -1
- package/clis/pixiv/search.test.js +1 -1
- package/clis/pubmed/article.js +50 -0
- package/clis/pubmed/author.js +64 -0
- package/clis/pubmed/citations.js +36 -0
- package/clis/pubmed/pubmed.test.js +276 -0
- package/clis/pubmed/related.js +45 -0
- package/clis/pubmed/search.js +75 -0
- package/clis/pubmed/utils.js +309 -0
- package/clis/pypi/downloads.js +66 -0
- package/clis/pypi/package.js +79 -0
- package/clis/pypi/utils.js +55 -0
- package/clis/quark/mv.js +1 -1
- package/clis/quark/save.js +1 -1
- package/clis/qwen/ask.js +85 -0
- package/clis/qwen/detail.js +62 -0
- package/clis/qwen/history.js +61 -0
- package/clis/qwen/image.js +179 -0
- package/clis/qwen/new.js +23 -0
- package/clis/qwen/read.js +41 -0
- package/clis/qwen/send.js +55 -0
- package/clis/qwen/status.js +37 -0
- package/clis/qwen/utils.js +409 -0
- package/clis/qwen/utils.test.js +45 -0
- package/clis/rest-countries/country.js +65 -0
- package/clis/rest-countries/region.js +64 -0
- package/clis/rest-countries/rest-countries.test.js +83 -0
- package/clis/rest-countries/utils.js +126 -0
- package/clis/reuters/article-detail.js +53 -0
- package/clis/reuters/reuters.test.js +299 -0
- package/clis/reuters/search.js +45 -34
- package/clis/reuters/utils.js +159 -0
- package/clis/rfc/rfc.js +52 -0
- package/clis/rfc/rfc.test.js +74 -0
- package/clis/rfc/utils.js +72 -0
- package/clis/rubygems/gem.js +42 -0
- package/clis/rubygems/search.js +47 -0
- package/clis/rubygems/utils.js +86 -0
- package/clis/stackoverflow/related.js +66 -0
- package/clis/stackoverflow/stackoverflow.test.js +58 -0
- package/clis/stackoverflow/tag.js +60 -0
- package/clis/stackoverflow/user.js +50 -0
- package/clis/stackoverflow/utils.js +118 -0
- package/clis/steam/app.js +67 -0
- package/clis/steam/search.js +58 -0
- package/clis/steam/steam.test.js +46 -0
- package/clis/steam/utils.js +107 -0
- package/clis/taobao/commands.test.js +1 -24
- package/clis/test-utils.js +61 -0
- package/clis/tieba/hot.js +0 -1
- package/clis/tiktok/comment.js +128 -41
- package/clis/tiktok/creator-videos.js +270 -0
- package/clis/tiktok/creator-videos.test.js +113 -0
- package/clis/tiktok/explore.js +137 -29
- package/clis/tiktok/follow.js +115 -33
- package/clis/tiktok/following.js +157 -36
- package/clis/tiktok/friends.js +139 -37
- package/clis/tiktok/live.js +137 -41
- package/clis/tiktok/notifications.js +141 -38
- package/clis/tiktok/refactor.test.js +389 -0
- package/clis/tiktok/unfollow.js +124 -38
- package/clis/tiktok/user.js +203 -29
- package/clis/tiktok/utils.js +505 -0
- package/clis/tiktok/write-refactor.test.js +370 -0
- package/clis/toutiao/articles.js +36 -62
- package/clis/toutiao/hot.js +63 -0
- package/clis/toutiao/toutiao.test.js +378 -0
- package/clis/toutiao/utils.js +161 -0
- package/clis/tvmaze/search.js +61 -0
- package/clis/tvmaze/show.js +60 -0
- package/clis/tvmaze/tvmaze.test.js +93 -0
- package/clis/tvmaze/utils.js +110 -0
- package/clis/twitter/accept.js +1 -1
- package/clis/twitter/followers.js +134 -69
- package/clis/twitter/quote.js +139 -0
- package/clis/twitter/quote.test.js +106 -0
- package/clis/twitter/reply-dm.js +1 -1
- package/clis/twitter/reply.test.js +1 -29
- package/clis/twitter/retweet.js +99 -0
- package/clis/twitter/retweet.test.js +69 -0
- package/clis/twitter/shared.js +38 -0
- package/clis/twitter/shared.test.js +28 -1
- package/clis/twitter/unlike.js +87 -0
- package/clis/twitter/unlike.test.js +72 -0
- package/clis/twitter/unretweet.js +99 -0
- package/clis/twitter/unretweet.test.js +69 -0
- package/clis/uisdc/news.js +105 -0
- package/clis/uisdc/news.test.js +66 -0
- package/clis/wanfang/search.js +0 -1
- package/clis/web/read.js +47 -17
- package/clis/web/read.test.js +101 -1
- package/clis/weixin/create-draft.js +1 -1
- package/clis/weixin/drafts.js +1 -1
- package/clis/weixin/drafts.test.js +5 -1
- package/clis/weixin/search.js +157 -0
- package/clis/weixin/search.test.js +227 -0
- package/clis/wikidata/entity.js +60 -0
- package/clis/wikidata/search.js +50 -0
- package/clis/wikidata/utils.js +117 -0
- package/clis/wikidata/wikidata.test.js +83 -0
- package/clis/wikipedia/page.js +95 -0
- package/clis/wttr/current.js +63 -0
- package/clis/wttr/forecast.js +71 -0
- package/clis/wttr/utils.js +50 -0
- package/clis/wttr/wttr.test.js +84 -0
- package/clis/xianyu/chat.js +16 -4
- package/clis/xianyu/chat.test.js +64 -0
- package/clis/xianyu/publish.js +485 -0
- package/clis/xianyu/publish.test.js +220 -0
- package/clis/xiaoe/catalog.js +105 -40
- package/clis/xiaoe/content.js +164 -29
- package/clis/xiaoe/courses.js +86 -29
- package/clis/xiaoe/xiaoe.test.js +486 -0
- package/clis/xiaohongshu/creator-notes-summary.js +1 -1
- package/clis/xiaohongshu/publish.js +16 -3
- package/clis/xiaohongshu/publish.test.js +46 -1
- package/clis/youtube/transcript.js +13 -19
- package/clis/youtube/transcript.test.js +17 -0
- package/clis/yuanbao/ask.js +17 -66
- package/clis/yuanbao/ask.test.js +5 -5
- package/clis/yuanbao/detail.js +65 -0
- package/clis/yuanbao/history.js +51 -0
- package/clis/yuanbao/new.js +1 -0
- package/clis/yuanbao/read.js +38 -0
- package/clis/yuanbao/send.js +57 -0
- package/clis/yuanbao/shared.js +297 -5
- package/clis/yuanbao/shared.test.js +80 -0
- package/clis/yuanbao/status.js +44 -0
- package/clis/zlibrary/commands.test.js +1 -11
- package/dist/src/browser/base-page.d.ts +9 -0
- package/dist/src/browser/base-page.js +44 -1
- package/dist/src/browser/base-page.test.js +66 -0
- package/dist/src/browser/bridge.js +47 -45
- package/dist/src/browser/cdp.d.ts +1 -0
- package/dist/src/browser/cdp.js +51 -9
- package/dist/src/browser/daemon-client.d.ts +4 -0
- package/dist/src/browser/errors.js +1 -1
- package/dist/src/browser/page.d.ts +1 -1
- package/dist/src/browser/page.js +3 -1
- package/dist/src/browser/page.test.js +29 -0
- package/dist/src/browser/target-errors.d.ts +2 -1
- package/dist/src/browser/target-errors.js +1 -0
- package/dist/src/browser/target-resolver.d.ts +25 -0
- package/dist/src/browser/target-resolver.js +43 -0
- package/dist/src/browser.test.js +18 -0
- package/dist/src/build-manifest.js +9 -4
- package/dist/src/build-manifest.test.js +2 -8
- package/dist/src/capabilityRouting.d.ts +16 -1
- package/dist/src/capabilityRouting.js +24 -1
- package/dist/src/capabilityRouting.test.js +19 -1
- package/dist/src/cli.js +76 -11
- package/dist/src/cli.test.js +241 -1
- package/dist/src/commanderAdapter.js +23 -9
- package/dist/src/commanderAdapter.test.js +0 -1
- package/dist/src/discovery.js +2 -5
- package/dist/src/errors.js +1 -1
- package/dist/src/execution.d.ts +1 -1
- package/dist/src/execution.js +111 -27
- package/dist/src/execution.test.js +326 -17
- package/dist/src/help.d.ts +27 -2
- package/dist/src/help.js +196 -23
- package/dist/src/help.test.d.ts +1 -0
- package/dist/src/help.test.js +54 -0
- package/dist/src/main.js +14 -1
- package/dist/src/manifest-types.d.ts +5 -3
- package/dist/src/pipeline/executor.js +1 -1
- package/dist/src/pipeline/executor.test.js +8 -0
- package/dist/src/pipeline/registry.d.ts +9 -0
- package/dist/src/pipeline/registry.js +13 -1
- package/dist/src/pipeline/steps/browser.d.ts +1 -0
- package/dist/src/pipeline/steps/browser.js +10 -0
- package/dist/src/pipeline/steps/download.test.js +1 -0
- package/dist/src/registry-api.d.ts +1 -1
- package/dist/src/registry.d.ts +12 -11
- package/dist/src/registry.js +16 -6
- package/dist/src/registry.test.js +2 -2
- package/dist/src/runtime.d.ts +2 -1
- package/dist/src/runtime.js +1 -1
- package/dist/src/serialization.d.ts +2 -2
- package/dist/src/serialization.js +4 -6
- package/dist/src/serialization.test.js +17 -0
- package/dist/src/types.d.ts +17 -0
- package/dist/src/validate.js +15 -11
- package/dist/src/validate.test.d.ts +9 -0
- package/dist/src/validate.test.js +90 -0
- package/package.json +1 -1
- package/scripts/fetch-adapters.js +1 -1
- package/scripts/typed-error-lint-baseline.json +5 -77
- package/clis/ctrip/search.test.js +0 -64
- package/clis/gov-policy/commands.test.js +0 -27
- package/clis/linux-do/category.js +0 -37
- package/clis/linux-do/hot.js +0 -26
- package/clis/linux-do/latest.js +0 -19
- package/clis/pixiv/test-utils.js +0 -23
- package/clis/toutiao/articles.test.js +0 -30
- package/dist/src/analysis.d.ts +0 -40
- package/dist/src/analysis.js +0 -172
package/clis/xiaoe/courses.js
CHANGED
|
@@ -1,16 +1,51 @@
|
|
|
1
|
+
// Xiaoe (小鹅通) purchased-courses list — pulls "已购内容" tab cards from
|
|
2
|
+
// `study.xiaoe-tech.com`.
|
|
3
|
+
//
|
|
4
|
+
// Replaces the legacy `pipeline:[]` form. The in-page extraction logic
|
|
5
|
+
// (Vue `__vue__.$parent` walk to match a card title back to the original
|
|
6
|
+
// purchase entry) is kept byte-for-byte — Xiaoe's purchase list is not
|
|
7
|
+
// exposed as a public JSON endpoint, and Vue's private runtime tree is
|
|
8
|
+
// the only stable hook. JSDOM cannot reproduce the Vue runtime, so
|
|
9
|
+
// rewriting the IIFE without live verify would be silent-failure risk.
|
|
10
|
+
//
|
|
11
|
+
// What changes:
|
|
12
|
+
// - `func` form + `Strategy.COOKIE` + `browser:true`.
|
|
13
|
+
// - Typed errors: `EmptyResultError` when zero card rows are found
|
|
14
|
+
// (almost always means the cookie expired); `CommandExecutionError`
|
|
15
|
+
// when `page.evaluate` rejects.
|
|
16
|
+
// - One pure helper (`buildCourseUrl`) is extracted as a module-level
|
|
17
|
+
// export; the in-page IIFE embeds it via `${fn.toString()}` so the
|
|
18
|
+
// live and test paths share one source of truth. The helper covers
|
|
19
|
+
// the three URL fallbacks the legacy code had inline:
|
|
20
|
+
// 1. `entry.h5_url` if present
|
|
21
|
+
// 2. `entry.url` if present
|
|
22
|
+
// 3. otherwise build from `app_id` + `resource_id` + `resource_type`
|
|
23
|
+
// (column course `resource_type === 6` gets the `/v1/course/column/`
|
|
24
|
+
// path, everything else gets `/p/course/ecourse/`)
|
|
25
|
+
|
|
1
26
|
import { cli, Strategy } from '@jackwener/opencli/registry';
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
27
|
+
import { CommandExecutionError, EmptyResultError } from '@jackwener/opencli/errors';
|
|
28
|
+
|
|
29
|
+
// Pure: derive the canonical course URL for a single purchase entry.
|
|
30
|
+
// Returns '' when `entry` is missing the fields we'd need to construct
|
|
31
|
+
// any of the three forms — never makes up a partial URL.
|
|
32
|
+
export function buildCourseUrl(entry) {
|
|
33
|
+
if (!entry) return '';
|
|
34
|
+
if (entry.h5_url) return entry.h5_url;
|
|
35
|
+
if (entry.url) return entry.url;
|
|
36
|
+
if (entry.app_id && entry.resource_id) {
|
|
37
|
+
const base = 'https://' + entry.app_id + '.h5.xet.citv.cn';
|
|
38
|
+
if (entry.resource_type === 6) {
|
|
39
|
+
return base + '/v1/course/column/' + entry.resource_id + '?type=3';
|
|
40
|
+
}
|
|
41
|
+
return base + '/p/course/ecourse/' + entry.resource_id;
|
|
42
|
+
}
|
|
43
|
+
return '';
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export function buildCoursesScript() {
|
|
47
|
+
return `(async () => {
|
|
48
|
+
${buildCourseUrl.toString()}
|
|
14
49
|
// 切换到「内容」tab
|
|
15
50
|
var tabs = document.querySelectorAll('span, div');
|
|
16
51
|
for (var i = 0; i < tabs.length; i++) {
|
|
@@ -37,18 +72,6 @@ cli({
|
|
|
37
72
|
return vm.$parent ? matchEntry(title, vm.$parent, depth + 1) : null;
|
|
38
73
|
}
|
|
39
74
|
|
|
40
|
-
// 构造课程 URL
|
|
41
|
-
function buildUrl(entry) {
|
|
42
|
-
if (entry.h5_url) return entry.h5_url;
|
|
43
|
-
if (entry.url) return entry.url;
|
|
44
|
-
if (entry.app_id && entry.resource_id) {
|
|
45
|
-
var base = 'https://' + entry.app_id + '.h5.xet.citv.cn';
|
|
46
|
-
if (entry.resource_type === 6) return base + '/v1/course/column/' + entry.resource_id + '?type=3';
|
|
47
|
-
return base + '/p/course/ecourse/' + entry.resource_id;
|
|
48
|
-
}
|
|
49
|
-
return '';
|
|
50
|
-
}
|
|
51
|
-
|
|
52
75
|
var cards = document.querySelectorAll('.course-card-list');
|
|
53
76
|
var results = [];
|
|
54
77
|
for (var c = 0; c < cards.length; c++) {
|
|
@@ -59,12 +82,46 @@ cli({
|
|
|
59
82
|
results.push({
|
|
60
83
|
title: title,
|
|
61
84
|
shop: entry ? (entry.shop_name || entry.app_name || '') : '',
|
|
62
|
-
url: entry ?
|
|
85
|
+
url: entry ? buildCourseUrl(entry) : '',
|
|
63
86
|
});
|
|
64
87
|
}
|
|
65
88
|
return results;
|
|
66
|
-
})()
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
89
|
+
})()`;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
async function getXiaoeCourses(page) {
|
|
93
|
+
let rows;
|
|
94
|
+
try {
|
|
95
|
+
await page.goto('https://study.xiaoe-tech.com/', { waitUntil: 'load', settleMs: 8000 });
|
|
96
|
+
rows = await page.evaluate(buildCoursesScript());
|
|
97
|
+
} catch (error) {
|
|
98
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
99
|
+
throw new CommandExecutionError(
|
|
100
|
+
`Failed to list xiaoe courses: ${message}`,
|
|
101
|
+
'page may not have rendered or auth may be required',
|
|
102
|
+
);
|
|
103
|
+
}
|
|
104
|
+
if (!Array.isArray(rows) || rows.length === 0) {
|
|
105
|
+
throw new EmptyResultError(
|
|
106
|
+
'xiaoe/courses',
|
|
107
|
+
'No purchased courses found — login session may have expired or the "内容" tab has no items',
|
|
108
|
+
);
|
|
109
|
+
}
|
|
110
|
+
return rows;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
export const coursesCommand = cli({
|
|
114
|
+
site: 'xiaoe',
|
|
115
|
+
name: 'courses',
|
|
116
|
+
access: 'read',
|
|
117
|
+
description: '列出已购小鹅通课程(含 URL 和店铺名)',
|
|
118
|
+
domain: 'study.xiaoe-tech.com',
|
|
119
|
+
strategy: Strategy.COOKIE,
|
|
120
|
+
browser: true,
|
|
121
|
+
columns: ['title', 'shop', 'url'],
|
|
122
|
+
func: getXiaoeCourses,
|
|
70
123
|
});
|
|
124
|
+
|
|
125
|
+
export const __test__ = {
|
|
126
|
+
buildCoursesScript,
|
|
127
|
+
};
|
|
@@ -0,0 +1,486 @@
|
|
|
1
|
+
// Xiaoe adapter contract + helper tests.
|
|
2
|
+
//
|
|
3
|
+
// The IIFEs walk Vue's private runtime tree (`__vue__`, `$store`,
|
|
4
|
+
// `$children`) which JSDOM cannot reproduce — testing the full IIFE
|
|
5
|
+
// against a frozen fixture is not viable here. Instead this file
|
|
6
|
+
// exercises:
|
|
7
|
+
// 1. The pure helpers each IIFE embeds via `${fn.toString()}` — these
|
|
8
|
+
// are also called from JSDOM where the surface (URL building,
|
|
9
|
+
// simple DOM walks) does work without a Vue runtime.
|
|
10
|
+
// 2. Each adapter's `func`-form wiring: upfront validation, typed
|
|
11
|
+
// errors on empty results, rows passed through verbatim on the
|
|
12
|
+
// happy path, no wasted `page.goto` on bad input.
|
|
13
|
+
// 3. The `build*Script` outputs: the embedded helpers must be
|
|
14
|
+
// present, anti-patterns from the legacy adapters (silent column
|
|
15
|
+
// drop, silent slice truncation) must NOT regress.
|
|
16
|
+
|
|
17
|
+
import { JSDOM } from 'jsdom';
|
|
18
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
19
|
+
import {
|
|
20
|
+
ArgumentError,
|
|
21
|
+
CommandExecutionError,
|
|
22
|
+
EmptyResultError,
|
|
23
|
+
} from '@jackwener/opencli/errors';
|
|
24
|
+
import { getRegistry } from '@jackwener/opencli/registry';
|
|
25
|
+
import {
|
|
26
|
+
CONTENT_SELECTORS,
|
|
27
|
+
contentCommand,
|
|
28
|
+
countXiaoeImages,
|
|
29
|
+
pickContentText,
|
|
30
|
+
requireXiaoePageUrl,
|
|
31
|
+
__test__ as contentTest,
|
|
32
|
+
} from './content.js';
|
|
33
|
+
import {
|
|
34
|
+
buildItemUrl,
|
|
35
|
+
catalogCommand,
|
|
36
|
+
chapterUrlPath,
|
|
37
|
+
typeLabel,
|
|
38
|
+
__test__ as catalogTest,
|
|
39
|
+
} from './catalog.js';
|
|
40
|
+
import {
|
|
41
|
+
buildCourseUrl,
|
|
42
|
+
coursesCommand,
|
|
43
|
+
__test__ as coursesTest,
|
|
44
|
+
} from './courses.js';
|
|
45
|
+
|
|
46
|
+
// ─── Registration contract ──────────────────────────────────────────
|
|
47
|
+
|
|
48
|
+
describe('xiaoe — adapter registration', () => {
|
|
49
|
+
it('content registers as cookie + browser, with the new full column shape', () => {
|
|
50
|
+
const reg = getRegistry();
|
|
51
|
+
expect(reg.get('xiaoe/content')).toBe(contentCommand);
|
|
52
|
+
expect(contentCommand.browser).toBe(true);
|
|
53
|
+
expect(contentCommand.strategy).toBe('cookie');
|
|
54
|
+
expect(contentCommand.access).toBe('read');
|
|
55
|
+
expect(contentCommand.domain).toBe('h5.xet.citv.cn');
|
|
56
|
+
// `content` is the new column — restoring the silently-dropped
|
|
57
|
+
// text the IIFE always extracted but the legacy `columns`
|
|
58
|
+
// declaration discarded.
|
|
59
|
+
expect(contentCommand.columns).toEqual([
|
|
60
|
+
'title', 'content', 'content_length', 'image_count',
|
|
61
|
+
]);
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
it('catalog registers as cookie + browser, columns unchanged', () => {
|
|
65
|
+
const reg = getRegistry();
|
|
66
|
+
expect(reg.get('xiaoe/catalog')).toBe(catalogCommand);
|
|
67
|
+
expect(catalogCommand.browser).toBe(true);
|
|
68
|
+
expect(catalogCommand.strategy).toBe('cookie');
|
|
69
|
+
expect(catalogCommand.access).toBe('read');
|
|
70
|
+
expect(catalogCommand.columns).toEqual([
|
|
71
|
+
'ch', 'chapter', 'no', 'title', 'type', 'resource_id', 'url', 'status',
|
|
72
|
+
]);
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('courses registers as cookie + browser, columns unchanged', () => {
|
|
76
|
+
const reg = getRegistry();
|
|
77
|
+
expect(reg.get('xiaoe/courses')).toBe(coursesCommand);
|
|
78
|
+
expect(coursesCommand.browser).toBe(true);
|
|
79
|
+
expect(coursesCommand.strategy).toBe('cookie');
|
|
80
|
+
expect(coursesCommand.access).toBe('read');
|
|
81
|
+
expect(coursesCommand.domain).toBe('study.xiaoe-tech.com');
|
|
82
|
+
expect(coursesCommand.columns).toEqual(['title', 'shop', 'url']);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
// ─── content.js helpers ────────────────────────────────────────────
|
|
87
|
+
|
|
88
|
+
describe('xiaoe/content — pickContentText', () => {
|
|
89
|
+
function htmlDoc(body) {
|
|
90
|
+
return new JSDOM(`<html><body>${body}</body></html>`).window.document;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
it('returns the first selector whose text exceeds the min-length threshold', () => {
|
|
94
|
+
const doc = htmlDoc(`
|
|
95
|
+
<div class="rich-text-wrap">${'a'.repeat(60)}</div>
|
|
96
|
+
<div class="content-wrap">${'b'.repeat(120)}</div>
|
|
97
|
+
`);
|
|
98
|
+
const text = pickContentText(doc, CONTENT_SELECTORS, 50);
|
|
99
|
+
expect(text.startsWith('a')).toBe(true);
|
|
100
|
+
expect(text).toHaveLength(60);
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
it('skips selectors whose text is at or below the threshold (legacy used > 50)', () => {
|
|
104
|
+
const doc = htmlDoc(`
|
|
105
|
+
<div class="rich-text-wrap">${'a'.repeat(50)}</div>
|
|
106
|
+
<div class="content-wrap">${'b'.repeat(120)}</div>
|
|
107
|
+
`);
|
|
108
|
+
const text = pickContentText(doc, CONTENT_SELECTORS, 50);
|
|
109
|
+
expect(text.startsWith('b')).toBe(true);
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
it('falls back through main → #app → body when no selector qualifies', () => {
|
|
113
|
+
const doc1 = htmlDoc(`<main>${'m'.repeat(80)}</main>`);
|
|
114
|
+
expect(pickContentText(doc1, CONTENT_SELECTORS).startsWith('m')).toBe(true);
|
|
115
|
+
|
|
116
|
+
const doc2 = htmlDoc(`<div id="app">${'p'.repeat(80)}</div>`);
|
|
117
|
+
expect(pickContentText(doc2, CONTENT_SELECTORS).startsWith('p')).toBe(true);
|
|
118
|
+
|
|
119
|
+
const doc3 = htmlDoc(`${'q'.repeat(80)}`);
|
|
120
|
+
expect(pickContentText(doc3, CONTENT_SELECTORS).startsWith('q')).toBe(true);
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
it('returns "" when the body is genuinely empty (legitimate empty signal)', () => {
|
|
124
|
+
const dom = new JSDOM('<html><body></body></html>');
|
|
125
|
+
// Force the body to be missing (edge case JSDOM actually creates one)
|
|
126
|
+
const doc = dom.window.document;
|
|
127
|
+
// empty body → fallback chain returns ''
|
|
128
|
+
expect(pickContentText(doc, CONTENT_SELECTORS)).toBe('');
|
|
129
|
+
});
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
describe('xiaoe/content — countXiaoeImages', () => {
|
|
133
|
+
function imgDoc(srcs) {
|
|
134
|
+
const tags = srcs.map((s) => `<img src="${s}">`).join('');
|
|
135
|
+
return new JSDOM(`<html><body>${tags}</body></html>`).window.document;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
it('counts only xiaoe-hosted, non-data: images', () => {
|
|
139
|
+
const doc = imgDoc([
|
|
140
|
+
'https://commonresource-1252524126.cdn.xiaoe-tech.com/abc.jpg',
|
|
141
|
+
'data:image/png;base64,iVBOR',
|
|
142
|
+
'https://other-cdn.com/foo.jpg',
|
|
143
|
+
'https://app.xiaoe-tech.com/img/bar.png',
|
|
144
|
+
]);
|
|
145
|
+
expect(countXiaoeImages(doc)).toBe(2);
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
it('returns 0 when every image is excluded — never silently undefined', () => {
|
|
149
|
+
const doc = imgDoc([
|
|
150
|
+
'data:image/png;base64,xxx',
|
|
151
|
+
'https://avatar.cdn.com/u.jpg',
|
|
152
|
+
]);
|
|
153
|
+
expect(countXiaoeImages(doc)).toBe(0);
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it('returns 0 on a doc with no <img> elements', () => {
|
|
157
|
+
const doc = new JSDOM('<html><body><p>no images here</p></body></html>').window.document;
|
|
158
|
+
expect(countXiaoeImages(doc)).toBe(0);
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
describe('xiaoe/content — requireXiaoePageUrl', () => {
|
|
163
|
+
it('rejects empty, malformed, and non-xiaoe URLs upfront', () => {
|
|
164
|
+
expect(() => requireXiaoePageUrl('', 'content')).toThrow(ArgumentError);
|
|
165
|
+
expect(() => requireXiaoePageUrl('not a url', 'content')).toThrow(ArgumentError);
|
|
166
|
+
expect(() => requireXiaoePageUrl('http://h5.xet.citv.cn/p/course/ecourse/v_x', 'content')).toThrow(ArgumentError);
|
|
167
|
+
expect(() => requireXiaoePageUrl('https://example.com/p/course/ecourse/v_x', 'content')).toThrow(ArgumentError);
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
it('accepts root and shop h5.xet.citv.cn URLs', () => {
|
|
171
|
+
expect(requireXiaoePageUrl('https://h5.xet.citv.cn/p/course/ecourse/v_x', 'content'))
|
|
172
|
+
.toBe('https://h5.xet.citv.cn/p/course/ecourse/v_x');
|
|
173
|
+
expect(requireXiaoePageUrl('https://appxxxx.h5.xet.citv.cn/p/course/ecourse/v_x', 'content'))
|
|
174
|
+
.toBe('https://appxxxx.h5.xet.citv.cn/p/course/ecourse/v_x');
|
|
175
|
+
});
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
// ─── catalog.js helpers ────────────────────────────────────────────
|
|
179
|
+
|
|
180
|
+
describe('xiaoe/catalog — typeLabel', () => {
|
|
181
|
+
it('maps known resource types to Chinese labels', () => {
|
|
182
|
+
expect(typeLabel(1)).toBe('图文');
|
|
183
|
+
expect(typeLabel(2)).toBe('直播');
|
|
184
|
+
expect(typeLabel(3)).toBe('音频');
|
|
185
|
+
expect(typeLabel(4)).toBe('视频');
|
|
186
|
+
expect(typeLabel(6)).toBe('专栏');
|
|
187
|
+
expect(typeLabel(8)).toBe('大专栏');
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
it('coerces string-typed resource type ints', () => {
|
|
191
|
+
expect(typeLabel('1')).toBe('图文');
|
|
192
|
+
expect(typeLabel('8')).toBe('大专栏');
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
it('returns the raw string for unknown types — never silently swallows', () => {
|
|
196
|
+
expect(typeLabel(99)).toBe('99');
|
|
197
|
+
expect(typeLabel('custom')).toBe('custom');
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
it('returns "" for nullish input (no falsy crash)', () => {
|
|
201
|
+
expect(typeLabel(null)).toBe('');
|
|
202
|
+
expect(typeLabel(undefined)).toBe('');
|
|
203
|
+
expect(typeLabel(0)).toBe('');
|
|
204
|
+
});
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
describe('xiaoe/catalog — buildItemUrl', () => {
|
|
208
|
+
it('passes through fully-qualified URLs unchanged', () => {
|
|
209
|
+
expect(buildItemUrl({ jump_url: 'https://example.com/x' }, 'https://h5.xet.citv.cn'))
|
|
210
|
+
.toBe('https://example.com/x');
|
|
211
|
+
expect(buildItemUrl({ h5_url: 'http://x.test/p' }, 'https://h5.xet.citv.cn'))
|
|
212
|
+
.toBe('http://x.test/p');
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
it('prepends origin to relative URLs', () => {
|
|
216
|
+
expect(buildItemUrl({ jump_url: '/p/123' }, 'https://h5.xet.citv.cn'))
|
|
217
|
+
.toBe('https://h5.xet.citv.cn/p/123');
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
it('falls through jump_url → h5_url → url priority', () => {
|
|
221
|
+
expect(buildItemUrl({ jump_url: '/a', h5_url: '/b', url: '/c' }, 'https://x.test'))
|
|
222
|
+
.toBe('https://x.test/a');
|
|
223
|
+
expect(buildItemUrl({ h5_url: '/b', url: '/c' }, 'https://x.test'))
|
|
224
|
+
.toBe('https://x.test/b');
|
|
225
|
+
expect(buildItemUrl({ url: '/c' }, 'https://x.test'))
|
|
226
|
+
.toBe('https://x.test/c');
|
|
227
|
+
});
|
|
228
|
+
|
|
229
|
+
it('returns "" when no URL field is present (no synthetic URL)', () => {
|
|
230
|
+
expect(buildItemUrl({}, 'https://x.test')).toBe('');
|
|
231
|
+
expect(buildItemUrl({ jump_url: '' }, 'https://x.test')).toBe('');
|
|
232
|
+
});
|
|
233
|
+
});
|
|
234
|
+
|
|
235
|
+
describe('xiaoe/catalog — chapterUrlPath', () => {
|
|
236
|
+
it('returns the right path for known chapter types', () => {
|
|
237
|
+
expect(chapterUrlPath(1)).toBe('/v1/course/text/');
|
|
238
|
+
expect(chapterUrlPath(2)).toBe('/v2/course/alive/');
|
|
239
|
+
expect(chapterUrlPath(3)).toBe('/v1/course/audio/');
|
|
240
|
+
expect(chapterUrlPath(4)).toBe('/v1/course/video/');
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
it('coerces string ints', () => {
|
|
244
|
+
expect(chapterUrlPath('1')).toBe('/v1/course/text/');
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
it('returns undefined for unknown / nullish — caller decides empty-URL semantics', () => {
|
|
248
|
+
expect(chapterUrlPath(99)).toBeUndefined();
|
|
249
|
+
expect(chapterUrlPath(0)).toBeUndefined();
|
|
250
|
+
expect(chapterUrlPath(null)).toBeUndefined();
|
|
251
|
+
});
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
// ─── courses.js helper ─────────────────────────────────────────────
|
|
255
|
+
|
|
256
|
+
describe('xiaoe/courses — buildCourseUrl', () => {
|
|
257
|
+
it('returns h5_url when present (highest priority)', () => {
|
|
258
|
+
expect(buildCourseUrl({
|
|
259
|
+
h5_url: 'https://h5.xet.citv.cn/p/abc',
|
|
260
|
+
app_id: 'appXXX',
|
|
261
|
+
resource_id: 'p_999',
|
|
262
|
+
})).toBe('https://h5.xet.citv.cn/p/abc');
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
it('returns url when h5_url missing', () => {
|
|
266
|
+
expect(buildCourseUrl({ url: 'https://study.xiaoe-tech.com/u/y' }))
|
|
267
|
+
.toBe('https://study.xiaoe-tech.com/u/y');
|
|
268
|
+
});
|
|
269
|
+
|
|
270
|
+
it('builds column path for resource_type 6', () => {
|
|
271
|
+
expect(buildCourseUrl({
|
|
272
|
+
app_id: 'appAAA',
|
|
273
|
+
resource_id: 'p_111',
|
|
274
|
+
resource_type: 6,
|
|
275
|
+
})).toBe('https://appAAA.h5.xet.citv.cn/v1/course/column/p_111?type=3');
|
|
276
|
+
});
|
|
277
|
+
|
|
278
|
+
it('builds ecourse path for non-column types', () => {
|
|
279
|
+
expect(buildCourseUrl({
|
|
280
|
+
app_id: 'appAAA',
|
|
281
|
+
resource_id: 'p_222',
|
|
282
|
+
resource_type: 4,
|
|
283
|
+
})).toBe('https://appAAA.h5.xet.citv.cn/p/course/ecourse/p_222');
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
it('returns "" when entry has no URL fields and no app_id+resource_id pair (no synthetic URL)', () => {
|
|
287
|
+
expect(buildCourseUrl({})).toBe('');
|
|
288
|
+
expect(buildCourseUrl({ app_id: 'x' })).toBe(''); // missing resource_id
|
|
289
|
+
expect(buildCourseUrl({ resource_id: 'p_1' })).toBe(''); // missing app_id
|
|
290
|
+
expect(buildCourseUrl(null)).toBe('');
|
|
291
|
+
expect(buildCourseUrl(undefined)).toBe('');
|
|
292
|
+
});
|
|
293
|
+
});
|
|
294
|
+
|
|
295
|
+
// ─── buildScript invariants (anti-pattern regression guards) ──────
|
|
296
|
+
|
|
297
|
+
describe('xiaoe — buildScript embeds helpers + no anti-patterns', () => {
|
|
298
|
+
it('content script embeds pickContentText + countXiaoeImages and never silently slices images', () => {
|
|
299
|
+
const script = contentTest.buildContentScript();
|
|
300
|
+
expect(script).toContain('pickContentText');
|
|
301
|
+
expect(script).toContain('countXiaoeImages');
|
|
302
|
+
// The legacy adapter did `images.slice(0, 20)` and silently
|
|
303
|
+
// dropped everything past the 20th image. The new script
|
|
304
|
+
// exposes only `image_count` (counting all images via the
|
|
305
|
+
// helper), so a hard `.slice(0, 20)` should not appear.
|
|
306
|
+
expect(script).not.toMatch(/\.slice\(0,\s*20\)/);
|
|
307
|
+
// Embeds the selector list literally so the IIFE has no
|
|
308
|
+
// dependency on the host page's globals.
|
|
309
|
+
expect(script).toContain('rich-text-wrap');
|
|
310
|
+
});
|
|
311
|
+
|
|
312
|
+
it('catalog script embeds typeLabel + buildItemUrl + chapterUrlPath', () => {
|
|
313
|
+
const script = catalogTest.buildCatalogScript();
|
|
314
|
+
expect(script).toContain('typeLabel');
|
|
315
|
+
expect(script).toContain('buildItemUrl');
|
|
316
|
+
expect(script).toContain('chapterUrlPath');
|
|
317
|
+
// Vue private API anchors must remain — these are the only
|
|
318
|
+
// stable hook into Xiaoe's SPA. If a future refactor removes
|
|
319
|
+
// them, the test will fail and force a deliberate decision.
|
|
320
|
+
expect(script).toContain('__vue__');
|
|
321
|
+
expect(script).toContain('$store');
|
|
322
|
+
});
|
|
323
|
+
|
|
324
|
+
it('courses script embeds buildCourseUrl', () => {
|
|
325
|
+
const script = coursesTest.buildCoursesScript();
|
|
326
|
+
expect(script).toContain('buildCourseUrl');
|
|
327
|
+
expect(script).toContain('__vue__');
|
|
328
|
+
});
|
|
329
|
+
});
|
|
330
|
+
|
|
331
|
+
// ─── Wire tests for the func form ─────────────────────────────────
|
|
332
|
+
|
|
333
|
+
describe('xiaoe/content — getXiaoeContent func wiring', () => {
|
|
334
|
+
function pageMock(rows) {
|
|
335
|
+
return {
|
|
336
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
337
|
+
evaluate: vi.fn().mockResolvedValue(rows),
|
|
338
|
+
};
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
it('throws ArgumentError on missing url BEFORE calling page.goto', async () => {
|
|
342
|
+
const page = pageMock([]);
|
|
343
|
+
await expect(contentCommand.func(page, {})).rejects.toThrow(ArgumentError);
|
|
344
|
+
await expect(contentCommand.func(page, { url: '' })).rejects.toThrow(ArgumentError);
|
|
345
|
+
await expect(contentCommand.func(page, { url: ' ' })).rejects.toThrow(ArgumentError);
|
|
346
|
+
await expect(contentCommand.func(page, { url: 'https://example.com/p/x' })).rejects.toThrow(ArgumentError);
|
|
347
|
+
expect(page.goto).not.toHaveBeenCalled();
|
|
348
|
+
});
|
|
349
|
+
|
|
350
|
+
it('wraps browser navigation failures as CommandExecutionError', async () => {
|
|
351
|
+
const page = {
|
|
352
|
+
goto: vi.fn().mockRejectedValue(new Error('net::ERR_ABORTED')),
|
|
353
|
+
evaluate: vi.fn(),
|
|
354
|
+
};
|
|
355
|
+
await expect(contentCommand.func(page, { url: 'https://h5.xet.citv.cn/p/x' }))
|
|
356
|
+
.rejects.toThrow(CommandExecutionError);
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
it('throws EmptyResultError when no rows are returned', async () => {
|
|
360
|
+
const page = pageMock([]);
|
|
361
|
+
await expect(contentCommand.func(page, { url: 'https://h5.xet.citv.cn/p/x' }))
|
|
362
|
+
.rejects.toThrow(EmptyResultError);
|
|
363
|
+
});
|
|
364
|
+
|
|
365
|
+
it('throws EmptyResultError when content is empty (login likely expired — fail-fast not silent empty row)', async () => {
|
|
366
|
+
const page = pageMock([{ title: 'shell', content: '', content_length: 0, image_count: 0 }]);
|
|
367
|
+
await expect(contentCommand.func(page, { url: 'https://h5.xet.citv.cn/p/x' }))
|
|
368
|
+
.rejects.toThrow(EmptyResultError);
|
|
369
|
+
});
|
|
370
|
+
|
|
371
|
+
it('throws CommandExecutionError when page.evaluate rejects', async () => {
|
|
372
|
+
const page = {
|
|
373
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
374
|
+
evaluate: vi.fn().mockRejectedValue(new Error('CDP exploded')),
|
|
375
|
+
};
|
|
376
|
+
await expect(contentCommand.func(page, { url: 'https://h5.xet.citv.cn/p/x' }))
|
|
377
|
+
.rejects.toThrow(CommandExecutionError);
|
|
378
|
+
});
|
|
379
|
+
|
|
380
|
+
it('returns the rows verbatim on the happy path', async () => {
|
|
381
|
+
const fakeRow = {
|
|
382
|
+
title: 'demo',
|
|
383
|
+
content: 'hello world '.repeat(10),
|
|
384
|
+
content_length: 120,
|
|
385
|
+
image_count: 3,
|
|
386
|
+
};
|
|
387
|
+
const page = pageMock([fakeRow]);
|
|
388
|
+
const result = await contentCommand.func(page, { url: 'https://h5.xet.citv.cn/p/x' });
|
|
389
|
+
expect(result).toEqual([fakeRow]);
|
|
390
|
+
expect(page.goto).toHaveBeenCalledWith(
|
|
391
|
+
'https://h5.xet.citv.cn/p/x',
|
|
392
|
+
expect.objectContaining({ waitUntil: 'load' }),
|
|
393
|
+
);
|
|
394
|
+
});
|
|
395
|
+
});
|
|
396
|
+
|
|
397
|
+
describe('xiaoe/catalog — getXiaoeCatalog func wiring', () => {
|
|
398
|
+
function pageMock(rows) {
|
|
399
|
+
return {
|
|
400
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
401
|
+
evaluate: vi.fn().mockResolvedValue(rows),
|
|
402
|
+
};
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
it('throws ArgumentError on missing url BEFORE calling page.goto', async () => {
|
|
406
|
+
const page = pageMock([]);
|
|
407
|
+
await expect(catalogCommand.func(page, {})).rejects.toThrow(ArgumentError);
|
|
408
|
+
await expect(catalogCommand.func(page, { url: 'https://example.com/p/c' })).rejects.toThrow(ArgumentError);
|
|
409
|
+
expect(page.goto).not.toHaveBeenCalled();
|
|
410
|
+
});
|
|
411
|
+
|
|
412
|
+
it('wraps browser navigation failures as CommandExecutionError', async () => {
|
|
413
|
+
const page = {
|
|
414
|
+
goto: vi.fn().mockRejectedValue(new Error('Execution context was destroyed')),
|
|
415
|
+
evaluate: vi.fn(),
|
|
416
|
+
};
|
|
417
|
+
await expect(catalogCommand.func(page, { url: 'https://h5.xet.citv.cn/p/c' }))
|
|
418
|
+
.rejects.toThrow(CommandExecutionError);
|
|
419
|
+
});
|
|
420
|
+
|
|
421
|
+
it('throws EmptyResultError when no chapters extracted (cookie likely expired)', async () => {
|
|
422
|
+
const page = pageMock([]);
|
|
423
|
+
await expect(catalogCommand.func(page, { url: 'https://h5.xet.citv.cn/p/c' }))
|
|
424
|
+
.rejects.toThrow(EmptyResultError);
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
it('returns rows verbatim on the happy path', async () => {
|
|
428
|
+
const fake = [{
|
|
429
|
+
ch: 1, chapter: '入门', no: 1, title: '第一节',
|
|
430
|
+
type: '视频', resource_id: 'v_abc', url: 'https://x.test/v/abc', status: '未学',
|
|
431
|
+
}];
|
|
432
|
+
const page = pageMock(fake);
|
|
433
|
+
expect(await catalogCommand.func(page, { url: 'https://h5.xet.citv.cn/p/c' }))
|
|
434
|
+
.toEqual(fake);
|
|
435
|
+
});
|
|
436
|
+
|
|
437
|
+
it('throws CommandExecutionError when page.evaluate rejects', async () => {
|
|
438
|
+
const page = {
|
|
439
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
440
|
+
evaluate: vi.fn().mockRejectedValue(new Error('boom')),
|
|
441
|
+
};
|
|
442
|
+
await expect(catalogCommand.func(page, { url: 'https://h5.xet.citv.cn/p/c' }))
|
|
443
|
+
.rejects.toThrow(CommandExecutionError);
|
|
444
|
+
});
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
describe('xiaoe/courses — getXiaoeCourses func wiring', () => {
|
|
448
|
+
function pageMock(rows) {
|
|
449
|
+
return {
|
|
450
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
451
|
+
evaluate: vi.fn().mockResolvedValue(rows),
|
|
452
|
+
};
|
|
453
|
+
}
|
|
454
|
+
|
|
455
|
+
it('navigates to the study landing page (no positional arg required)', async () => {
|
|
456
|
+
const fake = [{ title: 'Course A', shop: 'Shop A', url: 'https://x.test/c/a' }];
|
|
457
|
+
const page = pageMock(fake);
|
|
458
|
+
const result = await coursesCommand.func(page, {});
|
|
459
|
+
expect(page.goto).toHaveBeenCalledWith(
|
|
460
|
+
'https://study.xiaoe-tech.com/',
|
|
461
|
+
expect.objectContaining({ waitUntil: 'load' }),
|
|
462
|
+
);
|
|
463
|
+
expect(result).toEqual(fake);
|
|
464
|
+
});
|
|
465
|
+
|
|
466
|
+
it('throws EmptyResultError when no cards found (cookie likely expired)', async () => {
|
|
467
|
+
const page = pageMock([]);
|
|
468
|
+
await expect(coursesCommand.func(page, {})).rejects.toThrow(EmptyResultError);
|
|
469
|
+
});
|
|
470
|
+
|
|
471
|
+
it('throws CommandExecutionError when page.evaluate rejects', async () => {
|
|
472
|
+
const page = {
|
|
473
|
+
goto: vi.fn().mockResolvedValue(undefined),
|
|
474
|
+
evaluate: vi.fn().mockRejectedValue(new Error('cdp')),
|
|
475
|
+
};
|
|
476
|
+
await expect(coursesCommand.func(page, {})).rejects.toThrow(CommandExecutionError);
|
|
477
|
+
});
|
|
478
|
+
|
|
479
|
+
it('wraps browser navigation failures as CommandExecutionError', async () => {
|
|
480
|
+
const page = {
|
|
481
|
+
goto: vi.fn().mockRejectedValue(new Error('net::ERR_ABORTED')),
|
|
482
|
+
evaluate: vi.fn(),
|
|
483
|
+
};
|
|
484
|
+
await expect(coursesCommand.func(page, {})).rejects.toThrow(CommandExecutionError);
|
|
485
|
+
});
|
|
486
|
+
});
|
|
@@ -54,9 +54,9 @@ cli({
|
|
|
54
54
|
navigateBefore: false,
|
|
55
55
|
args: [
|
|
56
56
|
{ name: 'limit', type: 'int', default: 3, help: 'Number of recent notes to summarize' },
|
|
57
|
+
{ name: 'timeout', type: 'int', required: false, default: 180, help: 'Max seconds for the overall command (default: 180)' },
|
|
57
58
|
],
|
|
58
59
|
columns: ['rank', 'id', 'title', 'views', 'likes', 'collects', 'comments', 'shares', 'avg_view_time', 'rise_fans', 'top_source', 'top_interest', 'url'],
|
|
59
|
-
timeoutSeconds: 180,
|
|
60
60
|
func: async (page, kwargs) => {
|
|
61
61
|
const limit = kwargs.limit || 3;
|
|
62
62
|
const notes = await fetchCreatorNotes(page, limit);
|
|
@@ -18,7 +18,7 @@
|
|
|
18
18
|
import * as fs from 'node:fs';
|
|
19
19
|
import * as path from 'node:path';
|
|
20
20
|
import { cli, Strategy } from '@jackwener/opencli/registry';
|
|
21
|
-
const PUBLISH_URL = 'https://creator.xiaohongshu.com/publish/publish?from=menu_left';
|
|
21
|
+
const PUBLISH_URL = 'https://creator.xiaohongshu.com/publish/publish?from=menu_left&target=image';
|
|
22
22
|
const MAX_IMAGES = 9;
|
|
23
23
|
const MAX_TITLE_LEN = 20;
|
|
24
24
|
const UPLOAD_SETTLE_MS = 3000;
|
|
@@ -95,7 +95,7 @@ async function uploadImages(page, absPaths) {
|
|
|
95
95
|
catch (err) {
|
|
96
96
|
// If set-file-input action is not supported by extension, fall through to legacy
|
|
97
97
|
const msg = err instanceof Error ? err.message : String(err);
|
|
98
|
-
if (msg.includes('Unknown action') || msg.includes('not supported')) {
|
|
98
|
+
if (msg.includes('Unknown action') || msg.includes('not supported') || msg.includes('Not allowed')) {
|
|
99
99
|
// Extension too old — fall through to legacy base64 method
|
|
100
100
|
}
|
|
101
101
|
else {
|
|
@@ -351,7 +351,20 @@ async function selectImageTextTab(page) {
|
|
|
351
351
|
if (!isVisible(node)) continue;
|
|
352
352
|
const text = normalize(node.innerText || node.textContent || '');
|
|
353
353
|
if (!text || text.includes('视频')) continue;
|
|
354
|
-
if (text === target
|
|
354
|
+
if (text === target) {
|
|
355
|
+
const clickable = node.closest('button, [role="tab"], [role="button"], a, label') || node;
|
|
356
|
+
clickable.click();
|
|
357
|
+
return { ok: true, target, text };
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
for (const target of targets) {
|
|
363
|
+
for (const node of nodes) {
|
|
364
|
+
if (!isVisible(node)) continue;
|
|
365
|
+
const text = normalize(node.innerText || node.textContent || '');
|
|
366
|
+
if (!text || text.includes('视频')) continue;
|
|
367
|
+
if (text.startsWith(target) || text.includes(target)) {
|
|
355
368
|
const clickable = node.closest('button, [role="tab"], [role="button"], a, label') || node;
|
|
356
369
|
clickable.click();
|
|
357
370
|
return { ok: true, target, text };
|