@mulsok/traders-client 0.1.0
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 +103 -0
- package/bin/cli.js +160 -0
- package/bin/postinstall.js +57 -0
- package/bin/preuninstall.js +36 -0
- package/dist/server/broker/kiwoom/cache.js +86 -0
- package/dist/server/broker/kiwoom/cache.js.map +1 -0
- package/dist/server/broker/kiwoom/client.js +256 -0
- package/dist/server/broker/kiwoom/client.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/_helpers.js +61 -0
- package/dist/server/broker/kiwoom/endpoints/_helpers.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/account.js +448 -0
- package/dist/server/broker/kiwoom/endpoints/account.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/detail.js +118 -0
- package/dist/server/broker/kiwoom/endpoints/detail.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/investor.js +139 -0
- package/dist/server/broker/kiwoom/endpoints/investor.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/order.js +134 -0
- package/dist/server/broker/kiwoom/endpoints/order.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/quote.js +165 -0
- package/dist/server/broker/kiwoom/endpoints/quote.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/ranking.js +180 -0
- package/dist/server/broker/kiwoom/endpoints/ranking.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/sector.js +135 -0
- package/dist/server/broker/kiwoom/endpoints/sector.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/theme.js +104 -0
- package/dist/server/broker/kiwoom/endpoints/theme.js.map +1 -0
- package/dist/server/broker/kiwoom/endpoints/universe.js +119 -0
- package/dist/server/broker/kiwoom/endpoints/universe.js.map +1 -0
- package/dist/server/broker/kiwoom/index.js +59 -0
- package/dist/server/broker/kiwoom/index.js.map +1 -0
- package/dist/server/broker/kiwoom/order-tracker.js +353 -0
- package/dist/server/broker/kiwoom/order-tracker.js.map +1 -0
- package/dist/server/broker/kiwoom/price-feed.js +119 -0
- package/dist/server/broker/kiwoom/price-feed.js.map +1 -0
- package/dist/server/broker/kiwoom/rate-limiter.js +97 -0
- package/dist/server/broker/kiwoom/rate-limiter.js.map +1 -0
- package/dist/server/broker/kiwoom/types.js +13 -0
- package/dist/server/broker/kiwoom/types.js.map +1 -0
- package/dist/server/broker/kiwoom/ws/client.js +370 -0
- package/dist/server/broker/kiwoom/ws/client.js.map +1 -0
- package/dist/server/broker/kiwoom/ws/endpoints/condition.js +146 -0
- package/dist/server/broker/kiwoom/ws/endpoints/condition.js.map +1 -0
- package/dist/server/broker/kiwoom/ws/realtime-bus.js +42 -0
- package/dist/server/broker/kiwoom/ws/realtime-bus.js.map +1 -0
- package/dist/server/broker/kiwoom/ws/types.js +19 -0
- package/dist/server/broker/kiwoom/ws/types.js.map +1 -0
- package/dist/server/broker/news.js +34 -0
- package/dist/server/broker/news.js.map +1 -0
- package/dist/server/bundle.js +43 -0
- package/dist/server/bundle.js.map +1 -0
- package/dist/server/calendar/krx-holidays.js +162 -0
- package/dist/server/calendar/krx-holidays.js.map +1 -0
- package/dist/server/config.js +263 -0
- package/dist/server/config.js.map +1 -0
- package/dist/server/db/sqlite.js +252 -0
- package/dist/server/db/sqlite.js.map +1 -0
- package/dist/server/diary/writer.js +266 -0
- package/dist/server/diary/writer.js.map +1 -0
- package/dist/server/index.js +316 -0
- package/dist/server/index.js.map +1 -0
- package/dist/server/jobs/universe-sync.js +132 -0
- package/dist/server/jobs/universe-sync.js.map +1 -0
- package/dist/server/jobs/watchdog.js +87 -0
- package/dist/server/jobs/watchdog.js.map +1 -0
- package/dist/server/journal/pnl-stats.js +108 -0
- package/dist/server/journal/pnl-stats.js.map +1 -0
- package/dist/server/journal/telemetry.js +174 -0
- package/dist/server/journal/telemetry.js.map +1 -0
- package/dist/server/journal/trade-journal.js +239 -0
- package/dist/server/journal/trade-journal.js.map +1 -0
- package/dist/server/llm/anthropic.js +98 -0
- package/dist/server/llm/anthropic.js.map +1 -0
- package/dist/server/llm/claude-cli.js +204 -0
- package/dist/server/llm/claude-cli.js.map +1 -0
- package/dist/server/llm/context-builder.js +229 -0
- package/dist/server/llm/context-builder.js.map +1 -0
- package/dist/server/llm/gemini.js +86 -0
- package/dist/server/llm/gemini.js.map +1 -0
- package/dist/server/llm/index.js +36 -0
- package/dist/server/llm/index.js.map +1 -0
- package/dist/server/llm/openai.js +87 -0
- package/dist/server/llm/openai.js.map +1 -0
- package/dist/server/personas/executor.js +318 -0
- package/dist/server/personas/executor.js.map +1 -0
- package/dist/server/personas/loader.js +165 -0
- package/dist/server/personas/loader.js.map +1 -0
- package/dist/server/personas/persona-agent.js +386 -0
- package/dist/server/personas/persona-agent.js.map +1 -0
- package/dist/server/personas/persona-state.js +170 -0
- package/dist/server/personas/persona-state.js.map +1 -0
- package/dist/server/personas/runner.js +162 -0
- package/dist/server/personas/runner.js.map +1 -0
- package/dist/server/personas/schema.js +123 -0
- package/dist/server/personas/schema.js.map +1 -0
- package/dist/server/personas/wake-plan.js +313 -0
- package/dist/server/personas/wake-plan.js.map +1 -0
- package/dist/server/readiness.js +414 -0
- package/dist/server/readiness.js.map +1 -0
- package/dist/server/routes.js +1216 -0
- package/dist/server/routes.js.map +1 -0
- package/dist/server/safety.js +153 -0
- package/dist/server/safety.js.map +1 -0
- package/dist/server/screener/engine.js +856 -0
- package/dist/server/screener/engine.js.map +1 -0
- package/dist/server/server-sync.js +427 -0
- package/dist/server/server-sync.js.map +1 -0
- package/dist/server/signing.js +39 -0
- package/dist/server/signing.js.map +1 -0
- package/dist/server/watchers/condition-watcher.js +519 -0
- package/dist/server/watchers/condition-watcher.js.map +1 -0
- package/dist/server/watchers/types.js +16 -0
- package/dist/server/watchers/types.js.map +1 -0
- package/dist/web/assets/index-62SMpbaf.js +79 -0
- package/dist/web/assets/index-BPLQR0wt.css +1 -0
- package/dist/web/index.html +14 -0
- package/package.json +93 -0
- package/scripts/com.mulsok.traders.client.plist.template +58 -0
- package/scripts/install-daemon.sh +156 -0
- package/scripts/uninstall-daemon.sh +62 -0
|
@@ -0,0 +1,316 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Client entrypoint · Fastify + static web serve + 브라우저 자동 오픈.
|
|
3
|
+
* ADR-008 · 사용자 PC 에서만 돌아감. 서버 송신 데이터는 익명 집계 뿐.
|
|
4
|
+
*
|
|
5
|
+
* 환경 변수 로드 (loadConfig 보다 먼저):
|
|
6
|
+
* 1. ~/.mulsok-traders/.env (사용자별 · 운영 환경 우선)
|
|
7
|
+
* 2. apps/client 의 .env (개발 / 보조)
|
|
8
|
+
* 둘 다 옵셔널. 없으면 OS env / launchd plist EnvironmentVariables 사용.
|
|
9
|
+
*/
|
|
10
|
+
import { config as loadDotenv } from "dotenv";
|
|
11
|
+
import os from "node:os";
|
|
12
|
+
import path0 from "node:path";
|
|
13
|
+
loadDotenv({ path: path0.join(os.homedir(), ".mulsok-traders/.env") });
|
|
14
|
+
loadDotenv(); // cwd .env (override 안 함)
|
|
15
|
+
import Fastify from "fastify";
|
|
16
|
+
import fastifyStatic from "@fastify/static";
|
|
17
|
+
import fs from "node:fs";
|
|
18
|
+
import path from "node:path";
|
|
19
|
+
import { fileURLToPath } from "node:url";
|
|
20
|
+
import open from "open";
|
|
21
|
+
import { registerRoutes } from "./routes.js";
|
|
22
|
+
import { getPriceFeed } from "./broker/kiwoom/index.js";
|
|
23
|
+
import { getOrderTracker } from "./broker/kiwoom/order-tracker.js";
|
|
24
|
+
import { getConditionWatcher } from "./watchers/condition-watcher.js";
|
|
25
|
+
import { executePersonaCycle } from "./personas/persona-agent.js";
|
|
26
|
+
import { startPeriodicSync, stopPeriodicSync } from "./server-sync.js";
|
|
27
|
+
// SPEC-27 (ADR-012): AI 행동 cron 폐기. 시스템 watchdog 만 인프라로 유지.
|
|
28
|
+
import { startWatchdog, stopWatchdog } from "./jobs/watchdog.js";
|
|
29
|
+
import { logOrderSubmitted, logOrderStatus, logOrderRejected, logWatcherTriggered, logDecision, } from "./journal/trade-journal.js";
|
|
30
|
+
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
31
|
+
// ── 포트 선택 ────────────────────────────────────────────────
|
|
32
|
+
const START_PORT = parseInt(process.env.CLIENT_PORT ?? "8787", 10);
|
|
33
|
+
const HOST = process.env.CLIENT_HOST ?? "127.0.0.1";
|
|
34
|
+
async function findPort(start) {
|
|
35
|
+
const net = await import("node:net");
|
|
36
|
+
return new Promise((resolve) => {
|
|
37
|
+
const probe = (p) => {
|
|
38
|
+
const srv = net.createServer().once("error", () => probe(p + 1)).once("listening", () => {
|
|
39
|
+
srv.close(() => resolve(p));
|
|
40
|
+
}).listen(p, HOST);
|
|
41
|
+
};
|
|
42
|
+
probe(start);
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
async function main() {
|
|
46
|
+
const port = await findPort(START_PORT);
|
|
47
|
+
const app = Fastify({
|
|
48
|
+
logger: {
|
|
49
|
+
level: process.env.LOG_LEVEL ?? "info",
|
|
50
|
+
transport: {
|
|
51
|
+
target: "pino-pretty",
|
|
52
|
+
options: { colorize: true, translateTime: "HH:MM:ss" },
|
|
53
|
+
},
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
// ── API 등록 ──────────────────────────────────────────────
|
|
57
|
+
await registerRoutes(app);
|
|
58
|
+
// ── PriceFeed + ConditionWatcher (Phase 3) ───────────────
|
|
59
|
+
// PriceFeed: 종목 폴링 신호기 (10s tick · 200ms stagger)
|
|
60
|
+
// ConditionWatcher: LLM wake_plan.watchers 등록 + 트리거 시 콜백
|
|
61
|
+
// 양쪽 모두 인메모리 · 프로세스 재시작 시 상태 소실 (P3 MVP)
|
|
62
|
+
getPriceFeed().start();
|
|
63
|
+
getConditionWatcher().start();
|
|
64
|
+
// OrderTracker → trade-journal 연결 (모든 상태 전이 영구 기록)
|
|
65
|
+
getOrderTracker().setStatusChangeCallback((order, prev) => {
|
|
66
|
+
if (order.status === "rejected") {
|
|
67
|
+
logOrderRejected({
|
|
68
|
+
personaSlug: order.personaSlug,
|
|
69
|
+
symbolCode: order.symbolCode,
|
|
70
|
+
side: order.side,
|
|
71
|
+
quantity: order.intendedQty,
|
|
72
|
+
priceKrw: order.intendedPriceKrw,
|
|
73
|
+
clientOrderId: order.clientOrderId,
|
|
74
|
+
error: order.error ?? "unknown",
|
|
75
|
+
});
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
if (prev === "submitting" && order.status === "pending") {
|
|
79
|
+
logOrderSubmitted({
|
|
80
|
+
personaSlug: order.personaSlug,
|
|
81
|
+
symbolCode: order.symbolCode,
|
|
82
|
+
side: order.side,
|
|
83
|
+
quantity: order.intendedQty,
|
|
84
|
+
priceKrw: order.intendedPriceKrw,
|
|
85
|
+
brokerOrderNo: order.brokerOrderNo,
|
|
86
|
+
clientOrderId: order.clientOrderId,
|
|
87
|
+
});
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
logOrderStatus({
|
|
91
|
+
personaSlug: order.personaSlug,
|
|
92
|
+
symbolCode: order.symbolCode,
|
|
93
|
+
side: order.side,
|
|
94
|
+
brokerOrderNo: order.brokerOrderNo,
|
|
95
|
+
clientOrderId: order.clientOrderId,
|
|
96
|
+
filledQty: order.filledQty,
|
|
97
|
+
// SPEC-9 PnL 추적: intendedPriceKrw 를 체결가 근사로 (limit 정확 · market 추정)
|
|
98
|
+
priceKrw: order.intendedPriceKrw,
|
|
99
|
+
meta: { prev, next: order.status, pollCount: order.pollCount },
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
// ConditionWatcher 트리거 → trade-journal + 페르소나 자율 재호출
|
|
103
|
+
//
|
|
104
|
+
// 흐름 (executor loop):
|
|
105
|
+
// 1. watcher 트리거 발생 (time · price · event)
|
|
106
|
+
// 2. logWatcherTriggered (trigger 발화 자체 기록)
|
|
107
|
+
// 3. SPEC-26 emergency stop 가드 (halted 시 skip + journal 기록)
|
|
108
|
+
// 4. executePersonaDecision(personaSlug, {triggeredBy}) 호출
|
|
109
|
+
// → 해당 페르소나 재실행 · 새 wake_plan 반영 · watcher 재등록
|
|
110
|
+
// 5. 결과 (ok=true/false) journal 기록 — 사후 추적 + 자율 사이클 회귀 진단
|
|
111
|
+
// 6. 자율성 보장 (SPEC-10 정정): LLM 호출 횟수·간격 제한 X
|
|
112
|
+
// 유일한 rate limit = 키움 REST API · 무한 루프 방어 = wake-plan zod schema 의 minimum interval
|
|
113
|
+
getConditionWatcher().setGlobalTriggerCallback(async (ctx) => {
|
|
114
|
+
logWatcherTriggered({
|
|
115
|
+
personaSlug: ctx.watcher.personaSlug,
|
|
116
|
+
watcherId: ctx.watcher.id,
|
|
117
|
+
intent: ctx.watcher.spec.intent,
|
|
118
|
+
triggerData: ctx.triggerData,
|
|
119
|
+
});
|
|
120
|
+
// SPEC-26 · Emergency Stop 시 trigger callback skip + journal 기록
|
|
121
|
+
const { checkTriggerSafety } = await import("./safety.js");
|
|
122
|
+
const guard = checkTriggerSafety();
|
|
123
|
+
if (!guard.ok) {
|
|
124
|
+
console.warn(`[trigger] safety:${guard.code} → skip · ${ctx.watcher.id}`);
|
|
125
|
+
// SPEC-26 차단도 journal 에 기록 (decision eventType + llmContext 메타)
|
|
126
|
+
// 사후 분석 시 "왜 trigger 가 LLM 호출 안 했나?" 답할 수 있어야 함.
|
|
127
|
+
logDecision({
|
|
128
|
+
personaSlug: ctx.watcher.personaSlug,
|
|
129
|
+
llmContext: {
|
|
130
|
+
followup_status: "skipped_by_safety",
|
|
131
|
+
stage: "trigger_safety",
|
|
132
|
+
error: `safety:${guard.code} · ${guard.reason}`,
|
|
133
|
+
triggered_by: {
|
|
134
|
+
watcherId: ctx.watcher.id,
|
|
135
|
+
intent: ctx.watcher.spec.intent,
|
|
136
|
+
},
|
|
137
|
+
},
|
|
138
|
+
});
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
// SPEC-27 (ADR-011): stateful agent cycle 호출 (fire-and-forget)
|
|
142
|
+
const startedAt = Date.now();
|
|
143
|
+
executePersonaCycle(ctx.watcher.personaSlug, {
|
|
144
|
+
kind: "watcher",
|
|
145
|
+
watcherId: ctx.watcher.id,
|
|
146
|
+
intent: ctx.watcher.spec.intent,
|
|
147
|
+
}).then((result) => {
|
|
148
|
+
const elapsedMs = Date.now() - startedAt;
|
|
149
|
+
if (!result.ok) {
|
|
150
|
+
console.warn(`[executor] trigger 재호출 실패 · ${ctx.watcher.personaSlug} · ${result.stage} · ${result.error}`);
|
|
151
|
+
// 실패도 반드시 journal 기록 (사후 자율 사이클 진단 · 사용자 추적 가능)
|
|
152
|
+
logDecision({
|
|
153
|
+
personaSlug: ctx.watcher.personaSlug,
|
|
154
|
+
llmContext: {
|
|
155
|
+
followup_status: "failed",
|
|
156
|
+
stage: result.stage,
|
|
157
|
+
error: result.error,
|
|
158
|
+
triggered_by: {
|
|
159
|
+
watcherId: ctx.watcher.id,
|
|
160
|
+
intent: ctx.watcher.spec.intent,
|
|
161
|
+
},
|
|
162
|
+
elapsedMs,
|
|
163
|
+
},
|
|
164
|
+
});
|
|
165
|
+
}
|
|
166
|
+
else {
|
|
167
|
+
const stateAfter = result.stateAfter;
|
|
168
|
+
console.log(`[agent] trigger cycle 완료 · ${ctx.watcher.personaSlug} · ${result.elapsedMs}ms · ` +
|
|
169
|
+
`watchlist ${stateAfter?.watchlist.length ?? 0} · positions ${stateAfter?.positions.length ?? 0} · ` +
|
|
170
|
+
`nextRunAt ${stateAfter?.nextRunAt ?? "(없음)"}`);
|
|
171
|
+
}
|
|
172
|
+
}).catch((e) => {
|
|
173
|
+
const elapsedMs = Date.now() - startedAt;
|
|
174
|
+
console.error(`[executor] trigger 재호출 예외 · ${ctx.watcher.personaSlug}:`, e);
|
|
175
|
+
// 예외도 journal 기록 (try/catch 누락 회귀 시 사후 진단)
|
|
176
|
+
logDecision({
|
|
177
|
+
personaSlug: ctx.watcher.personaSlug,
|
|
178
|
+
llmContext: {
|
|
179
|
+
followup_status: "exception",
|
|
180
|
+
stage: "trigger_callback",
|
|
181
|
+
error: String(e),
|
|
182
|
+
triggered_by: {
|
|
183
|
+
watcherId: ctx.watcher.id,
|
|
184
|
+
intent: ctx.watcher.spec.intent,
|
|
185
|
+
},
|
|
186
|
+
elapsedMs,
|
|
187
|
+
},
|
|
188
|
+
});
|
|
189
|
+
});
|
|
190
|
+
});
|
|
191
|
+
// ── 서버 구독 동기화 시작 (1시간 주기 + 즉시 1회) ─────────
|
|
192
|
+
startPeriodicSync(3600_000);
|
|
193
|
+
// ── 페르소나 일기 cron 폐기 (SPEC-27 ADR-012) ─────────
|
|
194
|
+
// AI 가 strategy.md 안에서 직접 일기 형식으로 갱신함.
|
|
195
|
+
// ── SQLite watchers 테이블 vacuum (P1-4) — 30일+ 된 non-active row 자동 정리 ──
|
|
196
|
+
// 1년+ 운영 시 누적으로 restoreFromDb 성능 ↓ 방지. status='active' row 는 절대 건드리지 않음.
|
|
197
|
+
try {
|
|
198
|
+
const { vacuumWatchersOlderThan } = await import("./db/sqlite.js");
|
|
199
|
+
const removed = vacuumWatchersOlderThan(30);
|
|
200
|
+
if (removed > 0)
|
|
201
|
+
console.log(`[startup] watchers vacuum · removed ${removed} old rows (>30d non-active)`);
|
|
202
|
+
}
|
|
203
|
+
catch (e) {
|
|
204
|
+
console.warn("[startup] watchers vacuum 실패:", e);
|
|
205
|
+
}
|
|
206
|
+
// ── Watcher SQLite 복원 (SPEC-11) — restart 시 active watcher 재무장 ──
|
|
207
|
+
await getConditionWatcher().restoreFromDb();
|
|
208
|
+
// ── Universe sync cron 폐기 (SPEC-27 ADR-012) ──
|
|
209
|
+
// AI 가 필요할 때 도구로 호출 (또는 사용자가 수동 트리거)
|
|
210
|
+
// ── System watchdog (SPEC-27 ADR-012) — 1분 polling
|
|
211
|
+
// nextRunAt 도달 시 cycle 발화 + 잠듦 감지 alert
|
|
212
|
+
// 이건 "AI 행동 cron" 이 아닌 자율 시스템의 인프라 안전망
|
|
213
|
+
startWatchdog();
|
|
214
|
+
// ── 정적 파일 (Vite 빌드 결과) ─────────────────────────────
|
|
215
|
+
const webDist = path.resolve(__dirname, "../web");
|
|
216
|
+
if (fs.existsSync(webDist)) {
|
|
217
|
+
await app.register(fastifyStatic, {
|
|
218
|
+
root: webDist,
|
|
219
|
+
prefix: "/",
|
|
220
|
+
index: ["index.html"],
|
|
221
|
+
});
|
|
222
|
+
// SPA fallback (Fastify 는 자동 fallback 없음 · setNotFoundHandler 로 index 반환)
|
|
223
|
+
app.setNotFoundHandler((req, reply) => {
|
|
224
|
+
if (req.url.startsWith("/api")) {
|
|
225
|
+
reply.code(404).send({ error: "API route not found" });
|
|
226
|
+
return;
|
|
227
|
+
}
|
|
228
|
+
reply.type("text/html").send(fs.readFileSync(path.join(webDist, "index.html")));
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
console.warn(`[client] 빌드된 web 없음: ${webDist}\n` +
|
|
233
|
+
` - 개발: 이 Fastify 는 API 전용으로 돌고 UI 는 'npm run dev:web' 의 Vite dev server (http://localhost:5173) 에서.\n` +
|
|
234
|
+
` - 배포: 'npm run build' 후 dist/web 생성되면 이 서버가 함께 서빙.`);
|
|
235
|
+
}
|
|
236
|
+
// ── Graceful shutdown (SIGINT / SIGTERM · SPEC-25 line 743-744 보강) ──
|
|
237
|
+
//
|
|
238
|
+
// 순서:
|
|
239
|
+
// 1. Fastify close (새 요청 accept 중지 + 진행 중 요청 완료 대기)
|
|
240
|
+
// 2. ConditionWatcher.pauseTimers() (timer 만 정리 · DB watchers row 보존 → restart 시 복원)
|
|
241
|
+
// 3. OrderTracker.pauseAllPolling() (polling timer 정리 · 주문 상태는 키움 측에서 진행)
|
|
242
|
+
// 4. PriceFeed.stop()
|
|
243
|
+
// 5. cron 정리 (server-sync · diary · universe-sync)
|
|
244
|
+
// 6. closeDb (마지막)
|
|
245
|
+
// 7. process.exit(0)
|
|
246
|
+
//
|
|
247
|
+
// Ctrl+C 직후 즉시 종료 시 위험:
|
|
248
|
+
// - 진행 중 키움 API 호출 끊김 (응답 손실)
|
|
249
|
+
// - 진행 중 주문 polling 중단 (체결 정보 누락)
|
|
250
|
+
// - DB write 중간 끊김 가능
|
|
251
|
+
let shuttingDown = false;
|
|
252
|
+
const shutdown = async (signal) => {
|
|
253
|
+
if (shuttingDown)
|
|
254
|
+
return;
|
|
255
|
+
shuttingDown = true;
|
|
256
|
+
console.log(`\n[shutdown] ${signal} received · graceful close 시작...`);
|
|
257
|
+
try {
|
|
258
|
+
await app.close();
|
|
259
|
+
console.log("[shutdown] Fastify closed");
|
|
260
|
+
}
|
|
261
|
+
catch (e) {
|
|
262
|
+
console.error("[shutdown] Fastify close err:", e);
|
|
263
|
+
}
|
|
264
|
+
try {
|
|
265
|
+
getConditionWatcher().pauseTimers();
|
|
266
|
+
getOrderTracker().pauseAllPolling();
|
|
267
|
+
getPriceFeed().stop();
|
|
268
|
+
stopPeriodicSync();
|
|
269
|
+
stopWatchdog();
|
|
270
|
+
}
|
|
271
|
+
catch (e) {
|
|
272
|
+
console.error("[shutdown] cleanup err:", e);
|
|
273
|
+
}
|
|
274
|
+
try {
|
|
275
|
+
const { closeDb } = await import("./db/sqlite.js");
|
|
276
|
+
closeDb();
|
|
277
|
+
console.log("[shutdown] SQLite closed");
|
|
278
|
+
}
|
|
279
|
+
catch (e) {
|
|
280
|
+
console.error("[shutdown] closeDb err:", e);
|
|
281
|
+
}
|
|
282
|
+
console.log("[shutdown] complete");
|
|
283
|
+
process.exit(0);
|
|
284
|
+
};
|
|
285
|
+
process.on("SIGINT", () => void shutdown("SIGINT"));
|
|
286
|
+
process.on("SIGTERM", () => void shutdown("SIGTERM"));
|
|
287
|
+
try {
|
|
288
|
+
await app.listen({ port, host: HOST });
|
|
289
|
+
const url = `http://${HOST}:${port}`;
|
|
290
|
+
console.log(`\n🚀 mulsok-traders client · ${url}\n`);
|
|
291
|
+
// 개발 모드 (web 빌드 없음) 에는 Vite dev 사이트를 열어주는 편이 UX 가 좋음
|
|
292
|
+
const isDev = !fs.existsSync(webDist);
|
|
293
|
+
const openTarget = isDev ? "http://localhost:5173" : url;
|
|
294
|
+
// 기본 안 열림 · 명시적으로 OPEN=1 일 때만 자동 오픈 (개발 중 빈번한 재시작 방해 방지)
|
|
295
|
+
if (process.env.OPEN === "1") {
|
|
296
|
+
try {
|
|
297
|
+
await open(openTarget);
|
|
298
|
+
}
|
|
299
|
+
catch (e) {
|
|
300
|
+
console.warn(`[client] 브라우저 자동 오픈 실패 · 수동으로 ${openTarget} 열어주세요. (${e})`);
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
else {
|
|
304
|
+
console.log(`[client] 브라우저 자동 오픈 비활성 (수동: ${openTarget} · 자동 원하면 OPEN=1)`);
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
catch (e) {
|
|
308
|
+
console.error("[client] listen 실패:", e);
|
|
309
|
+
process.exit(1);
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
main().catch((e) => {
|
|
313
|
+
console.error("[client] bootstrap 실패:", e);
|
|
314
|
+
process.exit(1);
|
|
315
|
+
});
|
|
316
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src-server/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 유니버스 + 일봉 배치 job (apps/client)
|
|
3
|
+
*
|
|
4
|
+
* 흐름:
|
|
5
|
+
* 1. ka10099 → KOSPI + KOSDAQ 전종목 → SQLite universe 테이블
|
|
6
|
+
* 2. 각 종목 ka10081 일봉 120일 → SQLite daily_candles 테이블
|
|
7
|
+
* 3. sync_meta 에 결과 기록
|
|
8
|
+
*
|
|
9
|
+
* 호출 시점:
|
|
10
|
+
* - 장 마감 후 수동 호출 (/api/jobs/universe-sync POST)
|
|
11
|
+
* - 혹은 daily cron (추후 스케줄러 추가 시)
|
|
12
|
+
*
|
|
13
|
+
* 소요:
|
|
14
|
+
* - 유니버스 2호출 (KOSPI + KOSDAQ) · ~3000건
|
|
15
|
+
* - 일봉 ~3000 * 1호출 = ~3000 req · live 5/s → 10분
|
|
16
|
+
* - 초기 실행은 분할 권장 (opts.maxCodes 로 제한)
|
|
17
|
+
*
|
|
18
|
+
* Rate limit 안전:
|
|
19
|
+
* - kiwoom-client 의 rate limiter 가 자동으로 stagger
|
|
20
|
+
* - 배치 루프는 단순 순차 await · 별도 sleep 불필요
|
|
21
|
+
*/
|
|
22
|
+
import { getDailyChart, getStandardUniverse } from "../broker/kiwoom/index.js";
|
|
23
|
+
import { upsertUniverseBulk, upsertCandlesBulk, recordSync, } from "../db/sqlite.js";
|
|
24
|
+
/**
|
|
25
|
+
* 유니버스 + 일봉 배치 전체 실행.
|
|
26
|
+
*/
|
|
27
|
+
export async function runUniverseSync(opts = {}) {
|
|
28
|
+
const start = Date.now();
|
|
29
|
+
const maxCodes = opts.maxCodes ?? 50;
|
|
30
|
+
const candleDays = opts.candleDays ?? 120;
|
|
31
|
+
/* ─── 1. 유니버스 ─── */
|
|
32
|
+
const uniRes = await getStandardUniverse();
|
|
33
|
+
if (!uniRes.ok) {
|
|
34
|
+
recordSync("universe", 0, "error", uniRes.error.message);
|
|
35
|
+
return { universeCount: 0, candlesCount: 0, errorCount: 1, elapsedMs: Date.now() - start };
|
|
36
|
+
}
|
|
37
|
+
const allItems = [
|
|
38
|
+
...uniRes.data.kospi.map((item) => ({ item, market: "kospi" })),
|
|
39
|
+
...uniRes.data.kosdaq.map((item) => ({ item, market: "kosdaq" })),
|
|
40
|
+
];
|
|
41
|
+
const universeRows = allItems.map(({ item, market }) => ({
|
|
42
|
+
code: item.code,
|
|
43
|
+
name: item.name,
|
|
44
|
+
market,
|
|
45
|
+
market_code: item.marketCode,
|
|
46
|
+
up_name: item.upName,
|
|
47
|
+
up_size_name: item.upSizeName,
|
|
48
|
+
list_count: item.listCount,
|
|
49
|
+
last_price_krw: item.lastPriceKrw,
|
|
50
|
+
reg_day: item.regDay,
|
|
51
|
+
state: item.state,
|
|
52
|
+
audit_info: item.auditInfo,
|
|
53
|
+
order_warning_code: item.orderWarningCode,
|
|
54
|
+
company_class_name: item.companyClassName,
|
|
55
|
+
nxt_enable: item.nxtEnable ? 1 : 0,
|
|
56
|
+
updated_at: new Date().toISOString(),
|
|
57
|
+
}));
|
|
58
|
+
upsertUniverseBulk(universeRows);
|
|
59
|
+
recordSync("universe", universeRows.length);
|
|
60
|
+
/* ─── 2. 일봉 (타겟 코드 우선 · 없으면 maxCodes 제한) ─── */
|
|
61
|
+
let target;
|
|
62
|
+
if (opts.targetCodes && opts.targetCodes.length > 0) {
|
|
63
|
+
const wanted = new Set(opts.targetCodes.map((c) => c.replace(/^A/, "")));
|
|
64
|
+
target = universeRows.filter((r) => wanted.has(r.code));
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
target = maxCodes > 0 ? universeRows.slice(0, maxCodes) : universeRows;
|
|
68
|
+
}
|
|
69
|
+
let candlesCount = 0;
|
|
70
|
+
let errorCount = 0;
|
|
71
|
+
for (let i = 0; i < target.length; i++) {
|
|
72
|
+
const row = target[i];
|
|
73
|
+
const chartRes = await getDailyChart(row.code, candleDays);
|
|
74
|
+
opts.onProgress?.(i + 1, target.length, row.code);
|
|
75
|
+
if (!chartRes.ok) {
|
|
76
|
+
errorCount++;
|
|
77
|
+
continue;
|
|
78
|
+
}
|
|
79
|
+
const candleRows = chartRes.data.candles.map((c) => ({
|
|
80
|
+
code: row.code,
|
|
81
|
+
date: c.date,
|
|
82
|
+
open_krw: Math.round(c.openKrw),
|
|
83
|
+
high_krw: Math.round(c.highKrw),
|
|
84
|
+
low_krw: Math.round(c.lowKrw),
|
|
85
|
+
close_krw: Math.round(c.closeKrw),
|
|
86
|
+
volume: Math.round(c.volume),
|
|
87
|
+
}));
|
|
88
|
+
upsertCandlesBulk(candleRows);
|
|
89
|
+
candlesCount += candleRows.length;
|
|
90
|
+
}
|
|
91
|
+
recordSync(`candles:${new Date().toISOString().slice(0, 10)}`, candlesCount, errorCount === 0 ? "ok" : "error", errorCount > 0 ? `${errorCount}건 실패` : undefined);
|
|
92
|
+
return {
|
|
93
|
+
universeCount: universeRows.length,
|
|
94
|
+
candlesCount,
|
|
95
|
+
errorCount,
|
|
96
|
+
elapsedMs: Date.now() - start,
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
/* ─────────── 자동 cron (SPEC-19 · 매일 16:30 KST 캔들 backfill) ─────────── */
|
|
100
|
+
let cronTimer = null;
|
|
101
|
+
/**
|
|
102
|
+
* 매 분 시각 체크 → 16:30 KST 도달 시 자동 sync.
|
|
103
|
+
* 평일만 (주말 KRX 휴장).
|
|
104
|
+
*/
|
|
105
|
+
export function startUniverseSyncCron() {
|
|
106
|
+
if (cronTimer)
|
|
107
|
+
return;
|
|
108
|
+
cronTimer = setInterval(async () => {
|
|
109
|
+
const now = new Date();
|
|
110
|
+
const kst = new Date(now.getTime() + 9 * 3600_000);
|
|
111
|
+
const day = kst.getUTCDay();
|
|
112
|
+
if (day === 0 || day === 6)
|
|
113
|
+
return; // 주말 skip
|
|
114
|
+
if (kst.getUTCHours() === 16 && kst.getUTCMinutes() === 30) {
|
|
115
|
+
console.log("[universe-sync cron] 16:30 KST · 자동 backfill 시작");
|
|
116
|
+
try {
|
|
117
|
+
const r = await runUniverseSync({});
|
|
118
|
+
console.log(`[universe-sync cron] 완료 · universe ${r.universeCount} · candles ${r.candlesCount} · err ${r.errorCount} · ${r.elapsedMs}ms`);
|
|
119
|
+
}
|
|
120
|
+
catch (e) {
|
|
121
|
+
console.error("[universe-sync cron] 실패:", e);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}, 60_000);
|
|
125
|
+
console.log("[universe-sync] cron started · 매일 16:30 KST 자동 backfill");
|
|
126
|
+
}
|
|
127
|
+
export function stopUniverseSyncCron() {
|
|
128
|
+
if (cronTimer)
|
|
129
|
+
clearInterval(cronTimer);
|
|
130
|
+
cronTimer = null;
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=universe-sync.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"universe-sync.js","sourceRoot":"","sources":["../../../src-server/jobs/universe-sync.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SPEC-27 · System Watchdog (ADR-012).
|
|
3
|
+
*
|
|
4
|
+
* 페르소나 자율 발화의 안전망. 1분마다 polling:
|
|
5
|
+
* 1. 모든 구독 페르소나의 state.json 의 nextRunAt 확인
|
|
6
|
+
* - 도달 → executePersonaCycle({ kind: "scheduled" })
|
|
7
|
+
* 2. 페르소나 잠듦 감지
|
|
8
|
+
* - lastRunAt 24시간 + nextRunAt 없음 + 활성 watcher 0
|
|
9
|
+
* → console.warn + journal warn 이벤트 (UI banner 노출)
|
|
10
|
+
*
|
|
11
|
+
* AI 의 행동을 발화시키는 cron 이 아님 — 자율 시스템의 인프라 안전망.
|
|
12
|
+
*/
|
|
13
|
+
import { loadConfig } from "../config.js";
|
|
14
|
+
import { loadState } from "../personas/persona-state.js";
|
|
15
|
+
import { executePersonaCycle } from "../personas/persona-agent.js";
|
|
16
|
+
import { getConditionWatcher } from "../watchers/condition-watcher.js";
|
|
17
|
+
const POLL_INTERVAL_MS = 60_000; // 1분
|
|
18
|
+
const ASLEEP_THRESHOLD_MS = 24 * 3600_000; // 24시간
|
|
19
|
+
let timer = null;
|
|
20
|
+
const lastFiredAt = {}; // slug → ISO (중복 방지)
|
|
21
|
+
const lastAlertedAt = {}; // slug → ISO (alert 중복 방지 · 6시간)
|
|
22
|
+
export function startWatchdog() {
|
|
23
|
+
if (timer)
|
|
24
|
+
return;
|
|
25
|
+
timer = setInterval(() => { void tick(); }, POLL_INTERVAL_MS);
|
|
26
|
+
// 첫 부팅 시 즉시 1회 (지각된 nextRunAt 처리)
|
|
27
|
+
void tick();
|
|
28
|
+
console.log("[watchdog] started · 1분 polling · nextRunAt + 잠듦 감지");
|
|
29
|
+
}
|
|
30
|
+
export function stopWatchdog() {
|
|
31
|
+
if (timer)
|
|
32
|
+
clearInterval(timer);
|
|
33
|
+
timer = null;
|
|
34
|
+
}
|
|
35
|
+
async function tick() {
|
|
36
|
+
const cfg = loadConfig();
|
|
37
|
+
if (cfg.halted)
|
|
38
|
+
return; // halted 시 아무것도 안 함 (SPEC-26)
|
|
39
|
+
const subs = cfg.subscriptions ?? [];
|
|
40
|
+
for (const slug of subs) {
|
|
41
|
+
try {
|
|
42
|
+
await checkPersona(slug);
|
|
43
|
+
}
|
|
44
|
+
catch (e) {
|
|
45
|
+
console.warn(`[watchdog] ${slug} tick 실패:`, e);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
async function checkPersona(slug) {
|
|
50
|
+
const state = loadState(slug);
|
|
51
|
+
const now = Date.now();
|
|
52
|
+
// 1. nextRunAt 도달 → cycle 발화
|
|
53
|
+
if (state.nextRunAt) {
|
|
54
|
+
const nextMs = new Date(state.nextRunAt).getTime();
|
|
55
|
+
if (!Number.isNaN(nextMs) && nextMs <= now) {
|
|
56
|
+
const fireKey = state.nextRunAt;
|
|
57
|
+
if (lastFiredAt[slug] !== fireKey) {
|
|
58
|
+
lastFiredAt[slug] = fireKey;
|
|
59
|
+
console.log(`[watchdog] ${slug} nextRunAt 도달 (${state.nextRunAt}) · cycle 발화`);
|
|
60
|
+
const r = await executePersonaCycle(slug, { kind: "scheduled" });
|
|
61
|
+
console.log(`[watchdog] ${slug} cycle ${r.ok ? "✓" : `✗ ${r.stage}`} · ${r.elapsedMs}ms`);
|
|
62
|
+
return; // 이번 tick 종료 (cycle 후 state 갱신됨)
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
// 2. 잠듦 감지
|
|
67
|
+
const lastMs = new Date(state.lastRunAt).getTime();
|
|
68
|
+
if (Number.isNaN(lastMs))
|
|
69
|
+
return;
|
|
70
|
+
const sinceLast = now - lastMs;
|
|
71
|
+
if (sinceLast < ASLEEP_THRESHOLD_MS)
|
|
72
|
+
return;
|
|
73
|
+
if (state.nextRunAt)
|
|
74
|
+
return; // 다음 깨우기 잡혀있음
|
|
75
|
+
// 활성 watcher 가 있으면 잠든 게 아님 (시세 발화 대기 중)
|
|
76
|
+
const activeWatchers = getConditionWatcher().list({ personaSlug: slug, activeOnly: true }).length;
|
|
77
|
+
if (activeWatchers > 0)
|
|
78
|
+
return;
|
|
79
|
+
// 진짜 잠듦 — alert (6시간 1회 dedupe)
|
|
80
|
+
const alertKey = new Date(now).toISOString().slice(0, 13); // 시간 단위
|
|
81
|
+
if (lastAlertedAt[slug] === alertKey)
|
|
82
|
+
return;
|
|
83
|
+
lastAlertedAt[slug] = alertKey;
|
|
84
|
+
console.warn(`[watchdog] ⚠ ${slug} 잠들음 · lastRunAt ${state.lastRunAt} (${Math.floor(sinceLast / 3600_000)}h 전) · ` +
|
|
85
|
+
`nextRunAt 없음 · 활성 watcher 0`);
|
|
86
|
+
}
|
|
87
|
+
//# sourceMappingURL=watchdog.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"watchdog.js","sourceRoot":"","sources":["../../../src-server/jobs/watchdog.ts"],"names":[],"mappings":"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"}
|