chaimi-keep-mcp 3.3.3-beta.1 → 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.
- package/README.md +6 -0
- package/package.json +1 -1
- package/server.js +107 -8
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
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;
|
|
@@ -1855,13 +1877,7 @@ function formatDateWithTimezone(dateInput) {
|
|
|
1855
1877
|
timestamp = parseInt(dateInput, 10);
|
|
1856
1878
|
}
|
|
1857
1879
|
|
|
1858
|
-
//
|
|
1859
|
-
if (timestamp && timestamp % 1000 === 0 && String(timestamp).length === 13) {
|
|
1860
|
-
console.error(`⚠️ 警告:检测到可能错误的时间戳(秒级转毫秒级):${timestamp},使用当前时间替代`);
|
|
1861
|
-
return Date.now();
|
|
1862
|
-
}
|
|
1863
|
-
|
|
1864
|
-
// 如果是数字时间戳且没问题,直接返回
|
|
1880
|
+
// 如果是数字时间戳,直接返回
|
|
1865
1881
|
if (timestamp) {
|
|
1866
1882
|
return timestamp;
|
|
1867
1883
|
}
|
|
@@ -1899,6 +1915,89 @@ function formatDateWithTimezone(dateInput) {
|
|
|
1899
1915
|
}
|
|
1900
1916
|
}
|
|
1901
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
|
+
|
|
1902
2001
|
function extractWeightInGrams(weightStr) {
|
|
1903
2002
|
if (!weightStr || typeof weightStr !== 'string') return 0;
|
|
1904
2003
|
|