chaimi-keep-mcp 3.3.3-beta.2 → 3.3.3-beta.3

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 (3) hide show
  1. package/README.md +6 -0
  2. package/package.json +1 -1
  3. package/server.js +106 -1
package/README.md CHANGED
@@ -89,6 +89,12 @@ export MCP_PROMPT_URL="你的Prompt服务地址"
89
89
 
90
90
  ## Changelog
91
91
 
92
+ ### v3.3.3-beta.3 (2026-04-29)
93
+ - **优化** 文本记账时间解析 - 使用 time_description 替代直接计算时间戳,解决 AI 计算时间错误的问题
94
+
95
+ ### v3.3.3-beta.2 (2026-04-29)
96
+ - **修复** 时间入库错误 - 解决消费时间被错误替换为当前时间的问题
97
+
92
98
  ### v3.3.1-beta.0 (2026-04-27)
93
99
  - **优化** 简化工具描述 - save_expense/save_receipt/save_income 描述更简洁,不暴露业务逻辑
94
100
  - **优化** 参数类型自动转换 - 支持 CLI 工具的字符串传参自动转为数字
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "chaimi-keep-mcp",
3
- "version": "3.3.3-beta.2",
3
+ "version": "3.3.3-beta.3",
4
4
  "description": "柴米AI记账 MCP Server - 支持 Claude、Cursor、OpenClaw、WorkBuddy 等 AI 工具直接记账",
5
5
  "main": "server.js",
6
6
  "bin": {
package/server.js CHANGED
@@ -999,6 +999,15 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
999
999
  break;
1000
1000
  }
1001
1001
 
1002
+ // 【新增】处理 time_description(文本记账新时间格式)
1003
+ let timeNote = '';
1004
+ if (processedArgs.time_description) {
1005
+ const parsedDate = parseTimeDescription(processedArgs.time_description, Date.now());
1006
+ processedArgs.date = parsedDate;
1007
+ timeNote = getTimeNote(processedArgs.time_description);
1008
+ console.log(`[save_expense] 时间描述解析: ${processedArgs.time_description} → ${parsedDate} (${new Date(parsedDate).toLocaleString('zh-CN')})`);
1009
+ }
1010
+
1002
1011
  // P1: 日期合理性检查
1003
1012
  if (processedArgs.date) {
1004
1013
  const validation = validateDate(processedArgs.date, '消费日期');
@@ -1011,6 +1020,18 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
1011
1020
  userMessage = validation.message;
1012
1021
  break;
1013
1022
  }
1023
+
1024
+ // 【新增】校验时间是否在合理范围内(±1年)
1025
+ const dateObj = new Date(processedArgs.date);
1026
+ const now = new Date();
1027
+ const oneYearAgo = new Date(now.getFullYear() - 1, 0, 1);
1028
+ const oneYearLater = new Date(now.getFullYear() + 1, 11, 31);
1029
+
1030
+ if (dateObj < oneYearAgo || dateObj > oneYearLater) {
1031
+ console.warn(`[save_expense] 时间异常:${processedArgs.date} (${dateObj.toLocaleString('zh-CN')}),使用当前时间`);
1032
+ processedArgs.date = Date.now();
1033
+ timeNote = '⏰ 时间说明:检测到时间异常,已使用当前时间';
1034
+ }
1014
1035
  }
1015
1036
 
1016
1037
  const mcpParams = convertParams('save_expense', processedArgs);
@@ -1057,7 +1078,8 @@ server.setRequestHandler(CallToolRequestSchema, async (request) => {
1057
1078
  日期: result.data?.date ? new Date(result.data.date).toLocaleDateString('zh-CN') : new Date().toLocaleDateString('zh-CN'),
1058
1079
  正能量祝福语: friendlyEnding,
1059
1080
  insightsText,
1060
- achievementsText
1081
+ achievementsText,
1082
+ timeNote // 【新增】时间说明
1061
1083
  };
1062
1084
  }
