OpenClaw Agent
5b546061c0
fix: 使用 Lazy 完全解决并发问题
...
问题分析:
1. GetOrAdd 在并发时可能创建多个 Value
2. 两个请求同时进入 GetPriceAsync,都检查内存缓存未命中
3. 然后同时查数据库,导致连接冲突
解决方案:
1. 使用 Lazy<Task<T>> 确保 Value Factory 只执行一次
2. 把整个流程(查数据库缓存 → 获取价格)放在 Lazy 里
3. 内存缓存命中时快速返回,不进入 Lazy
流程:
- 内存缓存命中 → 直接返回
- 内存缓存未命中 → 进入 Lazy(只执行一次)
- 查数据库缓存 → 命中则返回
- 数据库缓存未命中 → 从数据源获取
2026-03-24 10:17:53 +00:00
OpenClaw Agent
51b7372d81
fix: 优化 MySQL 连接池配置
...
问题:Cannot Open when State is Connecting
分析:
- SqlSugarScope 是线程安全的,使用 AsyncLocal 隔离上下文
- Singleton 注册符合官方建议
- 问题可能出在 MySQL 连接池配置
修复:
添加连接池参数:
- Pooling=true(开启连接池)
- MaximumPoolSize=100(最大连接数)
- MinimumPoolSize=5(最小连接数)
- ConnectionTimeout=30(连接超时)
- ConnectionIdleTimeout=180(空闲连接超时,避免使用陈旧连接)
配合内存缓存层,减少数据库查询次数
2026-03-24 10:08:35 +00:00
OpenClaw Agent
8d7e62f8af
fix: SqlSugar 注册改为 Scoped 避免并发冲突
...
问题:
- ISqlSugarClient 注册为 Singleton
- 多个请求共享同一个 SqlSugarScope 实例
- Task.WhenAll 并发查询时连接状态冲突
修复:
- Singleton → Scoped
- 每个 HTTP 请求独立的 SqlSugarScope 实例
- 避免跨请求共享连接对象
配合内存缓存层,双重保障:
1. 内存缓存:减少数据库查询次数
2. Scoped:隔离请求间的连接对象
2026-03-24 09:59:30 +00:00
OpenClaw Agent
9014363d6d
fix: 增强腾讯财经解析和日志
...
改进:
1. 修复引号解析逻辑(更健壮)
2. 减少字段验证从 36 到 5(只需价格字段)
3. 添加详细日志追踪解析过程
4. 记录请求 URL 和原始响应
日志关键词:
- [腾讯财经] 请求URL
- [腾讯财经] 原始响应
- [腾讯财经] 字段数量
- [腾讯财经] 成功
2026-03-24 09:52:25 +00:00
OpenClaw Agent
ec7ed6d686
fix: 添加内存缓存层解决数据库连接池冲突
...
根因:
- Task.WhenAll 并发获取多个股票价格
- 每个价格查询都访问数据库缓存
- SqlSugar 连接状态冲突:Cannot Open when State is Connecting
解决方案:
1. 添加 ConcurrentDictionary 内存缓存层
2. 先查内存缓存,命中则跳过数据库查询
3. 数据库缓存命中后写入内存缓存
4. API 获取成功后同时写入内存缓存
效果:
- 避免并发数据库查询
- 减少数据库连接压力
- 提高响应速度
2026-03-24 09:50:12 +00:00
OpenClaw Agent
89c6ca5397
fix: 增强批量获取价格日志级别
...
- LogWarning → LogError(价格获取失败是严重问题)
- 记录具体错误信息
- 记录 AssetType 用于排查数据源选择问题
2026-03-24 09:45:28 +00:00
OpenClaw Agent
8022731b34
fix: 增强价格获取降级链日志
...
记录每个数据源的尝试和结果:
- [数据源获取开始]
- [数据源] 尝试 Yahoo/腾讯/Tiingo
- [数据源] Yahoo/腾讯/Tiingo 成功/失败
- 具体错误信息
帮助定位 UPRO 等股票获取失败的原因
2026-03-24 09:44:15 +00:00
OpenClaw Agent
3fb2403e85
fix: 增强价格获取稳定性和日志追踪
...
问题排查:
1. 缓存命中时验证价格有效性(Price > 0)
2. 外部 API 返回无效价格时拒绝写入缓存
3. 缓存查询层添加详细日志
4. 捕获缓存查询异常并返回 null
改进:
- 缓存价格 <= 0 时忽略缓存重新获取
- 外部 API 价格 <= 0 时抛出异常,避免污染缓存
- 详细日志追踪价格获取全流程
2026-03-24 09:35:50 +00:00
OpenClaw Agent
3768f6e747
fix: 防止价格获取并发竞态条件
...
问题:多个并发请求同时获取同一股票价格,导致:
1. 外部 API 被重复调用(可能被限流)
2. 部分请求失败导致收益率显示 0%
解决方案:
- 使用 ConcurrentDictionary 存储进行中的请求
- GetOrAdd 模式确保同一时间只有一个请求在获取价格
- 其他并发请求等待第一个请求的结果
- 请求完成后移除 pending 任务
2026-03-24 08:57:45 +00:00
OpenClaw Agent
0579e2f47a
fix: 修复交易时间解析逻辑
...
- TransactionDate 和 TransactionTime 正确组合
- 移除 else if 改为独立 if,支持日期+时间
- CreateTransactionAsync 返回实际交易时间而非当前时间
2026-03-24 08:53:47 +00:00
OpenClaw Agent
7bf687323d
feat: 添加列表接口价格获取详细日志
...
定位组合收益率显示为 0 的问题:
- 记录每个股票的价格获取结果
- 记录价格字典查找情况
- 区分价格获取失败 vs 价格为 0 的情况
2026-03-24 08:39:49 +00:00
OpenClaw Agent
c0accdc903
feat: 用户访问时自动触发当日净值计算
...
修改 GetNavHistoryAsync:
- 检查是否存在今日净值记录
- 如果不存在,自动调用 CalculateAndSaveDailyNavAsync
- 捕获异常避免影响历史数据查询
收益:
- 用户每次访问收益曲线都能获取最新数据
- 无需手动触发或定时任务
2026-03-24 06:22:28 +00:00
OpenClaw Agent
1c3add27ef
fix: 修复收益曲线统计数据与组合详情不一致的问题
...
问题:
- 收益曲线接口 totalReturn 直接取净值历史最后一条的 cumulativeReturn
- 如果净值历史数据过时,统计信息会与当前持仓数据不一致
修复:
- 在 GetNavHistoryAsync 中检查净值历史最后一条是否为今天
- 如果不是,获取当前持仓数据实时计算累计收益率
- 更新 statistics.TotalReturn 为实时值
影响:
- 组合详情 historicalChange 和收益曲线 totalReturn 将保持一致
2026-03-24 06:14:27 +00:00
OpenClaw Agent
6b50d7fc50
refactor: SqlSugarScope注册为Singleton实现线程安全,移除手动信号量限制
2026-03-17 09:26:26 +00:00
OpenClaw Agent
d30c3076bd
fix: 添加数据库并发限制,解决SqlSugar连接状态冲突
2026-03-17 09:19:15 +00:00
OpenClaw Agent
7abb8796ec
fix: 恢复GetTotalAssets同步方法以满足接口定义
2026-03-17 08:23:31 +00:00
OpenClaw Agent
65abd50108
perf: 优化首页接口性能,批量并行获取价格
2026-03-17 08:18:56 +00:00
OpenClaw Agent
b39044bfe1
fix: 复用YahooQuotes实例,添加并发限制防止429错误
2026-03-17 06:56:42 +00:00
OpenClaw Agent
2edac30fd8
feat: Yahoo历史数据根据timeframe和limit动态计算开始日期
2026-03-17 06:48:27 +00:00
OpenClaw Agent
a91041e168
fix: Yahoo股票代码转换,BRK.B → BRK-B(类别股用连字符)
2026-03-17 05:47:27 +00:00
OpenClaw Agent
1830b93207
refactor: 移除腾讯历史K线降级逻辑,添加废弃标记和注释说明
2026-03-17 04:30:53 +00:00
OpenClaw Agent
aa4f63455b
refactor: 优化YahooMarketService,复用实例并改进异常类型
2026-03-17 04:22:00 +00:00
niannian zheng
2a6512ff48
feat(市场数据): 添加Yahoo财经服务并设为优先数据源
...
- 新增YahooMarketService实现股票实时价格和历史数据获取
- 更新MarketDataService优先使用Yahoo服务,腾讯财经降级为第二选择
- 添加YahooQuotesApi依赖并更新相关NuGet包版本
- 补充Yahoo服务测试用例
2026-03-17 12:06:47 +08:00
OpenClaw Agent
5bc318725d
feat: 添加组合更新接口,支持修改名称/策略/状态
2026-03-17 01:50:47 +00:00
OpenClaw Agent
9f82ad0a81
fix: 腾讯历史K线接口已废弃,改进错误处理优雅降级到Tiingo
2026-03-16 09:15:12 +00:00
OpenClaw Agent
79105c339d
fix: 买入后持仓数量和均价正确保存到数据库
2026-03-16 09:08:47 +00:00
OpenClaw Agent
71aa7211d6
fix: 首页组合收益率实时计算
2026-03-16 08:56:14 +00:00
OpenClaw Agent
c7712e57bb
feat: 组合列表增加今日涨跌额和持仓数量
...
- PortfolioListItem 新增 TodayProfit、TodayProfitCurrency 字段
- GetPortfolios 计算今日盈亏(从净值历史)
- Tags 显示持仓数量(如 '运行中 · USD · 3只')
2026-03-15 23:51:32 +00:00
OpenClaw Agent
dcd212efa7
feat: 自动触发净值计算
...
1. 创建组合后,如有持仓则自动计算净值
2. 创建交易后,自动更新当日净值
3. 异步执行,不阻塞主流程
2026-03-15 23:48:11 +00:00
OpenClaw Agent
74e5f85579
refactor: 移除 calculateDailyNav 接口
...
前端暂不需要,后续有需求再加
2026-03-15 14:43:35 +00:00
OpenClaw Agent
f7d346c0ae
fix: API路由重构以匹配前端期望
...
1. 创建交易: POST /api/v1/portfolio/{id}/transactions -> POST /api/v1/portfolio/transactions
- portfolioId 改为在 body 中传递
2. 新增计算净值端点: POST /api/v1/portfolio/{id}/nav-history/calculate
前端 API 格式已完全对齐
2026-03-15 14:35:26 +00:00
OpenClaw Agent
a0817081a8
refactor: 删除冗余的 /api/v1/portfolio/{id}/transactions 路由
...
保留 query parameter 方式: /api/v1/portfolio/transactions?portfolioId=xxx
2026-03-15 14:25:09 +00:00
OpenClaw Agent
7db098b5f9
fix: 添加 /api/v1/portfolio/transactions 路由支持 query parameter
...
前端请求格式: /api/v1/portfolio/transactions?portfolioId=xxx
后端原有路由: /api/v1/portfolio/{id}/transactions
添加 [HttpGet("transactions")] 支持 query parameter 方式
2026-03-15 14:20:11 +00:00
OpenClaw Agent
781f707419
fix: GetPortfolios 返回格式修正为 { items: [...] }
...
前端期望 data 是对象而非数组
2026-03-15 14:10:56 +00:00
OpenClaw Agent
d4c58c8a08
fix: 添加 /api/v1/portfolio/assets 端点
...
- IPortfolioFacade 添加 GetTotalAssetsAsync 方法
- PortfolioFacade 实现 GetTotalAssetsAsync
- PortfolioController 添加 [HttpGet("assets")] 端点(放在 {id} 路由之前)
2026-03-15 13:58:22 +00:00
OpenClaw Agent
41ff4a2135
fix: 修复 CurrentPrice 赋值错误
...
PortfolioService.cs: CurrentPrice 应该赋值为当前价格,而非今日盈亏
2026-03-15 13:49:54 +00:00
niannian zheng
49b9adfdab
fix 字段名称错误
2026-03-15 21:44:48 +08:00
OpenClaw Agent
60f3b487ff
fix: 修复 PortfolioService 和 PortfolioNavService 变量命名
...
- 统一局部变量使用 camelCase(C# 规范)
- 修复 DTO 属性名引用
- NavItem -> NavHistoryItem
- 修复 for 循环变量名
2026-03-15 13:31:49 +00:00
OpenClaw Agent
da12f6d163
fix: 修复编译错误和警告
...
- PortfolioFacade: 使用同步方法 GetStrategyById
- OkxMarketService: 响应模型属性改为可空类型
2026-03-15 13:18:24 +00:00
OpenClaw Agent
61e3023837
fix: 添加缺失的 StrategySignalResponse DTO
...
- 新增 StrategySignalResponse 类(策略信号响应)
- 新增 SignalAction 类(信号行动建议)
- 修正 PortfolioFacade 属性名为 PascalCase
2026-03-15 13:14:22 +00:00
OpenClaw Agent
625bfa1624
fix: 修复编译错误
...
- PortfolioRepository: Date -> NavDate 属性名修正
- MarketDataRepository: 修复 TiingoTicker.Ticker 空引用警告
2026-03-15 13:07:40 +00:00
niannian zheng
33392c4524
添加缺失的引用
2026-03-15 20:59:46 +08:00
niannian zheng
086d53f54d
添加单元测试项目
2026-03-15 20:57:10 +08:00
OpenClaw Agent
4ce29a1036
refactor: 架构优化 P0-P3
...
P0 - 安全修复:
- 移除硬编码 API Key,启动时校验必填环境变量
P1 - 高优先级:
- Entity 拆分:Position.cs, Transaction.cs 独立文件
- Controller Facade 封装:IPortfolioFacade 减少依赖注入
P2 - 中优先级:
- Repository 抽象:IPortfolioRepository, IMarketDataRepository
- MarketDataService 拆分:组合模式整合 Tencent/Tiingo/OKX
P3 - 低优先级:
- DTO 命名规范:统一 PascalCase
- 单元测试框架:xUnit + Moq + FluentAssertions
2026-03-15 12:54:05 +00:00
OpenClaw Agent
abcdc7110b
refactor: 后端架构优化 P0-P1
...
P0: 移除硬编码 API Key
- MarketDataService Tiingo Key 改为强制环境变量
P1: 拆分 Entity 文件
- 从 Portfolio.cs 提取 Position.cs
- 从 Portfolio.cs 提取 Transaction.cs
- 每个实体独立文件,符合单一职责原则
2026-03-15 11:44:06 +00:00
OpenClaw Agent
34f06ce941
perf: 优化历史价格获取,优先从缓存表查询
...
1. 先从 market_kline_cache 缓存表查特定日期价格
2. 缓存命中直接返回,减少 API 调用
3. 缓存未命中才调用 GetHistoricalDataAsync
4. 获取历史数据从 365 天减少到 30 天
5. 大幅减少 Tiingo API 消耗
2026-03-15 10:12:20 +00:00
OpenClaw Agent
267b0bd6ba
feat: 添加腾讯财经历史K线数据接口
...
- 新增 GetStockHistoricalFromTencentAsync 方法
- 腾讯财经历史数据免费无限制,解决 Tiingo 429 限流问题
- 获取历史数据时优先使用腾讯财经,失败后降级 Tiingo
- 支持日/周/月K线数据
- 数据格式: [日期, 开盘, 收盘, 最高, 最低, 成交量]
2026-03-15 10:06:14 +00:00
OpenClaw Agent
da86aa43e6
fix: 行情获取失败时不写入错误数据,添加 429 限流重试机制
...
1. GetHistoricalPriceAsync 返回 decimal? 而非 decimal
- 价格获取失败时返回 null 而非 0
2. BackfillNavHistoryInternalAsync 检查价格有效性
- 任何持仓价格获取失败时跳过该日期
- 不写入 totalValue=0 的错误数据
3. MarketDataService 添加 GetWithRetryAsync 方法
- 处理 429 Too Many Requests 限流
- 最多重试 3 次,指数退避
2026-03-15 10:02:52 +00:00
OpenClaw Agent
6a757f56da
fix: 添加 PortfolioNavHistory 到 CodeFirst 初始化,删除手动迁移脚本
...
- DatabaseService.InitializeDatabase() 添加 PortfolioNavHistory
- 删除 migrations/ 目录,使用 CodeFirst 自动建表
2026-03-15 08:29:05 +00:00
OpenClaw Agent
849db7d2b2
feat: 交易后自动触发净值历史重算
...
- CreateTransaction完成后删除该交易日期之后的净值记录
- 下次请求收益曲线时自动重新计算
- 修改CreateTransaction为async方法
- 注入IPortfolioNavService到PortfolioService
流程:
1. 用户买入/卖出 → 创建交易记录
2. 删除交易日期之后的净值历史
3. 下次请求收益曲线 → 自动回填缺失数据
2026-03-13 16:53:02 +00:00