@jackwener/opencli 1.7.4 → 1.7.5

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.
Files changed (126) hide show
  1. package/README.md +71 -49
  2. package/README.zh-CN.md +73 -60
  3. package/cli-manifest.json +3261 -1758
  4. package/clis/antigravity/serve.js +71 -25
  5. package/clis/baidu-scholar/search.js +87 -0
  6. package/clis/baidu-scholar/search.test.js +23 -0
  7. package/clis/deepseek/ask.js +74 -0
  8. package/clis/deepseek/history.js +25 -0
  9. package/clis/deepseek/new.js +20 -0
  10. package/clis/deepseek/read.js +22 -0
  11. package/clis/deepseek/status.js +24 -0
  12. package/clis/deepseek/utils.js +208 -0
  13. package/clis/eastmoney/_secid.js +78 -0
  14. package/clis/eastmoney/announcement.js +52 -0
  15. package/clis/eastmoney/convertible.js +73 -0
  16. package/clis/eastmoney/etf.js +65 -0
  17. package/clis/eastmoney/holders.js +78 -0
  18. package/clis/eastmoney/index-board.js +96 -0
  19. package/clis/eastmoney/kline.js +87 -0
  20. package/clis/eastmoney/kuaixun.js +54 -0
  21. package/clis/eastmoney/longhu.js +67 -0
  22. package/clis/eastmoney/money-flow.js +78 -0
  23. package/clis/eastmoney/northbound.js +57 -0
  24. package/clis/eastmoney/quote.js +107 -0
  25. package/clis/eastmoney/rank.js +94 -0
  26. package/clis/eastmoney/sectors.js +76 -0
  27. package/clis/google-scholar/search.js +58 -0
  28. package/clis/google-scholar/search.test.js +23 -0
  29. package/clis/gov-law/commands.test.js +39 -0
  30. package/clis/gov-law/recent.js +22 -0
  31. package/clis/gov-law/search.js +41 -0
  32. package/clis/gov-law/shared.js +51 -0
  33. package/clis/gov-policy/commands.test.js +27 -0
  34. package/clis/gov-policy/recent.js +47 -0
  35. package/clis/gov-policy/search.js +48 -0
  36. package/clis/nowcoder/companies.js +23 -0
  37. package/clis/nowcoder/creators.js +27 -0
  38. package/clis/nowcoder/detail.js +61 -0
  39. package/clis/nowcoder/experience.js +36 -0
  40. package/clis/nowcoder/hot.js +24 -0
  41. package/clis/nowcoder/jobs.js +21 -0
  42. package/clis/nowcoder/notifications.js +29 -0
  43. package/clis/nowcoder/papers.js +40 -0
  44. package/clis/nowcoder/practice.js +37 -0
  45. package/clis/nowcoder/recommend.js +30 -0
  46. package/clis/nowcoder/referral.js +39 -0
  47. package/clis/nowcoder/salary.js +40 -0
  48. package/clis/nowcoder/search.js +49 -0
  49. package/clis/nowcoder/suggest.js +33 -0
  50. package/clis/nowcoder/topics.js +27 -0
  51. package/clis/nowcoder/trending.js +25 -0
  52. package/clis/twitter/list-add.js +337 -0
  53. package/clis/twitter/list-add.test.js +15 -0
  54. package/clis/twitter/list-remove.js +297 -0
  55. package/clis/twitter/list-remove.test.js +14 -0
  56. package/clis/twitter/list-tweets.js +185 -0
  57. package/clis/twitter/list-tweets.test.js +108 -0
  58. package/clis/twitter/lists.js +134 -47
  59. package/clis/twitter/lists.test.js +105 -38
  60. package/clis/wanfang/search.js +66 -0
  61. package/clis/wanfang/search.test.js +23 -0
  62. package/clis/web/read.js +1 -1
  63. package/clis/weixin/download.js +3 -2
  64. package/clis/xiaohongshu/publish.js +149 -28
  65. package/clis/xiaohongshu/publish.test.js +319 -6
  66. package/clis/xiaoyuzhou/download.js +8 -4
  67. package/clis/xiaoyuzhou/download.test.js +23 -13
  68. package/clis/xiaoyuzhou/episode.js +9 -4
  69. package/clis/xiaoyuzhou/podcast-episodes.js +15 -11
  70. package/clis/xiaoyuzhou/podcast.js +9 -4
  71. package/clis/xiaoyuzhou/utils.js +0 -40
  72. package/clis/xiaoyuzhou/utils.test.js +15 -75
  73. package/clis/zsxq/dynamics.js +1 -1
  74. package/clis/zsxq/utils.js +6 -3
  75. package/clis/zsxq/utils.test.js +31 -0
  76. package/dist/src/browser/base-page.d.ts +1 -1
  77. package/dist/src/browser/bridge.d.ts +1 -0
  78. package/dist/src/browser/bridge.js +1 -1
  79. package/dist/src/browser/cdp.js +1 -1
  80. package/dist/src/browser/daemon-client.d.ts +6 -4
  81. package/dist/src/browser/daemon-client.js +6 -1
  82. package/dist/src/browser/daemon-client.test.js +40 -1
  83. package/dist/src/browser/dom-snapshot.js +7 -2
  84. package/dist/src/browser/page.d.ts +14 -4
  85. package/dist/src/browser/page.js +48 -7
  86. package/dist/src/browser/page.test.js +97 -0
  87. package/dist/src/cli.js +227 -150
  88. package/dist/src/cli.test.js +167 -90
  89. package/dist/src/commanderAdapter.d.ts +0 -1
  90. package/dist/src/commanderAdapter.js +2 -16
  91. package/dist/src/commanderAdapter.test.js +1 -1
  92. package/dist/src/completion-shared.js +2 -5
  93. package/dist/src/daemon.js +8 -0
  94. package/dist/src/download/article-download.d.ts +1 -0
  95. package/dist/src/download/article-download.js +3 -0
  96. package/dist/src/download/article-download.test.js +39 -0
  97. package/dist/src/plugin.d.ts +1 -8
  98. package/dist/src/plugin.js +1 -27
  99. package/dist/src/plugin.test.js +1 -59
  100. package/dist/src/registry.d.ts +1 -0
  101. package/dist/src/registry.js +3 -2
  102. package/dist/src/registry.test.js +22 -0
  103. package/dist/src/types.d.ts +14 -5
  104. package/package.json +1 -1
  105. package/clis/twitter/lists-parser.js +0 -77
  106. package/clis/twitter/lists.d.ts +0 -5
  107. package/dist/src/cascade.d.ts +0 -46
  108. package/dist/src/cascade.js +0 -135
  109. package/dist/src/explore.d.ts +0 -99
  110. package/dist/src/explore.js +0 -402
  111. package/dist/src/generate-verified.d.ts +0 -105
  112. package/dist/src/generate-verified.js +0 -696
  113. package/dist/src/generate-verified.test.js +0 -925
  114. package/dist/src/generate.d.ts +0 -46
  115. package/dist/src/generate.js +0 -117
  116. package/dist/src/record.d.ts +0 -96
  117. package/dist/src/record.js +0 -657
  118. package/dist/src/record.test.d.ts +0 -1
  119. package/dist/src/record.test.js +0 -293
  120. package/dist/src/skill-generate.d.ts +0 -30
  121. package/dist/src/skill-generate.js +0 -75
  122. package/dist/src/skill-generate.test.d.ts +0 -1
  123. package/dist/src/skill-generate.test.js +0 -173
  124. package/dist/src/synthesize.d.ts +0 -97
  125. package/dist/src/synthesize.js +0 -208
  126. /package/dist/src/{generate-verified.test.d.ts → download/article-download.test.d.ts} +0 -0
