@tienne/gestalt 0.19.2 → 0.19.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 (147) hide show
  1. package/dist/package.json +1 -1
  2. package/package.json +1 -1
  3. package/dist/skills/seed/SKILL.md +0 -92
  4. package/dist/src/cli/commands/monitor.d.ts +0 -2
  5. package/dist/src/cli/commands/monitor.d.ts.map +0 -1
  6. package/dist/src/cli/commands/monitor.js +0 -13
  7. package/dist/src/cli/commands/monitor.js.map +0 -1
  8. package/dist/src/cli/commands/seed.d.ts +0 -4
  9. package/dist/src/cli/commands/seed.d.ts.map +0 -1
  10. package/dist/src/cli/commands/seed.js +0 -34
  11. package/dist/src/cli/commands/seed.js.map +0 -1
  12. package/dist/src/interview/ambiguity.d.ts +0 -8
  13. package/dist/src/interview/ambiguity.d.ts.map +0 -1
  14. package/dist/src/interview/ambiguity.js +0 -69
  15. package/dist/src/interview/ambiguity.js.map +0 -1
  16. package/dist/src/mcp/tools/seed-passthrough.d.ts +0 -5
  17. package/dist/src/mcp/tools/seed-passthrough.d.ts.map +0 -1
  18. package/dist/src/mcp/tools/seed-passthrough.js +0 -29
  19. package/dist/src/mcp/tools/seed-passthrough.js.map +0 -1
  20. package/dist/src/mcp/tools/seed.d.ts +0 -5
  21. package/dist/src/mcp/tools/seed.d.ts.map +0 -1
  22. package/dist/src/mcp/tools/seed.js +0 -19
  23. package/dist/src/mcp/tools/seed.js.map +0 -1
  24. package/dist/src/recording/agg-converter.d.ts +0 -25
  25. package/dist/src/recording/agg-converter.d.ts.map +0 -1
  26. package/dist/src/recording/agg-converter.js +0 -80
  27. package/dist/src/recording/agg-converter.js.map +0 -1
  28. package/dist/src/recording/agg-installer.d.ts +0 -6
  29. package/dist/src/recording/agg-installer.d.ts.map +0 -1
  30. package/dist/src/recording/agg-installer.js +0 -50
  31. package/dist/src/recording/agg-installer.js.map +0 -1
  32. package/dist/src/recording/asciinema-installer.d.ts +0 -6
  33. package/dist/src/recording/asciinema-installer.d.ts.map +0 -1
  34. package/dist/src/recording/asciinema-installer.js +0 -50
  35. package/dist/src/recording/asciinema-installer.js.map +0 -1
  36. package/dist/src/recording/asciinema-recorder.d.ts +0 -26
  37. package/dist/src/recording/asciinema-recorder.d.ts.map +0 -1
  38. package/dist/src/recording/asciinema-recorder.js +0 -52
  39. package/dist/src/recording/asciinema-recorder.js.map +0 -1
  40. package/dist/src/recording/cast-generator.d.ts +0 -7
  41. package/dist/src/recording/cast-generator.d.ts.map +0 -1
  42. package/dist/src/recording/cast-generator.js +0 -97
  43. package/dist/src/recording/cast-generator.js.map +0 -1
  44. package/dist/src/recording/filename-generator.d.ts +0 -19
  45. package/dist/src/recording/filename-generator.d.ts.map +0 -1
  46. package/dist/src/recording/filename-generator.js +0 -67
  47. package/dist/src/recording/filename-generator.js.map +0 -1
  48. package/dist/src/recording/gif-generator.d.ts +0 -21
  49. package/dist/src/recording/gif-generator.d.ts.map +0 -1
  50. package/dist/src/recording/gif-generator.js +0 -121
  51. package/dist/src/recording/gif-generator.js.map +0 -1
  52. package/dist/src/recording/recording-dir.d.ts +0 -5
  53. package/dist/src/recording/recording-dir.d.ts.map +0 -1
  54. package/dist/src/recording/recording-dir.js +0 -13
  55. package/dist/src/recording/recording-dir.js.map +0 -1
  56. package/dist/src/recording/recording-orchestrator.d.ts +0 -50
  57. package/dist/src/recording/recording-orchestrator.d.ts.map +0 -1
  58. package/dist/src/recording/recording-orchestrator.js +0 -98
  59. package/dist/src/recording/recording-orchestrator.js.map +0 -1
  60. package/dist/src/recording/resume-detector.d.ts +0 -10
  61. package/dist/src/recording/resume-detector.d.ts.map +0 -1
  62. package/dist/src/recording/resume-detector.js +0 -14
  63. package/dist/src/recording/resume-detector.js.map +0 -1
  64. package/dist/src/recording/segment-merger.d.ts +0 -27
  65. package/dist/src/recording/segment-merger.d.ts.map +0 -1
  66. package/dist/src/recording/segment-merger.js +0 -65
  67. package/dist/src/recording/segment-merger.js.map +0 -1
  68. package/dist/src/recording/terminal-recorder.d.ts +0 -31
  69. package/dist/src/recording/terminal-recorder.d.ts.map +0 -1
  70. package/dist/src/recording/terminal-recorder.js +0 -111
  71. package/dist/src/recording/terminal-recorder.js.map +0 -1
  72. package/dist/src/scripts/postinstall.d.ts +0 -2
  73. package/dist/src/scripts/postinstall.d.ts.map +0 -1
  74. package/dist/src/scripts/postinstall.js +0 -29
  75. package/dist/src/scripts/postinstall.js.map +0 -1
  76. package/dist/src/seed/extractor.d.ts +0 -15
  77. package/dist/src/seed/extractor.d.ts.map +0 -1
  78. package/dist/src/seed/extractor.js +0 -88
  79. package/dist/src/seed/extractor.js.map +0 -1
  80. package/dist/src/seed/generator.d.ts +0 -12
  81. package/dist/src/seed/generator.d.ts.map +0 -1
  82. package/dist/src/seed/generator.js +0 -66
  83. package/dist/src/seed/generator.js.map +0 -1
  84. package/dist/src/seed/passthrough-generator.d.ts +0 -31
  85. package/dist/src/seed/passthrough-generator.d.ts.map +0 -1
  86. package/dist/src/seed/passthrough-generator.js +0 -80
  87. package/dist/src/seed/passthrough-generator.js.map +0 -1
  88. package/dist/src/seed/schema.d.ts +0 -145
  89. package/dist/src/seed/schema.d.ts.map +0 -1
  90. package/dist/src/seed/schema.js +0 -37
  91. package/dist/src/seed/schema.js.map +0 -1
  92. package/dist/src/tui/components/TUIApp.d.ts +0 -20
  93. package/dist/src/tui/components/TUIApp.d.ts.map +0 -1
  94. package/dist/src/tui/components/TUIApp.js +0 -84
  95. package/dist/src/tui/components/TUIApp.js.map +0 -1
  96. package/dist/src/tui/hooks/event-store-reader.d.ts +0 -28
  97. package/dist/src/tui/hooks/event-store-reader.d.ts.map +0 -1
  98. package/dist/src/tui/hooks/event-store-reader.js +0 -141
  99. package/dist/src/tui/hooks/event-store-reader.js.map +0 -1
  100. package/dist/src/tui/hooks/useEventStorePoller.d.ts +0 -12
  101. package/dist/src/tui/hooks/useEventStorePoller.d.ts.map +0 -1
  102. package/dist/src/tui/hooks/useEventStorePoller.js +0 -84
  103. package/dist/src/tui/hooks/useEventStorePoller.js.map +0 -1
  104. package/dist/src/tui/screens/DashboardScreen.d.ts +0 -4
  105. package/dist/src/tui/screens/DashboardScreen.d.ts.map +0 -1
  106. package/dist/src/tui/screens/DashboardScreen.js +0 -132
  107. package/dist/src/tui/screens/DashboardScreen.js.map +0 -1
  108. package/dist/src/tui/screens/DebugScreen.d.ts +0 -4
  109. package/dist/src/tui/screens/DebugScreen.d.ts.map +0 -1
  110. package/dist/src/tui/screens/DebugScreen.js +0 -40
  111. package/dist/src/tui/screens/DebugScreen.js.map +0 -1
  112. package/dist/src/tui/screens/EvolutionScreen.d.ts +0 -4
  113. package/dist/src/tui/screens/EvolutionScreen.d.ts.map +0 -1
  114. package/dist/src/tui/screens/EvolutionScreen.js +0 -136
  115. package/dist/src/tui/screens/EvolutionScreen.js.map +0 -1
  116. package/dist/src/tui/screens/HUDPanel.d.ts +0 -4
  117. package/dist/src/tui/screens/HUDPanel.d.ts.map +0 -1
  118. package/dist/src/tui/screens/HUDPanel.js +0 -13
  119. package/dist/src/tui/screens/HUDPanel.js.map +0 -1
  120. package/dist/src/tui/screens/InterviewScreen.d.ts +0 -4
  121. package/dist/src/tui/screens/InterviewScreen.d.ts.map +0 -1
  122. package/dist/src/tui/screens/InterviewScreen.js +0 -103
  123. package/dist/src/tui/screens/InterviewScreen.js.map +0 -1
  124. package/dist/src/tui/screens/LogScreen.d.ts +0 -4
  125. package/dist/src/tui/screens/LogScreen.d.ts.map +0 -1
  126. package/dist/src/tui/screens/LogScreen.js +0 -83
  127. package/dist/src/tui/screens/LogScreen.js.map +0 -1
  128. package/dist/src/tui/screens/SessionListScreen.d.ts +0 -4
  129. package/dist/src/tui/screens/SessionListScreen.d.ts.map +0 -1
  130. package/dist/src/tui/screens/SessionListScreen.js +0 -71
  131. package/dist/src/tui/screens/SessionListScreen.js.map +0 -1
  132. package/dist/src/tui/screens/SpecViewerScreen.d.ts +0 -4
  133. package/dist/src/tui/screens/SpecViewerScreen.d.ts.map +0 -1
  134. package/dist/src/tui/screens/SpecViewerScreen.js +0 -73
  135. package/dist/src/tui/screens/SpecViewerScreen.js.map +0 -1
  136. package/dist/src/tui/widgets/DriftMeter.d.ts +0 -15
  137. package/dist/src/tui/widgets/DriftMeter.d.ts.map +0 -1
  138. package/dist/src/tui/widgets/DriftMeter.js +0 -27
  139. package/dist/src/tui/widgets/DriftMeter.js.map +0 -1
  140. package/dist/src/tui/widgets/GestaltPrincipleBar.d.ts +0 -9
  141. package/dist/src/tui/widgets/GestaltPrincipleBar.d.ts.map +0 -1
  142. package/dist/src/tui/widgets/GestaltPrincipleBar.js +0 -35
  143. package/dist/src/tui/widgets/GestaltPrincipleBar.js.map +0 -1
  144. package/dist/src/tui/widgets/TaskDAGTree.d.ts +0 -15
  145. package/dist/src/tui/widgets/TaskDAGTree.d.ts.map +0 -1
  146. package/dist/src/tui/widgets/TaskDAGTree.js +0 -54
  147. package/dist/src/tui/widgets/TaskDAGTree.js.map +0 -1
