stock-sdk 1.0.0
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/.cursor/worktrees.json +5 -0
- package/.yarn/install-state.gz +0 -0
- package/README.md +330 -0
- package/dist/index.d.ts +59 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +293 -0
- package/dist/types.d.ts +177 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/new-doc.md +98 -0
- package/origin.md +480 -0
- package/package.json +32 -0
- package/src/index.test.ts +97 -0
- package/src/index.ts +281 -0
- package/src/types.ts +177 -0
- package/tsconfig.json +20 -0
- package/vitest.config.ts +9 -0
|
Binary file
|
package/README.md
ADDED
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
# Stock SDK
|
|
2
|
+
|
|
3
|
+
基于腾讯财经 `qt.gtimg.cn` 非官方接口封装的 TypeScript SDK,支持 A 股、港股、美股、公募基金实时行情查询。
|
|
4
|
+
|
|
5
|
+
## 安装
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install stock-sdk
|
|
9
|
+
# 或
|
|
10
|
+
yarn add stock-sdk
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
## 快速开始
|
|
14
|
+
|
|
15
|
+
```typescript
|
|
16
|
+
import StockSDK from 'stock-sdk';
|
|
17
|
+
|
|
18
|
+
const sdk = new StockSDK();
|
|
19
|
+
|
|
20
|
+
// A 股全量行情
|
|
21
|
+
const fullQuotes = await sdk.getFullQuotes(['sz000858', 'sh600000']);
|
|
22
|
+
console.log(fullQuotes);
|
|
23
|
+
|
|
24
|
+
// 简要行情
|
|
25
|
+
const simpleQuotes = await sdk.getSimpleQuotes(['s_sz000858', 's_sh000001']);
|
|
26
|
+
console.log(simpleQuotes);
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
## API 文档
|
|
30
|
+
|
|
31
|
+
### `getFullQuotes(codes: string[]): Promise<FullQuote[]>`
|
|
32
|
+
|
|
33
|
+
获取 A 股 / 指数全量行情数据。
|
|
34
|
+
|
|
35
|
+
**参数**
|
|
36
|
+
|
|
37
|
+
- `codes` — 股票代码数组,格式 `<market><code>`,如 `['sz000858', 'sh600000']`
|
|
38
|
+
|
|
39
|
+
**返回**
|
|
40
|
+
|
|
41
|
+
```typescript
|
|
42
|
+
interface FullQuote {
|
|
43
|
+
marketId: string; // 市场标识
|
|
44
|
+
name: string; // 名称
|
|
45
|
+
code: string; // 股票代码
|
|
46
|
+
price: number; // 最新价
|
|
47
|
+
prevClose: number; // 昨收
|
|
48
|
+
open: number; // 今开
|
|
49
|
+
volume: number; // 成交量(手)
|
|
50
|
+
outerVolume: number; // 外盘
|
|
51
|
+
innerVolume: number; // 内盘
|
|
52
|
+
bid: { price: number; volume: number }[]; // 买一~买五
|
|
53
|
+
ask: { price: number; volume: number }[]; // 卖一~卖五
|
|
54
|
+
time: string; // 时间戳 yyyyMMddHHmmss
|
|
55
|
+
change: number; // 涨跌额
|
|
56
|
+
changePercent: number; // 涨跌幅%
|
|
57
|
+
high: number; // 最高
|
|
58
|
+
low: number; // 最低
|
|
59
|
+
volume2: number; // 成交量(手)
|
|
60
|
+
amount: number; // 成交额(万)
|
|
61
|
+
turnoverRate: number | null; // 换手率%
|
|
62
|
+
pe: number | null; // 市盈率
|
|
63
|
+
circulatingMarketCap: number | null; // 流通市值(亿)
|
|
64
|
+
totalMarketCap: number | null; // 总市值(亿)
|
|
65
|
+
pb: number | null; // 市净率
|
|
66
|
+
limitUp: number | null; // 涨停价
|
|
67
|
+
limitDown: number | null; // 跌停价
|
|
68
|
+
raw: string[]; // 原始字段数组
|
|
69
|
+
}
|
|
70
|
+
```
|
|
71
|
+
|
|
72
|
+
**示例**
|
|
73
|
+
|
|
74
|
+
```typescript
|
|
75
|
+
const quotes = await sdk.getFullQuotes(['sz000858']);
|
|
76
|
+
console.log(quotes[0].name); // 五 粮 液
|
|
77
|
+
console.log(quotes[0].price); // 111.70
|
|
78
|
+
```
|
|
79
|
+
|
|
80
|
+
---
|
|
81
|
+
|
|
82
|
+
### `getSimpleQuotes(codes: string[]): Promise<SimpleQuote[]>`
|
|
83
|
+
|
|
84
|
+
获取简要行情(股票 / 指数)。
|
|
85
|
+
|
|
86
|
+
**参数**
|
|
87
|
+
|
|
88
|
+
- `codes` — 代码数组,格式 `s_<market><code>`,如 `['s_sz000858', 's_sh000001']`
|
|
89
|
+
|
|
90
|
+
**返回**
|
|
91
|
+
|
|
92
|
+
```typescript
|
|
93
|
+
interface SimpleQuote {
|
|
94
|
+
marketId: string;
|
|
95
|
+
name: string;
|
|
96
|
+
code: string;
|
|
97
|
+
price: number;
|
|
98
|
+
change: number;
|
|
99
|
+
changePercent: number;
|
|
100
|
+
volume: number;
|
|
101
|
+
amount: number;
|
|
102
|
+
marketCap: number | null; // 总市值(亿)
|
|
103
|
+
marketType: string; // 如 GP-A / ZS
|
|
104
|
+
raw: string[];
|
|
105
|
+
}
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
**示例**
|
|
109
|
+
|
|
110
|
+
```typescript
|
|
111
|
+
const quotes = await sdk.getSimpleQuotes(['s_sh000001']);
|
|
112
|
+
console.log(quotes[0].name); // 上证指数
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
---
|
|
116
|
+
|
|
117
|
+
### `getFundFlow(codes: string[]): Promise<FundFlow[]>`
|
|
118
|
+
|
|
119
|
+
获取资金流向数据。
|
|
120
|
+
|
|
121
|
+
**参数**
|
|
122
|
+
|
|
123
|
+
- `codes` — 代码数组,格式 `ff_<market><code>`,如 `['ff_sz000858']`
|
|
124
|
+
|
|
125
|
+
**返回**
|
|
126
|
+
|
|
127
|
+
```typescript
|
|
128
|
+
interface FundFlow {
|
|
129
|
+
code: string;
|
|
130
|
+
mainInflow: number; // 主力流入
|
|
131
|
+
mainOutflow: number; // 主力流出
|
|
132
|
+
mainNet: number; // 主力净流入
|
|
133
|
+
mainNetRatio: number; // 主力净流入占比
|
|
134
|
+
retailInflow: number; // 散户流入
|
|
135
|
+
retailOutflow: number; // 散户流出
|
|
136
|
+
retailNet: number; // 散户净流入
|
|
137
|
+
retailNetRatio: number;// 散户净流入占比
|
|
138
|
+
totalFlow: number; // 总资金流
|
|
139
|
+
name: string;
|
|
140
|
+
date: string;
|
|
141
|
+
raw: string[];
|
|
142
|
+
}
|
|
143
|
+
```
|
|
144
|
+
|
|
145
|
+
**示例**
|
|
146
|
+
|
|
147
|
+
```typescript
|
|
148
|
+
const flows = await sdk.getFundFlow(['ff_sz000858']);
|
|
149
|
+
console.log(flows[0].mainNet);
|
|
150
|
+
```
|
|
151
|
+
|
|
152
|
+
---
|
|
153
|
+
|
|
154
|
+
### `getPanelLargeOrder(codes: string[]): Promise<PanelLargeOrder[]>`
|
|
155
|
+
|
|
156
|
+
获取盘口大单占比。
|
|
157
|
+
|
|
158
|
+
**参数**
|
|
159
|
+
|
|
160
|
+
- `codes` — 代码数组,格式 `s_pk<market><code>`,如 `['s_pksz000858']`
|
|
161
|
+
|
|
162
|
+
**返回**
|
|
163
|
+
|
|
164
|
+
```typescript
|
|
165
|
+
interface PanelLargeOrder {
|
|
166
|
+
buyLargeRatio: number; // 买盘大单占比
|
|
167
|
+
buySmallRatio: number; // 买盘小单占比
|
|
168
|
+
sellLargeRatio: number; // 卖盘大单占比
|
|
169
|
+
sellSmallRatio: number; // 卖盘小单占比
|
|
170
|
+
raw: string[];
|
|
171
|
+
}
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
**示例**
|
|
175
|
+
|
|
176
|
+
```typescript
|
|
177
|
+
const orders = await sdk.getPanelLargeOrder(['s_pksz000858']);
|
|
178
|
+
console.log(orders[0].buyLargeRatio);
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
---
|
|
182
|
+
|
|
183
|
+
### `getHKQuotes(codes: string[]): Promise<HKQuote[]>`
|
|
184
|
+
|
|
185
|
+
获取港股扩展行情。
|
|
186
|
+
|
|
187
|
+
**参数**
|
|
188
|
+
|
|
189
|
+
- `codes` — 代码数组,格式 `r_hk<code>`,如 `['r_hk09988']`
|
|
190
|
+
|
|
191
|
+
**返回**
|
|
192
|
+
|
|
193
|
+
```typescript
|
|
194
|
+
interface HKQuote {
|
|
195
|
+
marketId: string;
|
|
196
|
+
name: string;
|
|
197
|
+
code: string;
|
|
198
|
+
price: number;
|
|
199
|
+
prevClose: number;
|
|
200
|
+
open: number;
|
|
201
|
+
volume: number;
|
|
202
|
+
time: string;
|
|
203
|
+
change: number;
|
|
204
|
+
changePercent: number;
|
|
205
|
+
high: number;
|
|
206
|
+
low: number;
|
|
207
|
+
amount: number;
|
|
208
|
+
lotSize: number | null;
|
|
209
|
+
circulatingMarketCap: number | null;
|
|
210
|
+
totalMarketCap: number | null;
|
|
211
|
+
currency: string;
|
|
212
|
+
raw: string[];
|
|
213
|
+
}
|
|
214
|
+
```
|
|
215
|
+
|
|
216
|
+
**示例**
|
|
217
|
+
|
|
218
|
+
```typescript
|
|
219
|
+
const quotes = await sdk.getHKQuotes(['r_hk09988']);
|
|
220
|
+
console.log(quotes[0].name); // 阿里巴巴-W
|
|
221
|
+
```
|
|
222
|
+
|
|
223
|
+
---
|
|
224
|
+
|
|
225
|
+
### `getUSQuotes(codes: string[]): Promise<USQuote[]>`
|
|
226
|
+
|
|
227
|
+
获取美股简要行情。
|
|
228
|
+
|
|
229
|
+
**参数**
|
|
230
|
+
|
|
231
|
+
- `codes` — 代码数组,格式 `s_us<code>`,如 `['s_usBABA']`
|
|
232
|
+
|
|
233
|
+
**返回**
|
|
234
|
+
|
|
235
|
+
```typescript
|
|
236
|
+
interface USQuote {
|
|
237
|
+
marketId: string;
|
|
238
|
+
name: string;
|
|
239
|
+
code: string;
|
|
240
|
+
price: number;
|
|
241
|
+
change: number;
|
|
242
|
+
changePercent: number;
|
|
243
|
+
volume: number;
|
|
244
|
+
amount: number;
|
|
245
|
+
marketCap: number | null;
|
|
246
|
+
raw: string[];
|
|
247
|
+
}
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
**示例**
|
|
251
|
+
|
|
252
|
+
```typescript
|
|
253
|
+
const quotes = await sdk.getUSQuotes(['s_usBABA']);
|
|
254
|
+
console.log(quotes[0].code); // BABA.N
|
|
255
|
+
```
|
|
256
|
+
|
|
257
|
+
---
|
|
258
|
+
|
|
259
|
+
### `getFundQuotes(codes: string[]): Promise<FundQuote[]>`
|
|
260
|
+
|
|
261
|
+
获取公募基金行情。
|
|
262
|
+
|
|
263
|
+
**参数**
|
|
264
|
+
|
|
265
|
+
- `codes` — 代码数组,格式 `jj<六位代码>`,如 `['jj000001']`
|
|
266
|
+
|
|
267
|
+
**返回**
|
|
268
|
+
|
|
269
|
+
```typescript
|
|
270
|
+
interface FundQuote {
|
|
271
|
+
code: string;
|
|
272
|
+
name: string;
|
|
273
|
+
nav: number; // 最新单位净值
|
|
274
|
+
accNav: number; // 累计净值
|
|
275
|
+
change: number; // 当日涨跌额
|
|
276
|
+
navDate: string; // 净值日期
|
|
277
|
+
raw: string[];
|
|
278
|
+
}
|
|
279
|
+
```
|
|
280
|
+
|
|
281
|
+
**示例**
|
|
282
|
+
|
|
283
|
+
```typescript
|
|
284
|
+
const funds = await sdk.getFundQuotes(['jj000001']);
|
|
285
|
+
console.log(funds[0].name); // 华夏成长混合
|
|
286
|
+
```
|
|
287
|
+
|
|
288
|
+
---
|
|
289
|
+
|
|
290
|
+
### `batchRaw(params: string): Promise<{ key: string; fields: string[] }[]>`
|
|
291
|
+
|
|
292
|
+
批量混合查询,返回原始解析结果。
|
|
293
|
+
|
|
294
|
+
**参数**
|
|
295
|
+
|
|
296
|
+
- `params` — 逗号分隔的多个查询参数,如 `'sz000858,s_sh000001,jj000001'`
|
|
297
|
+
|
|
298
|
+
**返回**
|
|
299
|
+
|
|
300
|
+
```typescript
|
|
301
|
+
{ key: string; fields: string[] }[]
|
|
302
|
+
```
|
|
303
|
+
|
|
304
|
+
**示例**
|
|
305
|
+
|
|
306
|
+
```typescript
|
|
307
|
+
const raw = await sdk.batchRaw('sz000858,s_sh000001');
|
|
308
|
+
console.log(raw[0].key); // sz000858
|
|
309
|
+
console.log(raw[0].fields); // ['51', '五 粮 液', '000858', ...]
|
|
310
|
+
```
|
|
311
|
+
|
|
312
|
+
---
|
|
313
|
+
|
|
314
|
+
## 开发
|
|
315
|
+
|
|
316
|
+
```bash
|
|
317
|
+
# 安装依赖
|
|
318
|
+
npm install
|
|
319
|
+
|
|
320
|
+
# 运行测试
|
|
321
|
+
npm test
|
|
322
|
+
|
|
323
|
+
# 构建
|
|
324
|
+
npm run build
|
|
325
|
+
```
|
|
326
|
+
|
|
327
|
+
## 许可证
|
|
328
|
+
|
|
329
|
+
ISC
|
|
330
|
+
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { FullQuote, SimpleQuote, FundFlow, PanelLargeOrder, HKQuote, USQuote, FundQuote } from './types';
|
|
2
|
+
export * from './types';
|
|
3
|
+
export declare class TencentStockSDK {
|
|
4
|
+
private client;
|
|
5
|
+
constructor();
|
|
6
|
+
private fetch;
|
|
7
|
+
/**
|
|
8
|
+
* 获取 A 股 / 指数 全量行情
|
|
9
|
+
* @param codes 如 ['sz000858', 'sh600000']
|
|
10
|
+
*/
|
|
11
|
+
getFullQuotes(codes: string[]): Promise<FullQuote[]>;
|
|
12
|
+
private parseFullQuote;
|
|
13
|
+
/**
|
|
14
|
+
* 获取简要行情
|
|
15
|
+
* @param codes 如 ['s_sz000858', 's_sh000001']
|
|
16
|
+
*/
|
|
17
|
+
getSimpleQuotes(codes: string[]): Promise<SimpleQuote[]>;
|
|
18
|
+
private parseSimpleQuote;
|
|
19
|
+
/**
|
|
20
|
+
* 获取资金流向
|
|
21
|
+
* @param codes 如 ['ff_sz000858']
|
|
22
|
+
*/
|
|
23
|
+
getFundFlow(codes: string[]): Promise<FundFlow[]>;
|
|
24
|
+
private parseFundFlow;
|
|
25
|
+
/**
|
|
26
|
+
* 获取盘口大单占比
|
|
27
|
+
* @param codes 如 ['s_pksz000858']
|
|
28
|
+
*/
|
|
29
|
+
getPanelLargeOrder(codes: string[]): Promise<PanelLargeOrder[]>;
|
|
30
|
+
private parsePanelLargeOrder;
|
|
31
|
+
/**
|
|
32
|
+
* 获取港股扩展行情
|
|
33
|
+
* @param codes 如 ['r_hk09988']
|
|
34
|
+
*/
|
|
35
|
+
getHKQuotes(codes: string[]): Promise<HKQuote[]>;
|
|
36
|
+
private parseHKQuote;
|
|
37
|
+
/**
|
|
38
|
+
* 获取美股简要行情
|
|
39
|
+
* @param codes 如 ['s_usBABA']
|
|
40
|
+
*/
|
|
41
|
+
getUSQuotes(codes: string[]): Promise<USQuote[]>;
|
|
42
|
+
private parseUSQuote;
|
|
43
|
+
/**
|
|
44
|
+
* 获取公募基金行情
|
|
45
|
+
* @param codes 如 ['jj000001']
|
|
46
|
+
*/
|
|
47
|
+
getFundQuotes(codes: string[]): Promise<FundQuote[]>;
|
|
48
|
+
private parseFundQuote;
|
|
49
|
+
/**
|
|
50
|
+
* 批量混合查询,返回原始解析结果(key + fields)
|
|
51
|
+
* @param params 如 'sz000858,s_sh000001,jj000001'
|
|
52
|
+
*/
|
|
53
|
+
batchRaw(params: string): Promise<{
|
|
54
|
+
key: string;
|
|
55
|
+
fields: string[];
|
|
56
|
+
}[]>;
|
|
57
|
+
}
|
|
58
|
+
export default TencentStockSDK;
|
|
59
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,SAAS,EACT,WAAW,EACX,QAAQ,EACR,eAAe,EACf,OAAO,EACP,OAAO,EACP,SAAS,EACV,MAAM,SAAS,CAAC;AAEjB,cAAc,SAAS,CAAC;AAqCxB,qBAAa,eAAe;IAC1B,OAAO,CAAC,MAAM,CAAgB;;YAUhB,KAAK;IAOnB;;;OAGG;IACG,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,SAAS,EAAE,CAAC;IAK1D,OAAO,CAAC,cAAc;IAwCtB;;;OAGG;IACG,eAAe,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC;IAK9D,OAAO,CAAC,gBAAgB;IAiBxB;;;OAGG;IACG,WAAW,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAKvD,OAAO,CAAC,aAAa;IAmBrB;;;OAGG;IACG,kBAAkB,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,eAAe,EAAE,CAAC;IAKrE,OAAO,CAAC,oBAAoB;IAW5B;;;OAGG;IACG,WAAW,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;IAKtD,OAAO,CAAC,YAAY;IAwBpB;;;OAGG;IACG,WAAW,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;IAKtD,OAAO,CAAC,YAAY;IAgBpB;;;OAGG;IACG,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,SAAS,EAAE,CAAC;IAK1D,OAAO,CAAC,cAAc;IAatB;;;OAGG;IACG,QAAQ,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;QAAE,GAAG,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,EAAE,CAAA;KAAE,EAAE,CAAC;CAG7E;AAED,eAAe,eAAe,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
36
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
37
|
+
};
|
|
38
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
39
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
40
|
+
};
|
|
41
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
42
|
+
exports.TencentStockSDK = void 0;
|
|
43
|
+
const axios_1 = __importDefault(require("axios"));
|
|
44
|
+
const iconv = __importStar(require("iconv-lite"));
|
|
45
|
+
__exportStar(require("./types"), exports);
|
|
46
|
+
const BASE_URL = 'http://qt.gtimg.cn';
|
|
47
|
+
function safeNumber(val) {
|
|
48
|
+
if (!val || val === '')
|
|
49
|
+
return 0;
|
|
50
|
+
const n = parseFloat(val);
|
|
51
|
+
return Number.isNaN(n) ? 0 : n;
|
|
52
|
+
}
|
|
53
|
+
function safeNumberOrNull(val) {
|
|
54
|
+
if (!val || val === '')
|
|
55
|
+
return null;
|
|
56
|
+
const n = parseFloat(val);
|
|
57
|
+
return Number.isNaN(n) ? null : n;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* 解析响应文本,按 `;` 拆行,提取 `v_xxx="..."` 里的内容,返回 { key, fields }[]
|
|
61
|
+
*/
|
|
62
|
+
function parseResponse(text) {
|
|
63
|
+
const lines = text.split(';').map((l) => l.trim()).filter(Boolean);
|
|
64
|
+
const results = [];
|
|
65
|
+
for (const line of lines) {
|
|
66
|
+
const eqIdx = line.indexOf('=');
|
|
67
|
+
if (eqIdx < 0)
|
|
68
|
+
continue;
|
|
69
|
+
let key = line.slice(0, eqIdx).trim();
|
|
70
|
+
if (key.startsWith('v_'))
|
|
71
|
+
key = key.slice(2);
|
|
72
|
+
let raw = line.slice(eqIdx + 1).trim();
|
|
73
|
+
if (raw.startsWith('"') && raw.endsWith('"')) {
|
|
74
|
+
raw = raw.slice(1, -1);
|
|
75
|
+
}
|
|
76
|
+
const fields = raw.split('~');
|
|
77
|
+
results.push({ key, fields });
|
|
78
|
+
}
|
|
79
|
+
return results;
|
|
80
|
+
}
|
|
81
|
+
class TencentStockSDK {
|
|
82
|
+
constructor() {
|
|
83
|
+
this.client = axios_1.default.create({
|
|
84
|
+
baseURL: BASE_URL,
|
|
85
|
+
responseType: 'arraybuffer',
|
|
86
|
+
timeout: 10000,
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
async fetch(params) {
|
|
90
|
+
const resp = await this.client.get('/', { params: { q: params } });
|
|
91
|
+
const text = iconv.decode(Buffer.from(resp.data), 'gbk');
|
|
92
|
+
return parseResponse(text);
|
|
93
|
+
}
|
|
94
|
+
// ---------- 实时全量行情 ----------
|
|
95
|
+
/**
|
|
96
|
+
* 获取 A 股 / 指数 全量行情
|
|
97
|
+
* @param codes 如 ['sz000858', 'sh600000']
|
|
98
|
+
*/
|
|
99
|
+
async getFullQuotes(codes) {
|
|
100
|
+
const data = await this.fetch(codes.join(','));
|
|
101
|
+
return data.map((d) => this.parseFullQuote(d.fields));
|
|
102
|
+
}
|
|
103
|
+
parseFullQuote(f) {
|
|
104
|
+
const bid = [];
|
|
105
|
+
for (let i = 0; i < 5; i++) {
|
|
106
|
+
bid.push({ price: safeNumber(f[9 + i * 2]), volume: safeNumber(f[10 + i * 2]) });
|
|
107
|
+
}
|
|
108
|
+
const ask = [];
|
|
109
|
+
for (let i = 0; i < 5; i++) {
|
|
110
|
+
ask.push({ price: safeNumber(f[19 + i * 2]), volume: safeNumber(f[20 + i * 2]) });
|
|
111
|
+
}
|
|
112
|
+
return {
|
|
113
|
+
marketId: f[0] ?? '',
|
|
114
|
+
name: f[1] ?? '',
|
|
115
|
+
code: f[2] ?? '',
|
|
116
|
+
price: safeNumber(f[3]),
|
|
117
|
+
prevClose: safeNumber(f[4]),
|
|
118
|
+
open: safeNumber(f[5]),
|
|
119
|
+
volume: safeNumber(f[6]),
|
|
120
|
+
outerVolume: safeNumber(f[7]),
|
|
121
|
+
innerVolume: safeNumber(f[8]),
|
|
122
|
+
bid,
|
|
123
|
+
ask,
|
|
124
|
+
time: f[30] ?? '',
|
|
125
|
+
change: safeNumber(f[31]),
|
|
126
|
+
changePercent: safeNumber(f[32]),
|
|
127
|
+
high: safeNumber(f[33]),
|
|
128
|
+
low: safeNumber(f[34]),
|
|
129
|
+
volume2: safeNumber(f[36]),
|
|
130
|
+
amount: safeNumber(f[37]),
|
|
131
|
+
turnoverRate: safeNumberOrNull(f[38]),
|
|
132
|
+
pe: safeNumberOrNull(f[39]),
|
|
133
|
+
circulatingMarketCap: safeNumberOrNull(f[44]),
|
|
134
|
+
totalMarketCap: safeNumberOrNull(f[45]),
|
|
135
|
+
pb: safeNumberOrNull(f[46]),
|
|
136
|
+
limitUp: safeNumberOrNull(f[47]),
|
|
137
|
+
limitDown: safeNumberOrNull(f[48]),
|
|
138
|
+
raw: f,
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
// ---------- 简要行情 ----------
|
|
142
|
+
/**
|
|
143
|
+
* 获取简要行情
|
|
144
|
+
* @param codes 如 ['s_sz000858', 's_sh000001']
|
|
145
|
+
*/
|
|
146
|
+
async getSimpleQuotes(codes) {
|
|
147
|
+
const data = await this.fetch(codes.join(','));
|
|
148
|
+
return data.map((d) => this.parseSimpleQuote(d.fields));
|
|
149
|
+
}
|
|
150
|
+
parseSimpleQuote(f) {
|
|
151
|
+
return {
|
|
152
|
+
marketId: f[0] ?? '',
|
|
153
|
+
name: f[1] ?? '',
|
|
154
|
+
code: f[2] ?? '',
|
|
155
|
+
price: safeNumber(f[3]),
|
|
156
|
+
change: safeNumber(f[4]),
|
|
157
|
+
changePercent: safeNumber(f[5]),
|
|
158
|
+
volume: safeNumber(f[6]),
|
|
159
|
+
amount: safeNumber(f[7]),
|
|
160
|
+
marketCap: safeNumberOrNull(f[9]),
|
|
161
|
+
marketType: f[10] ?? '',
|
|
162
|
+
raw: f,
|
|
163
|
+
};
|
|
164
|
+
}
|
|
165
|
+
// ---------- 资金流向 ----------
|
|
166
|
+
/**
|
|
167
|
+
* 获取资金流向
|
|
168
|
+
* @param codes 如 ['ff_sz000858']
|
|
169
|
+
*/
|
|
170
|
+
async getFundFlow(codes) {
|
|
171
|
+
const data = await this.fetch(codes.join(','));
|
|
172
|
+
return data.map((d) => this.parseFundFlow(d.fields));
|
|
173
|
+
}
|
|
174
|
+
parseFundFlow(f) {
|
|
175
|
+
return {
|
|
176
|
+
code: f[0] ?? '',
|
|
177
|
+
mainInflow: safeNumber(f[1]),
|
|
178
|
+
mainOutflow: safeNumber(f[2]),
|
|
179
|
+
mainNet: safeNumber(f[3]),
|
|
180
|
+
mainNetRatio: safeNumber(f[4]),
|
|
181
|
+
retailInflow: safeNumber(f[5]),
|
|
182
|
+
retailOutflow: safeNumber(f[6]),
|
|
183
|
+
retailNet: safeNumber(f[7]),
|
|
184
|
+
retailNetRatio: safeNumber(f[8]),
|
|
185
|
+
totalFlow: safeNumber(f[9]),
|
|
186
|
+
name: f[12] ?? '',
|
|
187
|
+
date: f[13] ?? '',
|
|
188
|
+
raw: f,
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
// ---------- 盘口大单占比 ----------
|
|
192
|
+
/**
|
|
193
|
+
* 获取盘口大单占比
|
|
194
|
+
* @param codes 如 ['s_pksz000858']
|
|
195
|
+
*/
|
|
196
|
+
async getPanelLargeOrder(codes) {
|
|
197
|
+
const data = await this.fetch(codes.join(','));
|
|
198
|
+
return data.map((d) => this.parsePanelLargeOrder(d.fields));
|
|
199
|
+
}
|
|
200
|
+
parsePanelLargeOrder(f) {
|
|
201
|
+
return {
|
|
202
|
+
buyLargeRatio: safeNumber(f[0]),
|
|
203
|
+
buySmallRatio: safeNumber(f[1]),
|
|
204
|
+
sellLargeRatio: safeNumber(f[2]),
|
|
205
|
+
sellSmallRatio: safeNumber(f[3]),
|
|
206
|
+
raw: f,
|
|
207
|
+
};
|
|
208
|
+
}
|
|
209
|
+
// ---------- 港股扩展行情 ----------
|
|
210
|
+
/**
|
|
211
|
+
* 获取港股扩展行情
|
|
212
|
+
* @param codes 如 ['r_hk09988']
|
|
213
|
+
*/
|
|
214
|
+
async getHKQuotes(codes) {
|
|
215
|
+
const data = await this.fetch(codes.join(','));
|
|
216
|
+
return data.map((d) => this.parseHKQuote(d.fields));
|
|
217
|
+
}
|
|
218
|
+
parseHKQuote(f) {
|
|
219
|
+
return {
|
|
220
|
+
marketId: f[0] ?? '',
|
|
221
|
+
name: f[1] ?? '',
|
|
222
|
+
code: f[2] ?? '',
|
|
223
|
+
price: safeNumber(f[3]),
|
|
224
|
+
prevClose: safeNumber(f[4]),
|
|
225
|
+
open: safeNumber(f[5]),
|
|
226
|
+
volume: safeNumber(f[6]),
|
|
227
|
+
time: f[30] ?? '',
|
|
228
|
+
change: safeNumber(f[31]),
|
|
229
|
+
changePercent: safeNumber(f[32]),
|
|
230
|
+
high: safeNumber(f[33]),
|
|
231
|
+
low: safeNumber(f[34]),
|
|
232
|
+
amount: safeNumber(f[36]),
|
|
233
|
+
lotSize: safeNumberOrNull(f[40]),
|
|
234
|
+
circulatingMarketCap: safeNumberOrNull(f[46]),
|
|
235
|
+
totalMarketCap: safeNumberOrNull(f[47]),
|
|
236
|
+
currency: f[f.length - 3] ?? '',
|
|
237
|
+
raw: f,
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
// ---------- 美股简要行情 ----------
|
|
241
|
+
/**
|
|
242
|
+
* 获取美股简要行情
|
|
243
|
+
* @param codes 如 ['s_usBABA']
|
|
244
|
+
*/
|
|
245
|
+
async getUSQuotes(codes) {
|
|
246
|
+
const data = await this.fetch(codes.join(','));
|
|
247
|
+
return data.map((d) => this.parseUSQuote(d.fields));
|
|
248
|
+
}
|
|
249
|
+
parseUSQuote(f) {
|
|
250
|
+
return {
|
|
251
|
+
marketId: f[0] ?? '',
|
|
252
|
+
name: f[1] ?? '',
|
|
253
|
+
code: f[2] ?? '',
|
|
254
|
+
price: safeNumber(f[3]),
|
|
255
|
+
change: safeNumber(f[4]),
|
|
256
|
+
changePercent: safeNumber(f[5]),
|
|
257
|
+
volume: safeNumber(f[6]),
|
|
258
|
+
amount: safeNumber(f[7]),
|
|
259
|
+
marketCap: safeNumberOrNull(f[8]),
|
|
260
|
+
raw: f,
|
|
261
|
+
};
|
|
262
|
+
}
|
|
263
|
+
// ---------- 公募基金行情 ----------
|
|
264
|
+
/**
|
|
265
|
+
* 获取公募基金行情
|
|
266
|
+
* @param codes 如 ['jj000001']
|
|
267
|
+
*/
|
|
268
|
+
async getFundQuotes(codes) {
|
|
269
|
+
const data = await this.fetch(codes.join(','));
|
|
270
|
+
return data.map((d) => this.parseFundQuote(d.fields));
|
|
271
|
+
}
|
|
272
|
+
parseFundQuote(f) {
|
|
273
|
+
return {
|
|
274
|
+
code: f[0] ?? '',
|
|
275
|
+
name: f[1] ?? '',
|
|
276
|
+
nav: safeNumber(f[5]),
|
|
277
|
+
accNav: safeNumber(f[6]),
|
|
278
|
+
change: safeNumber(f[7]),
|
|
279
|
+
navDate: f[8] ?? '',
|
|
280
|
+
raw: f,
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
// ---------- 批量混合查询 ----------
|
|
284
|
+
/**
|
|
285
|
+
* 批量混合查询,返回原始解析结果(key + fields)
|
|
286
|
+
* @param params 如 'sz000858,s_sh000001,jj000001'
|
|
287
|
+
*/
|
|
288
|
+
async batchRaw(params) {
|
|
289
|
+
return this.fetch(params);
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
exports.TencentStockSDK = TencentStockSDK;
|
|
293
|
+
exports.default = TencentStockSDK;
|