@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.
Files changed (119) hide show
  1. package/README.md +103 -0
  2. package/bin/cli.js +160 -0
  3. package/bin/postinstall.js +57 -0
  4. package/bin/preuninstall.js +36 -0
  5. package/dist/server/broker/kiwoom/cache.js +86 -0
  6. package/dist/server/broker/kiwoom/cache.js.map +1 -0
  7. package/dist/server/broker/kiwoom/client.js +256 -0
  8. package/dist/server/broker/kiwoom/client.js.map +1 -0
  9. package/dist/server/broker/kiwoom/endpoints/_helpers.js +61 -0
  10. package/dist/server/broker/kiwoom/endpoints/_helpers.js.map +1 -0
  11. package/dist/server/broker/kiwoom/endpoints/account.js +448 -0
  12. package/dist/server/broker/kiwoom/endpoints/account.js.map +1 -0
  13. package/dist/server/broker/kiwoom/endpoints/detail.js +118 -0
  14. package/dist/server/broker/kiwoom/endpoints/detail.js.map +1 -0
  15. package/dist/server/broker/kiwoom/endpoints/investor.js +139 -0
  16. package/dist/server/broker/kiwoom/endpoints/investor.js.map +1 -0
  17. package/dist/server/broker/kiwoom/endpoints/order.js +134 -0
  18. package/dist/server/broker/kiwoom/endpoints/order.js.map +1 -0
  19. package/dist/server/broker/kiwoom/endpoints/quote.js +165 -0
  20. package/dist/server/broker/kiwoom/endpoints/quote.js.map +1 -0
  21. package/dist/server/broker/kiwoom/endpoints/ranking.js +180 -0
  22. package/dist/server/broker/kiwoom/endpoints/ranking.js.map +1 -0
  23. package/dist/server/broker/kiwoom/endpoints/sector.js +135 -0
  24. package/dist/server/broker/kiwoom/endpoints/sector.js.map +1 -0
  25. package/dist/server/broker/kiwoom/endpoints/theme.js +104 -0
  26. package/dist/server/broker/kiwoom/endpoints/theme.js.map +1 -0
  27. package/dist/server/broker/kiwoom/endpoints/universe.js +119 -0
  28. package/dist/server/broker/kiwoom/endpoints/universe.js.map +1 -0
  29. package/dist/server/broker/kiwoom/index.js +59 -0
  30. package/dist/server/broker/kiwoom/index.js.map +1 -0
  31. package/dist/server/broker/kiwoom/order-tracker.js +353 -0
  32. package/dist/server/broker/kiwoom/order-tracker.js.map +1 -0
  33. package/dist/server/broker/kiwoom/price-feed.js +119 -0
  34. package/dist/server/broker/kiwoom/price-feed.js.map +1 -0
  35. package/dist/server/broker/kiwoom/rate-limiter.js +97 -0
  36. package/dist/server/broker/kiwoom/rate-limiter.js.map +1 -0
  37. package/dist/server/broker/kiwoom/types.js +13 -0
  38. package/dist/server/broker/kiwoom/types.js.map +1 -0
  39. package/dist/server/broker/kiwoom/ws/client.js +370 -0
  40. package/dist/server/broker/kiwoom/ws/client.js.map +1 -0
  41. package/dist/server/broker/kiwoom/ws/endpoints/condition.js +146 -0
  42. package/dist/server/broker/kiwoom/ws/endpoints/condition.js.map +1 -0
  43. package/dist/server/broker/kiwoom/ws/realtime-bus.js +42 -0
  44. package/dist/server/broker/kiwoom/ws/realtime-bus.js.map +1 -0
  45. package/dist/server/broker/kiwoom/ws/types.js +19 -0
  46. package/dist/server/broker/kiwoom/ws/types.js.map +1 -0
  47. package/dist/server/broker/news.js +34 -0
  48. package/dist/server/broker/news.js.map +1 -0
  49. package/dist/server/bundle.js +43 -0
  50. package/dist/server/bundle.js.map +1 -0
  51. package/dist/server/calendar/krx-holidays.js +162 -0
  52. package/dist/server/calendar/krx-holidays.js.map +1 -0
  53. package/dist/server/config.js +263 -0
  54. package/dist/server/config.js.map +1 -0
  55. package/dist/server/db/sqlite.js +252 -0
  56. package/dist/server/db/sqlite.js.map +1 -0
  57. package/dist/server/diary/writer.js +266 -0
  58. package/dist/server/diary/writer.js.map +1 -0
  59. package/dist/server/index.js +316 -0
  60. package/dist/server/index.js.map +1 -0
  61. package/dist/server/jobs/universe-sync.js +132 -0
  62. package/dist/server/jobs/universe-sync.js.map +1 -0
  63. package/dist/server/jobs/watchdog.js +87 -0
  64. package/dist/server/jobs/watchdog.js.map +1 -0
  65. package/dist/server/journal/pnl-stats.js +108 -0
  66. package/dist/server/journal/pnl-stats.js.map +1 -0
  67. package/dist/server/journal/telemetry.js +174 -0
  68. package/dist/server/journal/telemetry.js.map +1 -0
  69. package/dist/server/journal/trade-journal.js +239 -0
  70. package/dist/server/journal/trade-journal.js.map +1 -0
  71. package/dist/server/llm/anthropic.js +98 -0
  72. package/dist/server/llm/anthropic.js.map +1 -0
  73. package/dist/server/llm/claude-cli.js +204 -0
  74. package/dist/server/llm/claude-cli.js.map +1 -0
  75. package/dist/server/llm/context-builder.js +229 -0
  76. package/dist/server/llm/context-builder.js.map +1 -0
  77. package/dist/server/llm/gemini.js +86 -0
  78. package/dist/server/llm/gemini.js.map +1 -0
  79. package/dist/server/llm/index.js +36 -0
  80. package/dist/server/llm/index.js.map +1 -0
  81. package/dist/server/llm/openai.js +87 -0
  82. package/dist/server/llm/openai.js.map +1 -0
  83. package/dist/server/personas/executor.js +318 -0
  84. package/dist/server/personas/executor.js.map +1 -0
  85. package/dist/server/personas/loader.js +165 -0
  86. package/dist/server/personas/loader.js.map +1 -0
  87. package/dist/server/personas/persona-agent.js +386 -0
  88. package/dist/server/personas/persona-agent.js.map +1 -0
  89. package/dist/server/personas/persona-state.js +170 -0
  90. package/dist/server/personas/persona-state.js.map +1 -0
  91. package/dist/server/personas/runner.js +162 -0
  92. package/dist/server/personas/runner.js.map +1 -0
  93. package/dist/server/personas/schema.js +123 -0
  94. package/dist/server/personas/schema.js.map +1 -0
  95. package/dist/server/personas/wake-plan.js +313 -0
  96. package/dist/server/personas/wake-plan.js.map +1 -0
  97. package/dist/server/readiness.js +414 -0
  98. package/dist/server/readiness.js.map +1 -0
  99. package/dist/server/routes.js +1216 -0
  100. package/dist/server/routes.js.map +1 -0
  101. package/dist/server/safety.js +153 -0
  102. package/dist/server/safety.js.map +1 -0
  103. package/dist/server/screener/engine.js +856 -0
  104. package/dist/server/screener/engine.js.map +1 -0
  105. package/dist/server/server-sync.js +427 -0
  106. package/dist/server/server-sync.js.map +1 -0
  107. package/dist/server/signing.js +39 -0
  108. package/dist/server/signing.js.map +1 -0
  109. package/dist/server/watchers/condition-watcher.js +519 -0
  110. package/dist/server/watchers/condition-watcher.js.map +1 -0
  111. package/dist/server/watchers/types.js +16 -0
  112. package/dist/server/watchers/types.js.map +1 -0
  113. package/dist/web/assets/index-62SMpbaf.js +79 -0
  114. package/dist/web/assets/index-BPLQR0wt.css +1 -0
  115. package/dist/web/index.html +14 -0
  116. package/package.json +93 -0
  117. package/scripts/com.mulsok.traders.client.plist.template +58 -0
  118. package/scripts/install-daemon.sh +156 -0
  119. 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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"}