@@ -0,0 +1,78 @@
1
+ // Shared helpers for resolving eastmoney "secid" (市场.代码).
2
+ //
3
+ // Markets:
4
+ // 1.XXXXXX → Shanghai A (SSE)
5
+ // 0.XXXXXX → Shenzhen A (SZSE) or Beijing (BSE) — eastmoney groups both under 0
6
+ // 116.XXXXX → Hong Kong
7
+ // 105.SYMBOL → NASDAQ
8
+ // 106.SYMBOL → NYSE
9
+ // 107.SYMBOL → AMEX (US)
10
+
11
+ const A_PREFIX_TO_MARKET = /** @param {string} c */ (c) => {
12
+ if (/^(60|68|90|113|900)/.test(c)) return '1'; // SH (A + STAR + old B)
13
+ if (/^(00|30|20)/.test(c)) return '0'; // SZ (A + ChiNext + B)
14
+ if (/^(4|8|920|83|87)/.test(c)) return '0'; // BJ (eastmoney uses 0.)
15
+ return '0';
16
+ };
17
+
18
+ /**
19
+ * Resolve various user inputs to an eastmoney `secid`.
20
+ * - "600000" → "1.600000"
21
+ * - "sh600000" → "1.600000"
22
+ * - "sz000001" → "0.000001"
23
+ * - "bj430047" → "0.430047"
24
+ * - "hk00700" / "00700.HK" → "116.00700"
25
+ * - "us.AAPL" / "AAPL" → "105.AAPL"
26
+ * - "1.600000" → passed through
27
+ * @param {string} input
28
+ * @returns {string}
29
+ */
30
+ // Known eastmoney market numeric prefixes. Narrow whitelist so that inputs like
31
+ // "00700.HK" are NOT mistakenly treated as secids just because they look like
32
+ // "<digits>.<alphanumeric>".
33
+ const KNOWN_MARKET_PREFIXES = new Set(['0', '1', '100', '105', '106', '107', '116', '140', '150', '151', '152', '155', '156']);
34
+
35
+ export function resolveSecid(input) {
36
+ const raw = String(input || '').trim();
37
+ if (!raw) throw new Error('empty symbol');
38
+ const secidMatch = raw.match(/^(\d{1,3})\.([A-Za-z0-9]+)$/);
39
+ if (secidMatch && KNOWN_MARKET_PREFIXES.has(secidMatch[1])) return raw; // already a secid
40
+ const lower = raw.toLowerCase();
41
+
42
+ // market-prefixed Chinese code
43
+ const pref = lower.match(/^(sh|sz|bj)(\d{6})$/);
44
+ if (pref) {
45
+ const [, mk, code] = pref;
46
+ return (mk === 'sh' ? '1' : '0') + '.' + code;
47
+ }
48
+
49
+ // hk prefix
50
+ const hk = lower.match(/^hk(\d{4,5})$/) || lower.match(/^(\d{4,5})\.hk$/);
51
+ if (hk) return '116.' + hk[1].padStart(5, '0');
52
+
53
+ // us.SYMBOL or SYMBOL.N/.O (treat all as NASDAQ by default; .N as NYSE)
54
+ const usDot = lower.match(/^([a-z.\-]+)\.([no])$/);
55
+ if (usDot) return (usDot[2] === 'n' ? '106' : '105') + '.' + usDot[1].toUpperCase();
56
+ const usPref = lower.match(/^us\.([a-z.\-]+)$/);
57
+ if (usPref) return '105.' + usPref[1].toUpperCase();
58
+
59
+ // bare 6-digit Chinese code
60
+ if (/^\d{6}$/.test(raw)) return A_PREFIX_TO_MARKET(raw) + '.' + raw;
61
+
62
+ // bare US ticker — uppercase letters only
63
+ if (/^[A-Z.\-]{1,8}$/.test(raw)) return '105.' + raw;
64
+
65
+ throw new Error(`Unrecognized symbol: ${input}`);
66
+ }
67
+
68
+ /**
69
+ * Normalize a list of user inputs separated by comma / space / Chinese comma.
70
+ * @param {string} s
71
+ * @returns {string[]}
72
+ */
73
+ export function splitSymbols(s) {
74
+ return String(s || '')
75
+ .split(/[,,\s]+/)
76
+ .map((x) => x.trim())
77
+ .filter(Boolean);
78
+ }
@@ -0,0 +1,52 @@
1
+ // eastmoney announcement — listed company filings/announcements feed.
2
+ //
3
+ // opencli eastmoney announcement
4
+ // opencli eastmoney announcement --market SHA --limit 30
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ cli({
10
+ site: 'eastmoney',
11
+ name: 'announcement',
12
+ description: '上市公司公告(按交易所筛选)',
13
+ domain: 'np-anotice-stock.eastmoney.com',
14
+ strategy: Strategy.PUBLIC,
15
+ browser: false,
16
+ args: [
17
+ { name: 'market', type: 'string', default: 'SHA,SZA,BJA', help: '交易所:SHA (沪) / SZA (深) / BJA (北) 可逗号分隔' },
18
+ { name: 'limit', type: 'int', default: 20, help: '返回数量 (max 100)' },
19
+ ],
20
+ columns: ['time', 'code', 'name', 'title', 'category', 'url'],
21
+ func: async (_page, args) => {
22
+ const market = String(args.market ?? 'SHA,SZA,BJA').trim() || 'SHA,SZA,BJA';
23
+ const limit = Math.max(1, Math.min(Number(args.limit) || 20, 100));
24
+
25
+ const url = new URL('https://np-anotice-stock.eastmoney.com/api/security/ann');
26
+ url.searchParams.set('page_size', String(limit));
27
+ url.searchParams.set('page_index', '1');
28
+ url.searchParams.set('ann_type', market);
29
+ url.searchParams.set('client_source', 'web');
30
+ url.searchParams.set('f_node', '0');
31
+ url.searchParams.set('s_node', '0');
32
+
33
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
34
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `announcement failed: HTTP ${resp.status}`);
35
+ const data = await resp.json();
36
+ const list = Array.isArray(data?.data?.list) ? data.data.list : [];
37
+ if (list.length === 0) throw new CliError('NO_DATA', 'eastmoney returned no announcement data');
38
+
39
+ return list.slice(0, limit).map((it) => {
40
+ const primary = Array.isArray(it.codes) && it.codes.length > 0 ? it.codes[0] : {};
41
+ const cat = Array.isArray(it.columns) && it.columns.length > 0 ? it.columns[0]?.column_name : '';
42
+ return {
43
+ time: String(it.notice_date || it.display_time || '').slice(0, 19),
44
+ code: primary.stock_code || '',
45
+ name: primary.short_name || '',
46
+ title: it.title || it.title_ch || '',
47
+ category: cat || '',
48
+ url: `https://data.eastmoney.com/notices/detail/${primary.stock_code || ''}/${it.art_code || ''}.html`,
49
+ };
50
+ });
51
+ },
52
+ });
@@ -0,0 +1,73 @@
1
+ // eastmoney convertible — on-market convertible bond listing.
2
+ //
3
+ // opencli eastmoney convertible
4
+ // opencli eastmoney convertible --sort premium --limit 30
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ const SORTS = {
10
+ change: { fid: 'f3', order: 'desc' },
11
+ drop: { fid: 'f3', order: 'asc' },
12
+ turnover: { fid: 'f6', order: 'desc' },
13
+ price: { fid: 'f2', order: 'desc' },
14
+ premium: { fid: 'f237', order: 'desc' }, // 转股溢价率
15
+ value: { fid: 'f236', order: 'desc' }, // 转股价值
16
+ ytm: { fid: 'f239', order: 'desc' }, // 到期收益率
17
+ };
18
+
19
+ cli({
20
+ site: 'eastmoney',
21
+ name: 'convertible',
22
+ description: '可转债行情列表(默认按成交额排序)',
23
+ domain: 'push2.eastmoney.com',
24
+ strategy: Strategy.PUBLIC,
25
+ browser: false,
26
+ args: [
27
+ { name: 'sort', type: 'string', default: 'turnover', help: '排序:turnover / change / drop / price / premium' },
28
+ { name: 'limit', type: 'int', default: 20, help: '返回数量 (max 100)' },
29
+ ],
30
+ columns: ['rank', 'bondCode', 'bondName', 'bondPrice', 'bondChangePct', 'stockCode', 'stockName', 'stockPrice', 'stockChangePct', 'convPrice', 'convValue', 'convPremiumPct', 'remainingYears', 'ytm', 'listDate'],
31
+ func: async (_page, args) => {
32
+ const sortKey = String(args.sort ?? 'turnover').toLowerCase();
33
+ const sort = SORTS[sortKey];
34
+ if (!sort) throw new CliError('INVALID_ARGUMENT', `Unknown sort "${sortKey}". Valid: ${Object.keys(SORTS).join(', ')}`);
35
+ const limit = Math.max(1, Math.min(Number(args.limit) || 20, 100));
36
+
37
+ const url = new URL('https://push2.eastmoney.com/api/qt/clist/get');
38
+ url.searchParams.set('pn', '1');
39
+ url.searchParams.set('pz', String(limit));
40
+ url.searchParams.set('po', sort.order === 'desc' ? '1' : '0');
41
+ url.searchParams.set('np', '1');
42
+ url.searchParams.set('fltt', '2');
43
+ url.searchParams.set('invt', '2');
44
+ url.searchParams.set('fid', sort.fid);
45
+ url.searchParams.set('fs', 'b:MK0354');
46
+ url.searchParams.set('fields', 'f12,f14,f2,f3,f6,f229,f230,f232,f234,f235,f236,f237,f238,f239,f243');
47
+ url.searchParams.set('ut', 'bd1d9ddb04089700cf9c27f6f7426281');
48
+
49
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
50
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `convertible failed: HTTP ${resp.status}`);
51
+ const data = await resp.json();
52
+ const diff = Array.isArray(data?.data?.diff) ? data.data.diff : [];
53
+ if (diff.length === 0) throw new CliError('NO_DATA', 'eastmoney returned no convertible data');
54
+
55
+ return diff.slice(0, limit).map((it, i) => ({
56
+ rank: i + 1,
57
+ bondCode: it.f12,
58
+ bondName: it.f14,
59
+ bondPrice: it.f2,
60
+ bondChangePct: it.f3,
61
+ stockCode: it.f232,
62
+ stockName: it.f234,
63
+ stockPrice: it.f229,
64
+ stockChangePct: it.f230,
65
+ convPrice: it.f235,
66
+ convValue: it.f236,
67
+ convPremiumPct: it.f237,
68
+ remainingYears: it.f238,
69
+ ytm: it.f239,
70
+ listDate: String(it.f243 ?? ''),
71
+ }));
72
+ },
73
+ });
@@ -0,0 +1,65 @@
1
+ // eastmoney etf — ETF ranking by change / turnover.
2
+ //
3
+ // opencli eastmoney etf
4
+ // opencli eastmoney etf --sort change --limit 30
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ const SORTS = {
10
+ turnover: { fid: 'f6', order: 'desc' },
11
+ change: { fid: 'f3', order: 'desc' },
12
+ drop: { fid: 'f3', order: 'asc' },
13
+ volume: { fid: 'f5', order: 'desc' },
14
+ rate: { fid: 'f8', order: 'desc' },
15
+ };
16
+
17
+ cli({
18
+ site: 'eastmoney',
19
+ name: 'etf',
20
+ description: 'ETF 列表按成交额/涨跌幅排行',
21
+ domain: 'push2.eastmoney.com',
22
+ strategy: Strategy.PUBLIC,
23
+ browser: false,
24
+ args: [
25
+ { name: 'sort', type: 'string', default: 'turnover', help: '排序:turnover / change / drop / volume / rate' },
26
+ { name: 'limit', type: 'int', default: 20, help: '返回数量 (max 100)' },
27
+ ],
28
+ columns: ['rank', 'code', 'name', 'price', 'changePercent', 'change', 'turnover', 'volume', 'turnoverRate'],
29
+ func: async (_page, args) => {
30
+ const sortKey = String(args.sort ?? 'turnover').toLowerCase();
31
+ const sort = SORTS[sortKey];
32
+ if (!sort) throw new CliError('INVALID_ARGUMENT', `Unknown sort "${sortKey}". Valid: ${Object.keys(SORTS).join(', ')}`);
33
+ const limit = Math.max(1, Math.min(Number(args.limit) || 20, 100));
34
+
35
+ const url = new URL('https://push2.eastmoney.com/api/qt/clist/get');
36
+ url.searchParams.set('pn', '1');
37
+ url.searchParams.set('pz', String(limit));
38
+ url.searchParams.set('po', sort.order === 'desc' ? '1' : '0');
39
+ url.searchParams.set('np', '1');
40
+ url.searchParams.set('fltt', '2');
41
+ url.searchParams.set('invt', '2');
42
+ url.searchParams.set('fid', sort.fid);
43
+ url.searchParams.set('fs', 'b:MK0021'); // 场内ETF
44
+ url.searchParams.set('fields', 'f12,f14,f2,f3,f4,f5,f6,f8');
45
+ url.searchParams.set('ut', 'bd1d9ddb04089700cf9c27f6f7426281');
46
+
47
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
48
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `etf failed: HTTP ${resp.status}`);
49
+ const data = await resp.json();
50
+ const diff = Array.isArray(data?.data?.diff) ? data.data.diff : [];
51
+ if (diff.length === 0) throw new CliError('NO_DATA', 'eastmoney returned no ETF data');
52
+
53
+ return diff.slice(0, limit).map((it, i) => ({
54
+ rank: i + 1,
55
+ code: it.f12,
56
+ name: it.f14,
57
+ price: it.f2,
58
+ changePercent: it.f3,
59
+ change: it.f4,
60
+ turnover: it.f6,
61
+ volume: it.f5,
62
+ turnoverRate: it.f8,
63
+ }));
64
+ },
65
+ });
@@ -0,0 +1,78 @@
1
+ // eastmoney holders — top-10 float shareholders of an A-share (F10 data).
2
+ //
3
+ // opencli eastmoney holders 600519
4
+ // opencli eastmoney holders sh600519 --limit 10
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ /**
10
+ * Convert a bare A-share symbol to eastmoney's SECUCODE form ("600519.SH").
11
+ * Accepts "600519", "sh600519", "sz000001", "bj430047", or full "600519.SH".
12
+ * @param {string} input
13
+ * @returns {string}
14
+ */
15
+ function toSecucode(input) {
16
+ const raw = String(input || '').trim().toUpperCase();
17
+ if (/^\d{6}\.(SH|SZ|BJ)$/.test(raw)) return raw;
18
+ const pref = raw.match(/^(SH|SZ|BJ)(\d{6})$/);
19
+ if (pref) return `${pref[2]}.${pref[1]}`;
20
+ if (/^\d{6}$/.test(raw)) {
21
+ if (/^(60|68|90|113|900)/.test(raw)) return `${raw}.SH`;
22
+ if (/^(4|8|920|83|87)/.test(raw)) return `${raw}.BJ`;
23
+ return `${raw}.SZ`;
24
+ }
25
+ throw new Error(`Unrecognized A-share symbol: ${input}`);
26
+ }
27
+
28
+ cli({
29
+ site: 'eastmoney',
30
+ name: 'holders',
31
+ description: '十大流通股东(A股 F10 数据)',
32
+ domain: 'datacenter-web.eastmoney.com',
33
+ strategy: Strategy.PUBLIC,
34
+ browser: false,
35
+ args: [
36
+ { name: 'symbol', required: true, positional: true, help: 'A股代码(600519 / sh600519 等)' },
37
+ { name: 'limit', type: 'int', default: 10, help: '返回股东数(默认十大流通股东)' },
38
+ ],
39
+ columns: ['rank', 'reportDate', 'name', 'holdNum', 'floatRatio', 'change'],
40
+ func: async (_page, args) => {
41
+ /** @type {string} */
42
+ let secucode;
43
+ try { secucode = toSecucode(args.symbol); }
44
+ catch (err) { throw new CliError('INVALID_ARGUMENT', `${err instanceof Error ? err.message : err}`); }
45
+ const limit = Math.max(1, Math.min(Number(args.limit) || 10, 50));
46
+
47
+ const url = new URL('https://datacenter-web.eastmoney.com/api/data/v1/get');
48
+ url.searchParams.set('sortColumns', 'END_DATE,HOLDER_RANK');
49
+ url.searchParams.set('sortTypes', '-1,1');
50
+ url.searchParams.set('pageSize', String(Math.max(limit, 10)));
51
+ url.searchParams.set('pageNumber', '1');
52
+ url.searchParams.set('reportName', 'RPT_F10_EH_FREEHOLDERS');
53
+ url.searchParams.set('columns', 'SECUCODE,SECURITY_CODE,END_DATE,HOLDER_RANK,HOLDER_NAME,HOLD_NUM,FREE_HOLDNUM_RATIO,HOLD_NUM_CHANGE');
54
+ url.searchParams.set('source', 'HSF10');
55
+ url.searchParams.set('client', 'PC');
56
+ url.searchParams.set('filter', `(SECUCODE="${secucode}")`);
57
+
58
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
59
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `holders failed: HTTP ${resp.status}`);
60
+ const data = await resp.json();
61
+ const rows = Array.isArray(data?.result?.data) ? data.result.data : [];
62
+ if (rows.length === 0) throw new CliError('NO_DATA', `No shareholder data for ${secucode}`);
63
+
64
+ // Only the most recent reporting period
65
+ const latest = String(rows[0].END_DATE || '').slice(0, 10);
66
+ return rows
67
+ .filter((it) => String(it.END_DATE || '').slice(0, 10) === latest)
68
+ .slice(0, limit)
69
+ .map((it) => ({
70
+ rank: it.HOLDER_RANK,
71
+ reportDate: latest,
72
+ name: it.HOLDER_NAME,
73
+ holdNum: it.HOLD_NUM,
74
+ floatRatio: it.FREE_HOLDNUM_RATIO,
75
+ change: it.HOLD_NUM_CHANGE,
76
+ }));
77
+ },
78
+ });
@@ -0,0 +1,96 @@
1
+ // eastmoney index-board — live quotes for key Chinese market indices.
2
+ //
3
+ // Data source: push2.eastmoney.com (Tier 1 public JSON, no auth).
4
+ // opencli eastmoney index-board
5
+ // opencli eastmoney index-board --group all
6
+
7
+ import { cli, Strategy } from '@jackwener/opencli/registry';
8
+ import { CliError } from '@jackwener/opencli/errors';
9
+
10
+ const INDEX_GROUPS = {
11
+ main: [
12
+ ['1.000001', '上证指数'],
13
+ ['0.399001', '深证成指'],
14
+ ['0.399006', '创业板指'],
15
+ ['1.000688', '科创50'],
16
+ ['1.000300', '沪深300'],
17
+ ['1.000905', '中证500'],
18
+ ],
19
+ hk: [
20
+ ['100.HSI', '恒生指数'],
21
+ ['100.HSCEI', '恒生国企'],
22
+ ['100.HSTECH', '恒生科技'],
23
+ ],
24
+ us: [
25
+ ['100.DJIA', '道琼斯'],
26
+ ['100.SPX', '标普500'],
27
+ ['100.NDX', '纳斯达克100'],
28
+ ['100.IXIC', '纳斯达克综指'],
29
+ ],
30
+ };
31
+
32
+ const FIELDS = 'f2,f3,f4,f12,f13,f14,f15,f16,f17,f18';
33
+
34
+ cli({
35
+ site: 'eastmoney',
36
+ name: 'index-board',
37
+ description: '主要市场指数行情(A股 / 港股 / 美股)',
38
+ domain: 'push2.eastmoney.com',
39
+ strategy: Strategy.PUBLIC,
40
+ browser: false,
41
+ args: [
42
+ {
43
+ name: 'group',
44
+ type: 'string',
45
+ default: 'main',
46
+ help: '指数分组:main (A股主要), hk (港股), us (美股), all',
47
+ },
48
+ ],
49
+ columns: ['code', 'name', 'price', 'changePercent', 'change', 'open', 'high', 'low', 'prevClose'],
50
+ func: async (_page, args) => {
51
+ const group = String(args.group ?? 'main').toLowerCase();
52
+ /** @type {[string,string][]} */
53
+ let entries;
54
+ if (group === 'all') {
55
+ entries = [...INDEX_GROUPS.main, ...INDEX_GROUPS.hk, ...INDEX_GROUPS.us];
56
+ } else if (INDEX_GROUPS[group]) {
57
+ entries = INDEX_GROUPS[group];
58
+ } else {
59
+ throw new CliError('INVALID_ARGUMENT', `Unknown group "${group}". Valid: main, hk, us, all`);
60
+ }
61
+
62
+ const secids = entries.map(([secid]) => secid).join(',');
63
+ const url = new URL('https://push2.eastmoney.com/api/qt/ulist.np/get');
64
+ url.searchParams.set('secids', secids);
65
+ url.searchParams.set('fltt', '2');
66
+ url.searchParams.set('fields', FIELDS);
67
+ url.searchParams.set('ut', 'bd1d9ddb04089700cf9c27f6f7426281');
68
+
69
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0', Accept: 'application/json' } });
70
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `eastmoney index-board failed: HTTP ${resp.status}`);
71
+ const data = await resp.json();
72
+ const diff = Array.isArray(data?.data?.diff) ? data.data.diff : [];
73
+ if (diff.length === 0) throw new CliError('NO_DATA', 'eastmoney returned no index data');
74
+
75
+ // Preserve the order defined in INDEX_GROUPS regardless of API ordering
76
+ const byCode = new Map(diff.map((it) => [String(it.f12), it]));
77
+ return entries
78
+ .map(([secid, fallbackName]) => {
79
+ const code = secid.split('.')[1];
80
+ const it = byCode.get(code);
81
+ if (!it) return null;
82
+ return {
83
+ code,
84
+ name: it.f14 || fallbackName,
85
+ price: it.f2,
86
+ changePercent: it.f3,
87
+ change: it.f4,
88
+ open: it.f17,
89
+ high: it.f15,
90
+ low: it.f16,
91
+ prevClose: it.f18,
92
+ };
93
+ })
94
+ .filter(Boolean);
95
+ },
96
+ });
@@ -0,0 +1,87 @@
1
+ // eastmoney kline — historical OHLCV for one stock, any timeframe.
2
+ //
3
+ // Data source: push2his.eastmoney.com (Tier 1 public JSON).
4
+ // opencli eastmoney kline 600519 --period day --limit 30
5
+ // opencli eastmoney kline sh600519 --period week --adjust forward
6
+
7
+ import { cli, Strategy } from '@jackwener/opencli/registry';
8
+ import { CliError } from '@jackwener/opencli/errors';
9
+ import { resolveSecid } from './_secid.js';
10
+
11
+ const PERIOD_MAP = {
12
+ '1m': 1, '5m': 5, '15m': 15, '30m': 30, '60m': 60,
13
+ minute: 1, hour: 60,
14
+ day: 101, daily: 101,
15
+ week: 102, weekly: 102,
16
+ month: 103, monthly: 103,
17
+ };
18
+
19
+ const ADJUST_MAP = {
20
+ none: 0, no: 0, off: 0,
21
+ forward: 1, front: 1, 'pre': 1,
22
+ backward: 2, back: 2, 'post': 2,
23
+ };
24
+
25
+ cli({
26
+ site: 'eastmoney',
27
+ name: 'kline',
28
+ description: 'K线历史数据(分/日/周/月/前复权/后复权)',
29
+ domain: 'push2his.eastmoney.com',
30
+ strategy: Strategy.PUBLIC,
31
+ browser: false,
32
+ args: [
33
+ { name: 'symbol', required: true, positional: true, help: '股票代码(A/HK/US 均可)' },
34
+ { name: 'period', type: 'string', default: 'day', help: '周期:1m/5m/15m/30m/60m/day/week/month' },
35
+ { name: 'adjust', type: 'string', default: 'forward', help: '复权:none / forward / backward' },
36
+ { name: 'limit', type: 'int', default: 30, help: '返回最近 N 根(末尾)' },
37
+ ],
38
+ columns: ['date', 'open', 'close', 'high', 'low', 'volume', 'turnover', 'amplitude', 'changePercent', 'change', 'turnoverRate'],
39
+ func: async (_page, args) => {
40
+ const secid = resolveSecid(args.symbol);
41
+ const periodKey = String(args.period ?? 'day').toLowerCase();
42
+ const klt = PERIOD_MAP[periodKey];
43
+ if (klt == null) {
44
+ throw new CliError('INVALID_ARGUMENT', `Unknown period "${periodKey}". Valid: ${Object.keys(PERIOD_MAP).join(', ')}`);
45
+ }
46
+ const adjustKey = String(args.adjust ?? 'forward').toLowerCase();
47
+ const fqt = ADJUST_MAP[adjustKey];
48
+ if (fqt == null) {
49
+ throw new CliError('INVALID_ARGUMENT', `Unknown adjust "${adjustKey}". Valid: none / forward / backward`);
50
+ }
51
+ const limit = Math.max(1, Math.min(Number(args.limit) || 30, 1000));
52
+
53
+ const url = new URL('https://push2his.eastmoney.com/api/qt/stock/kline/get');
54
+ url.searchParams.set('secid', secid);
55
+ url.searchParams.set('klt', String(klt));
56
+ url.searchParams.set('fqt', String(fqt));
57
+ url.searchParams.set('beg', '0');
58
+ url.searchParams.set('end', '20500101');
59
+ url.searchParams.set('fields1', 'f1,f2,f3,f4,f5,f6');
60
+ url.searchParams.set('fields2', 'f51,f52,f53,f54,f55,f56,f57,f58,f59,f60,f61');
61
+ url.searchParams.set('ut', 'b2884a393a59ad64002292a3e90d46a5');
62
+
63
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
64
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `eastmoney kline failed: HTTP ${resp.status}`);
65
+ const data = await resp.json();
66
+ /** @type {string[]} */
67
+ const raw = Array.isArray(data?.data?.klines) ? data.data.klines : [];
68
+ if (raw.length === 0) throw new CliError('NO_DATA', `No kline data for ${args.symbol}`);
69
+
70
+ return raw.slice(-limit).map((line) => {
71
+ const [date, open, close, high, low, volume, turnover, amplitude, changePercent, change, turnoverRate] = line.split(',');
72
+ return {
73
+ date,
74
+ open: Number(open),
75
+ close: Number(close),
76
+ high: Number(high),
77
+ low: Number(low),
78
+ volume: Number(volume),
79
+ turnover: Number(turnover),
80
+ amplitude: Number(amplitude),
81
+ changePercent: Number(changePercent),
82
+ change: Number(change),
83
+ turnoverRate: Number(turnoverRate),
84
+ };
85
+ });
86
+ },
87
+ });
@@ -0,0 +1,54 @@
1
+ // eastmoney kuaixun — 7x24 real-time market news feed.
2
+ //
3
+ // opencli eastmoney kuaixun
4
+ // opencli eastmoney kuaixun --column 102 --limit 30
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ // Known columns on eastmoney 7x24:
10
+ // 102 = 重要 (default)
11
+ // 101 = 全部
12
+ // 104 = 公司
13
+ // 105 = 市场
14
+ // 106 = 机构
15
+ // 107 = 宏观
16
+
17
+ cli({
18
+ site: 'eastmoney',
19
+ name: 'kuaixun',
20
+ description: '东方财富 7x24 财经快讯',
21
+ domain: 'np-listapi.eastmoney.com',
22
+ strategy: Strategy.PUBLIC,
23
+ browser: false,
24
+ args: [
25
+ { name: 'column', type: 'string', default: '102', help: '频道:102 (重要) / 101 (全部) / 104 / 105 / 106 / 107' },
26
+ { name: 'limit', type: 'int', default: 20, help: '返回数量 (max 100)' },
27
+ ],
28
+ columns: ['time', 'title', 'summary', 'stocks'],
29
+ func: async (_page, args) => {
30
+ const column = String(args.column ?? '102').trim();
31
+ const limit = Math.max(1, Math.min(Number(args.limit) || 20, 100));
32
+
33
+ const url = new URL('https://np-listapi.eastmoney.com/comm/web/getFastNewsList');
34
+ url.searchParams.set('client', 'web');
35
+ url.searchParams.set('biz', 'web_724');
36
+ url.searchParams.set('fastColumn', column);
37
+ url.searchParams.set('sortEnd', '');
38
+ url.searchParams.set('pageSize', String(limit));
39
+ url.searchParams.set('req_trace', '1');
40
+
41
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0', Accept: 'application/json' } });
42
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `kuaixun failed: HTTP ${resp.status}`);
43
+ const data = await resp.json();
44
+ const list = Array.isArray(data?.data?.fastNewsList) ? data.data.fastNewsList : [];
45
+ if (list.length === 0) throw new CliError('NO_DATA', 'eastmoney returned no kuaixun data');
46
+
47
+ return list.slice(0, limit).map((it) => ({
48
+ time: it.showTime,
49
+ title: it.title,
50
+ summary: (it.summary || '').replace(/\s+/g, ' ').slice(0, 400),
51
+ stocks: Array.isArray(it.stockList) ? it.stockList.join(', ') : '',
52
+ }));
53
+ },
54
+ });
@@ -0,0 +1,67 @@
1
+ // eastmoney longhu — dragon & tiger list (龙虎榜).
2
+ //
3
+ // opencli eastmoney longhu
4
+ // opencli eastmoney longhu --date 2025-12-10 --limit 20
5
+
6
+ import { cli, Strategy } from '@jackwener/opencli/registry';
7
+ import { CliError } from '@jackwener/opencli/errors';
8
+
9
+ function defaultTradeDate() {
10
+ // Default window = 30 days back; results sorted DESC so latest comes first.
11
+ // This avoids missing data on weekends/holidays when "yesterday" had no trading.
12
+ const d = new Date(Date.now() + 8 * 3600 * 1000);
13
+ d.setUTCDate(d.getUTCDate() - 30);
14
+ return d.toISOString().slice(0, 10);
15
+ }
16
+
17
+ cli({
18
+ site: 'eastmoney',
19
+ name: 'longhu',
20
+ description: '龙虎榜明细(A股交易所公开披露榜单)',
21
+ domain: 'datacenter-web.eastmoney.com',
22
+ strategy: Strategy.PUBLIC,
23
+ browser: false,
24
+ args: [
25
+ { name: 'date', type: 'string', default: '', help: '开始交易日 YYYY-MM-DD (默认昨天)' },
26
+ { name: 'limit', type: 'int', default: 20, help: '返回数量 (max 100)' },
27
+ ],
28
+ columns: ['tradeDate', 'code', 'name', 'closePrice', 'changeRate', 'boardAmt', 'buyAmt', 'sellAmt', 'netAmt', 'turnover', 'dealRatio', 'market', 'reason'],
29
+ func: async (_page, args) => {
30
+ const sinceDate = String(args.date || '').trim() || defaultTradeDate();
31
+ const limit = Math.max(1, Math.min(Number(args.limit) || 20, 100));
32
+
33
+ const url = new URL('https://datacenter-web.eastmoney.com/api/data/v1/get');
34
+ url.searchParams.set('sortColumns', 'TRADE_DATE,SECURITY_CODE');
35
+ url.searchParams.set('sortTypes', '-1,1');
36
+ url.searchParams.set('pageSize', String(limit));
37
+ url.searchParams.set('pageNumber', '1');
38
+ url.searchParams.set('reportName', 'RPT_DAILYBILLBOARD_DETAILS');
39
+ url.searchParams.set('columns', 'ALL');
40
+ url.searchParams.set('source', 'WEB');
41
+ url.searchParams.set('client', 'WEB');
42
+ url.searchParams.set('filter', `(TRADE_DATE>='${sinceDate}')`);
43
+
44
+ const resp = await fetch(url, { headers: { 'User-Agent': 'Mozilla/5.0' } });
45
+ if (!resp.ok) throw new CliError('HTTP_ERROR', `longhu failed: HTTP ${resp.status}`);
46
+ const data = await resp.json();
47
+ /** @type {any[]} */
48
+ const rows = Array.isArray(data?.result?.data) ? data.result.data : [];
49
+ if (rows.length === 0) throw new CliError('NO_DATA', `No longhu data since ${sinceDate}`);
50
+
51
+ return rows.slice(0, limit).map((it) => ({
52
+ tradeDate: String(it.TRADE_DATE || '').slice(0, 10),
53
+ code: it.SECURITY_CODE,
54
+ name: it.SECURITY_NAME_ABBR,
55
+ closePrice: it.CLOSE_PRICE,
56
+ changeRate: it.CHANGE_RATE,
57
+ boardAmt: it.BILLBOARD_DEAL_AMT,
58
+ buyAmt: it.BILLBOARD_BUY_AMT,
59
+ sellAmt: it.BILLBOARD_SELL_AMT,
60
+ netAmt: it.BILLBOARD_NET_AMT,
61
+ turnover: it.ACCUM_AMOUNT,
62
+ dealRatio: it.DEAL_AMOUNT_RATIO,
63
+ market: it.TRADE_MARKET,
64
+ reason: it.EXPLANATION,
65
+ }));
66
+ },
67
+ });