@jackwener/opencli 1.3.3 → 1.4.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/.github/actions/setup-chrome/action.yml +5 -4
- package/.github/pull_request_template.md +3 -1
- package/.github/workflows/build-extension.yml +7 -1
- package/.github/workflows/ci.yml +46 -6
- package/.github/workflows/docs.yml +1 -1
- package/.github/workflows/e2e-headed.yml +36 -3
- package/.github/workflows/release.yml +1 -1
- package/.github/workflows/security.yml +0 -3
- package/CHANGELOG.md +78 -0
- package/CONTRIBUTING.md +6 -3
- package/PRIVACY.md +57 -0
- package/README.md +31 -4
- package/README.zh-CN.md +31 -4
- package/SKILL.md +107 -2
- package/TESTING.md +1 -0
- package/chatwise-opencli.ps1 +82 -0
- package/dist/analysis.d.ts +38 -0
- package/dist/analysis.js +166 -0
- package/dist/browser/cdp.d.ts +0 -4
- package/dist/browser/cdp.js +53 -41
- package/dist/browser/cdp.test.d.ts +1 -0
- package/dist/browser/cdp.test.js +52 -0
- package/dist/browser/dom-snapshot.d.ts +2 -2
- package/dist/browser/dom-snapshot.js +54 -1
- package/dist/browser/dom-snapshot.test.js +36 -0
- package/dist/browser/index.d.ts +2 -2
- package/dist/browser/index.js +1 -1
- package/dist/browser/mcp.d.ts +0 -2
- package/dist/browser/mcp.js +2 -3
- package/dist/browser/page.d.ts +4 -3
- package/dist/browser/page.js +34 -37
- package/dist/browser/stealth.d.ts +0 -2
- package/dist/browser/stealth.js +24 -9
- package/dist/browser.test.js +2 -2
- package/dist/build-manifest.js +15 -9
- package/dist/build-manifest.test.js +12 -0
- package/dist/cascade.js +4 -2
- package/dist/cli-manifest.json +1325 -256
- package/dist/cli.js +57 -29
- package/dist/clis/_shared/desktop-commands.d.ts +22 -0
- package/dist/clis/_shared/desktop-commands.js +108 -0
- package/dist/clis/antigravity/serve.js +5 -2
- package/dist/clis/apple-podcasts/search.js +2 -1
- package/dist/clis/arxiv/search.js +3 -3
- package/dist/clis/bbc/news.js +0 -1
- package/dist/clis/bilibili/dynamic.test.d.ts +1 -0
- package/dist/clis/bilibili/dynamic.test.js +68 -0
- package/dist/clis/bilibili/favorite.js +4 -2
- package/dist/clis/bilibili/following.js +3 -2
- package/dist/clis/bilibili/subtitle.js +8 -7
- package/dist/clis/bilibili/utils.js +2 -2
- package/dist/clis/boss/batchgreet.js +1 -1
- package/dist/clis/boss/chatlist.js +1 -1
- package/dist/clis/boss/chatmsg.js +1 -1
- package/dist/clis/boss/detail.js +1 -1
- package/dist/clis/boss/exchange.js +1 -1
- package/dist/clis/boss/greet.js +1 -1
- package/dist/clis/boss/invite.js +1 -1
- package/dist/clis/boss/joblist.js +1 -1
- package/dist/clis/boss/mark.js +4 -3
- package/dist/clis/boss/recommend.js +1 -1
- package/dist/clis/boss/resume.js +1 -1
- package/dist/clis/boss/search.js +1 -1
- package/dist/clis/boss/send.js +5 -4
- package/dist/clis/boss/stats.js +1 -1
- package/dist/clis/chatgpt/ask.js +4 -0
- package/dist/clis/chatgpt/new.js +5 -1
- package/dist/clis/chatgpt/read.js +5 -1
- package/dist/clis/chatgpt/send.js +2 -1
- package/dist/clis/chatgpt/status.js +5 -1
- package/dist/clis/chatwise/ask.js +8 -2
- package/dist/clis/chatwise/export.js +2 -0
- package/dist/clis/chatwise/history.js +2 -0
- package/dist/clis/chatwise/model.js +8 -3
- package/dist/clis/chatwise/new.js +3 -18
- package/dist/clis/chatwise/read.js +2 -0
- package/dist/clis/chatwise/screenshot.js +3 -27
- package/dist/clis/chatwise/send.js +8 -2
- package/dist/clis/chatwise/shared.d.ts +2 -0
- package/dist/clis/chatwise/shared.js +6 -0
- package/dist/clis/chatwise/status.js +3 -22
- package/dist/clis/codex/ask.js +6 -2
- package/dist/clis/codex/dump.js +2 -25
- package/dist/clis/codex/new.js +2 -25
- package/dist/clis/codex/screenshot.js +2 -27
- package/dist/clis/codex/send.js +6 -4
- package/dist/clis/codex/status.js +2 -22
- package/dist/clis/ctrip/search.js +0 -1
- package/dist/clis/cursor/ask.js +2 -1
- package/dist/clis/cursor/composer.js +2 -1
- package/dist/clis/cursor/dump.js +2 -25
- package/dist/clis/cursor/new.js +2 -18
- package/dist/clis/cursor/read.js +2 -1
- package/dist/clis/cursor/screenshot.js +1 -30
- package/dist/clis/cursor/send.js +2 -1
- package/dist/clis/cursor/status.js +2 -21
- package/dist/clis/dictionary/examples.yaml +25 -0
- package/dist/clis/dictionary/search.yaml +27 -0
- package/dist/clis/dictionary/synonyms.yaml +25 -0
- package/dist/clis/douban/book-hot.js +1 -1
- package/dist/clis/douban/movie-hot.js +1 -1
- package/dist/clis/douban/search.js +1 -1
- package/dist/clis/douban/utils.d.ts +4 -1
- package/dist/clis/douban/utils.js +156 -1
- package/dist/clis/doubao/ask.js +1 -1
- package/dist/clis/doubao/new.js +1 -1
- package/dist/clis/doubao/read.js +1 -1
- package/dist/clis/doubao/send.js +1 -1
- package/dist/clis/doubao/status.js +1 -1
- package/dist/clis/doubao-app/ask.js +1 -1
- package/dist/clis/doubao-app/new.js +1 -1
- package/dist/clis/doubao-app/read.js +1 -1
- package/dist/clis/doubao-app/send.js +1 -1
- package/dist/clis/douyin/_shared/browser-fetch.d.ts +10 -0
- package/dist/clis/douyin/_shared/browser-fetch.js +30 -0
- package/dist/clis/douyin/_shared/browser-fetch.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/browser-fetch.test.js +31 -0
- package/dist/clis/douyin/_shared/creation-id.d.ts +1 -0
- package/dist/clis/douyin/_shared/creation-id.js +5 -0
- package/dist/clis/douyin/_shared/creation-id.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/creation-id.test.js +22 -0
- package/dist/clis/douyin/_shared/imagex-upload.d.ts +20 -0
- package/dist/clis/douyin/_shared/imagex-upload.js +53 -0
- package/dist/clis/douyin/_shared/imagex-upload.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/imagex-upload.test.js +87 -0
- package/dist/clis/douyin/_shared/sts2.d.ts +8 -0
- package/dist/clis/douyin/_shared/sts2.js +15 -0
- package/dist/clis/douyin/_shared/text-extra.d.ts +18 -0
- package/dist/clis/douyin/_shared/text-extra.js +15 -0
- package/dist/clis/douyin/_shared/text-extra.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/text-extra.test.js +37 -0
- package/dist/clis/douyin/_shared/timing.d.ts +2 -0
- package/dist/clis/douyin/_shared/timing.js +22 -0
- package/dist/clis/douyin/_shared/timing.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/timing.test.js +28 -0
- package/dist/clis/douyin/_shared/tos-upload-short-read.test.d.ts +11 -0
- package/dist/clis/douyin/_shared/tos-upload-short-read.test.js +83 -0
- package/dist/clis/douyin/_shared/tos-upload.d.ts +53 -0
- package/dist/clis/douyin/_shared/tos-upload.js +295 -0
- package/dist/clis/douyin/_shared/tos-upload.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/tos-upload.test.js +229 -0
- package/dist/clis/douyin/_shared/transcode.d.ts +27 -0
- package/dist/clis/douyin/_shared/transcode.js +45 -0
- package/dist/clis/douyin/_shared/transcode.test.d.ts +1 -0
- package/dist/clis/douyin/_shared/transcode.test.js +93 -0
- package/dist/clis/douyin/_shared/types.d.ts +26 -0
- package/dist/clis/douyin/_shared/types.js +1 -0
- package/dist/clis/douyin/activities.d.ts +1 -0
- package/dist/clis/douyin/activities.js +20 -0
- package/dist/clis/douyin/activities.test.d.ts +1 -0
- package/dist/clis/douyin/activities.test.js +22 -0
- package/dist/clis/douyin/collections.d.ts +1 -0
- package/dist/clis/douyin/collections.js +22 -0
- package/dist/clis/douyin/collections.test.d.ts +1 -0
- package/dist/clis/douyin/collections.test.js +23 -0
- package/dist/clis/douyin/delete.d.ts +1 -0
- package/dist/clis/douyin/delete.js +18 -0
- package/dist/clis/douyin/delete.test.d.ts +1 -0
- package/dist/clis/douyin/delete.test.js +11 -0
- package/dist/clis/douyin/draft.d.ts +14 -0
- package/dist/clis/douyin/draft.js +237 -0
- package/dist/clis/douyin/draft.test.d.ts +1 -0
- package/dist/clis/douyin/draft.test.js +11 -0
- package/dist/clis/douyin/drafts.d.ts +1 -0
- package/dist/clis/douyin/drafts.js +23 -0
- package/dist/clis/douyin/drafts.test.d.ts +1 -0
- package/dist/clis/douyin/drafts.test.js +11 -0
- package/dist/clis/douyin/hashtag.d.ts +1 -0
- package/dist/clis/douyin/hashtag.js +45 -0
- package/dist/clis/douyin/hashtag.test.d.ts +1 -0
- package/dist/clis/douyin/hashtag.test.js +25 -0
- package/dist/clis/douyin/location.d.ts +1 -0
- package/dist/clis/douyin/location.js +24 -0
- package/dist/clis/douyin/location.test.d.ts +1 -0
- package/dist/clis/douyin/location.test.js +23 -0
- package/dist/clis/douyin/profile.d.ts +1 -0
- package/dist/clis/douyin/profile.js +28 -0
- package/dist/clis/douyin/profile.test.d.ts +1 -0
- package/dist/clis/douyin/profile.test.js +11 -0
- package/dist/clis/douyin/publish.d.ts +14 -0
- package/dist/clis/douyin/publish.js +288 -0
- package/dist/clis/douyin/publish.test.d.ts +1 -0
- package/dist/clis/douyin/publish.test.js +38 -0
- package/dist/clis/douyin/stats.d.ts +1 -0
- package/dist/clis/douyin/stats.js +27 -0
- package/dist/clis/douyin/stats.test.d.ts +1 -0
- package/dist/clis/douyin/stats.test.js +22 -0
- package/dist/clis/douyin/update.d.ts +1 -0
- package/dist/clis/douyin/update.js +31 -0
- package/dist/clis/douyin/update.test.d.ts +1 -0
- package/dist/clis/douyin/update.test.js +11 -0
- package/dist/clis/douyin/videos.d.ts +1 -0
- package/dist/clis/douyin/videos.js +34 -0
- package/dist/clis/douyin/videos.test.d.ts +1 -0
- package/dist/clis/douyin/videos.test.js +11 -0
- package/dist/clis/grok/ask.d.ts +4 -0
- package/dist/clis/grok/ask.js +28 -10
- package/dist/clis/grok/ask.test.js +18 -0
- package/dist/clis/hackernews/search.yaml +1 -1
- package/dist/clis/instagram/search.yaml +2 -1
- package/dist/clis/jd/item.d.ts +1 -0
- package/dist/clis/jd/item.js +96 -0
- package/dist/clis/jd/item.test.d.ts +1 -0
- package/dist/clis/jd/item.test.js +28 -0
- package/dist/clis/jike/feed.js +1 -1
- package/dist/clis/jike/search.js +1 -1
- package/dist/clis/linkedin/search.js +5 -4
- package/dist/clis/linkedin/timeline.d.ts +21 -0
- package/dist/clis/linkedin/timeline.js +503 -0
- package/dist/clis/linkedin/timeline.test.d.ts +1 -0
- package/dist/clis/linkedin/timeline.test.js +81 -0
- package/dist/clis/linux-do/search.yaml +3 -1
- package/dist/clis/medium/feed.js +1 -1
- package/dist/clis/medium/search.js +2 -2
- package/dist/clis/medium/user.js +1 -1
- package/dist/clis/medium/{shared.js → utils.js} +2 -1
- package/dist/clis/pixiv/detail.yaml +49 -0
- package/dist/clis/pixiv/download.d.ts +7 -0
- package/dist/clis/pixiv/download.js +78 -0
- package/dist/clis/pixiv/download.test.d.ts +1 -0
- package/dist/clis/pixiv/download.test.js +87 -0
- package/dist/clis/pixiv/illusts.d.ts +8 -0
- package/dist/clis/pixiv/illusts.js +65 -0
- package/dist/clis/pixiv/illusts.test.d.ts +1 -0
- package/dist/clis/pixiv/illusts.test.js +99 -0
- package/dist/clis/pixiv/ranking.yaml +53 -0
- package/dist/clis/pixiv/search.d.ts +6 -0
- package/dist/clis/pixiv/search.js +43 -0
- package/dist/clis/pixiv/search.test.d.ts +1 -0
- package/dist/clis/pixiv/search.test.js +83 -0
- package/dist/clis/pixiv/test-utils.d.ts +12 -0
- package/dist/clis/pixiv/test-utils.js +23 -0
- package/dist/clis/pixiv/user.yaml +46 -0
- package/dist/clis/pixiv/utils.d.ts +27 -0
- package/dist/clis/pixiv/utils.js +49 -0
- package/dist/clis/reddit/comment.js +2 -1
- package/dist/clis/reddit/read.js +4 -3
- package/dist/clis/reddit/read.test.d.ts +1 -0
- package/dist/clis/reddit/read.test.js +28 -0
- package/dist/clis/reddit/save.js +2 -1
- package/dist/clis/reddit/saved.js +7 -3
- package/dist/clis/reddit/subscribe.js +2 -1
- package/dist/clis/reddit/upvote.js +2 -1
- package/dist/clis/reddit/upvoted.js +7 -3
- package/dist/clis/reuters/search.js +0 -1
- package/dist/clis/sinablog/article.js +1 -1
- package/dist/clis/sinablog/hot.js +1 -1
- package/dist/clis/sinablog/user.js +1 -1
- package/dist/clis/substack/feed.js +1 -1
- package/dist/clis/substack/publication.js +1 -1
- package/dist/clis/substack/search.js +3 -2
- package/dist/clis/substack/{shared.js → utils.js} +3 -2
- package/dist/clis/tiktok/search.yaml +2 -1
- package/dist/clis/twitter/accept.js +2 -1
- package/dist/clis/twitter/article.js +4 -1
- package/dist/clis/twitter/block.js +2 -1
- package/dist/clis/twitter/bookmark.js +2 -1
- package/dist/clis/twitter/bookmarks.js +3 -2
- package/dist/clis/twitter/delete.js +2 -1
- package/dist/clis/twitter/follow.js +2 -1
- package/dist/clis/twitter/followers.js +3 -2
- package/dist/clis/twitter/following.js +3 -2
- package/dist/clis/twitter/hide-reply.js +2 -1
- package/dist/clis/twitter/like.js +2 -1
- package/dist/clis/twitter/notifications.js +2 -1
- package/dist/clis/twitter/post.js +2 -1
- package/dist/clis/twitter/profile.js +5 -2
- package/dist/clis/twitter/reply-dm.js +2 -1
- package/dist/clis/twitter/reply.js +2 -1
- package/dist/clis/twitter/search.js +32 -13
- package/dist/clis/twitter/search.test.d.ts +1 -0
- package/dist/clis/twitter/search.test.js +156 -0
- package/dist/clis/twitter/thread.js +2 -2
- package/dist/clis/twitter/timeline.js +3 -2
- package/dist/clis/twitter/trending.js +3 -2
- package/dist/clis/twitter/unblock.js +2 -1
- package/dist/clis/twitter/unbookmark.js +2 -1
- package/dist/clis/twitter/unfollow.js +2 -1
- package/dist/clis/v2ex/daily.js +3 -2
- package/dist/clis/v2ex/me.js +3 -2
- package/dist/clis/v2ex/notifications.js +4 -4
- package/dist/clis/web/read.d.ts +16 -0
- package/dist/clis/web/read.js +202 -0
- package/dist/clis/weibo/comments.d.ts +1 -0
- package/dist/clis/weibo/comments.js +53 -0
- package/dist/clis/weibo/feed.d.ts +1 -0
- package/dist/clis/weibo/feed.js +56 -0
- package/dist/clis/weibo/hot.js +0 -1
- package/dist/clis/weibo/me.d.ts +1 -0
- package/dist/clis/weibo/me.js +76 -0
- package/dist/clis/weibo/post.d.ts +1 -0
- package/dist/clis/weibo/post.js +75 -0
- package/dist/clis/weibo/user.d.ts +1 -0
- package/dist/clis/weibo/user.js +63 -0
- package/dist/clis/weibo/utils.d.ts +6 -0
- package/dist/clis/weibo/utils.js +30 -0
- package/dist/clis/weread/search.js +3 -2
- package/dist/clis/xueqiu/danjuan-utils.d.ts +55 -0
- package/dist/clis/xueqiu/danjuan-utils.js +126 -0
- package/dist/clis/xueqiu/danjuan-utils.test.d.ts +1 -0
- package/dist/clis/xueqiu/danjuan-utils.test.js +41 -0
- package/dist/clis/xueqiu/fund-holdings.d.ts +1 -0
- package/dist/clis/xueqiu/fund-holdings.js +28 -0
- package/dist/clis/xueqiu/fund-snapshot.d.ts +1 -0
- package/dist/clis/xueqiu/fund-snapshot.js +25 -0
- package/dist/clis/xueqiu/search.yaml +2 -1
- package/dist/clis/yahoo-finance/quote.js +0 -1
- package/dist/clis/youtube/channel.d.ts +1 -0
- package/dist/clis/youtube/channel.js +150 -0
- package/dist/clis/youtube/comments.d.ts +1 -0
- package/dist/clis/youtube/comments.js +95 -0
- package/dist/clis/youtube/search.js +0 -1
- package/dist/clis/youtube/transcript.js +5 -4
- package/dist/clis/youtube/video.js +3 -2
- package/dist/clis/zhihu/search.yaml +2 -1
- package/dist/daemon.js +7 -3
- package/dist/discovery.js +11 -10
- package/dist/doctor.js +2 -1
- package/dist/download/index.d.ts +4 -12
- package/dist/download/index.js +33 -12
- package/dist/download/index.test.js +79 -2
- package/dist/download/media-download.js +4 -2
- package/dist/engine.test.js +76 -4
- package/dist/execution.d.ts +1 -9
- package/dist/execution.js +56 -46
- package/dist/explore.js +12 -111
- package/dist/external-clis.yaml +0 -25
- package/dist/external.js +7 -5
- package/dist/external.test.js +4 -0
- package/dist/generate.d.ts +0 -9
- package/dist/generate.js +4 -20
- package/dist/hooks.d.ts +46 -0
- package/dist/hooks.js +56 -0
- package/dist/hooks.test.d.ts +4 -0
- package/dist/hooks.test.js +92 -0
- package/dist/interceptor.js +70 -23
- package/dist/main.js +2 -0
- package/dist/output.js +12 -6
- package/dist/pipeline/executor.js +1 -1
- package/dist/pipeline/steps/browser.js +1 -3
- package/dist/pipeline/steps/download.js +42 -26
- package/dist/pipeline/steps/download.test.d.ts +1 -0
- package/dist/pipeline/steps/download.test.js +101 -0
- package/dist/pipeline/steps/fetch.js +40 -22
- package/dist/pipeline/steps/fetch.test.d.ts +1 -0
- package/dist/pipeline/steps/fetch.test.js +123 -0
- package/dist/pipeline/steps/transform.js +2 -6
- package/dist/pipeline/template.js +66 -52
- package/dist/pipeline/template.test.js +28 -0
- package/dist/pipeline/transform.test.js +18 -0
- package/dist/plugin.d.ts +40 -1
- package/dist/plugin.js +214 -17
- package/dist/plugin.test.d.ts +1 -1
- package/dist/plugin.test.js +219 -3
- package/dist/record.js +6 -98
- package/dist/registry-api.d.ts +2 -0
- package/dist/registry-api.js +1 -0
- package/dist/registry.d.ts +5 -2
- package/dist/registry.js +1 -2
- package/dist/runtime.d.ts +0 -1
- package/dist/runtime.js +14 -4
- package/dist/snapshotFormatter.d.ts +7 -14
- package/dist/snapshotFormatter.js +38 -78
- package/dist/utils.d.ts +9 -0
- package/dist/utils.js +29 -0
- package/dist/validate.js +3 -5
- package/dist/weread-search-regression.test.d.ts +1 -0
- package/dist/weread-search-regression.test.js +39 -0
- package/dist/yaml-schema.d.ts +26 -0
- package/dist/yaml-schema.js +5 -0
- package/docs/.vitepress/config.mts +16 -0
- package/docs/adapters/browser/dictionary.md +27 -0
- package/docs/adapters/browser/douyin.md +75 -0
- package/docs/adapters/browser/jd.md +27 -0
- package/docs/adapters/browser/linkedin.md +6 -0
- package/docs/adapters/browser/pixiv.md +92 -0
- package/docs/adapters/browser/twitter.md +6 -0
- package/docs/adapters/browser/web.md +30 -0
- package/docs/adapters/browser/xueqiu.md +27 -9
- package/docs/adapters/index.md +9 -2
- package/docs/comparison.md +125 -0
- package/docs/developer/contributing.md +21 -2
- package/docs/developer/testing.md +14 -8
- package/docs/developer/ts-adapter.md +18 -0
- package/docs/developer/yaml-adapter.md +16 -0
- package/docs/guide/plugins.md +10 -0
- package/docs/zh/guide/plugins.md +10 -0
- package/extension/dist/background.js +100 -35
- package/extension/manifest.json +6 -2
- package/extension/package.json +1 -1
- package/extension/popup.html +84 -0
- package/extension/popup.js +25 -0
- package/extension/src/background.test.ts +46 -1
- package/extension/src/background.ts +128 -34
- package/extension/src/cdp.ts +9 -9
- package/package.json +3 -2
- package/scripts/check-doc-coverage.sh +2 -0
- package/src/analysis.ts +170 -0
- package/src/browser/cdp.test.ts +66 -0
- package/src/browser/cdp.ts +59 -44
- package/src/browser/dom-snapshot.test.ts +42 -0
- package/src/browser/dom-snapshot.ts +56 -3
- package/src/browser/index.ts +2 -2
- package/src/browser/mcp.ts +2 -4
- package/src/browser/page.ts +34 -37
- package/src/browser/stealth.ts +24 -10
- package/src/browser.test.ts +2 -2
- package/src/build-manifest.test.ts +14 -0
- package/src/build-manifest.ts +13 -31
- package/src/cascade.ts +5 -3
- package/src/cli.ts +66 -34
- package/src/clis/_shared/desktop-commands.ts +121 -0
- package/src/clis/antigravity/serve.ts +6 -3
- package/src/clis/apple-podcasts/search.ts +2 -1
- package/src/clis/arxiv/search.ts +3 -3
- package/src/clis/bbc/news.ts +0 -1
- package/src/clis/bilibili/dynamic.test.ts +79 -0
- package/src/clis/bilibili/favorite.ts +5 -2
- package/src/clis/bilibili/following.ts +3 -2
- package/src/clis/bilibili/subtitle.ts +8 -7
- package/src/clis/bilibili/utils.ts +2 -2
- package/src/clis/boss/batchgreet.ts +1 -1
- package/src/clis/boss/chatlist.ts +1 -1
- package/src/clis/boss/chatmsg.ts +1 -1
- package/src/clis/boss/detail.ts +1 -1
- package/src/clis/boss/exchange.ts +1 -1
- package/src/clis/boss/greet.ts +1 -1
- package/src/clis/boss/invite.ts +1 -1
- package/src/clis/boss/joblist.ts +1 -1
- package/src/clis/boss/mark.ts +4 -3
- package/src/clis/boss/recommend.ts +1 -1
- package/src/clis/boss/resume.ts +1 -1
- package/src/clis/boss/search.ts +1 -1
- package/src/clis/boss/send.ts +5 -4
- package/src/clis/boss/stats.ts +1 -1
- package/src/clis/chatgpt/ask.ts +5 -0
- package/src/clis/chatgpt/new.ts +7 -2
- package/src/clis/chatgpt/read.ts +7 -2
- package/src/clis/chatgpt/send.ts +3 -2
- package/src/clis/chatgpt/status.ts +6 -1
- package/src/clis/chatwise/ask.ts +7 -2
- package/src/clis/chatwise/export.ts +2 -0
- package/src/clis/chatwise/history.ts +2 -0
- package/src/clis/chatwise/model.ts +7 -3
- package/src/clis/chatwise/new.ts +3 -20
- package/src/clis/chatwise/read.ts +2 -0
- package/src/clis/chatwise/screenshot.ts +3 -32
- package/src/clis/chatwise/send.ts +7 -2
- package/src/clis/chatwise/shared.ts +8 -0
- package/src/clis/chatwise/status.ts +3 -24
- package/src/clis/codex/ask.ts +5 -2
- package/src/clis/codex/dump.ts +2 -27
- package/src/clis/codex/new.ts +2 -28
- package/src/clis/codex/screenshot.ts +2 -32
- package/src/clis/codex/send.ts +5 -4
- package/src/clis/codex/status.ts +2 -24
- package/src/clis/ctrip/search.ts +0 -1
- package/src/clis/cursor/ask.ts +2 -1
- package/src/clis/cursor/composer.ts +2 -1
- package/src/clis/cursor/dump.ts +2 -27
- package/src/clis/cursor/new.ts +2 -20
- package/src/clis/cursor/read.ts +2 -1
- package/src/clis/cursor/screenshot.ts +1 -36
- package/src/clis/cursor/send.ts +2 -1
- package/src/clis/cursor/status.ts +2 -22
- package/src/clis/dictionary/examples.yaml +25 -0
- package/src/clis/dictionary/search.yaml +27 -0
- package/src/clis/dictionary/synonyms.yaml +25 -0
- package/src/clis/douban/book-hot.ts +1 -1
- package/src/clis/douban/movie-hot.ts +1 -1
- package/src/clis/douban/search.ts +1 -1
- package/src/clis/douban/utils.ts +165 -1
- package/src/clis/doubao/ask.ts +1 -1
- package/src/clis/doubao/new.ts +1 -1
- package/src/clis/doubao/read.ts +1 -1
- package/src/clis/doubao/send.ts +1 -1
- package/src/clis/doubao/status.ts +1 -1
- package/src/clis/doubao-app/ask.ts +1 -1
- package/src/clis/doubao-app/new.ts +1 -1
- package/src/clis/doubao-app/read.ts +1 -1
- package/src/clis/doubao-app/send.ts +1 -1
- package/src/clis/douyin/_shared/browser-fetch.test.ts +38 -0
- package/src/clis/douyin/_shared/browser-fetch.ts +45 -0
- package/src/clis/douyin/_shared/creation-id.test.ts +26 -0
- package/src/clis/douyin/_shared/creation-id.ts +8 -0
- package/src/clis/douyin/_shared/imagex-upload.test.ts +113 -0
- package/src/clis/douyin/_shared/imagex-upload.ts +76 -0
- package/src/clis/douyin/_shared/sts2.ts +20 -0
- package/src/clis/douyin/_shared/text-extra.test.ts +42 -0
- package/src/clis/douyin/_shared/text-extra.ts +33 -0
- package/src/clis/douyin/_shared/timing.test.ts +38 -0
- package/src/clis/douyin/_shared/timing.ts +22 -0
- package/src/clis/douyin/_shared/tos-upload-short-read.test.ts +102 -0
- package/src/clis/douyin/_shared/tos-upload.test.ts +281 -0
- package/src/clis/douyin/_shared/tos-upload.ts +444 -0
- package/src/clis/douyin/_shared/transcode.test.ts +117 -0
- package/src/clis/douyin/_shared/transcode.ts +78 -0
- package/src/clis/douyin/_shared/types.ts +29 -0
- package/src/clis/douyin/activities.test.ts +25 -0
- package/src/clis/douyin/activities.ts +23 -0
- package/src/clis/douyin/collections.test.ts +26 -0
- package/src/clis/douyin/collections.ts +25 -0
- package/src/clis/douyin/delete.test.ts +12 -0
- package/src/clis/douyin/delete.ts +20 -0
- package/src/clis/douyin/draft.test.ts +12 -0
- package/src/clis/douyin/draft.ts +282 -0
- package/src/clis/douyin/drafts.test.ts +12 -0
- package/src/clis/douyin/drafts.ts +27 -0
- package/src/clis/douyin/hashtag.test.ts +28 -0
- package/src/clis/douyin/hashtag.ts +56 -0
- package/src/clis/douyin/location.test.ts +26 -0
- package/src/clis/douyin/location.ts +27 -0
- package/src/clis/douyin/profile.test.ts +12 -0
- package/src/clis/douyin/profile.ts +37 -0
- package/src/clis/douyin/publish.test.ts +45 -0
- package/src/clis/douyin/publish.ts +340 -0
- package/src/clis/douyin/stats.test.ts +25 -0
- package/src/clis/douyin/stats.ts +30 -0
- package/src/clis/douyin/update.test.ts +12 -0
- package/src/clis/douyin/update.ts +43 -0
- package/src/clis/douyin/videos.test.ts +12 -0
- package/src/clis/douyin/videos.ts +49 -0
- package/src/clis/grok/ask.test.ts +25 -0
- package/src/clis/grok/ask.ts +25 -12
- package/src/clis/hackernews/search.yaml +1 -1
- package/src/clis/instagram/search.yaml +2 -1
- package/src/clis/jd/item.test.ts +35 -0
- package/src/clis/jd/item.ts +101 -0
- package/src/clis/jike/feed.ts +1 -1
- package/src/clis/jike/search.ts +1 -1
- package/src/clis/linkedin/search.ts +5 -4
- package/src/clis/linkedin/timeline.test.ts +99 -0
- package/src/clis/linkedin/timeline.ts +532 -0
- package/src/clis/linux-do/search.yaml +3 -1
- package/src/clis/medium/feed.ts +1 -1
- package/src/clis/medium/search.ts +2 -2
- package/src/clis/medium/user.ts +1 -1
- package/src/clis/medium/{shared.ts → utils.ts} +2 -1
- package/src/clis/pixiv/detail.yaml +49 -0
- package/src/clis/pixiv/download.test.ts +114 -0
- package/src/clis/pixiv/download.ts +91 -0
- package/src/clis/pixiv/illusts.test.ts +115 -0
- package/src/clis/pixiv/illusts.ts +78 -0
- package/src/clis/pixiv/ranking.yaml +53 -0
- package/src/clis/pixiv/search.test.ts +97 -0
- package/src/clis/pixiv/search.ts +53 -0
- package/src/clis/pixiv/test-utils.ts +29 -0
- package/src/clis/pixiv/user.yaml +46 -0
- package/src/clis/pixiv/utils.ts +62 -0
- package/src/clis/reddit/comment.ts +2 -1
- package/src/clis/reddit/read.test.ts +34 -0
- package/src/clis/reddit/read.ts +4 -3
- package/src/clis/reddit/save.ts +2 -1
- package/src/clis/reddit/saved.ts +6 -2
- package/src/clis/reddit/subscribe.ts +2 -1
- package/src/clis/reddit/upvote.ts +2 -1
- package/src/clis/reddit/upvoted.ts +6 -2
- package/src/clis/reuters/search.ts +0 -1
- package/src/clis/sinablog/article.ts +1 -1
- package/src/clis/sinablog/hot.ts +1 -1
- package/src/clis/sinablog/user.ts +1 -1
- package/src/clis/substack/feed.ts +1 -1
- package/src/clis/substack/publication.ts +1 -1
- package/src/clis/substack/search.ts +3 -2
- package/src/clis/substack/{shared.ts → utils.ts} +3 -2
- package/src/clis/tiktok/search.yaml +2 -1
- package/src/clis/twitter/accept.ts +2 -1
- package/src/clis/twitter/article.ts +3 -1
- package/src/clis/twitter/block.ts +2 -1
- package/src/clis/twitter/bookmark.ts +2 -1
- package/src/clis/twitter/bookmarks.ts +3 -2
- package/src/clis/twitter/delete.ts +2 -1
- package/src/clis/twitter/follow.ts +2 -1
- package/src/clis/twitter/followers.ts +3 -2
- package/src/clis/twitter/following.ts +3 -2
- package/src/clis/twitter/hide-reply.ts +2 -1
- package/src/clis/twitter/like.ts +2 -1
- package/src/clis/twitter/notifications.ts +2 -1
- package/src/clis/twitter/post.ts +2 -1
- package/src/clis/twitter/profile.ts +4 -2
- package/src/clis/twitter/reply-dm.ts +2 -1
- package/src/clis/twitter/reply.ts +2 -1
- package/src/clis/twitter/search.test.ts +180 -0
- package/src/clis/twitter/search.ts +40 -14
- package/src/clis/twitter/thread.ts +2 -2
- package/src/clis/twitter/timeline.ts +3 -2
- package/src/clis/twitter/trending.ts +3 -2
- package/src/clis/twitter/unblock.ts +2 -1
- package/src/clis/twitter/unbookmark.ts +2 -1
- package/src/clis/twitter/unfollow.ts +2 -1
- package/src/clis/v2ex/daily.ts +3 -2
- package/src/clis/v2ex/me.ts +3 -2
- package/src/clis/v2ex/notifications.ts +3 -4
- package/src/clis/web/read.ts +210 -0
- package/src/clis/weibo/comments.ts +54 -0
- package/src/clis/weibo/feed.ts +57 -0
- package/src/clis/weibo/hot.ts +0 -1
- package/src/clis/weibo/me.ts +77 -0
- package/src/clis/weibo/post.ts +77 -0
- package/src/clis/weibo/user.ts +64 -0
- package/src/clis/weibo/utils.ts +32 -0
- package/src/clis/weread/search.ts +3 -2
- package/src/clis/xueqiu/danjuan-utils.test.ts +49 -0
- package/src/clis/xueqiu/danjuan-utils.ts +176 -0
- package/src/clis/xueqiu/fund-holdings.ts +32 -0
- package/src/clis/xueqiu/fund-snapshot.ts +27 -0
- package/src/clis/xueqiu/search.yaml +2 -1
- package/src/clis/yahoo-finance/quote.ts +0 -1
- package/src/clis/youtube/channel.ts +155 -0
- package/src/clis/youtube/comments.ts +97 -0
- package/src/clis/youtube/search.ts +0 -1
- package/src/clis/youtube/transcript.ts +5 -4
- package/src/clis/youtube/video.ts +3 -2
- package/src/clis/zhihu/search.yaml +2 -1
- package/src/daemon.ts +5 -4
- package/src/discovery.ts +12 -34
- package/src/doctor.ts +3 -2
- package/src/download/index.test.ts +93 -2
- package/src/download/index.ts +44 -23
- package/src/download/media-download.ts +5 -3
- package/src/engine.test.ts +84 -3
- package/src/execution.ts +62 -46
- package/src/explore.ts +21 -90
- package/src/external-clis.yaml +0 -25
- package/src/external.test.ts +9 -0
- package/src/external.ts +12 -10
- package/src/generate.ts +4 -41
- package/src/hooks.test.ts +126 -0
- package/src/hooks.ts +90 -0
- package/src/interceptor.ts +73 -23
- package/src/main.ts +2 -0
- package/src/output.ts +14 -6
- package/src/pipeline/executor.ts +1 -1
- package/src/pipeline/steps/browser.ts +1 -3
- package/src/pipeline/steps/download.test.ts +136 -0
- package/src/pipeline/steps/download.ts +47 -34
- package/src/pipeline/steps/fetch.test.ts +179 -0
- package/src/pipeline/steps/fetch.ts +39 -23
- package/src/pipeline/steps/transform.ts +2 -6
- package/src/pipeline/template.test.ts +28 -0
- package/src/pipeline/template.ts +67 -79
- package/src/pipeline/transform.test.ts +20 -0
- package/src/plugin.test.ts +251 -3
- package/src/plugin.ts +265 -21
- package/src/record.ts +12 -84
- package/src/registry-api.ts +2 -0
- package/src/registry.ts +7 -4
- package/src/runtime.ts +14 -4
- package/src/snapshotFormatter.ts +43 -121
- package/src/utils.ts +39 -0
- package/src/validate.ts +3 -5
- package/src/weread-search-regression.test.ts +44 -0
- package/src/yaml-schema.ts +28 -0
- package/tests/e2e/browser-auth.test.ts +25 -0
- package/tests/e2e/browser-public-extended.test.ts +162 -0
- package/tests/e2e/browser-public.test.ts +7 -146
- package/tests/e2e/plugin-management.test.ts +137 -0
- package/tests/e2e/public-commands.test.ts +34 -1
- package/vitest.config.ts +33 -8
- package/.github/workflows/pkg-pr-new.yml +0 -30
- package/dist/clis/douban/shared.d.ts +0 -4
- package/dist/clis/douban/shared.js +0 -155
- package/src/clis/douban/shared.ts +0 -165
- /package/dist/clis/boss/{common.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/boss/{common.js → utils.js} +0 -0
- /package/dist/clis/doubao/{common.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/doubao/{common.js → utils.js} +0 -0
- /package/dist/clis/doubao-app/{common.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/doubao-app/{common.js → utils.js} +0 -0
- /package/dist/clis/jike/{shared.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/jike/{shared.js → utils.js} +0 -0
- /package/dist/clis/medium/{shared.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/sinablog/{shared.d.ts → utils.d.ts} +0 -0
- /package/dist/clis/sinablog/{shared.js → utils.js} +0 -0
- /package/dist/clis/substack/{shared.d.ts → utils.d.ts} +0 -0
- /package/src/clis/boss/{common.ts → utils.ts} +0 -0
- /package/src/clis/doubao/{common.ts → utils.ts} +0 -0
- /package/src/clis/doubao-app/{common.ts → utils.ts} +0 -0
- /package/src/clis/jike/{shared.ts → utils.ts} +0 -0
- /package/src/clis/sinablog/{shared.ts → utils.ts} +0 -0
package/src/clis/douban/utils.ts
CHANGED
|
@@ -1,9 +1,173 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Douban
|
|
2
|
+
* Douban adapter utilities.
|
|
3
3
|
*/
|
|
4
4
|
|
|
5
|
+
import { CliError } from '../../errors.js';
|
|
5
6
|
import type { IPage } from '../../types.js';
|
|
6
7
|
|
|
8
|
+
function clampLimit(limit: number): number {
|
|
9
|
+
return Math.max(1, Math.min(limit || 20, 50));
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
async function ensureDoubanReady(page: IPage): Promise<void> {
|
|
13
|
+
const state = await page.evaluate(`
|
|
14
|
+
(() => {
|
|
15
|
+
const title = (document.title || '').trim();
|
|
16
|
+
const href = (location.href || '').trim();
|
|
17
|
+
const blocked = href.includes('sec.douban.com') || /登录跳转/.test(title) || /异常请求/.test(document.body?.innerText || '');
|
|
18
|
+
return { blocked, title, href };
|
|
19
|
+
})()
|
|
20
|
+
`);
|
|
21
|
+
if (state?.blocked) {
|
|
22
|
+
throw new CliError(
|
|
23
|
+
'AUTH_REQUIRED',
|
|
24
|
+
'Douban requires a logged-in browser session before these commands can load data.',
|
|
25
|
+
'Please sign in to douban.com in the browser that opencli reuses, then rerun the command.',
|
|
26
|
+
);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export async function loadDoubanBookHot(page: IPage, limit: number): Promise<any[]> {
|
|
31
|
+
const safeLimit = clampLimit(limit);
|
|
32
|
+
await page.goto('https://book.douban.com/chart');
|
|
33
|
+
await page.wait(4);
|
|
34
|
+
await ensureDoubanReady(page);
|
|
35
|
+
const data = await page.evaluate(`
|
|
36
|
+
(() => {
|
|
37
|
+
const normalize = (value) => (value || '').replace(/\\s+/g, ' ').trim();
|
|
38
|
+
const books = [];
|
|
39
|
+
for (const el of Array.from(document.querySelectorAll('.media.clearfix'))) {
|
|
40
|
+
try {
|
|
41
|
+
const titleEl = el.querySelector('h2 a[href*="/subject/"]');
|
|
42
|
+
const title = normalize(titleEl?.textContent);
|
|
43
|
+
let url = titleEl?.getAttribute('href') || '';
|
|
44
|
+
if (!title || !url) continue;
|
|
45
|
+
if (!url.startsWith('http')) url = 'https://book.douban.com' + url;
|
|
46
|
+
|
|
47
|
+
const info = normalize(el.querySelector('.subject-abstract, .pl, .pub')?.textContent);
|
|
48
|
+
const infoParts = info.split('/').map((part) => part.trim()).filter(Boolean);
|
|
49
|
+
const ratingText = normalize(el.querySelector('.subject-rating .font-small, .rating_nums, .rating')?.textContent);
|
|
50
|
+
const quote = Array.from(el.querySelectorAll('.subject-tags .tag'))
|
|
51
|
+
.map((node) => normalize(node.textContent))
|
|
52
|
+
.filter(Boolean)
|
|
53
|
+
.join(' / ');
|
|
54
|
+
|
|
55
|
+
books.push({
|
|
56
|
+
rank: parseInt(normalize(el.querySelector('.green-num-box')?.textContent), 10) || books.length + 1,
|
|
57
|
+
title,
|
|
58
|
+
rating: parseFloat(ratingText) || 0,
|
|
59
|
+
quote,
|
|
60
|
+
author: infoParts[0] || '',
|
|
61
|
+
publisher: infoParts.find((part) => /出版社|出版公司|Press/i.test(part)) || infoParts[2] || '',
|
|
62
|
+
year: infoParts.find((part) => /\\d{4}(?:-\\d{1,2})?/.test(part))?.match(/\\d{4}/)?.[0] || '',
|
|
63
|
+
price: infoParts.find((part) => /元|USD|\\$|¥/.test(part)) || '',
|
|
64
|
+
url,
|
|
65
|
+
cover: el.querySelector('img')?.getAttribute('src') || '',
|
|
66
|
+
});
|
|
67
|
+
} catch {}
|
|
68
|
+
}
|
|
69
|
+
return books.slice(0, ${safeLimit});
|
|
70
|
+
})()
|
|
71
|
+
`);
|
|
72
|
+
return Array.isArray(data) ? data : [];
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
export async function loadDoubanMovieHot(page: IPage, limit: number): Promise<any[]> {
|
|
76
|
+
const safeLimit = clampLimit(limit);
|
|
77
|
+
await page.goto('https://movie.douban.com/chart');
|
|
78
|
+
await page.wait(4);
|
|
79
|
+
await ensureDoubanReady(page);
|
|
80
|
+
const data = await page.evaluate(`
|
|
81
|
+
(() => {
|
|
82
|
+
const normalize = (value) => (value || '').replace(/\\s+/g, ' ').trim();
|
|
83
|
+
const results = [];
|
|
84
|
+
for (const el of Array.from(document.querySelectorAll('.item'))) {
|
|
85
|
+
const titleEl = el.querySelector('.pl2 a');
|
|
86
|
+
const title = normalize(titleEl?.textContent);
|
|
87
|
+
let url = titleEl?.getAttribute('href') || '';
|
|
88
|
+
if (!title || !url) continue;
|
|
89
|
+
if (!url.startsWith('http')) url = 'https://movie.douban.com' + url;
|
|
90
|
+
|
|
91
|
+
const info = normalize(el.querySelector('.pl2 p')?.textContent);
|
|
92
|
+
const infoParts = info.split('/').map((part) => part.trim()).filter(Boolean);
|
|
93
|
+
const releaseIndex = (() => {
|
|
94
|
+
for (let i = infoParts.length - 1; i >= 0; i -= 1) {
|
|
95
|
+
if (/\\d{4}-\\d{2}-\\d{2}|\\d{4}\\/\\d{2}\\/\\d{2}/.test(infoParts[i])) return i;
|
|
96
|
+
}
|
|
97
|
+
return -1;
|
|
98
|
+
})();
|
|
99
|
+
const directorPart = releaseIndex >= 1 ? infoParts[releaseIndex - 1] : '';
|
|
100
|
+
const regionPart = releaseIndex >= 2 ? infoParts[releaseIndex - 2] : '';
|
|
101
|
+
const yearMatch = info.match(/\\b(19|20)\\d{2}\\b/);
|
|
102
|
+
results.push({
|
|
103
|
+
rank: results.length + 1,
|
|
104
|
+
title,
|
|
105
|
+
rating: parseFloat(normalize(el.querySelector('.rating_nums')?.textContent)) || 0,
|
|
106
|
+
quote: normalize(el.querySelector('.inq')?.textContent),
|
|
107
|
+
director: directorPart.replace(/^导演:\\s*/, ''),
|
|
108
|
+
year: yearMatch?.[0] || '',
|
|
109
|
+
region: regionPart,
|
|
110
|
+
url,
|
|
111
|
+
cover: el.querySelector('img')?.getAttribute('src') || '',
|
|
112
|
+
});
|
|
113
|
+
if (results.length >= ${safeLimit}) break;
|
|
114
|
+
}
|
|
115
|
+
return results;
|
|
116
|
+
})()
|
|
117
|
+
`);
|
|
118
|
+
return Array.isArray(data) ? data : [];
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
export async function searchDouban(page: IPage, type: string, keyword: string, limit: number): Promise<any[]> {
|
|
122
|
+
const safeLimit = clampLimit(limit);
|
|
123
|
+
await page.goto(`https://search.douban.com/${encodeURIComponent(type)}/subject_search?search_text=${encodeURIComponent(keyword)}`);
|
|
124
|
+
await page.wait(2);
|
|
125
|
+
await ensureDoubanReady(page);
|
|
126
|
+
const data = await page.evaluate(`
|
|
127
|
+
(async () => {
|
|
128
|
+
const type = ${JSON.stringify(type)};
|
|
129
|
+
const normalize = (value) => (value || '').replace(/\\s+/g, ' ').trim();
|
|
130
|
+
const seen = new Set();
|
|
131
|
+
const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
132
|
+
|
|
133
|
+
for (let i = 0; i < 20; i += 1) {
|
|
134
|
+
if (document.querySelector('.item-root .title-text, .item-root .title a')) break;
|
|
135
|
+
await sleep(300);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
const items = Array.from(document.querySelectorAll('.item-root'));
|
|
139
|
+
|
|
140
|
+
const results = [];
|
|
141
|
+
for (const el of items) {
|
|
142
|
+
const titleEl = el.querySelector('.title-text, .title a, a[title]');
|
|
143
|
+
const title = normalize(titleEl?.textContent) || normalize(titleEl?.getAttribute('title'));
|
|
144
|
+
let url = titleEl?.getAttribute('href') || '';
|
|
145
|
+
if (!title || !url) continue;
|
|
146
|
+
if (!url.startsWith('http')) url = 'https://search.douban.com' + url;
|
|
147
|
+
if (!url.includes('/subject/') || seen.has(url)) continue;
|
|
148
|
+
seen.add(url);
|
|
149
|
+
const ratingText = normalize(el.querySelector('.rating_nums')?.textContent);
|
|
150
|
+
const abstract = normalize(
|
|
151
|
+
el.querySelector('.meta.abstract, .meta, .abstract, p')?.textContent,
|
|
152
|
+
);
|
|
153
|
+
results.push({
|
|
154
|
+
rank: results.length + 1,
|
|
155
|
+
id: url.match(/subject\\/(\\d+)/)?.[1] || '',
|
|
156
|
+
type,
|
|
157
|
+
title,
|
|
158
|
+
rating: ratingText.includes('.') ? parseFloat(ratingText) : 0,
|
|
159
|
+
abstract: abstract.slice(0, 100) + (abstract.length > 100 ? '...' : ''),
|
|
160
|
+
url,
|
|
161
|
+
cover: el.querySelector('img')?.getAttribute('src') || '',
|
|
162
|
+
});
|
|
163
|
+
if (results.length >= ${safeLimit}) break;
|
|
164
|
+
}
|
|
165
|
+
return results;
|
|
166
|
+
})()
|
|
167
|
+
`);
|
|
168
|
+
return Array.isArray(data) ? data : [];
|
|
169
|
+
}
|
|
170
|
+
|
|
7
171
|
/**
|
|
8
172
|
* Get current user's Douban ID from movie.douban.com/mine page
|
|
9
173
|
*/
|
package/src/clis/doubao/ask.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
2
|
import type { IPage } from '../../types.js';
|
|
3
|
-
import { DOUBAO_DOMAIN, getDoubaoTranscriptLines, getDoubaoVisibleTurns, sendDoubaoMessage, waitForDoubaoResponse } from './
|
|
3
|
+
import { DOUBAO_DOMAIN, getDoubaoTranscriptLines, getDoubaoVisibleTurns, sendDoubaoMessage, waitForDoubaoResponse } from './utils.js';
|
|
4
4
|
|
|
5
5
|
export const askCommand = cli({
|
|
6
6
|
site: 'doubao',
|
package/src/clis/doubao/new.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
2
|
import type { IPage } from '../../types.js';
|
|
3
|
-
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, startNewDoubaoChat } from './
|
|
3
|
+
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, startNewDoubaoChat } from './utils.js';
|
|
4
4
|
|
|
5
5
|
export const newCommand = cli({
|
|
6
6
|
site: 'doubao',
|
package/src/clis/doubao/read.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
2
|
import type { IPage } from '../../types.js';
|
|
3
|
-
import { DOUBAO_DOMAIN, getDoubaoVisibleTurns } from './
|
|
3
|
+
import { DOUBAO_DOMAIN, getDoubaoVisibleTurns } from './utils.js';
|
|
4
4
|
|
|
5
5
|
export const readCommand = cli({
|
|
6
6
|
site: 'doubao',
|
package/src/clis/doubao/send.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
2
|
import type { IPage } from '../../types.js';
|
|
3
|
-
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, sendDoubaoMessage } from './
|
|
3
|
+
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, sendDoubaoMessage } from './utils.js';
|
|
4
4
|
|
|
5
5
|
export const sendCommand = cli({
|
|
6
6
|
site: 'doubao',
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
2
|
import type { IPage } from '../../types.js';
|
|
3
|
-
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, getDoubaoPageState } from './
|
|
3
|
+
import { DOUBAO_DOMAIN, DOUBAO_CHAT_URL, getDoubaoPageState } from './utils.js';
|
|
4
4
|
|
|
5
5
|
export const statusCommand = cli({
|
|
6
6
|
site: 'doubao',
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { cli, Strategy } from '../../registry.js';
|
|
2
|
-
import { SEL, injectTextScript, clickSendScript, pollResponseScript } from './
|
|
2
|
+
import { SEL, injectTextScript, clickSendScript, pollResponseScript } from './utils.js';
|
|
3
3
|
|
|
4
4
|
export const askCommand = cli({
|
|
5
5
|
site: 'doubao-app',
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import type { IPage } from '../../../types.js';
|
|
3
|
+
import { browserFetch } from './browser-fetch.js';
|
|
4
|
+
|
|
5
|
+
function makePage(result: unknown): IPage {
|
|
6
|
+
return {
|
|
7
|
+
goto: vi.fn(), evaluate: vi.fn().mockResolvedValue(result),
|
|
8
|
+
getCookies: vi.fn(), snapshot: vi.fn(), click: vi.fn(),
|
|
9
|
+
typeText: vi.fn(), pressKey: vi.fn(), scrollTo: vi.fn(),
|
|
10
|
+
getFormState: vi.fn(), wait: vi.fn(), tabs: vi.fn(),
|
|
11
|
+
closeTab: vi.fn(), newTab: vi.fn(), selectTab: vi.fn(),
|
|
12
|
+
networkRequests: vi.fn(), consoleMessages: vi.fn(),
|
|
13
|
+
scroll: vi.fn(), autoScroll: vi.fn(),
|
|
14
|
+
installInterceptor: vi.fn(), getInterceptedRequests: vi.fn(),
|
|
15
|
+
screenshot: vi.fn(),
|
|
16
|
+
} as unknown as IPage;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
describe('browserFetch', () => {
|
|
20
|
+
it('returns parsed JSON on success', async () => {
|
|
21
|
+
const page = makePage({ status_code: 0, data: { ak: 'KEY' } });
|
|
22
|
+
const result = await browserFetch(page, 'GET', 'https://creator.douyin.com/api/test');
|
|
23
|
+
expect(result).toEqual({ status_code: 0, data: { ak: 'KEY' } });
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it('throws when status_code is non-zero', async () => {
|
|
27
|
+
const page = makePage({ status_code: 8, message: 'fail' });
|
|
28
|
+
await expect(
|
|
29
|
+
browserFetch(page, 'GET', 'https://creator.douyin.com/api/test')
|
|
30
|
+
).rejects.toThrow('Douyin API error 8');
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it('returns result even when no status_code field', async () => {
|
|
34
|
+
const page = makePage({ some_field: 'value' });
|
|
35
|
+
const result = await browserFetch(page, 'GET', 'https://creator.douyin.com/api/test');
|
|
36
|
+
expect(result).toEqual({ some_field: 'value' });
|
|
37
|
+
});
|
|
38
|
+
});
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import type { IPage } from '../../../types.js';
|
|
2
|
+
import { CommandExecutionError } from '../../../errors.js';
|
|
3
|
+
|
|
4
|
+
export interface FetchOptions {
|
|
5
|
+
body?: unknown;
|
|
6
|
+
headers?: Record<string, string>;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Execute a fetch() call inside the Chrome browser context via page.evaluate.
|
|
11
|
+
* This ensures a_bogus signing and cookies are handled automatically by the browser.
|
|
12
|
+
*/
|
|
13
|
+
export async function browserFetch(
|
|
14
|
+
page: IPage,
|
|
15
|
+
method: 'GET' | 'POST',
|
|
16
|
+
url: string,
|
|
17
|
+
options: FetchOptions = {}
|
|
18
|
+
): Promise<unknown> {
|
|
19
|
+
const js = `
|
|
20
|
+
(async () => {
|
|
21
|
+
const res = await fetch(${JSON.stringify(url)}, {
|
|
22
|
+
method: ${JSON.stringify(method)},
|
|
23
|
+
credentials: 'include',
|
|
24
|
+
headers: {
|
|
25
|
+
'Content-Type': 'application/json',
|
|
26
|
+
...${JSON.stringify(options.headers ?? {})}
|
|
27
|
+
},
|
|
28
|
+
${options.body ? `body: JSON.stringify(${JSON.stringify(options.body)}),` : ''}
|
|
29
|
+
});
|
|
30
|
+
return res.json();
|
|
31
|
+
})()
|
|
32
|
+
`;
|
|
33
|
+
|
|
34
|
+
const result = await page.evaluate(js);
|
|
35
|
+
|
|
36
|
+
if (result && typeof result === 'object' && 'status_code' in result) {
|
|
37
|
+
const code = (result as { status_code: number }).status_code;
|
|
38
|
+
if (code !== 0) {
|
|
39
|
+
const msg = (result as { status_msg?: string }).status_msg ?? 'unknown error';
|
|
40
|
+
throw new CommandExecutionError(`Douyin API error ${code}: ${msg}`);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
return result;
|
|
45
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { generateCreationId } from './creation-id.js';
|
|
3
|
+
|
|
4
|
+
describe('generateCreationId', () => {
|
|
5
|
+
it('starts with "pin"', () => {
|
|
6
|
+
expect(generateCreationId()).toMatch(/^pin/);
|
|
7
|
+
});
|
|
8
|
+
|
|
9
|
+
it('has 4 random lowercase-alphanumeric chars after "pin"', () => {
|
|
10
|
+
expect(generateCreationId()).toMatch(/^pin[a-z0-9]{4}/);
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it('ends with a numeric timestamp (ms)', () => {
|
|
14
|
+
const before = Date.now();
|
|
15
|
+
const id = generateCreationId();
|
|
16
|
+
const after = Date.now();
|
|
17
|
+
const ts = parseInt(id.replace(/^pin[a-z0-9]{4}/, ''), 10);
|
|
18
|
+
expect(ts).toBeGreaterThanOrEqual(before);
|
|
19
|
+
expect(ts).toBeLessThanOrEqual(after);
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it('generates unique IDs', () => {
|
|
23
|
+
const ids = new Set(Array.from({ length: 100 }, generateCreationId));
|
|
24
|
+
expect(ids.size).toBe(100);
|
|
25
|
+
});
|
|
26
|
+
});
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import * as fs from 'node:fs';
|
|
2
|
+
import * as os from 'node:os';
|
|
3
|
+
import * as path from 'node:path';
|
|
4
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
5
|
+
import { CommandExecutionError } from '../../../errors.js';
|
|
6
|
+
import { imagexUpload } from './imagex-upload.js';
|
|
7
|
+
import type { ImageXUploadInfo } from './imagex-upload.js';
|
|
8
|
+
|
|
9
|
+
// ── Helpers ──────────────────────────────────────────────────────────────────
|
|
10
|
+
|
|
11
|
+
function makeTempImage(ext = '.jpg'): string {
|
|
12
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), 'imagex-test-'));
|
|
13
|
+
const filePath = path.join(dir, `cover${ext}`);
|
|
14
|
+
fs.writeFileSync(filePath, Buffer.from([0xff, 0xd8, 0xff, 0xe0])); // minimal JPEG header bytes
|
|
15
|
+
return filePath;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const FAKE_UPLOAD_INFO: ImageXUploadInfo = {
|
|
19
|
+
upload_url: 'https://imagex.bytedance.com/upload/presigned/fake',
|
|
20
|
+
store_uri: 'tos-cn-i-alisg.example.com/cover/abc123',
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
// ── Tests ─────────────────────────────────────────────────────────────────────
|
|
24
|
+
|
|
25
|
+
describe('imagexUpload', () => {
|
|
26
|
+
let imagePath: string;
|
|
27
|
+
|
|
28
|
+
beforeEach(() => {
|
|
29
|
+
imagePath = makeTempImage('.jpg');
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
afterEach(() => {
|
|
33
|
+
// Clean up temp files
|
|
34
|
+
try {
|
|
35
|
+
fs.unlinkSync(imagePath);
|
|
36
|
+
fs.rmdirSync(path.dirname(imagePath));
|
|
37
|
+
} catch {
|
|
38
|
+
// ignore cleanup errors
|
|
39
|
+
}
|
|
40
|
+
vi.restoreAllMocks();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it('throws CommandExecutionError when image file does not exist', async () => {
|
|
44
|
+
await expect(
|
|
45
|
+
imagexUpload('/nonexistent/path/cover.jpg', FAKE_UPLOAD_INFO),
|
|
46
|
+
).rejects.toThrow(CommandExecutionError);
|
|
47
|
+
|
|
48
|
+
await expect(
|
|
49
|
+
imagexUpload('/nonexistent/path/cover.jpg', FAKE_UPLOAD_INFO),
|
|
50
|
+
).rejects.toThrow('Cover image file not found');
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it('PUTs the image and returns store_uri on success', async () => {
|
|
54
|
+
const mockFetch = vi.fn().mockResolvedValue({
|
|
55
|
+
ok: true,
|
|
56
|
+
status: 200,
|
|
57
|
+
text: vi.fn().mockResolvedValue(''),
|
|
58
|
+
});
|
|
59
|
+
vi.stubGlobal('fetch', mockFetch);
|
|
60
|
+
|
|
61
|
+
const result = await imagexUpload(imagePath, FAKE_UPLOAD_INFO);
|
|
62
|
+
|
|
63
|
+
expect(result).toBe(FAKE_UPLOAD_INFO.store_uri);
|
|
64
|
+
expect(mockFetch).toHaveBeenCalledOnce();
|
|
65
|
+
const [url, init] = mockFetch.mock.calls[0] as [string, RequestInit];
|
|
66
|
+
expect(url).toBe(FAKE_UPLOAD_INFO.upload_url);
|
|
67
|
+
expect(init.method).toBe('PUT');
|
|
68
|
+
expect((init.headers as Record<string, string>)['Content-Type']).toBe(
|
|
69
|
+
'image/jpeg',
|
|
70
|
+
);
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it('uses image/png Content-Type for .png files', async () => {
|
|
74
|
+
const pngPath = makeTempImage('.png');
|
|
75
|
+
const mockFetch = vi.fn().mockResolvedValue({
|
|
76
|
+
ok: true,
|
|
77
|
+
status: 200,
|
|
78
|
+
text: vi.fn().mockResolvedValue(''),
|
|
79
|
+
});
|
|
80
|
+
vi.stubGlobal('fetch', mockFetch);
|
|
81
|
+
|
|
82
|
+
try {
|
|
83
|
+
await imagexUpload(pngPath, FAKE_UPLOAD_INFO);
|
|
84
|
+
const [, init] = mockFetch.mock.calls[0] as [string, RequestInit];
|
|
85
|
+
expect((init.headers as Record<string, string>)['Content-Type']).toBe(
|
|
86
|
+
'image/png',
|
|
87
|
+
);
|
|
88
|
+
} finally {
|
|
89
|
+
try {
|
|
90
|
+
fs.unlinkSync(pngPath);
|
|
91
|
+
fs.rmdirSync(path.dirname(pngPath));
|
|
92
|
+
} catch {
|
|
93
|
+
// ignore
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('throws CommandExecutionError on non-2xx PUT response', async () => {
|
|
99
|
+
const mockFetch = vi.fn().mockResolvedValue({
|
|
100
|
+
ok: false,
|
|
101
|
+
status: 403,
|
|
102
|
+
text: vi.fn().mockResolvedValue('Forbidden'),
|
|
103
|
+
});
|
|
104
|
+
vi.stubGlobal('fetch', mockFetch);
|
|
105
|
+
|
|
106
|
+
await expect(imagexUpload(imagePath, FAKE_UPLOAD_INFO)).rejects.toThrow(
|
|
107
|
+
CommandExecutionError,
|
|
108
|
+
);
|
|
109
|
+
await expect(imagexUpload(imagePath, FAKE_UPLOAD_INFO)).rejects.toThrow(
|
|
110
|
+
'ImageX upload failed with status 403',
|
|
111
|
+
);
|
|
112
|
+
});
|
|
113
|
+
});
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ImageX cover image uploader.
|
|
3
|
+
*
|
|
4
|
+
* Uploads a JPEG/PNG image to ByteDance ImageX via a pre-signed PUT URL
|
|
5
|
+
* obtained from the Douyin "apply cover upload" API.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import * as fs from 'node:fs';
|
|
9
|
+
import * as path from 'node:path';
|
|
10
|
+
import { CommandExecutionError } from '../../../errors.js';
|
|
11
|
+
|
|
12
|
+
export interface ImageXUploadInfo {
|
|
13
|
+
/** Pre-signed PUT target URL (provided by the apply cover upload API) */
|
|
14
|
+
upload_url: string;
|
|
15
|
+
/** Image URI to use in create_v2 (returned from the apply step) */
|
|
16
|
+
store_uri: string;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Detect MIME type from file extension.
|
|
21
|
+
* Falls back to image/jpeg for unknown extensions.
|
|
22
|
+
*/
|
|
23
|
+
function detectContentType(filePath: string): string {
|
|
24
|
+
const ext = path.extname(filePath).toLowerCase();
|
|
25
|
+
switch (ext) {
|
|
26
|
+
case '.png':
|
|
27
|
+
return 'image/png';
|
|
28
|
+
case '.gif':
|
|
29
|
+
return 'image/gif';
|
|
30
|
+
case '.webp':
|
|
31
|
+
return 'image/webp';
|
|
32
|
+
default:
|
|
33
|
+
return 'image/jpeg';
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Upload a cover image to ByteDance ImageX via a pre-signed PUT URL.
|
|
39
|
+
*
|
|
40
|
+
* @param imagePath - Local file path to the image (JPEG/PNG/etc.)
|
|
41
|
+
* @param uploadInfo - Upload URL and store_uri from the apply cover upload API
|
|
42
|
+
* @returns The store_uri (= image_uri for use in create_v2)
|
|
43
|
+
*/
|
|
44
|
+
export async function imagexUpload(
|
|
45
|
+
imagePath: string,
|
|
46
|
+
uploadInfo: ImageXUploadInfo,
|
|
47
|
+
): Promise<string> {
|
|
48
|
+
if (!fs.existsSync(imagePath)) {
|
|
49
|
+
throw new CommandExecutionError(
|
|
50
|
+
`Cover image file not found: ${imagePath}`,
|
|
51
|
+
'Ensure the file path is correct and accessible.',
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const imageBuffer = fs.readFileSync(imagePath);
|
|
56
|
+
const contentType = detectContentType(imagePath);
|
|
57
|
+
|
|
58
|
+
const res = await fetch(uploadInfo.upload_url, {
|
|
59
|
+
method: 'PUT',
|
|
60
|
+
headers: {
|
|
61
|
+
'Content-Type': contentType,
|
|
62
|
+
'Content-Length': String(imageBuffer.byteLength),
|
|
63
|
+
},
|
|
64
|
+
body: imageBuffer as unknown as BodyInit,
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
if (!res.ok) {
|
|
68
|
+
const body = await res.text().catch(() => '');
|
|
69
|
+
throw new CommandExecutionError(
|
|
70
|
+
`ImageX upload failed with status ${res.status}: ${body}`,
|
|
71
|
+
'Check that the upload URL is valid and has not expired.',
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
return uploadInfo.store_uri;
|
|
76
|
+
}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { IPage } from '../../../types.js';
|
|
2
|
+
import type { Sts2Credentials } from './types.js';
|
|
3
|
+
import { AuthRequiredError } from '../../../errors.js';
|
|
4
|
+
|
|
5
|
+
const STS2_URL =
|
|
6
|
+
'https://creator.douyin.com/aweme/mid/video/sts2/?scene=web&aid=1128&cookie_enabled=true&device_platform=web';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Fetch STS2 temporary credentials from the creator center.
|
|
10
|
+
* These are used to authenticate Node.js-side TOS multipart uploads.
|
|
11
|
+
* Returns: { access_key_id, secret_access_key, session_token, expired_time }
|
|
12
|
+
*/
|
|
13
|
+
export async function getSts2Credentials(page: IPage): Promise<Sts2Credentials> {
|
|
14
|
+
const js = `fetch(${JSON.stringify(STS2_URL)}, { credentials: 'include' }).then(r => r.json())`;
|
|
15
|
+
const res = await page.evaluate(js) as { data: Sts2Credentials };
|
|
16
|
+
if (!res?.data?.access_key_id) {
|
|
17
|
+
throw new AuthRequiredError('creator.douyin.com', 'STS2 credentials missing');
|
|
18
|
+
}
|
|
19
|
+
return res.data;
|
|
20
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { parseTextExtra, extractHashtagNames, type HashtagInfo } from './text-extra.js';
|
|
3
|
+
|
|
4
|
+
describe('parseTextExtra', () => {
|
|
5
|
+
it('returns empty array for text with no hashtags', () => {
|
|
6
|
+
const result = parseTextExtra('普通文本内容', []);
|
|
7
|
+
expect(result).toEqual([]);
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
it('produces type-1 entry for each hashtag', () => {
|
|
11
|
+
const hashtags: HashtagInfo[] = [
|
|
12
|
+
{ name: '话题', id: 12345, start: 5, end: 8 },
|
|
13
|
+
];
|
|
14
|
+
const result = parseTextExtra('普通文本 #话题', hashtags);
|
|
15
|
+
expect(result).toHaveLength(1);
|
|
16
|
+
expect(result[0]).toMatchObject({
|
|
17
|
+
type: 1,
|
|
18
|
+
hashtag_name: '话题',
|
|
19
|
+
hashtag_id: 12345,
|
|
20
|
+
start: 5,
|
|
21
|
+
end: 8,
|
|
22
|
+
});
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it('sets hashtag_id to 0 when not found', () => {
|
|
26
|
+
const hashtags: HashtagInfo[] = [
|
|
27
|
+
{ name: '未知话题', id: 0, start: 0, end: 5 },
|
|
28
|
+
];
|
|
29
|
+
const result = parseTextExtra('#未知话题', hashtags);
|
|
30
|
+
expect(result[0].hashtag_id).toBe(0);
|
|
31
|
+
});
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
describe('extractHashtagNames', () => {
|
|
35
|
+
it('extracts hashtag names from text', () => {
|
|
36
|
+
expect(extractHashtagNames('hello #foo and #bar')).toEqual(['foo', 'bar']);
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it('returns empty array when no hashtags', () => {
|
|
40
|
+
expect(extractHashtagNames('no hashtags here')).toEqual([]);
|
|
41
|
+
});
|
|
42
|
+
});
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
export interface HashtagInfo {
|
|
2
|
+
name: string;
|
|
3
|
+
id: number;
|
|
4
|
+
start: number;
|
|
5
|
+
end: number;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export interface TextExtraItem {
|
|
9
|
+
type: number;
|
|
10
|
+
hashtag_id: number;
|
|
11
|
+
hashtag_name: string;
|
|
12
|
+
start: number;
|
|
13
|
+
end: number;
|
|
14
|
+
caption_start: number;
|
|
15
|
+
caption_end: number;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export function parseTextExtra(_text: string, hashtags: HashtagInfo[]): TextExtraItem[] {
|
|
19
|
+
return hashtags.map((h) => ({
|
|
20
|
+
type: 1,
|
|
21
|
+
hashtag_id: h.id,
|
|
22
|
+
hashtag_name: h.name,
|
|
23
|
+
start: h.start,
|
|
24
|
+
end: h.end,
|
|
25
|
+
caption_start: 0,
|
|
26
|
+
caption_end: h.end - h.start,
|
|
27
|
+
}));
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/** Extract hashtag names from text (e.g. "#话题" → ["话题"]) */
|
|
31
|
+
export function extractHashtagNames(text: string): string[] {
|
|
32
|
+
return [...text.matchAll(/#([^\s#]+)/g)].map((m) => m[1]);
|
|
33
|
+
}
|