1063
1085
  break;
@@ -1893,6 +1915,89 @@ function formatDateWithTimezone(dateInput) {
1893
1915
  }
1894
1916
  }
1895
1917
 
1918
+ /**
1919
+ * 解析时间描述为时间戳
1920
+ * @param {string} timeDesc - 时间描述(如 'yesterday', 'today')
1921
+ * @param {number} currentTimestamp - 当前时间戳
1922
+ * @returns {number} 时间戳
1923
+ */
1924
+ function parseTimeDescription(timeDesc, currentTimestamp) {
1925
+ const now = new Date(currentTimestamp);
1926
+ const year = now.getFullYear();
1927
+ const month = now.getMonth();
1928
+ const day = now.getDate();
1929
+
1930
+ switch(timeDesc) {
1931
+ case 'just_now':
1932
+ case 'today':
1933
+ return currentTimestamp;
1934
+
1935
+ case 'yesterday':
1936
+ return new Date(year, month, day - 1, 12, 0, 0).getTime();
1937
+
1938
+ case 'yesterday_evening':
1939
+ return new Date(year, month, day - 1, 19, 0, 0).getTime();
1940
+
1941
+ case 'this_morning':
1942
+ return new Date(year, month, day, 8, 0, 0).getTime();
1943
+
1944
+ case 'this_noon':
1945
+ return new Date(year, month, day, 12, 0, 0).getTime();
1946
+
1947
+ case 'this_afternoon':
1948
+ return new Date(year, month, day, 14, 0, 0).getTime();
1949
+
1950
+ case 'this_evening':
1951
+ return new Date(year, month, day, 19, 0, 0).getTime();
1952
+
1953
+ default:
1954
+ // 尝试解析具体日期格式:2026-03-15 或 2026-03-15T15:00
1955
+ if (/^\d{4}-\d{2}-\d{2}$/.test(timeDesc)) {
1956
+ const [y, m, d] = timeDesc.split('-').map(Number);
1957
+ return new Date(y, m - 1, d, 12, 0, 0).getTime();
1958
+ }
1959
+
1960
+ if (/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}$/.test(timeDesc)) {
1961
+ return new Date(timeDesc).getTime();
1962
+ }
1963
+
1964
+ // 兜底:返回当前时间
1965
+ return currentTimestamp;
1966
+ }
1967
+ }
1968
+
1969
+ /**
1970
+ * 获取时间说明文本
1971
+ * @param {string} timeDesc - 时间描述
1972
+ * @returns {string} 时间说明
1973
+ */
1974
+ function getTimeNote(timeDesc) {
1975
+ const descMap = {
1976
+ 'just_now': '刚刚',
1977
+ 'today': '今天',
1978
+ 'yesterday': '昨天',
1979
+ 'yesterday_evening': '昨晚',
1980
+ 'this_morning': '今早',
1981
+ 'this_noon': '今天中午',
1982
+ 'this_afternoon': '今天下午',
1983
+ 'this_evening': '今晚'
1984
+ };
1985
+
1986
+ if (descMap[timeDesc]) {
1987
+ return `⏰ 时间说明:系统使用"${descMap[timeDesc]}"作为默认时间`;
1988
+ }
1989
+
1990
+ if (timeDesc && timeDesc.match(/^\d{4}-\d{2}-\d{2}$/)) {
1991
+ return `⏰ 时间说明:记录日期为 ${timeDesc}`;
1992
+ }
1993
+
1994
+ if (timeDesc && timeDesc.match(/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}$/)) {
1995
+ return `⏰ 时间说明:记录精确时间为 ${timeDesc}`;
1996
+ }
1997
+
1998
+ return '';
1999
+ }
2000
+
1896
2001
  function extractWeightInGrams(weightStr) {
1897
2002
  if (!weightStr || typeof weightStr !== 'string') return 0;
1898
2003