feat: 实现GUID主键与service_mask索引改造

- 将主键从自增id改为GUID格式并添加格式校验
- 为service_mask添加表达式索引优化首位查询性能
- 更新相关文档说明改造方案与验证步骤
- 添加统计模块记录数据库写入与Kafka消费量
- 重构Redis心跳协议改用LIST类型存储项目状态
- 修复部署脚本中的服务名称不一致问题
This commit is contained in:
2026-01-17 18:37:44 +08:00
parent 662eeee380
commit 41301f9ce5
21 changed files with 828 additions and 106 deletions

102
src/stats/statsManager.js Normal file
View File

@@ -0,0 +1,102 @@
class StatsCounters {
constructor() {
this._minuteBuf = new SharedArrayBuffer(BigInt64Array.BYTES_PER_ELEMENT * 3);
this._minute = new BigInt64Array(this._minuteBuf);
}
incDbWritten(n = 1) {
const v = BigInt(Math.max(0, Number(n) || 0));
if (v === 0n) return;
Atomics.add(this._minute, 0, v);
}
incFiltered(n = 1) {
const v = BigInt(Math.max(0, Number(n) || 0));
if (v === 0n) return;
Atomics.add(this._minute, 1, v);
}
incKafkaPulled(n = 1) {
const v = BigInt(Math.max(0, Number(n) || 0));
if (v === 0n) return;
Atomics.add(this._minute, 2, v);
}
snapshotAndResetMinute() {
const dbWritten = Atomics.exchange(this._minute, 0, 0n);
const filtered = Atomics.exchange(this._minute, 1, 0n);
const kafkaPulled = Atomics.exchange(this._minute, 2, 0n);
return { dbWritten, filtered, kafkaPulled };
}
}
const pad2 = (n) => String(n).padStart(2, '0');
const pad3 = (n) => String(n).padStart(3, '0');
const formatTimestamp = (d) => {
const year = d.getFullYear();
const month = pad2(d.getMonth() + 1);
const day = pad2(d.getDate());
const hour = pad2(d.getHours());
const minute = pad2(d.getMinutes());
const second = pad2(d.getSeconds());
const ms = pad3(d.getMilliseconds());
return `${year}-${month}-${day} ${hour}:${minute}:${second}.${ms}`;
};
class StatsReporter {
constructor({ redis, stats }) {
this.redis = redis;
this.stats = stats;
this._timer = null;
this._running = false;
}
start() {
if (this._running) return;
this._running = true;
this._scheduleNext();
}
stop() {
this._running = false;
if (this._timer) {
clearTimeout(this._timer);
this._timer = null;
}
}
flushOnce() {
if (!this.redis?.isEnabled?.()) return;
const { dbWritten, filtered, kafkaPulled } = this.stats.snapshotAndResetMinute();
const ts = formatTimestamp(new Date());
this.redis.pushConsoleLog?.({ level: 'info', message: `[STATS] ${ts} 数据库写入量: ${dbWritten}`, metadata: { module: 'stats' } });
this.redis.pushConsoleLog?.({ level: 'info', message: `[STATS] ${ts} 数据过滤量: ${filtered}`, metadata: { module: 'stats' } });
this.redis.pushConsoleLog?.({ level: 'info', message: `[STATS] ${ts} Kafka拉取量: ${kafkaPulled}`, metadata: { module: 'stats' } });
}
_scheduleNext() {
if (!this._running) return;
if (this._timer) return;
const now = Date.now();
const delay = 60_000 - (now % 60_000);
this._timer = setTimeout(() => {
this._timer = null;
try {
this.flushOnce();
} catch (err) {
this.redis?.pushConsoleLog?.({
level: 'warn',
message: `[ERROR] ${formatTimestamp(new Date())} 统计任务异常: ${String(err?.message ?? err)}`,
metadata: { module: 'stats' },
});
} finally {
this._scheduleNext();
}
}, delay);
}
}
export { StatsCounters, StatsReporter, formatTimestamp };