@helloxiaohu/plugin-stock-backtest-sim 0.0.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/README.md +3 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +44 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/backtest-engine.d.ts +84 -0
- package/dist/lib/backtest-engine.d.ts.map +1 -0
- package/dist/lib/backtest-engine.js +339 -0
- package/dist/lib/backtest-engine.js.map +1 -0
- package/dist/lib/market-data-adapter.d.ts +12 -0
- package/dist/lib/market-data-adapter.d.ts.map +1 -0
- package/dist/lib/market-data-adapter.js +98 -0
- package/dist/lib/market-data-adapter.js.map +1 -0
- package/dist/lib/stock-backtest-sim.plugin.d.ts +3 -0
- package/dist/lib/stock-backtest-sim.plugin.d.ts.map +1 -0
- package/dist/lib/stock-backtest-sim.plugin.js +15 -0
- package/dist/lib/stock-backtest-sim.plugin.js.map +1 -0
- package/dist/lib/stock-backtest-sim.strategy.d.ts +333 -0
- package/dist/lib/stock-backtest-sim.strategy.d.ts.map +1 -0
- package/dist/lib/stock-backtest-sim.strategy.js +69 -0
- package/dist/lib/stock-backtest-sim.strategy.js.map +1 -0
- package/dist/lib/tools/backtest-momentum-auto.tool.d.ts +73 -0
- package/dist/lib/tools/backtest-momentum-auto.tool.d.ts.map +1 -0
- package/dist/lib/tools/backtest-momentum-auto.tool.js +72 -0
- package/dist/lib/tools/backtest-momentum-auto.tool.js.map +1 -0
- package/dist/lib/tools/backtest-momentum.tool.d.ts +127 -0
- package/dist/lib/tools/backtest-momentum.tool.d.ts.map +1 -0
- package/dist/lib/tools/backtest-momentum.tool.js +60 -0
- package/dist/lib/tools/backtest-momentum.tool.js.map +1 -0
- package/dist/lib/tools/stress-test.tool.d.ts +88 -0
- package/dist/lib/tools/stress-test.tool.d.ts.map +1 -0
- package/dist/lib/tools/stress-test.tool.js +51 -0
- package/dist/lib/tools/stress-test.tool.js.map +1 -0
- package/dist/lib/toolset.d.ts +7 -0
- package/dist/lib/toolset.d.ts.map +1 -0
- package/dist/lib/toolset.js +21 -0
- package/dist/lib/toolset.js.map +1 -0
- package/dist/lib/types.d.ts +21 -0
- package/dist/lib/types.d.ts.map +1 -0
- package/dist/lib/types.js +3 -0
- package/dist/lib/types.js.map +1 -0
- package/package.json +36 -0
package/README.md
ADDED
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { XpertPlugin } from '@xpert-ai/plugin-sdk';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
declare const ConfigSchema: z.ZodObject<{
|
|
4
|
+
defaultFeeRate: z.ZodDefault<z.ZodOptional<z.ZodNumber>>;
|
|
5
|
+
defaultSlippageRate: z.ZodDefault<z.ZodOptional<z.ZodNumber>>;
|
|
6
|
+
defaultRebalanceDays: z.ZodDefault<z.ZodOptional<z.ZodNumber>>;
|
|
7
|
+
}, "strip", z.ZodTypeAny, {
|
|
8
|
+
defaultFeeRate?: number;
|
|
9
|
+
defaultSlippageRate?: number;
|
|
10
|
+
defaultRebalanceDays?: number;
|
|
11
|
+
}, {
|
|
12
|
+
defaultFeeRate?: number;
|
|
13
|
+
defaultSlippageRate?: number;
|
|
14
|
+
defaultRebalanceDays?: number;
|
|
15
|
+
}>;
|
|
16
|
+
declare const plugin: XpertPlugin<z.infer<typeof ConfigSchema>>;
|
|
17
|
+
export default plugin;
|
|
18
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAA;AAIvD,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AAYvB,QAAA,MAAM,YAAY;;;;;;;;;;;;EAIhB,CAAA;AAEF,QAAA,MAAM,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,YAAY,CAAC,CA2BrD,CAAA;AAED,eAAe,MAAM,CAAA"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { readFileSync } from 'fs';
|
|
2
|
+
import { fileURLToPath } from 'url';
|
|
3
|
+
import { dirname, join } from 'path';
|
|
4
|
+
import { z } from 'zod';
|
|
5
|
+
import { StockBacktestSimPlugin } from './lib/stock-backtest-sim.plugin.js';
|
|
6
|
+
import { icon } from './lib/types.js';
|
|
7
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
8
|
+
const __dirname = dirname(__filename);
|
|
9
|
+
const packageJson = JSON.parse(readFileSync(join(__dirname, '../package.json'), 'utf8'));
|
|
10
|
+
const ConfigSchema = z.object({
|
|
11
|
+
defaultFeeRate: z.number().min(0).max(0.01).optional().default(0.0005),
|
|
12
|
+
defaultSlippageRate: z.number().min(0).max(0.01).optional().default(0.0005),
|
|
13
|
+
defaultRebalanceDays: z.number().int().min(1).max(60).optional().default(5)
|
|
14
|
+
});
|
|
15
|
+
const plugin = {
|
|
16
|
+
meta: {
|
|
17
|
+
name: packageJson.name,
|
|
18
|
+
version: packageJson.version,
|
|
19
|
+
category: 'tools',
|
|
20
|
+
icon: {
|
|
21
|
+
type: 'image',
|
|
22
|
+
value: icon
|
|
23
|
+
},
|
|
24
|
+
displayName: 'Stock Backtest & Simulation',
|
|
25
|
+
description: 'Run historical backtests and scenario stress simulation for stock strategies',
|
|
26
|
+
keywords: ['backtest', 'simulation', 'stress test', '回测', '仿真'],
|
|
27
|
+
author: 'XpertAI Team',
|
|
28
|
+
},
|
|
29
|
+
config: {
|
|
30
|
+
schema: ConfigSchema,
|
|
31
|
+
},
|
|
32
|
+
register(ctx) {
|
|
33
|
+
ctx.logger.log('register stock-backtest-sim plugin');
|
|
34
|
+
return { module: StockBacktestSimPlugin, global: true };
|
|
35
|
+
},
|
|
36
|
+
async onStart(ctx) {
|
|
37
|
+
ctx.logger.log('stock-backtest-sim plugin started');
|
|
38
|
+
},
|
|
39
|
+
async onStop(ctx) {
|
|
40
|
+
ctx.logger.log('stock-backtest-sim plugin stopped');
|
|
41
|
+
},
|
|
42
|
+
};
|
|
43
|
+
export default plugin;
|
|
44
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,YAAY,EAAE,MAAM,IAAI,CAAA;AACjC,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAA;AACnC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAA;AACpC,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AACvB,OAAO,EAAE,sBAAsB,EAAE,MAAM,oCAAoC,CAAA;AAC3E,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAA;AAErC,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AACjD,MAAM,SAAS,GAAG,OAAO,CAAC,UAAU,CAAC,CAAA;AAErC,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,EAAE,iBAAiB,CAAC,EAAE,MAAM,CAAC,CAGtF,CAAA;AAED,MAAM,YAAY,GAAG,CAAC,CAAC,MAAM,CAAC;IAC5B,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC;IACtE,mBAAmB,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC;IAC3E,oBAAoB,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;CAC5E,CAAC,CAAA;AAEF,MAAM,MAAM,GAA8C;IACxD,IAAI,EAAE;QACJ,IAAI,EAAE,WAAW,CAAC,IAAI;QACtB,OAAO,EAAE,WAAW,CAAC,OAAO;QAC5B,QAAQ,EAAE,OAAO;QACjB,IAAI,EAAE;YACJ,IAAI,EAAE,OAAO;YACb,KAAK,EAAE,IAAI;SACZ;QACD,WAAW,EAAE,6BAA6B;QAC1C,WAAW,EAAE,8EAA8E;QAC3F,QAAQ,EAAE,CAAC,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,IAAI,EAAE,IAAI,CAAC;QAC/D,MAAM,EAAE,cAAc;KACvB;IACD,MAAM,EAAE;QACN,MAAM,EAAE,YAAY;KACrB;IACD,QAAQ,CAAC,GAAG;QACV,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,oCAAoC,CAAC,CAAA;QACpD,OAAO,EAAE,MAAM,EAAE,sBAAsB,EAAE,MAAM,EAAE,IAAI,EAAE,CAAA;IACzD,CAAC;IACD,KAAK,CAAC,OAAO,CAAC,GAAG;QACf,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAA;IACrD,CAAC;IACD,KAAK,CAAC,MAAM,CAAC,GAAG;QACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAA;IACrD,CAAC;CACF,CAAA;AAED,eAAe,MAAM,CAAA"}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { PricePoint, StressPosition, SymbolSeries } from './types.js';
|
|
2
|
+
interface BacktestOptions {
|
|
3
|
+
initialCapital: number;
|
|
4
|
+
lookbackDays?: number;
|
|
5
|
+
rebalanceEveryDays?: number;
|
|
6
|
+
topN?: number;
|
|
7
|
+
feeRate?: number;
|
|
8
|
+
slippageRate?: number;
|
|
9
|
+
maxSingleWeight?: number;
|
|
10
|
+
cashReserveRatio?: number;
|
|
11
|
+
benchmarkSeries?: PricePoint[];
|
|
12
|
+
}
|
|
13
|
+
interface EquityPoint {
|
|
14
|
+
date: string;
|
|
15
|
+
equity: number;
|
|
16
|
+
}
|
|
17
|
+
interface TradeLog {
|
|
18
|
+
date: string;
|
|
19
|
+
code: string;
|
|
20
|
+
side: 'buy' | 'sell';
|
|
21
|
+
quantity: number;
|
|
22
|
+
midPrice: number;
|
|
23
|
+
price: number;
|
|
24
|
+
amount: number;
|
|
25
|
+
fee: number;
|
|
26
|
+
slippageCost: number;
|
|
27
|
+
}
|
|
28
|
+
interface AttributionRow {
|
|
29
|
+
code: string;
|
|
30
|
+
pnl: number;
|
|
31
|
+
contribution: number;
|
|
32
|
+
endingValue: number;
|
|
33
|
+
}
|
|
34
|
+
export declare function runMomentumBacktest(series: SymbolSeries[], options: BacktestOptions): {
|
|
35
|
+
summary: {
|
|
36
|
+
startDate: string;
|
|
37
|
+
endDate: string;
|
|
38
|
+
initialCapital: number;
|
|
39
|
+
finalCapital: number;
|
|
40
|
+
totalReturn: number;
|
|
41
|
+
annualizedReturn: number;
|
|
42
|
+
annualizedVolatility: number;
|
|
43
|
+
sharpe: number;
|
|
44
|
+
maxDrawdown: number;
|
|
45
|
+
tradeCount: number;
|
|
46
|
+
benchmarkTotalReturn: number;
|
|
47
|
+
excessReturn: number;
|
|
48
|
+
};
|
|
49
|
+
benchmark: {
|
|
50
|
+
benchmarkTotalReturn: number;
|
|
51
|
+
benchmarkAnnualizedReturn: number;
|
|
52
|
+
excessReturn: number;
|
|
53
|
+
beta: number;
|
|
54
|
+
alphaAnnualized: number;
|
|
55
|
+
trackingError: number;
|
|
56
|
+
informationRatio: number;
|
|
57
|
+
};
|
|
58
|
+
costBreakdown: {
|
|
59
|
+
turnover: number;
|
|
60
|
+
totalFees: number;
|
|
61
|
+
totalSlippageCost: number;
|
|
62
|
+
totalCost: number;
|
|
63
|
+
};
|
|
64
|
+
attribution: AttributionRow[];
|
|
65
|
+
equityCurve: EquityPoint[];
|
|
66
|
+
trades: TradeLog[];
|
|
67
|
+
};
|
|
68
|
+
export declare function runStressTest(positions: StressPosition[], scenarios: Array<{
|
|
69
|
+
name: string;
|
|
70
|
+
marketShock: number;
|
|
71
|
+
betaScale?: number;
|
|
72
|
+
perSymbolShock?: Record<string, number>;
|
|
73
|
+
}>): {
|
|
74
|
+
baseValue: number;
|
|
75
|
+
scenarios: {
|
|
76
|
+
name: string;
|
|
77
|
+
marketShock: number;
|
|
78
|
+
pnl: number;
|
|
79
|
+
stressedValue: number;
|
|
80
|
+
returnRate: number;
|
|
81
|
+
}[];
|
|
82
|
+
};
|
|
83
|
+
export {};
|
|
84
|
+
//# sourceMappingURL=backtest-engine.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"backtest-engine.d.ts","sourceRoot":"","sources":["../../src/lib/backtest-engine.ts"],"names":[],"mappings":"AAAA,OAAO,EAAW,UAAU,EAAE,cAAc,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAE9E,UAAU,eAAe;IACvB,cAAc,EAAE,MAAM,CAAA;IACtB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,kBAAkB,CAAC,EAAE,MAAM,CAAA;IAC3B,IAAI,CAAC,EAAE,MAAM,CAAA;IACb,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,eAAe,CAAC,EAAE,MAAM,CAAA;IACxB,gBAAgB,CAAC,EAAE,MAAM,CAAA;IACzB,eAAe,CAAC,EAAE,UAAU,EAAE,CAAA;CAC/B;AAED,UAAU,WAAW;IACnB,IAAI,EAAE,MAAM,CAAA;IACZ,MAAM,EAAE,MAAM,CAAA;CACf;AAED,UAAU,QAAQ;IAChB,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,KAAK,GAAG,MAAM,CAAA;IACpB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,KAAK,EAAE,MAAM,CAAA;IACb,MAAM,EAAE,MAAM,CAAA;IACd,GAAG,EAAE,MAAM,CAAA;IACX,YAAY,EAAE,MAAM,CAAA;CACrB;AAED,UAAU,cAAc;IACtB,IAAI,EAAE,MAAM,CAAA;IACZ,GAAG,EAAE,MAAM,CAAA;IACX,YAAY,EAAE,MAAM,CAAA;IACpB,WAAW,EAAE,MAAM,CAAA;CACpB;AAkJD,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,YAAY,EAAE,EAAE,OAAO,EAAE,eAAe;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA2MnF;AAED,wBAAgB,aAAa,CAC3B,SAAS,EAAE,cAAc,EAAE,EAC3B,SAAS,EAAE,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,SAAS,CAAC,EAAE,MAAM,CAAC;IAAC,cAAc,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,CAAC;;;;;;;;;EA0BrH"}
|
|
@@ -0,0 +1,339 @@
|
|
|
1
|
+
const SQRT_252 = Math.sqrt(252);
|
|
2
|
+
function safePrice(value) {
|
|
3
|
+
return Number.isFinite(value) && value > 0 ? value : 0;
|
|
4
|
+
}
|
|
5
|
+
function alignSeries(series) {
|
|
6
|
+
const minLen = Math.min(...series.map((item) => item.prices.length));
|
|
7
|
+
if (!Number.isFinite(minLen) || minLen <= 1) {
|
|
8
|
+
return { dates: [], closesByCode: {} };
|
|
9
|
+
}
|
|
10
|
+
const trimmed = series.map((item) => ({
|
|
11
|
+
code: item.code.toUpperCase(),
|
|
12
|
+
prices: item.prices.slice(-minLen)
|
|
13
|
+
}));
|
|
14
|
+
const dates = trimmed[0].prices.map((item) => item.date);
|
|
15
|
+
const closesByCode = {};
|
|
16
|
+
for (const item of trimmed) {
|
|
17
|
+
closesByCode[item.code] = item.prices.map((point) => safePrice(point.close));
|
|
18
|
+
}
|
|
19
|
+
return { dates, closesByCode };
|
|
20
|
+
}
|
|
21
|
+
function portfolioValue(holdings, prices, cash) {
|
|
22
|
+
let value = cash;
|
|
23
|
+
for (const item of holdings.values()) {
|
|
24
|
+
const price = prices[item.code] || 0;
|
|
25
|
+
value += item.quantity * price;
|
|
26
|
+
}
|
|
27
|
+
return value;
|
|
28
|
+
}
|
|
29
|
+
function mean(values) {
|
|
30
|
+
if (values.length === 0)
|
|
31
|
+
return 0;
|
|
32
|
+
return values.reduce((sum, item) => sum + item, 0) / values.length;
|
|
33
|
+
}
|
|
34
|
+
function sampleVariance(values) {
|
|
35
|
+
if (values.length <= 1)
|
|
36
|
+
return 0;
|
|
37
|
+
const m = mean(values);
|
|
38
|
+
return values.reduce((sum, item) => sum + (item - m) * (item - m), 0) / (values.length - 1);
|
|
39
|
+
}
|
|
40
|
+
function annualizedReturn(totalReturn, days) {
|
|
41
|
+
if (!Number.isFinite(totalReturn) || days <= 0)
|
|
42
|
+
return 0;
|
|
43
|
+
return Math.pow(1 + totalReturn, 252 / days) - 1;
|
|
44
|
+
}
|
|
45
|
+
function computeBenchmarkStats(equityCurve, benchmarkSeries, strategyAnnualizedReturn) {
|
|
46
|
+
if (!benchmarkSeries || benchmarkSeries.length < 2 || equityCurve.length < 2) {
|
|
47
|
+
return null;
|
|
48
|
+
}
|
|
49
|
+
const benchmarkByDate = new Map();
|
|
50
|
+
for (const point of benchmarkSeries) {
|
|
51
|
+
const close = safePrice(point.close);
|
|
52
|
+
if (close > 0) {
|
|
53
|
+
benchmarkByDate.set(point.date, close);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
const alignedBenchmark = [];
|
|
57
|
+
for (const point of equityCurve) {
|
|
58
|
+
const close = benchmarkByDate.get(point.date);
|
|
59
|
+
if (close && close > 0) {
|
|
60
|
+
alignedBenchmark.push({ date: point.date, close });
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
if (alignedBenchmark.length < 2) {
|
|
64
|
+
return null;
|
|
65
|
+
}
|
|
66
|
+
const benchStart = alignedBenchmark[0].close;
|
|
67
|
+
const benchEnd = alignedBenchmark[alignedBenchmark.length - 1].close;
|
|
68
|
+
const benchmarkTotalReturn = benchStart > 0 ? benchEnd / benchStart - 1 : 0;
|
|
69
|
+
const benchmarkAnnualizedReturn = annualizedReturn(benchmarkTotalReturn, alignedBenchmark.length - 1);
|
|
70
|
+
const strategyReturns = [];
|
|
71
|
+
const benchmarkReturns = [];
|
|
72
|
+
for (let i = 1; i < alignedBenchmark.length; i++) {
|
|
73
|
+
const date = alignedBenchmark[i].date;
|
|
74
|
+
const prevDate = alignedBenchmark[i - 1].date;
|
|
75
|
+
const strategyNow = equityCurve.find((item) => item.date === date)?.equity || 0;
|
|
76
|
+
const strategyPrev = equityCurve.find((item) => item.date === prevDate)?.equity || 0;
|
|
77
|
+
const benchNow = alignedBenchmark[i].close;
|
|
78
|
+
const benchPrev = alignedBenchmark[i - 1].close;
|
|
79
|
+
if (strategyNow > 0 && strategyPrev > 0 && benchNow > 0 && benchPrev > 0) {
|
|
80
|
+
strategyReturns.push(strategyNow / strategyPrev - 1);
|
|
81
|
+
benchmarkReturns.push(benchNow / benchPrev - 1);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
if (strategyReturns.length < 2 || benchmarkReturns.length < 2) {
|
|
85
|
+
return {
|
|
86
|
+
benchmarkTotalReturn,
|
|
87
|
+
benchmarkAnnualizedReturn,
|
|
88
|
+
excessReturn: strategyAnnualizedReturn - benchmarkAnnualizedReturn,
|
|
89
|
+
beta: null,
|
|
90
|
+
alphaAnnualized: null,
|
|
91
|
+
trackingError: null,
|
|
92
|
+
informationRatio: null
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
const strategyMean = mean(strategyReturns);
|
|
96
|
+
const benchmarkMean = mean(benchmarkReturns);
|
|
97
|
+
const benchmarkVar = sampleVariance(benchmarkReturns);
|
|
98
|
+
let covariance = 0;
|
|
99
|
+
for (let i = 0; i < strategyReturns.length; i++) {
|
|
100
|
+
covariance += (strategyReturns[i] - strategyMean) * (benchmarkReturns[i] - benchmarkMean);
|
|
101
|
+
}
|
|
102
|
+
covariance = covariance / Math.max(1, strategyReturns.length - 1);
|
|
103
|
+
const beta = benchmarkVar > 0 ? covariance / benchmarkVar : 0;
|
|
104
|
+
const alphaAnnualized = strategyAnnualizedReturn - beta * benchmarkAnnualizedReturn;
|
|
105
|
+
const excessDaily = strategyReturns.map((value, idx) => value - benchmarkReturns[idx]);
|
|
106
|
+
const trackingErrorDaily = Math.sqrt(Math.max(0, sampleVariance(excessDaily)));
|
|
107
|
+
const trackingError = trackingErrorDaily * SQRT_252;
|
|
108
|
+
const informationRatio = trackingErrorDaily > 0 ? (mean(excessDaily) / trackingErrorDaily) * SQRT_252 : 0;
|
|
109
|
+
return {
|
|
110
|
+
benchmarkTotalReturn,
|
|
111
|
+
benchmarkAnnualizedReturn,
|
|
112
|
+
excessReturn: strategyAnnualizedReturn - benchmarkAnnualizedReturn,
|
|
113
|
+
beta,
|
|
114
|
+
alphaAnnualized,
|
|
115
|
+
trackingError,
|
|
116
|
+
informationRatio
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
export function runMomentumBacktest(series, options) {
|
|
120
|
+
const lookbackDays = options.lookbackDays ?? 20;
|
|
121
|
+
const rebalanceEveryDays = options.rebalanceEveryDays ?? 5;
|
|
122
|
+
const topN = options.topN ?? 3;
|
|
123
|
+
const feeRate = options.feeRate ?? 0.0005;
|
|
124
|
+
const slippageRate = options.slippageRate ?? 0.0005;
|
|
125
|
+
const maxSingleWeight = options.maxSingleWeight ?? 0.4;
|
|
126
|
+
const cashReserveRatio = options.cashReserveRatio ?? 0.05;
|
|
127
|
+
const aligned = alignSeries(series.filter((item) => item.prices.length > lookbackDays + 2));
|
|
128
|
+
if (aligned.dates.length <= lookbackDays + 1) {
|
|
129
|
+
throw new Error('not_enough_price_history');
|
|
130
|
+
}
|
|
131
|
+
let cash = options.initialCapital;
|
|
132
|
+
const holdings = new Map();
|
|
133
|
+
const equityCurve = [];
|
|
134
|
+
const trades = [];
|
|
135
|
+
let totalFees = 0;
|
|
136
|
+
let totalSlippageCost = 0;
|
|
137
|
+
let turnover = 0;
|
|
138
|
+
const pnlLedger = new Map();
|
|
139
|
+
const updateLedger = (code, side, quantity, netCashEffect) => {
|
|
140
|
+
const key = code.toUpperCase();
|
|
141
|
+
const current = pnlLedger.get(key) || { buyCost: 0, sellProceeds: 0, quantity: 0 };
|
|
142
|
+
if (side === 'buy') {
|
|
143
|
+
current.buyCost += Math.max(0, -netCashEffect);
|
|
144
|
+
current.quantity += quantity;
|
|
145
|
+
}
|
|
146
|
+
else {
|
|
147
|
+
current.sellProceeds += Math.max(0, netCashEffect);
|
|
148
|
+
current.quantity -= quantity;
|
|
149
|
+
}
|
|
150
|
+
pnlLedger.set(key, current);
|
|
151
|
+
};
|
|
152
|
+
for (let day = lookbackDays; day < aligned.dates.length; day++) {
|
|
153
|
+
const date = aligned.dates[day];
|
|
154
|
+
const dailyPrices = {};
|
|
155
|
+
for (const [code, closes] of Object.entries(aligned.closesByCode)) {
|
|
156
|
+
dailyPrices[code] = closes[day];
|
|
157
|
+
}
|
|
158
|
+
if ((day - lookbackDays) % rebalanceEveryDays === 0) {
|
|
159
|
+
const momentum = Object.entries(aligned.closesByCode)
|
|
160
|
+
.map(([code, closes]) => {
|
|
161
|
+
const past = closes[day - lookbackDays];
|
|
162
|
+
const now = closes[day];
|
|
163
|
+
if (past <= 0 || now <= 0)
|
|
164
|
+
return { code, score: Number.NEGATIVE_INFINITY };
|
|
165
|
+
return { code, score: (now - past) / past };
|
|
166
|
+
})
|
|
167
|
+
.filter((item) => Number.isFinite(item.score) && item.score > 0)
|
|
168
|
+
.sort((a, b) => b.score - a.score)
|
|
169
|
+
.slice(0, topN);
|
|
170
|
+
const selectedCodes = new Set(momentum.map((item) => item.code));
|
|
171
|
+
const totalEquityBefore = portfolioValue(holdings, dailyPrices, cash);
|
|
172
|
+
const investable = totalEquityBefore * (1 - cashReserveRatio);
|
|
173
|
+
const baseWeight = momentum.length > 0 ? Math.min(maxSingleWeight, 1 / momentum.length) : 0;
|
|
174
|
+
const targetValueByCode = {};
|
|
175
|
+
for (const item of momentum) {
|
|
176
|
+
targetValueByCode[item.code] = investable * baseWeight;
|
|
177
|
+
}
|
|
178
|
+
for (const [code, holding] of Array.from(holdings.entries())) {
|
|
179
|
+
if (selectedCodes.has(code))
|
|
180
|
+
continue;
|
|
181
|
+
const midPrice = dailyPrices[code] || 0;
|
|
182
|
+
if (midPrice <= 0 || holding.quantity <= 0)
|
|
183
|
+
continue;
|
|
184
|
+
const tradePrice = midPrice * (1 - slippageRate);
|
|
185
|
+
const amount = holding.quantity * tradePrice;
|
|
186
|
+
const fee = amount * feeRate;
|
|
187
|
+
const slippageCost = holding.quantity * Math.max(0, midPrice - tradePrice);
|
|
188
|
+
const netCash = amount - fee;
|
|
189
|
+
cash += netCash;
|
|
190
|
+
totalFees += fee;
|
|
191
|
+
totalSlippageCost += slippageCost;
|
|
192
|
+
turnover += amount;
|
|
193
|
+
updateLedger(code, 'sell', holding.quantity, netCash);
|
|
194
|
+
trades.push({ date, code, side: 'sell', quantity: holding.quantity, midPrice, price: tradePrice, amount, fee, slippageCost });
|
|
195
|
+
holdings.delete(code);
|
|
196
|
+
}
|
|
197
|
+
for (const [code, targetValue] of Object.entries(targetValueByCode)) {
|
|
198
|
+
const midPrice = dailyPrices[code] || 0;
|
|
199
|
+
if (midPrice <= 0)
|
|
200
|
+
continue;
|
|
201
|
+
const currentQty = holdings.get(code)?.quantity || 0;
|
|
202
|
+
const currentValue = currentQty * midPrice;
|
|
203
|
+
const delta = targetValue - currentValue;
|
|
204
|
+
if (Math.abs(delta) < midPrice)
|
|
205
|
+
continue;
|
|
206
|
+
if (delta < 0) {
|
|
207
|
+
const qtyToSell = Math.min(currentQty, Math.floor(Math.abs(delta) / midPrice));
|
|
208
|
+
if (qtyToSell <= 0)
|
|
209
|
+
continue;
|
|
210
|
+
const tradePrice = midPrice * (1 - slippageRate);
|
|
211
|
+
const amount = qtyToSell * tradePrice;
|
|
212
|
+
const fee = amount * feeRate;
|
|
213
|
+
const slippageCost = qtyToSell * Math.max(0, midPrice - tradePrice);
|
|
214
|
+
const netCash = amount - fee;
|
|
215
|
+
cash += netCash;
|
|
216
|
+
totalFees += fee;
|
|
217
|
+
totalSlippageCost += slippageCost;
|
|
218
|
+
turnover += amount;
|
|
219
|
+
holdings.set(code, { code, quantity: currentQty - qtyToSell });
|
|
220
|
+
if ((holdings.get(code)?.quantity || 0) <= 0)
|
|
221
|
+
holdings.delete(code);
|
|
222
|
+
updateLedger(code, 'sell', qtyToSell, netCash);
|
|
223
|
+
trades.push({ date, code, side: 'sell', quantity: qtyToSell, midPrice, price: tradePrice, amount, fee, slippageCost });
|
|
224
|
+
}
|
|
225
|
+
else {
|
|
226
|
+
const qtyToBuy = Math.floor(delta / midPrice);
|
|
227
|
+
if (qtyToBuy <= 0)
|
|
228
|
+
continue;
|
|
229
|
+
const tradePrice = midPrice * (1 + slippageRate);
|
|
230
|
+
const amount = qtyToBuy * tradePrice;
|
|
231
|
+
const fee = amount * feeRate;
|
|
232
|
+
const slippageCost = qtyToBuy * Math.max(0, tradePrice - midPrice);
|
|
233
|
+
const totalCost = amount + fee;
|
|
234
|
+
if (totalCost > cash)
|
|
235
|
+
continue;
|
|
236
|
+
cash -= totalCost;
|
|
237
|
+
totalFees += fee;
|
|
238
|
+
totalSlippageCost += slippageCost;
|
|
239
|
+
turnover += amount;
|
|
240
|
+
holdings.set(code, { code, quantity: currentQty + qtyToBuy });
|
|
241
|
+
updateLedger(code, 'buy', qtyToBuy, -totalCost);
|
|
242
|
+
trades.push({ date, code, side: 'buy', quantity: qtyToBuy, midPrice, price: tradePrice, amount, fee, slippageCost });
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
const equity = portfolioValue(holdings, dailyPrices, cash);
|
|
247
|
+
equityCurve.push({ date, equity });
|
|
248
|
+
}
|
|
249
|
+
const first = equityCurve[0]?.equity || options.initialCapital;
|
|
250
|
+
const last = equityCurve[equityCurve.length - 1]?.equity || first;
|
|
251
|
+
const totalReturn = first > 0 ? last / first - 1 : 0;
|
|
252
|
+
const dailyReturns = [];
|
|
253
|
+
for (let i = 1; i < equityCurve.length; i++) {
|
|
254
|
+
const prev = equityCurve[i - 1].equity;
|
|
255
|
+
const now = equityCurve[i].equity;
|
|
256
|
+
if (prev > 0)
|
|
257
|
+
dailyReturns.push(now / prev - 1);
|
|
258
|
+
}
|
|
259
|
+
const dailyVol = Math.sqrt(Math.max(0, sampleVariance(dailyReturns)));
|
|
260
|
+
const annualVol = dailyVol * SQRT_252;
|
|
261
|
+
const annualReturn = annualizedReturn(totalReturn, Math.max(1, dailyReturns.length));
|
|
262
|
+
const sharpe = annualVol > 0 ? annualReturn / annualVol : 0;
|
|
263
|
+
let peak = equityCurve[0]?.equity || 0;
|
|
264
|
+
let maxDrawdown = 0;
|
|
265
|
+
for (const point of equityCurve) {
|
|
266
|
+
peak = Math.max(peak, point.equity);
|
|
267
|
+
if (peak > 0) {
|
|
268
|
+
maxDrawdown = Math.max(maxDrawdown, (peak - point.equity) / peak);
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
const finalPrices = equityCurve.length > 0
|
|
272
|
+
? Object.fromEntries(Object.entries(aligned.closesByCode).map(([code, values]) => [code, values[values.length - 1] || 0]))
|
|
273
|
+
: {};
|
|
274
|
+
const attribution = [];
|
|
275
|
+
for (const [code, stats] of pnlLedger.entries()) {
|
|
276
|
+
const endingValue = Math.max(0, stats.quantity) * (finalPrices[code] || 0);
|
|
277
|
+
const pnl = stats.sellProceeds + endingValue - stats.buyCost;
|
|
278
|
+
attribution.push({
|
|
279
|
+
code,
|
|
280
|
+
pnl,
|
|
281
|
+
contribution: options.initialCapital > 0 ? pnl / options.initialCapital : 0,
|
|
282
|
+
endingValue
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
attribution.sort((a, b) => b.pnl - a.pnl);
|
|
286
|
+
const benchmark = computeBenchmarkStats(equityCurve, options.benchmarkSeries, annualReturn);
|
|
287
|
+
return {
|
|
288
|
+
summary: {
|
|
289
|
+
startDate: equityCurve[0]?.date || null,
|
|
290
|
+
endDate: equityCurve[equityCurve.length - 1]?.date || null,
|
|
291
|
+
initialCapital: options.initialCapital,
|
|
292
|
+
finalCapital: last,
|
|
293
|
+
totalReturn,
|
|
294
|
+
annualizedReturn: annualReturn,
|
|
295
|
+
annualizedVolatility: annualVol,
|
|
296
|
+
sharpe,
|
|
297
|
+
maxDrawdown,
|
|
298
|
+
tradeCount: trades.length,
|
|
299
|
+
benchmarkTotalReturn: benchmark?.benchmarkTotalReturn ?? null,
|
|
300
|
+
excessReturn: benchmark?.excessReturn ?? null
|
|
301
|
+
},
|
|
302
|
+
benchmark,
|
|
303
|
+
costBreakdown: {
|
|
304
|
+
turnover,
|
|
305
|
+
totalFees,
|
|
306
|
+
totalSlippageCost,
|
|
307
|
+
totalCost: totalFees + totalSlippageCost
|
|
308
|
+
},
|
|
309
|
+
attribution,
|
|
310
|
+
equityCurve,
|
|
311
|
+
trades: trades.slice(-200)
|
|
312
|
+
};
|
|
313
|
+
}
|
|
314
|
+
export function runStressTest(positions, scenarios) {
|
|
315
|
+
const baseValue = positions.reduce((sum, item) => sum + Math.max(0, item.quantity * item.price), 0);
|
|
316
|
+
const results = scenarios.map((scenario) => {
|
|
317
|
+
let pnl = 0;
|
|
318
|
+
for (const position of positions) {
|
|
319
|
+
const code = position.code.toUpperCase();
|
|
320
|
+
const beta = Number.isFinite(position.beta) ? Number(position.beta) : 1;
|
|
321
|
+
const betaScale = Number.isFinite(scenario.betaScale) ? Number(scenario.betaScale) : 1;
|
|
322
|
+
const symbolShock = scenario.perSymbolShock?.[code] ?? scenario.marketShock * beta * betaScale;
|
|
323
|
+
pnl += position.quantity * position.price * symbolShock;
|
|
324
|
+
}
|
|
325
|
+
const stressedValue = baseValue + pnl;
|
|
326
|
+
return {
|
|
327
|
+
name: scenario.name,
|
|
328
|
+
marketShock: scenario.marketShock,
|
|
329
|
+
pnl,
|
|
330
|
+
stressedValue,
|
|
331
|
+
returnRate: baseValue > 0 ? pnl / baseValue : 0
|
|
332
|
+
};
|
|
333
|
+
});
|
|
334
|
+
return {
|
|
335
|
+
baseValue,
|
|
336
|
+
scenarios: results
|
|
337
|
+
};
|
|
338
|
+
}
|
|
339
|
+
//# sourceMappingURL=backtest-engine.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"backtest-engine.js","sourceRoot":"","sources":["../../src/lib/backtest-engine.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { SymbolSeries } from './types.js';
|
|
2
|
+
interface AdapterOptions {
|
|
3
|
+
timeoutMs?: number;
|
|
4
|
+
}
|
|
5
|
+
export declare class MarketDataAdapter {
|
|
6
|
+
private timeoutMs;
|
|
7
|
+
constructor(options?: AdapterOptions);
|
|
8
|
+
fetchDailySeries(code: string, count?: number): Promise<SymbolSeries>;
|
|
9
|
+
fetchSeriesBatch(codes: string[], count?: number): Promise<SymbolSeries[]>;
|
|
10
|
+
}
|
|
11
|
+
export {};
|
|
12
|
+
//# sourceMappingURL=market-data-adapter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"market-data-adapter.d.ts","sourceRoot":"","sources":["../../src/lib/market-data-adapter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAc,YAAY,EAAE,MAAM,YAAY,CAAA;AAIrD,UAAU,cAAc;IACtB,SAAS,CAAC,EAAE,MAAM,CAAA;CACnB;AAwCD,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,SAAS,CAAQ;gBAEb,OAAO,GAAE,cAAmB;IAIlC,gBAAgB,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,SAAM,GAAG,OAAO,CAAC,YAAY,CAAC;IAuClE,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,KAAK,SAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC;CAc9E"}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
const SINA_KLINE_URL = 'https://quotes.sina.cn/cn/api/json_v2.php/CN_MarketDataService.getKLineData';
|
|
2
|
+
function detectMarket(code) {
|
|
3
|
+
const normalized = code.trim().toLowerCase();
|
|
4
|
+
if (normalized.startsWith('sh'))
|
|
5
|
+
return 'sh';
|
|
6
|
+
if (normalized.startsWith('sz'))
|
|
7
|
+
return 'sz';
|
|
8
|
+
if (normalized.startsWith('hk'))
|
|
9
|
+
return 'hk';
|
|
10
|
+
const clean = normalized.replace(/^(sh|sz|hk)/, '');
|
|
11
|
+
if (/^6\d{5}$/.test(clean))
|
|
12
|
+
return 'sh';
|
|
13
|
+
if (/^(0|3)\d{5}$/.test(clean))
|
|
14
|
+
return 'sz';
|
|
15
|
+
if (/^\d{5}$/.test(clean))
|
|
16
|
+
return 'hk';
|
|
17
|
+
return 'sh';
|
|
18
|
+
}
|
|
19
|
+
function formatSinaSymbol(code) {
|
|
20
|
+
const normalized = code.trim().toLowerCase();
|
|
21
|
+
if (/^(sh|sz|hk)\w+/.test(normalized))
|
|
22
|
+
return normalized;
|
|
23
|
+
const clean = normalized.replace(/^(sh|sz|hk)/, '');
|
|
24
|
+
const market = detectMarket(clean);
|
|
25
|
+
if (market === 'hk')
|
|
26
|
+
return `hk${clean.padStart(5, '0')}`;
|
|
27
|
+
if (market === 'sh')
|
|
28
|
+
return `sh${clean.padStart(6, '0')}`;
|
|
29
|
+
return `sz${clean.padStart(6, '0')}`;
|
|
30
|
+
}
|
|
31
|
+
function normalizeCode(code) {
|
|
32
|
+
return code.trim().toUpperCase().replace(/^(SH|SZ|HK)/, '');
|
|
33
|
+
}
|
|
34
|
+
function parsePricePoints(raw) {
|
|
35
|
+
if (!Array.isArray(raw))
|
|
36
|
+
return [];
|
|
37
|
+
return raw
|
|
38
|
+
.map((item) => ({
|
|
39
|
+
date: String(item?.day || item?.date || ''),
|
|
40
|
+
close: Number(item?.close || 0)
|
|
41
|
+
}))
|
|
42
|
+
.filter((item) => item.date && Number.isFinite(item.close) && item.close > 0);
|
|
43
|
+
}
|
|
44
|
+
export class MarketDataAdapter {
|
|
45
|
+
constructor(options = {}) {
|
|
46
|
+
this.timeoutMs = options.timeoutMs ?? 10000;
|
|
47
|
+
}
|
|
48
|
+
async fetchDailySeries(code, count = 200) {
|
|
49
|
+
const symbol = formatSinaSymbol(code);
|
|
50
|
+
const params = new URLSearchParams({
|
|
51
|
+
symbol,
|
|
52
|
+
scale: '240',
|
|
53
|
+
datalen: String(Math.max(30, Math.min(1000, Math.floor(count))))
|
|
54
|
+
});
|
|
55
|
+
const controller = new AbortController();
|
|
56
|
+
const timer = setTimeout(() => controller.abort(), this.timeoutMs);
|
|
57
|
+
try {
|
|
58
|
+
const response = await fetch(`${SINA_KLINE_URL}?${params.toString()}`, {
|
|
59
|
+
method: 'GET',
|
|
60
|
+
headers: {
|
|
61
|
+
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
|
|
62
|
+
'Referer': 'https://finance.sina.com.cn'
|
|
63
|
+
},
|
|
64
|
+
signal: controller.signal
|
|
65
|
+
});
|
|
66
|
+
if (!response.ok) {
|
|
67
|
+
throw new Error(`http_${response.status}`);
|
|
68
|
+
}
|
|
69
|
+
const payload = await response.json();
|
|
70
|
+
const prices = parsePricePoints(payload);
|
|
71
|
+
if (prices.length < 30) {
|
|
72
|
+
throw new Error('insufficient_series_points');
|
|
73
|
+
}
|
|
74
|
+
return {
|
|
75
|
+
code: normalizeCode(code),
|
|
76
|
+
prices
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
finally {
|
|
80
|
+
clearTimeout(timer);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
async fetchSeriesBatch(codes, count = 200) {
|
|
84
|
+
const uniqueCodes = Array.from(new Set(codes.map((item) => item.trim()).filter(Boolean)));
|
|
85
|
+
const results = await Promise.all(uniqueCodes.map(async (code) => {
|
|
86
|
+
try {
|
|
87
|
+
return await this.fetchDailySeries(code, count);
|
|
88
|
+
}
|
|
89
|
+
catch (error) {
|
|
90
|
+
const message = String(error?.message || 'unknown');
|
|
91
|
+
console.warn(`[stock-backtest-sim] fetch series failed for ${code}: ${message}`);
|
|
92
|
+
return null;
|
|
93
|
+
}
|
|
94
|
+
}));
|
|
95
|
+
return results.filter((item) => !!item);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
//# sourceMappingURL=market-data-adapter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"market-data-adapter.js","sourceRoot":"","sources":["../../src/lib/market-data-adapter.ts"],"names":[],"mappings":"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"}
|