@@ -1 +0,0 @@
1
- {"version":3,"file":"recording-orchestrator.js","sourceRoot":"","sources":["../../../src/recording/recording-orchestrator.ts"],"names":[],"mappings":"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"}
@@ -1,10 +0,0 @@
1
- export interface ResumeInfo {
2
- isResuming: boolean;
3
- framesPath: string;
4
- }
5
- /**
6
- * sessionId에 해당하는 .frames 임시 파일이 존재하는지 확인한다.
7
- * 존재하면 이전 녹화를 이어붙일 수 있음을 의미한다.
8
- */
9
- export declare function detectResume(sessionId: string): ResumeInfo;
10
- //# sourceMappingURL=resume-detector.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"resume-detector.d.ts","sourceRoot":"","sources":["../../../src/recording/resume-detector.ts"],"names":[],"mappings":"AAGA,MAAM,WAAW,UAAU;IACzB,UAAU,EAAE,OAAO,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED;;;GAGG;AACH,wBAAgB,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG,UAAU,CAM1D"}
@@ -1,14 +0,0 @@
1
- import { existsSync } from 'node:fs';
2
- import { getFramesPath } from './recording-dir.js';
3
- /**
4
- * sessionId에 해당하는 .frames 임시 파일이 존재하는지 확인한다.
5
- * 존재하면 이전 녹화를 이어붙일 수 있음을 의미한다.
6
- */
7
- export function detectResume(sessionId) {
8
- const framesPath = getFramesPath(sessionId);
9
- return {
10
- isResuming: existsSync(framesPath),
11
- framesPath,
12
- };
13
- }
14
- //# sourceMappingURL=resume-detector.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"resume-detector.js","sourceRoot":"","sources":["../../../src/recording/resume-detector.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAOnD;;;GAGG;AACH,MAAM,UAAU,YAAY,CAAC,SAAiB;IAC5C,MAAM,UAAU,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC;IAC5C,OAAO;QACL,UAAU,EAAE,UAAU,CAAC,UAAU,CAAC;QAClC,UAAU;KACX,CAAC;AACJ,CAAC"}
@@ -1,27 +0,0 @@
1
- import type { TerminalFrame } from '../core/types.js';
2
- import { GifGenerator } from './gif-generator.js';
3
- /**
4
- * SegmentMerger: 동일 sessionId의 여러 .frames 파일 세그먼트를
5
- * 시간순으로 병합해 GifGenerator에 전달한다.
6
- *
7
- * 현재 아키텍처에서는 sessionId당 .frames 파일이 1개이므로,
8
- * 단일 파일 읽기도 이 클래스를 경유해 일관된 인터페이스를 유지한다.
9
- */
10
- export declare class SegmentMerger {
11
- private readonly gifGenerator;
12
- constructor(gifGenerator?: GifGenerator);
13
- /**
14
- * 여러 .frames 파일을 읽어 타임스탬프 기준 오름차순으로 병합한다.
15
- * 각 세그먼트 사이의 긴 갭(5초 이상)은 3초로 압축한다.
16
- */
17
- mergeFrameFiles(framesPaths: string[]): Promise<TerminalFrame[]>;
18
- /**
19
- * 단일 .frames 파일 읽기 (공통 인터페이스 유지)
20
- */
21
- readSingleFile(framesPath: string): Promise<TerminalFrame[]>;
22
- /**
23
- * 타임스탬프를 오름차순 정렬하고 큰 갭을 압축한다.
24
- */
25
- private normalizeTimestamps;
26
- }
27
- //# sourceMappingURL=segment-merger.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"segment-merger.d.ts","sourceRoot":"","sources":["../../../src/recording/segment-merger.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACtD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAElD;;;;;;GAMG;AACH,qBAAa,aAAa;IACxB,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAe;gBAEhC,YAAY,CAAC,EAAE,YAAY;IAIvC;;;OAGG;IACG,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC;IAYtE;;OAEG;IACG,cAAc,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC;IAIlE;;OAEG;IACH,OAAO,CAAC,mBAAmB;CA4B5B"}
@@ -1,65 +0,0 @@
1
- import { existsSync } from 'node:fs';
2
- import { GifGenerator } from './gif-generator.js';
3
- /**
4
- * SegmentMerger: 동일 sessionId의 여러 .frames 파일 세그먼트를
5
- * 시간순으로 병합해 GifGenerator에 전달한다.
6
- *
7
- * 현재 아키텍처에서는 sessionId당 .frames 파일이 1개이므로,
8
- * 단일 파일 읽기도 이 클래스를 경유해 일관된 인터페이스를 유지한다.
9
- */
10
- export class SegmentMerger {
11
- gifGenerator;
12
- constructor(gifGenerator) {
13
- this.gifGenerator = gifGenerator ?? new GifGenerator();
14
- }
15
- /**
16
- * 여러 .frames 파일을 읽어 타임스탬프 기준 오름차순으로 병합한다.
17
- * 각 세그먼트 사이의 긴 갭(5초 이상)은 3초로 압축한다.
18
- */
19
- async mergeFrameFiles(framesPaths) {
20
- const allFrames = [];
21
- for (const framesPath of framesPaths) {
22
- if (!existsSync(framesPath))
23
- continue;
24
- const frames = await this.gifGenerator.readFrames(framesPath);
25
- allFrames.push(...frames);
26
- }
27
- return this.normalizeTimestamps(allFrames);
28
- }
29
- /**
30
- * 단일 .frames 파일 읽기 (공통 인터페이스 유지)
31
- */
32
- async readSingleFile(framesPath) {
33
- return this.mergeFrameFiles([framesPath]);
34
- }
35
- /**
36
- * 타임스탬프를 오름차순 정렬하고 큰 갭을 압축한다.
37
- */
38
- normalizeTimestamps(frames) {
39
- if (frames.length === 0)
40
- return [];
41
- // 타임스탬프 오름차순 정렬
42
- frames.sort((a, b) => a.timestamp - b.timestamp);
43
- // 세그먼트 갭 압축: 5초 이상 빈 구간은 3초로 압축
44
- const GAP_THRESHOLD_MS = 5000;
45
- const MAX_GAP_MS = 3000;
46
- const normalized = [frames[0]];
47
- for (let i = 1; i < frames.length; i++) {
48
- const curr = frames[i];
49
- const prev_frame = frames[i - 1];
50
- const gap = curr.timestamp - prev_frame.timestamp;
51
- if (gap > GAP_THRESHOLD_MS) {
52
- const prev = normalized[normalized.length - 1];
53
- normalized.push({
54
- ...curr,
55
- timestamp: prev.timestamp + MAX_GAP_MS,
56
- });
57
- }
58
- else {
59
- normalized.push(curr);
60
- }
61
- }
62
- return normalized;
63
- }
64
- }
65
- //# sourceMappingURL=segment-merger.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"segment-merger.js","sourceRoot":"","sources":["../../../src/recording/segment-merger.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAErC,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAElD;;;;;;GAMG;AACH,MAAM,OAAO,aAAa;IACP,YAAY,CAAe;IAE5C,YAAY,YAA2B;QACrC,IAAI,CAAC,YAAY,GAAG,YAAY,IAAI,IAAI,YAAY,EAAE,CAAC;IACzD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,eAAe,CAAC,WAAqB;QACzC,MAAM,SAAS,GAAoB,EAAE,CAAC;QAEtC,KAAK,MAAM,UAAU,IAAI,WAAW,EAAE,CAAC;YACrC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;gBAAE,SAAS;YACtC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;YAC9D,SAAS,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,CAAC;QAC5B,CAAC;QAED,OAAO,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;IAC7C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,cAAc,CAAC,UAAkB;QACrC,OAAO,IAAI,CAAC,eAAe,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;IAC5C,CAAC;IAED;;OAEG;IACK,mBAAmB,CAAC,MAAuB;QACjD,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,CAAC;QAEnC,gBAAgB;QAChB,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC,CAAC;QAEjD,gCAAgC;QAChC,MAAM,gBAAgB,GAAG,IAAI,CAAC;QAC9B,MAAM,UAAU,GAAG,IAAI,CAAC;QAExB,MAAM,UAAU,GAAoB,CAAC,MAAM,CAAC,CAAC,CAAE,CAAC,CAAC;QACjD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACvC,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAE,CAAC;YACxB,MAAM,UAAU,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,CAAE,CAAC;YAClC,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,GAAG,UAAU,CAAC,SAAS,CAAC;YAClD,IAAI,GAAG,GAAG,gBAAgB,EAAE,CAAC;gBAC3B,MAAM,IAAI,GAAG,UAAU,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,CAAE,CAAC;gBAChD,UAAU,CAAC,IAAI,CAAC;oBACd,GAAG,IAAI;oBACP,SAAS,EAAE,IAAI,CAAC,SAAS,GAAG,UAAU;iBACvC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACxB,CAAC;QACH,CAAC;QAED,OAAO,UAAU,CAAC;IACpB,CAAC;CACF"}
@@ -1,31 +0,0 @@
1
- import type { RecordingSegment } from '../core/types.js';
2
- /**
3
- * TerminalRecorder: process.stdout.write를 인터셉트하여
4
- * TerminalFrame을 .frames NDJSON 파일에 실시간 append한다.
5
- *
6
- * node-pty 없이도 동작하지만, PTY 기반 캡처가 필요한 경우
7
- * 향후 node-pty 통합으로 확장 가능하도록 설계한다.
8
- */
9
- export declare class TerminalRecorder {
10
- private readonly sessionId;
11
- private readonly framesPath;
12
- private segment;
13
- private isRecording;
14
- private frameBuffer;
15
- private originalStdoutWrite;
16
- constructor(sessionId: string);
17
- /** 녹화 시작. 기존 .frames 파일이 있으면 이어서 append (resume) */
18
- start(): RecordingSegment;
19
- /** 녹화 일시 중지 (세션 중단 시) */
20
- pause(): void;
21
- /** 녹화 완전 종료 */
22
- stop(): RecordingSegment | null;
23
- get recording(): boolean;
24
- get framesFilePath(): string;
25
- /** stdout.write 인터셉트 */
26
- private interceptStdout;
27
- private restoreStdout;
28
- private captureFrame;
29
- private flushBuffer;
30
- }
31
- //# sourceMappingURL=terminal-recorder.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"terminal-recorder.d.ts","sourceRoot":"","sources":["../../../src/recording/terminal-recorder.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAiB,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAExE;;;;;;GAMG;AACH,qBAAa,gBAAgB;IASf,OAAO,CAAC,QAAQ,CAAC,SAAS;IARtC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAS;IACpC,OAAO,CAAC,OAAO,CAAiC;IAChD,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,WAAW,CAAuB;IAG1C,OAAO,CAAC,mBAAmB,CAA+B;gBAE7B,SAAS,EAAE,MAAM;IAK9C,oDAAoD;IACpD,KAAK,IAAI,gBAAgB;IAoBzB,yBAAyB;IACzB,KAAK,IAAI,IAAI;IAUb,eAAe;IACf,IAAI,IAAI,gBAAgB,GAAG,IAAI;IAO/B,IAAI,SAAS,IAAI,OAAO,CAEvB;IAED,IAAI,cAAc,IAAI,MAAM,CAE3B;IAED,wBAAwB;IACxB,OAAO,CAAC,eAAe;IA2BvB,OAAO,CAAC,aAAa;IAMrB,OAAO,CAAC,YAAY;IAkBpB,OAAO,CAAC,WAAW;CAMpB"}
@@ -1,111 +0,0 @@
1
- import { appendFileSync, existsSync } from 'node:fs';
2
- import { ensureRecordingsDir, getFramesPath } from './recording-dir.js';
3
- /**
4
- * TerminalRecorder: process.stdout.write를 인터셉트하여
5
- * TerminalFrame을 .frames NDJSON 파일에 실시간 append한다.
6
- *
7
- * node-pty 없이도 동작하지만, PTY 기반 캡처가 필요한 경우
8
- * 향후 node-pty 통합으로 확장 가능하도록 설계한다.
9
- */
10
- export class TerminalRecorder {
11
- sessionId;
12
- framesPath;
13
- segment = null;
14
- isRecording = false;
15
- frameBuffer = [];
16
- // 원본 stdout.write 저장
17
- originalStdoutWrite;
18
- constructor(sessionId) {
19
- this.sessionId = sessionId;
20
- ensureRecordingsDir();
21
- this.framesPath = getFramesPath(sessionId);
22
- }
23
- /** 녹화 시작. 기존 .frames 파일이 있으면 이어서 append (resume) */
24
- start() {
25
- if (this.isRecording)
26
- return this.segment;
27
- const isResuming = existsSync(this.framesPath);
28
- this.segment = {
29
- sessionId: this.sessionId,
30
- framesPath: this.framesPath,
31
- startedAt: Date.now(),
32
- };
33
- this.isRecording = true;
34
- this.interceptStdout();
35
- if (!isResuming) {
36
- // 새 녹화: 빈 파일로 시작
37
- }
38
- return this.segment;
39
- }
40
- /** 녹화 일시 중지 (세션 중단 시) */
41
- pause() {
42
- if (!this.isRecording)
43
- return;
44
- this.restoreStdout();
45
- this.flushBuffer();
46
- if (this.segment) {
47
- this.segment.endedAt = Date.now();
48
- }
49
- this.isRecording = false;
50
- }
51
- /** 녹화 완전 종료 */
52
- stop() {
53
- if (this.isRecording) {
54
- this.pause();
55
- }
56
- return this.segment;
57
- }
58
- get recording() {
59
- return this.isRecording;
60
- }
61
- get framesFilePath() {
62
- return this.framesPath;
63
- }
64
- /** stdout.write 인터셉트 */
65
- interceptStdout() {
66
- const self = this;
67
- this.originalStdoutWrite = process.stdout.write.bind(process.stdout);
68
- // overwrite with interceptor
69
- const intercepted = function (chunk, encodingOrCallback, callback) {
70
- const data = typeof chunk === 'string' ? chunk : Buffer.from(chunk).toString('utf8');
71
- self.captureFrame(data);
72
- // 원본 write 호출
73
- if (typeof encodingOrCallback === 'function') {
74
- return self.originalStdoutWrite(chunk, encodingOrCallback);
75
- }
76
- if (typeof encodingOrCallback === 'string') {
77
- return self.originalStdoutWrite(chunk, encodingOrCallback, callback);
78
- }
79
- return self.originalStdoutWrite(chunk);
80
- };
81
- process.stdout.write = intercepted;
82
- }
83
- restoreStdout() {
84
- if (this.originalStdoutWrite) {
85
- process.stdout.write = this.originalStdoutWrite;
86
- }
87
- }
88
- captureFrame(data) {
89
- if (!data || !this.isRecording)
90
- return;
91
- const frame = {
92
- timestamp: Date.now(),
93
- data,
94
- cols: process.stdout.columns ?? 80,
95
- rows: process.stdout.rows ?? 24,
96
- };
97
- this.frameBuffer.push(frame);
98
- // 버퍼가 10개 이상 쌓이면 즉시 flush
99
- if (this.frameBuffer.length >= 10) {
100
- this.flushBuffer();
101
- }
102
- }
103
- flushBuffer() {
104
- if (this.frameBuffer.length === 0)
105
- return;
106
- const lines = this.frameBuffer.map((f) => JSON.stringify(f)).join('\n') + '\n';
107
- appendFileSync(this.framesPath, lines, 'utf8');
108
- this.frameBuffer = [];
109
- }
110
- }
111
- //# sourceMappingURL=terminal-recorder.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"terminal-recorder.js","sourceRoot":"","sources":["../../../src/recording/terminal-recorder.ts"],"names":[],"mappings":"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"}
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=postinstall.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"postinstall.d.ts","sourceRoot":"","sources":["../../../src/scripts/postinstall.ts"],"names":[],"mappings":""}
@@ -1,29 +0,0 @@
1
- /**
2
- * Postinstall script: auto-setup agg for GIF recording support.
3
- * Runs after: npm install @tienne/gestalt
4
- *
5
- * - Silently skips in CI or if GESTALT_SKIP_DEPS=1
6
- * - Always exits 0 (never fails the install)
7
- */
8
- import { AggInstaller } from '../recording/agg-installer.js';
9
- async function main() {
10
- if (process.env['CI'] || process.env['GESTALT_SKIP_DEPS'] === '1') {
11
- return;
12
- }
13
- const installer = new AggInstaller();
14
- if (installer.isInstalled())
15
- return;
16
- process.stdout.write('\n[gestalt] agg not found — setting up GIF recording support...\n');
17
- try {
18
- await installer.ensureInstalled();
19
- }
20
- catch (err) {
21
- const msg = err instanceof Error ? err.message : String(err);
22
- process.stdout.write(`[gestalt] Skipping agg setup: ${msg}\n`);
23
- process.stdout.write('[gestalt] Install manually later: https://github.com/asciinema/agg\n\n');
24
- }
25
- }
26
- main()
27
- .then(() => process.exit(0))
28
- .catch(() => process.exit(0));
29
- //# sourceMappingURL=postinstall.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"postinstall.js","sourceRoot":"","sources":["../../../src/scripts/postinstall.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AACH,OAAO,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAE7D,KAAK,UAAU,IAAI;IACjB,IAAI,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,KAAK,GAAG,EAAE,CAAC;QAClE,OAAO;IACT,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,YAAY,EAAE,CAAC;IACrC,IAAI,SAAS,CAAC,WAAW,EAAE;QAAE,OAAO;IAEpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,mEAAmE,CAAC,CAAC;IAE1F,IAAI,CAAC;QACH,MAAM,SAAS,CAAC,eAAe,EAAE,CAAC;IACpC,CAAC;IAAC,OAAO,GAAY,EAAE,CAAC;QACtB,MAAM,GAAG,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC7D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,iCAAiC,GAAG,IAAI,CAAC,CAAC;QAC/D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,wEAAwE,CAAC,CAAC;IACjG,CAAC;AACH,CAAC;AAED,IAAI,EAAE;KACH,IAAI,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;KAC3B,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC"}
@@ -1,15 +0,0 @@
1
- import type { InterviewSession, GestaltAnalysis, OntologySchema } from '../core/types.js';
2
- import type { LLMAdapter } from '../llm/types.js';
3
- export interface ExtractedData {
4
- goal: string;
5
- constraints: string[];
6
- acceptanceCriteria: string[];
7
- ontologySchema: OntologySchema;
8
- gestaltAnalysis: GestaltAnalysis[];
9
- }
10
- export declare class SeedExtractor {
11
- private llm;
12
- constructor(llm: LLMAdapter);
13
- extract(session: InterviewSession): Promise<ExtractedData>;
14
- }
15
- //# sourceMappingURL=extractor.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"extractor.d.ts","sourceRoot":"","sources":["../../../src/seed/extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAG1F,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAElD,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,kBAAkB,EAAE,MAAM,EAAE,CAAC;IAC7B,cAAc,EAAE,cAAc,CAAC;IAC/B,eAAe,EAAE,eAAe,EAAE,CAAC;CACpC;AAED,qBAAa,aAAa;IACZ,OAAO,CAAC,GAAG;gBAAH,GAAG,EAAE,UAAU;IAE7B,OAAO,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,aAAa,CAAC;CAsBjE"}
@@ -1,88 +0,0 @@
1
- import { buildSeedPrompt, INTERVIEW_SYSTEM_PROMPT } from '../llm/prompts.js';
2
- import { SeedGenerationError } from '../core/errors.js';
3
- export class SeedExtractor {
4
- llm;
5
- constructor(llm) {
6
- this.llm = llm;
7
- }
8
- async extract(session) {
9
- const completedRounds = session.rounds
10
- .filter((r) => r.userResponse)
11
- .map((r) => ({
12
- question: r.question,
13
- response: r.userResponse,
14
- }));
15
- if (completedRounds.length === 0) {
16
- throw new SeedGenerationError('No completed interview rounds to extract from');
17
- }
18
- const prompt = buildSeedPrompt(session.topic, completedRounds, session.projectType);
19
- const response = await this.llm.chat({
20
- system: INTERVIEW_SYSTEM_PROMPT,
21
- messages: [{ role: 'user', content: prompt }],
22
- temperature: 0.3,
23
- });
24
- return parseSeedResponse(response.content);
25
- }
26
- }
27
- function parseSeedResponse(content) {
28
- const jsonMatch = content.match(/\{[\s\S]*\}/);
29
- if (!jsonMatch) {
30
- throw new SeedGenerationError('Failed to parse LLM response as JSON');
31
- }
32
- try {
33
- const parsed = JSON.parse(jsonMatch[0]);
34
- return {
35
- goal: asString(parsed['goal'], 'Undefined goal'),
36
- constraints: asStringArray(parsed['constraints']),
37
- acceptanceCriteria: asStringArray(parsed['acceptanceCriteria']),
38
- ontologySchema: parseOntology(parsed['ontologySchema']),
39
- gestaltAnalysis: parseGestaltAnalysis(parsed['gestaltAnalysis']),
40
- };
41
- }
42
- catch (e) {
43
- if (e instanceof SeedGenerationError)
44
- throw e;
45
- throw new SeedGenerationError(`Failed to parse seed data: ${e instanceof Error ? e.message : String(e)}`);
46
- }
47
- }
48
- function asString(value, fallback) {
49
- return typeof value === 'string' && value.length > 0 ? value : fallback;
50
- }
51
- function asStringArray(value) {
52
- if (!Array.isArray(value))
53
- return [];
54
- return value.filter((v) => typeof v === 'string');
55
- }
56
- function parseOntology(value) {
57
- if (!value || typeof value !== 'object') {
58
- return { entities: [], relations: [] };
59
- }
60
- const obj = value;
61
- const entities = Array.isArray(obj['entities'])
62
- ? obj['entities'].map((e) => ({
63
- name: asString(e['name'], 'Unknown'),
64
- description: asString(e['description'], ''),
65
- attributes: asStringArray(e['attributes']),
66
- }))
67
- : [];
68
- const relations = Array.isArray(obj['relations'])
69
- ? obj['relations'].map((r) => ({
70
- from: asString(r['from'], ''),
71
- to: asString(r['to'], ''),
72
- type: asString(r['type'], ''),
73
- }))
74
- : [];
75
- return { entities, relations };
76
- }
77
- function parseGestaltAnalysis(value) {
78
- if (!Array.isArray(value))
79
- return [];
80
- return value
81
- .filter((v) => typeof v === 'object' && v !== null)
82
- .map((v) => ({
83
- principle: asString(v['principle'], 'closure'),
84
- finding: asString(v['finding'], ''),
85
- confidence: typeof v['confidence'] === 'number' ? v['confidence'] : 0.5,
86
- }));
87
- }
88
- //# sourceMappingURL=extractor.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"extractor.js","sourceRoot":"","sources":["../../../src/seed/extractor.ts"],"names":[],"mappings":"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"}
@@ -1,12 +0,0 @@
1
- import type { InterviewSession, Seed } from '../core/types.js';
2
- import { AmbiguityThresholdError, SeedGenerationError } from '../core/errors.js';
3
- import { type Result } from '../core/result.js';
4
- import type { LLMAdapter } from '../llm/types.js';
5
- import { EventStore } from '../events/store.js';
6
- export declare class SeedGenerator {
7
- private eventStore;
8
- private extractor;
9
- constructor(llm: LLMAdapter, eventStore: EventStore);
10
- generate(session: InterviewSession, force?: boolean): Promise<Result<Seed, SeedGenerationError | AmbiguityThresholdError>>;
11
- }
12
- //# sourceMappingURL=generator.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"generator.d.ts","sourceRoot":"","sources":["../../../src/seed/generator.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,gBAAgB,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAC/D,OAAO,EAAE,uBAAuB,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEjF,OAAO,EAAE,KAAK,MAAM,EAAW,MAAM,mBAAmB,CAAC;AAGzD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAGhD,qBAAa,aAAa;IAKtB,OAAO,CAAC,UAAU;IAJpB,OAAO,CAAC,SAAS,CAAgB;gBAG/B,GAAG,EAAE,UAAU,EACP,UAAU,EAAE,UAAU;IAK1B,QAAQ,CACZ,OAAO,EAAE,gBAAgB,EACzB,KAAK,UAAQ,GACZ,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,mBAAmB,GAAG,uBAAuB,CAAC,CAAC;CAmExE"}
@@ -1,66 +0,0 @@
1
- import { randomUUID } from 'node:crypto';
2
- import { AmbiguityThresholdError, SeedGenerationError } from '../core/errors.js';
3
- import { AMBIGUITY_THRESHOLD, MAX_SEED_RETRIES } from '../core/constants.js';
4
- import { ok, err } from '../core/result.js';
5
- import { seedSchema } from './schema.js';
6
- import { SeedExtractor } from './extractor.js';
7
- import { EventType } from '../events/types.js';
8
- export class SeedGenerator {
9
- eventStore;
10
- extractor;
11
- constructor(llm, eventStore) {
12
- this.eventStore = eventStore;
13
- this.extractor = new SeedExtractor(llm);
14
- }
15
- async generate(session, force = false) {
16
- // Validate ambiguity threshold
17
- if (!force) {
18
- const ambiguity = session.ambiguityScore?.overall ?? 1.0;
19
- if (ambiguity > AMBIGUITY_THRESHOLD) {
20
- return err(new AmbiguityThresholdError(ambiguity, AMBIGUITY_THRESHOLD));
21
- }
22
- }
23
- if (session.status !== 'completed') {
24
- return err(new SeedGenerationError('Interview session must be completed before generating a seed'));
25
- }
26
- // Retry up to MAX_SEED_RETRIES
27
- let lastError = null;
28
- for (let attempt = 0; attempt < MAX_SEED_RETRIES; attempt++) {
29
- try {
30
- const extracted = await this.extractor.extract(session);
31
- const seed = {
32
- version: '1.0.0',
33
- goal: extracted.goal,
34
- constraints: extracted.constraints,
35
- acceptanceCriteria: extracted.acceptanceCriteria,
36
- ontologySchema: extracted.ontologySchema,
37
- gestaltAnalysis: extracted.gestaltAnalysis,
38
- metadata: {
39
- seedId: randomUUID(),
40
- interviewSessionId: session.sessionId,
41
- ambiguityScore: session.ambiguityScore?.overall ?? 1.0,
42
- generatedAt: new Date().toISOString(),
43
- },
44
- };
45
- // Validate against schema
46
- const validation = seedSchema.safeParse(seed);
47
- if (!validation.success) {
48
- lastError = new SeedGenerationError(`Seed validation failed: ${validation.error.message}`);
49
- continue;
50
- }
51
- this.eventStore.append('seed', seed.metadata.seedId, EventType.SEED_SPEC_GENERATED, {
52
- sessionId: session.sessionId,
53
- goal: seed.goal,
54
- constraintCount: seed.constraints.length,
55
- criteriaCount: seed.acceptanceCriteria.length,
56
- });
57
- return ok(seed);
58
- }
59
- catch (e) {
60
- lastError = e instanceof Error ? e : new Error(String(e));
61
- }
62
- }
63
- return err(new SeedGenerationError(`Failed after ${MAX_SEED_RETRIES} attempts: ${lastError?.message ?? 'unknown error'}`));
64
- }
65
- }
66
- //# sourceMappingURL=generator.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"generator.js","sourceRoot":"","sources":["../../../src/seed/generator.ts"],"names":[],"mappings":"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"}
@@ -1,31 +0,0 @@
1
- import type { InterviewSession, Seed } from '../core/types.js';
2
- import { AmbiguityThresholdError, SeedGenerationError } from '../core/errors.js';
3
- import { type Result } from '../core/result.js';
4
- import { EventStore } from '../events/store.js';
5
- export interface SeedContext {
6
- systemPrompt: string;
7
- seedPrompt: string;
8
- allRounds: {
9
- roundNumber: number;
10
- question: string;
11
- response: string;
12
- gestaltFocus: string;
13
- }[];
14
- }
15
- export interface ExternalSeed {
16
- goal: string;
17
- constraints: string[];
18
- acceptanceCriteria: string[];
19
- ontologySchema: {
20
- entities: unknown[];
21
- relations: unknown[];
22
- };
23
- gestaltAnalysis: unknown[];
24
- }
25
- export declare class PassthroughSeedGenerator {
26
- private eventStore;
27
- constructor(eventStore: EventStore);
28
- buildSeedContext(session: InterviewSession): SeedContext;
29
- validateAndStore(session: InterviewSession, externalSeed: ExternalSeed, force?: boolean): Result<Seed, SeedGenerationError | AmbiguityThresholdError>;
30
- }
31
- //# sourceMappingURL=passthrough-generator.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"passthrough-generator.d.ts","sourceRoot":"","sources":["../../../src/seed/passthrough-generator.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,gBAAgB,EAAE,IAAI,EAAkB,MAAM,kBAAkB,CAAC;AAC/E,OAAO,EAAE,uBAAuB,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEjF,OAAO,EAAE,KAAK,MAAM,EAAW,MAAM,mBAAmB,CAAC;AAEzD,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAMhD,MAAM,WAAW,WAAW;IAC1B,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,SAAS,EAAE;QAAE,WAAW,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,YAAY,EAAE,MAAM,CAAA;KAAE,EAAE,CAAC;CAChG;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,kBAAkB,EAAE,MAAM,EAAE,CAAC;IAC7B,cAAc,EAAE;QAAE,QAAQ,EAAE,OAAO,EAAE,CAAC;QAAC,SAAS,EAAE,OAAO,EAAE,CAAA;KAAE,CAAC;IAC9D,eAAe,EAAE,OAAO,EAAE,CAAC;CAC5B;AAID,qBAAa,wBAAwB;IACvB,OAAO,CAAC,UAAU;gBAAV,UAAU,EAAE,UAAU;IAE1C,gBAAgB,CAAC,OAAO,EAAE,gBAAgB,GAAG,WAAW;IAwBxD,gBAAgB,CACd,OAAO,EAAE,gBAAgB,EACzB,YAAY,EAAE,YAAY,EAC1B,KAAK,UAAQ,GACZ,MAAM,CAAC,IAAI,EAAE,mBAAmB,GAAG,uBAAuB,CAAC;CA6D/D"}