@jackwener/opencli 1.7.12 → 1.7.13
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 +12194 -6843
- 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/reply-dm.js +1 -1
- package/clis/twitter/reply.test.js +1 -29
- 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/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/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 +150 -0
- package/dist/src/commanderAdapter.js +0 -5
- 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 +23 -2
- package/dist/src/help.js +41 -19
- 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/tiktok/user.js
CHANGED
|
@@ -1,10 +1,201 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
// Get recent videos from a TikTok user's public profile via page-context APIs.
|
|
2
|
+
//
|
|
3
|
+
// This is the `tiktok user` counterpart to the Phase-3 read refactor: resolve
|
|
4
|
+
// the profile `secUid`, page `/api/post/item_list/`, and use exact-author search
|
|
5
|
+
// only as a lower-authority fallback. All requests run inside the live page so
|
|
6
|
+
// cookies + msToken are forwarded by the browser.
|
|
7
|
+
|
|
8
|
+
import { cli, Strategy } from '@jackwener/opencli/registry';
|
|
9
|
+
import { EmptyResultError } from '@jackwener/opencli/errors';
|
|
10
|
+
import {
|
|
11
|
+
BROWSER_HELPERS,
|
|
12
|
+
MAX_PAGES,
|
|
13
|
+
SERVER_PAGE_MAX,
|
|
14
|
+
TIKTOK_AID,
|
|
15
|
+
VIDEO_ITEM_NORMALIZER,
|
|
16
|
+
normalizeUsername,
|
|
17
|
+
requireLimit,
|
|
18
|
+
throwTikTokPageContextError,
|
|
19
|
+
} from './utils.js';
|
|
20
|
+
|
|
21
|
+
const DEFAULT_LIMIT = 20;
|
|
22
|
+
const MAX_LIMIT = 120;
|
|
23
|
+
|
|
24
|
+
function buildUserScript(username, limit) {
|
|
25
|
+
return `
|
|
26
|
+
(async () => {
|
|
27
|
+
const username = ${JSON.stringify(username)};
|
|
28
|
+
const usernameLower = username.toLowerCase();
|
|
29
|
+
const limit = ${Number(limit)};
|
|
30
|
+
const maxPages = ${MAX_PAGES};
|
|
31
|
+
const SERVER_PAGE_MAX = ${SERVER_PAGE_MAX};
|
|
32
|
+
const pageSize = limit < SERVER_PAGE_MAX ? limit : SERVER_PAGE_MAX;
|
|
33
|
+
const aid = ${JSON.stringify(TIKTOK_AID)};
|
|
34
|
+
|
|
35
|
+
${BROWSER_HELPERS}
|
|
36
|
+
${VIDEO_ITEM_NORMALIZER}
|
|
37
|
+
|
|
38
|
+
function findProfileUser(root) {
|
|
39
|
+
if (!root) return null;
|
|
40
|
+
let found = null;
|
|
41
|
+
walkObjects(root, (node) => {
|
|
42
|
+
if (Array.isArray(node)) return false;
|
|
43
|
+
const user = node?.userInfo?.user || node?.user;
|
|
44
|
+
if (!user || typeof user !== 'object') return false;
|
|
45
|
+
const uniqueId = String(user.uniqueId || user.unique_id || '').toLowerCase();
|
|
46
|
+
if (uniqueId === usernameLower && (user.secUid || user.sec_uid)) {
|
|
47
|
+
found = user;
|
|
48
|
+
return true;
|
|
49
|
+
}
|
|
50
|
+
return false;
|
|
51
|
+
});
|
|
52
|
+
return found;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function collectProfileItems(root, secUid) {
|
|
56
|
+
if (!root) return [];
|
|
57
|
+
const out = [];
|
|
58
|
+
walkObjects(root, (node) => {
|
|
59
|
+
if (!node || typeof node !== 'object' || Array.isArray(node)) return false;
|
|
60
|
+
const item = node.itemStruct || node.item || node;
|
|
61
|
+
const id = item?.id || item?.item_id || item?.video_id;
|
|
62
|
+
const author = item?.author || item?.authorInfo || {};
|
|
63
|
+
const authorName = String(author.uniqueId || author.unique_id || item?.author_unique_id || '').toLowerCase();
|
|
64
|
+
const authorSecUid = String(author.secUid || author.sec_uid || '').trim();
|
|
65
|
+
if (id && (authorName === usernameLower || (secUid && authorSecUid === secUid))) {
|
|
66
|
+
out.push(item);
|
|
67
|
+
}
|
|
68
|
+
return false;
|
|
69
|
+
});
|
|
70
|
+
return out;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
function addVideo(dedup, item, source) {
|
|
74
|
+
const row = normalizeVideoItem(item, dedup.size + 1);
|
|
75
|
+
if (!row || dedup.has(row.id)) return;
|
|
76
|
+
dedup.set(row.id, { ...row, source });
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
const universal = findUniversalData();
|
|
80
|
+
let secUid = '';
|
|
81
|
+
const profileUser = findProfileUser(universal);
|
|
82
|
+
if (profileUser) {
|
|
83
|
+
secUid = String(profileUser.secUid || profileUser.sec_uid || '').trim();
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
const msToken = getCookie('msToken');
|
|
87
|
+
if (!secUid) {
|
|
88
|
+
const params = new URLSearchParams({ uniqueId: username, aid });
|
|
89
|
+
if (msToken) params.set('msToken', msToken);
|
|
90
|
+
const detail = await fetchJson('/api/user/detail/?' + params.toString());
|
|
91
|
+
assertTikTokApiSuccess(detail, 'user-detail');
|
|
92
|
+
secUid = String(detail?.userInfo?.user?.secUid || detail?.user?.secUid || '').trim();
|
|
93
|
+
}
|
|
94
|
+
if (!secUid) {
|
|
95
|
+
throw new Error('No videos found for @' + username);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
const dedup = new Map();
|
|
99
|
+
for (const item of collectProfileItems(universal, secUid)) {
|
|
100
|
+
addVideo(dedup, item, 'bootstrap');
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
let cursor = 0;
|
|
104
|
+
let primaryFailure = null;
|
|
105
|
+
for (let page = 0; page < maxPages && dedup.size < limit; page += 1) {
|
|
106
|
+
const params = new URLSearchParams({
|
|
107
|
+
secUid,
|
|
108
|
+
count: String(pageSize),
|
|
109
|
+
cursor: String(cursor),
|
|
110
|
+
aid,
|
|
111
|
+
});
|
|
112
|
+
if (msToken) params.set('msToken', msToken);
|
|
113
|
+
try {
|
|
114
|
+
const data = await fetchJson('/api/post/item_list/?' + params.toString());
|
|
115
|
+
assertTikTokApiSuccess(data, 'post-list');
|
|
116
|
+
const items = Array.isArray(data.itemList) ? data.itemList : [];
|
|
117
|
+
for (const item of items) addVideo(dedup, item, 'profile-api');
|
|
118
|
+
if (data.hasMore !== true || items.length === 0) break;
|
|
119
|
+
cursor = asNumber(data.cursor) ?? cursor + items.length;
|
|
120
|
+
} catch (error) {
|
|
121
|
+
primaryFailure = error instanceof Error ? error.message : String(error);
|
|
122
|
+
break;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
let searchFailure = null;
|
|
127
|
+
for (let offset = 0; offset < limit * 4 && dedup.size < limit; offset += pageSize) {
|
|
128
|
+
const params = new URLSearchParams({
|
|
129
|
+
keyword: username,
|
|
130
|
+
offset: String(offset),
|
|
131
|
+
count: String(pageSize),
|
|
132
|
+
aid,
|
|
133
|
+
});
|
|
134
|
+
if (msToken) params.set('msToken', msToken);
|
|
135
|
+
try {
|
|
136
|
+
const data = await fetchJson('/api/search/general/full/?' + params.toString());
|
|
137
|
+
assertTikTokApiSuccess(data, 'search');
|
|
138
|
+
const entries = Array.isArray(data.data) ? data.data : [];
|
|
139
|
+
if (entries.length === 0) break;
|
|
140
|
+
for (const entry of entries) {
|
|
141
|
+
if (entry?.type !== undefined && entry.type !== 1) continue;
|
|
142
|
+
const item = entry?.item || entry?.itemStruct || entry;
|
|
143
|
+
const author = item?.author || item?.authorInfo || {};
|
|
144
|
+
const authorName = String(author.uniqueId || author.unique_id || '').toLowerCase();
|
|
145
|
+
if (authorName !== usernameLower) continue;
|
|
146
|
+
addVideo(dedup, item, 'search-fallback');
|
|
147
|
+
}
|
|
148
|
+
} catch (error) {
|
|
149
|
+
searchFailure = error instanceof Error ? error.message : String(error);
|
|
150
|
+
break;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const rows = Array.from(dedup.values())
|
|
155
|
+
.sort((a, b) => (Number(b.createTime) || 0) - (Number(a.createTime) || 0))
|
|
156
|
+
.slice(0, limit)
|
|
157
|
+
.map((row, index) => ({ ...row, index: index + 1 }));
|
|
158
|
+
|
|
159
|
+
if (rows.length === 0) {
|
|
160
|
+
const suffix = primaryFailure || searchFailure
|
|
161
|
+
? ' (profile/search API failed: ' + (primaryFailure || searchFailure) + ')'
|
|
162
|
+
: '';
|
|
163
|
+
throw new Error('No videos found for @' + username + suffix);
|
|
164
|
+
}
|
|
165
|
+
return rows;
|
|
166
|
+
})()
|
|
167
|
+
`;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
async function listUserVideos(page, args) {
|
|
171
|
+
const username = normalizeUsername(args.username);
|
|
172
|
+
const limit = requireLimit(args.limit, { fallback: DEFAULT_LIMIT, max: MAX_LIMIT });
|
|
173
|
+
await page.goto(`https://www.tiktok.com/@${encodeURIComponent(username)}`, { waitUntil: 'load', settleMs: 6000 });
|
|
174
|
+
let rows;
|
|
175
|
+
try {
|
|
176
|
+
rows = await page.evaluate(buildUserScript(username, limit));
|
|
177
|
+
} catch (error) {
|
|
178
|
+
throwTikTokPageContextError(error, {
|
|
179
|
+
authMessage: 'TikTok requires browser access to load user videos',
|
|
180
|
+
emptyPattern: /No videos found/,
|
|
181
|
+
emptyTarget: 'tiktok user',
|
|
182
|
+
failureMessage: 'Failed to fetch TikTok user videos',
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
if (!Array.isArray(rows) || rows.length === 0) {
|
|
186
|
+
throw new EmptyResultError('tiktok user', `No videos found for @${username}`);
|
|
187
|
+
}
|
|
188
|
+
return rows;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
export const userCommand = cli({
|
|
3
192
|
site: 'tiktok',
|
|
4
193
|
name: 'user',
|
|
5
194
|
access: 'read',
|
|
6
|
-
description: 'Get recent videos from a TikTok user',
|
|
195
|
+
description: 'Get recent videos from a TikTok user via page-context APIs',
|
|
7
196
|
domain: 'www.tiktok.com',
|
|
197
|
+
strategy: Strategy.COOKIE,
|
|
198
|
+
browser: true,
|
|
8
199
|
args: [
|
|
9
200
|
{
|
|
10
201
|
name: 'username',
|
|
@@ -12,31 +203,14 @@ cli({
|
|
|
12
203
|
positional: true,
|
|
13
204
|
help: 'TikTok username (without @)',
|
|
14
205
|
},
|
|
15
|
-
{ name: 'limit', type: 'int', default:
|
|
16
|
-
],
|
|
17
|
-
columns: ['index', 'views', 'url'],
|
|
18
|
-
pipeline: [
|
|
19
|
-
{ navigate: { url: 'https://www.tiktok.com/@${{ args.username }}', settleMs: 6000 } },
|
|
20
|
-
{ evaluate: `(() => {
|
|
21
|
-
const limit = \${{ args.limit }};
|
|
22
|
-
const username = \${{ args.username | json }};
|
|
23
|
-
const links = Array.from(document.querySelectorAll('a[href*="/video/"]'));
|
|
24
|
-
const seen = {};
|
|
25
|
-
const results = [];
|
|
26
|
-
for (const a of links) {
|
|
27
|
-
const href = a.href;
|
|
28
|
-
if (seen[href]) continue;
|
|
29
|
-
seen[href] = true;
|
|
30
|
-
results.push({
|
|
31
|
-
index: results.length + 1,
|
|
32
|
-
views: a.textContent.trim() || '-',
|
|
33
|
-
url: href,
|
|
34
|
-
});
|
|
35
|
-
if (results.length >= limit) break;
|
|
36
|
-
}
|
|
37
|
-
if (results.length === 0) throw new Error('No videos found for @' + username);
|
|
38
|
-
return results;
|
|
39
|
-
})()
|
|
40
|
-
` },
|
|
206
|
+
{ name: 'limit', type: 'int', default: DEFAULT_LIMIT, help: `Number of videos to return (max ${MAX_LIMIT})` },
|
|
41
207
|
],
|
|
208
|
+
columns: ['index', 'id', 'source', 'author', 'url', 'cover', 'title', 'desc', 'plays', 'likes', 'comments', 'shares', 'createTime'],
|
|
209
|
+
func: listUserVideos,
|
|
42
210
|
});
|
|
211
|
+
|
|
212
|
+
export const __test__ = {
|
|
213
|
+
buildUserScript,
|
|
214
|
+
DEFAULT_LIMIT,
|
|
215
|
+
MAX_LIMIT,
|
|
216
|
+
};
|