@love-moon/tui-driver 0.2.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 (113) hide show
  1. package/README.md +142 -0
  2. package/dist/driver/StateMachine.d.ts +28 -0
  3. package/dist/driver/StateMachine.d.ts.map +1 -0
  4. package/dist/driver/StateMachine.js +56 -0
  5. package/dist/driver/StateMachine.js.map +1 -0
  6. package/dist/driver/TuiDriver.d.ts +73 -0
  7. package/dist/driver/TuiDriver.d.ts.map +1 -0
  8. package/dist/driver/TuiDriver.js +506 -0
  9. package/dist/driver/TuiDriver.js.map +1 -0
  10. package/dist/driver/TuiProfile.d.ts +59 -0
  11. package/dist/driver/TuiProfile.d.ts.map +1 -0
  12. package/dist/driver/TuiProfile.js +13 -0
  13. package/dist/driver/TuiProfile.js.map +1 -0
  14. package/dist/driver/index.d.ts +5 -0
  15. package/dist/driver/index.d.ts.map +1 -0
  16. package/dist/driver/index.js +13 -0
  17. package/dist/driver/index.js.map +1 -0
  18. package/dist/driver/profiles/claudeCode.profile.d.ts +4 -0
  19. package/dist/driver/profiles/claudeCode.profile.d.ts.map +1 -0
  20. package/dist/driver/profiles/claudeCode.profile.js +91 -0
  21. package/dist/driver/profiles/claudeCode.profile.js.map +1 -0
  22. package/dist/driver/profiles/codex.profile.d.ts +4 -0
  23. package/dist/driver/profiles/codex.profile.d.ts.map +1 -0
  24. package/dist/driver/profiles/codex.profile.js +82 -0
  25. package/dist/driver/profiles/codex.profile.js.map +1 -0
  26. package/dist/driver/profiles/index.d.ts +3 -0
  27. package/dist/driver/profiles/index.d.ts.map +1 -0
  28. package/dist/driver/profiles/index.js +8 -0
  29. package/dist/driver/profiles/index.js.map +1 -0
  30. package/dist/example.d.ts +2 -0
  31. package/dist/example.d.ts.map +1 -0
  32. package/dist/example.js +43 -0
  33. package/dist/example.js.map +1 -0
  34. package/dist/expect/ExpectEngine.d.ts +34 -0
  35. package/dist/expect/ExpectEngine.d.ts.map +1 -0
  36. package/dist/expect/ExpectEngine.js +121 -0
  37. package/dist/expect/ExpectEngine.js.map +1 -0
  38. package/dist/expect/Matchers.d.ts +24 -0
  39. package/dist/expect/Matchers.d.ts.map +1 -0
  40. package/dist/expect/Matchers.js +71 -0
  41. package/dist/expect/Matchers.js.map +1 -0
  42. package/dist/expect/index.d.ts +3 -0
  43. package/dist/expect/index.d.ts.map +1 -0
  44. package/dist/expect/index.js +8 -0
  45. package/dist/expect/index.js.map +1 -0
  46. package/dist/extract/Diff.d.ts +10 -0
  47. package/dist/extract/Diff.d.ts.map +1 -0
  48. package/dist/extract/Diff.js +44 -0
  49. package/dist/extract/Diff.js.map +1 -0
  50. package/dist/extract/OutputExtractor.d.ts +16 -0
  51. package/dist/extract/OutputExtractor.d.ts.map +1 -0
  52. package/dist/extract/OutputExtractor.js +71 -0
  53. package/dist/extract/OutputExtractor.js.map +1 -0
  54. package/dist/extract/index.d.ts +3 -0
  55. package/dist/extract/index.d.ts.map +1 -0
  56. package/dist/extract/index.js +11 -0
  57. package/dist/extract/index.js.map +1 -0
  58. package/dist/index.d.ts +11 -0
  59. package/dist/index.d.ts.map +1 -0
  60. package/dist/index.js +50 -0
  61. package/dist/index.js.map +1 -0
  62. package/dist/pty/PtySession.d.ts +38 -0
  63. package/dist/pty/PtySession.d.ts.map +1 -0
  64. package/dist/pty/PtySession.js +231 -0
  65. package/dist/pty/PtySession.js.map +1 -0
  66. package/dist/pty/index.d.ts +2 -0
  67. package/dist/pty/index.d.ts.map +1 -0
  68. package/dist/pty/index.js +6 -0
  69. package/dist/pty/index.js.map +1 -0
  70. package/dist/term/HeadlessScreen.d.ts +29 -0
  71. package/dist/term/HeadlessScreen.d.ts.map +1 -0
  72. package/dist/term/HeadlessScreen.js +126 -0
  73. package/dist/term/HeadlessScreen.js.map +1 -0
  74. package/dist/term/ScreenSnapshot.d.ts +37 -0
  75. package/dist/term/ScreenSnapshot.d.ts.map +1 -0
  76. package/dist/term/ScreenSnapshot.js +68 -0
  77. package/dist/term/ScreenSnapshot.js.map +1 -0
  78. package/dist/term/index.d.ts +3 -0
  79. package/dist/term/index.d.ts.map +1 -0
  80. package/dist/term/index.js +8 -0
  81. package/dist/term/index.js.map +1 -0
  82. package/docs/tui-driver_implementation_plan.md +307 -0
  83. package/package.json +33 -0
  84. package/pnpm-workspace.yaml +1 -0
  85. package/src/driver/StateMachine.ts +90 -0
  86. package/src/driver/TuiDriver.ts +624 -0
  87. package/src/driver/TuiProfile.ts +72 -0
  88. package/src/driver/index.ts +4 -0
  89. package/src/driver/profiles/claudeCode.profile.ts +96 -0
  90. package/src/driver/profiles/codex.profile.ts +87 -0
  91. package/src/driver/profiles/index.ts +2 -0
  92. package/src/example.ts +45 -0
  93. package/src/expect/ExpectEngine.ts +171 -0
  94. package/src/expect/Matchers.ts +92 -0
  95. package/src/expect/index.ts +2 -0
  96. package/src/extract/Diff.ts +51 -0
  97. package/src/extract/OutputExtractor.ts +88 -0
  98. package/src/extract/index.ts +2 -0
  99. package/src/index.ts +67 -0
  100. package/src/pty/PtySession.ts +234 -0
  101. package/src/pty/index.ts +1 -0
  102. package/src/term/HeadlessScreen.ts +151 -0
  103. package/src/term/ScreenSnapshot.ts +89 -0
  104. package/src/term/index.ts +2 -0
  105. package/test/claude-profile.test.ts +11 -0
  106. package/test/codex-profile.test.ts +108 -0
  107. package/test/debug-claude.ts +51 -0
  108. package/test/integration.ts +174 -0
  109. package/test/output-extractor.test.ts +49 -0
  110. package/test/state-diff.test.ts +120 -0
  111. package/test/unit.test.ts +136 -0
  112. package/tsconfig.json +20 -0
  113. package/vitest.config.ts +13 -0
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.claudeCodeProfile = void 0;
4
+ const TuiProfile_js_1 = require("../TuiProfile.js");
5
+ exports.claudeCodeProfile = (0, TuiProfile_js_1.createProfile)({
6
+ name: "claude-code",
7
+ command: "claude",
8
+ args: [],
9
+ env: {
10
+ TERM: "xterm-256color",
11
+ LANG: "en_US.UTF-8",
12
+ LC_ALL: "en_US.UTF-8",
13
+ },
14
+ anchors: {
15
+ ready: [
16
+ /❯\s*$/m, // Empty prompt (ready for input)
17
+ /❯\s*(?!\d+\.)/m, // Prompt line (avoid trust menu options)
18
+ ],
19
+ trust: [
20
+ /Accessing workspace:/i,
21
+ /Quick safety check/i,
22
+ /Yes, I trust this folder/i,
23
+ ],
24
+ busy: [
25
+ /⏺/, // Claude's streaming indicator
26
+ /Thinking/i,
27
+ /Working/i,
28
+ ],
29
+ error: [
30
+ /Error:/i,
31
+ /Failed to/i,
32
+ /Connection refused/i,
33
+ /Authentication failed/i,
34
+ /Rate limit/i,
35
+ /API.*error/i,
36
+ ],
37
+ },
38
+ keys: {
39
+ submit: ["ENTER"],
40
+ newChat: ["CTRL_L"],
41
+ cancel: ["CTRL_C"],
42
+ },
43
+ extraction: {
44
+ mode: "diff-scrollback",
45
+ includeLinePatterns: [
46
+ /^⏺\s(?!.*(tool|mcp|running|calling|call))/i,
47
+ ],
48
+ stopLinePatterns: [
49
+ /^❯\s*/m,
50
+ ],
51
+ stripPatterns: [
52
+ /^❯\s*/gm, // Prompt indicator
53
+ /^⏺\s*/gm, // Claude reply/tool indicator
54
+ /^\s*\d+\s*│/gm, // Line numbers
55
+ /Press .* to/gi,
56
+ /\? for shortcuts/gi, // Help hint
57
+ /────+/g, // Horizontal lines
58
+ /╭.*╮/g, // Box top
59
+ /╰.*╯/g, // Box bottom
60
+ /│.*│/g, // Box sides (be careful with this)
61
+ ],
62
+ stripEcho: true,
63
+ bottomUiLines: 4,
64
+ },
65
+ signals: {
66
+ prompt: [/^❯\s*/m],
67
+ promptHint: [/^❯\s+(?!\d+\.)\S+/m],
68
+ replyStart: [/^⏺\s(?!.*(tool|mcp|running|calling|call))/i],
69
+ replyStop: [/^❯\s*/m],
70
+ status: [
71
+ /^⏺\s.*(tool|mcp|running|calling|call)/i,
72
+ /^✻\s.*\(\d+s\s*•/i, // Working status with updating time: "✻ ... (5s • esc to interrupt)"
73
+ ],
74
+ statusDone: [
75
+ /^✻\s.*for\s+\d+(\.\d+)?s/i, // Final status with fixed time: "✻ Cooked for 33s"
76
+ ],
77
+ },
78
+ requireReplyStart: true,
79
+ cols: 120,
80
+ rows: 40,
81
+ scrollback: 5000,
82
+ timeouts: {
83
+ boot: 0,
84
+ ready: 0,
85
+ streamStart: 0,
86
+ streamEnd: 0,
87
+ idle: 1000,
88
+ },
89
+ });
90
+ exports.default = exports.claudeCodeProfile;
91
+ //# sourceMappingURL=claudeCode.profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claudeCode.profile.js","sourceRoot":"","sources":["../../../src/driver/profiles/claudeCode.profile.ts"],"names":[],"mappings":";;;AAAA,oDAA6D;AAEhD,QAAA,iBAAiB,GAAe,IAAA,6BAAa,EAAC;IACzD,IAAI,EAAE,aAAa;IACnB,OAAO,EAAE,QAAQ;IACjB,IAAI,EAAE,EAAE;IACR,GAAG,EAAE;QACH,IAAI,EAAE,gBAAgB;QACtB,IAAI,EAAE,aAAa;QACnB,MAAM,EAAE,aAAa;KACtB;IAED,OAAO,EAAE;QACP,KAAK,EAAE;YACL,QAAQ,EAAqB,iCAAiC;YAC9D,gBAAgB,EAAa,yCAAyC;SACvE;QACD,KAAK,EAAE;YACL,uBAAuB;YACvB,qBAAqB;YACrB,2BAA2B;SAC5B;QACD,IAAI,EAAE;YACJ,GAAG,EAA0B,+BAA+B;YAC5D,WAAW;YACX,UAAU;SACX;QACD,KAAK,EAAE;YACL,SAAS;YACT,YAAY;YACZ,qBAAqB;YACrB,wBAAwB;YACxB,aAAa;YACb,aAAa;SACd;KACF;IAED,IAAI,EAAE;QACJ,MAAM,EAAE,CAAC,OAAO,CAAC;QACjB,OAAO,EAAE,CAAC,QAAQ,CAAC;QACnB,MAAM,EAAE,CAAC,QAAQ,CAAC;KACnB;IAED,UAAU,EAAE;QACV,IAAI,EAAE,iBAAiB;QACvB,mBAAmB,EAAE;YACnB,4CAA4C;SAC7C;QACD,gBAAgB,EAAE;YAChB,QAAQ;SACT;QACD,aAAa,EAAE;YACb,SAAS,EAAoB,mBAAmB;YAChD,SAAS,EAAoB,8BAA8B;YAC3D,eAAe,EAAc,eAAe;YAC5C,eAAe;YACf,oBAAoB,EAAS,YAAY;YACzC,QAAQ,EAAqB,mBAAmB;YAChD,OAAO,EAAsB,UAAU;YACvC,OAAO,EAAsB,aAAa;YAC1C,OAAO,EAAsB,mCAAmC;SACjE;QACD,SAAS,EAAE,IAAI;QACf,aAAa,EAAE,CAAC;KACjB;IAED,OAAO,EAAE;QACP,MAAM,EAAE,CAAC,QAAQ,CAAC;QAClB,UAAU,EAAE,CAAC,oBAAoB,CAAC;QAClC,UAAU,EAAE,CAAC,4CAA4C,CAAC;QAC1D,SAAS,EAAE,CAAC,QAAQ,CAAC;QACrB,MAAM,EAAE;YACN,wCAAwC;YACxC,mBAAmB,EAAG,qEAAqE;SAC5F;QACD,UAAU,EAAE;YACV,2BAA2B,EAAG,mDAAmD;SAClF;KACF;IAED,iBAAiB,EAAE,IAAI;IAEvB,IAAI,EAAE,GAAG;IACT,IAAI,EAAE,EAAE;IACR,UAAU,EAAE,IAAI;IAEhB,QAAQ,EAAE;QACR,IAAI,EAAE,CAAC;QACP,KAAK,EAAE,CAAC;QACR,WAAW,EAAE,CAAC;QACd,SAAS,EAAE,CAAC;QACZ,IAAI,EAAE,IAAI;KACX;CACF,CAAC,CAAC;AAEH,kBAAe,yBAAiB,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { TuiProfile } from "../TuiProfile.js";
2
+ export declare const codexProfile: TuiProfile;
3
+ export default codexProfile;
4
+ //# sourceMappingURL=codex.profile.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codex.profile.d.ts","sourceRoot":"","sources":["../../../src/driver/profiles/codex.profile.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAiB,MAAM,kBAAkB,CAAC;AAE7D,eAAO,MAAM,YAAY,EAAE,UAkFzB,CAAC;AAEH,eAAe,YAAY,CAAC"}
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.codexProfile = void 0;
4
+ const TuiProfile_js_1 = require("../TuiProfile.js");
5
+ exports.codexProfile = (0, TuiProfile_js_1.createProfile)({
6
+ name: "codex",
7
+ command: "codex",
8
+ args: [],
9
+ env: {
10
+ TERM: "xterm-256color",
11
+ LANG: "en_US.UTF-8",
12
+ LC_ALL: "en_US.UTF-8",
13
+ },
14
+ anchors: {
15
+ ready: [
16
+ /[\s\S]*model:(?!\s*loading)[\s\S]*^›\s*/mi,
17
+ /^(?![\s\S]*model:\s*loading)(?![\s\S]*Press enter to continue)(?=[\s\S]*(?:^|\n)›\s*(?!\d+\.)[^\n]*)[\s\S]*$/i,
18
+ ],
19
+ trust: [
20
+ /You are running Codex in /i,
21
+ /Since this folder is not version controlled/i,
22
+ /Allow Codex to work in this folder without asking for approval/i,
23
+ /Require approval of edits and commands/i,
24
+ /Press enter to continue/i,
25
+ ],
26
+ busy: [
27
+ /^\s*•\s*(Working|Thinking)\b.*$/m,
28
+ /\b(Working|Thinking)\s*\(\d+s\s*•\s*esc to interrupt\)/i,
29
+ ],
30
+ error: [
31
+ /Error:/i,
32
+ /Failed/i,
33
+ /API.*error/i,
34
+ ],
35
+ },
36
+ keys: {
37
+ submit: ["ENTER"],
38
+ newChat: [],
39
+ cancel: ["CTRL_C"],
40
+ trustConfirm: ["UP", "ENTER"],
41
+ },
42
+ extraction: {
43
+ mode: "diff-scrollback",
44
+ includeLinePatterns: [
45
+ /^\s*•\s*(?!Working|Thinking).+/m,
46
+ ],
47
+ stopLinePatterns: [
48
+ /^\s*›\s*/m,
49
+ ],
50
+ stripPatterns: [
51
+ /^>\s*/gm,
52
+ /^\s*›\s*/gm,
53
+ /^\s*•\s*/gm,
54
+ /codex>\s*/gi,
55
+ ],
56
+ stripEcho: true,
57
+ bottomUiLines: 2,
58
+ },
59
+ signals: {
60
+ prompt: [/^\s*›\s*/m],
61
+ replyStart: [/^\s*•\s*(?!Working|Thinking).+/m],
62
+ replyStop: [/^\s*›\s*/m],
63
+ status: [
64
+ /^\s*•\s*(Working|Thinking)\b.*$/m,
65
+ /\b(Working|Thinking)\s*\(\d+s\s*•\s*esc to interrupt\)/i,
66
+ ],
67
+ statusDone: [/^\s*─\s*Worked for .*─+/m, /\bWorked for\b.*$/i],
68
+ },
69
+ requireReplyStart: true,
70
+ cols: 120,
71
+ rows: 40,
72
+ scrollback: 5000,
73
+ timeouts: {
74
+ boot: 0,
75
+ ready: 0,
76
+ streamStart: 0,
77
+ streamEnd: 0,
78
+ idle: 1200,
79
+ },
80
+ });
81
+ exports.default = exports.codexProfile;
82
+ //# sourceMappingURL=codex.profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codex.profile.js","sourceRoot":"","sources":["../../../src/driver/profiles/codex.profile.ts"],"names":[],"mappings":";;;AAAA,oDAA6D;AAEhD,QAAA,YAAY,GAAe,IAAA,6BAAa,EAAC;IACpD,IAAI,EAAE,OAAO;IACb,OAAO,EAAE,OAAO;IAChB,IAAI,EAAE,EAAE;IACR,GAAG,EAAE;QACH,IAAI,EAAE,gBAAgB;QACtB,IAAI,EAAE,aAAa;QACnB,MAAM,EAAE,aAAa;KACtB;IAED,OAAO,EAAE;QACP,KAAK,EAAE;YACL,2CAA2C;YAC3C,+GAA+G;SAChH;QACD,KAAK,EAAE;YACL,4BAA4B;YAC5B,8CAA8C;YAC9C,iEAAiE;YACjE,yCAAyC;YACzC,0BAA0B;SAC3B;QACD,IAAI,EAAE;YACJ,kCAAkC;YAClC,yDAAyD;SAC1D;QACD,KAAK,EAAE;YACL,SAAS;YACT,SAAS;YACT,aAAa;SACd;KACF;IAED,IAAI,EAAE;QACJ,MAAM,EAAE,CAAC,OAAO,CAAC;QACjB,OAAO,EAAE,EAAE;QACX,MAAM,EAAE,CAAC,QAAQ,CAAC;QAClB,YAAY,EAAE,CAAC,IAAI,EAAE,OAAO,CAAC;KAC9B;IAED,UAAU,EAAE;QACV,IAAI,EAAE,iBAAiB;QACvB,mBAAmB,EAAE;YACnB,iCAAiC;SAClC;QACD,gBAAgB,EAAE;YAChB,WAAW;SACZ;QACD,aAAa,EAAE;YACb,SAAS;YACT,YAAY;YACZ,YAAY;YACZ,aAAa;SACd;QACD,SAAS,EAAE,IAAI;QACf,aAAa,EAAE,CAAC;KACjB;IAED,OAAO,EAAE;QACP,MAAM,EAAE,CAAC,WAAW,CAAC;QACrB,UAAU,EAAE,CAAC,iCAAiC,CAAC;QAC/C,SAAS,EAAE,CAAC,WAAW,CAAC;QACxB,MAAM,EAAE;YACN,kCAAkC;YAClC,yDAAyD;SAC1D;QACD,UAAU,EAAE,CAAC,0BAA0B,EAAE,oBAAoB,CAAC;KAC/D;IAED,iBAAiB,EAAE,IAAI;IAEvB,IAAI,EAAE,GAAG;IACT,IAAI,EAAE,EAAE;IACR,UAAU,EAAE,IAAI;IAEhB,QAAQ,EAAE;QACR,IAAI,EAAE,CAAC;QACP,KAAK,EAAE,CAAC;QACR,WAAW,EAAE,CAAC;QACd,SAAS,EAAE,CAAC;QACZ,IAAI,EAAE,IAAI;KACX;CACF,CAAC,CAAC;AAEH,kBAAe,oBAAY,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { claudeCodeProfile } from "./claudeCode.profile.js";
2
+ export { codexProfile } from "./codex.profile.js";
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/driver/profiles/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAC5D,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC"}
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.codexProfile = exports.claudeCodeProfile = void 0;
4
+ var claudeCode_profile_js_1 = require("./claudeCode.profile.js");
5
+ Object.defineProperty(exports, "claudeCodeProfile", { enumerable: true, get: function () { return claudeCode_profile_js_1.claudeCodeProfile; } });
6
+ var codex_profile_js_1 = require("./codex.profile.js");
7
+ Object.defineProperty(exports, "codexProfile", { enumerable: true, get: function () { return codex_profile_js_1.codexProfile; } });
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/driver/profiles/index.ts"],"names":[],"mappings":";;;AAAA,iEAA4D;AAAnD,0HAAA,iBAAiB,OAAA;AAC1B,uDAAkD;AAAzC,gHAAA,YAAY,OAAA"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=example.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.d.ts","sourceRoot":"","sources":["../src/example.ts"],"names":[],"mappings":""}
@@ -0,0 +1,43 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const index_js_1 = require("./index.js");
4
+ async function main() {
5
+ // Method 1: Using the convenience factory
6
+ const driver1 = (0, index_js_1.createDriver)("claude-code", { debug: true });
7
+ // Method 2: Using TuiDriver directly with a profile
8
+ const driver2 = new index_js_1.TuiDriver({
9
+ profile: index_js_1.claudeCodeProfile,
10
+ debug: true,
11
+ onSnapshot: (snapshot, state) => {
12
+ console.log(`[${state}] Screen hash: ${snapshot.hash}`);
13
+ },
14
+ });
15
+ try {
16
+ // Boot the TUI
17
+ await driver2.boot();
18
+ console.log("TUI booted successfully");
19
+ // Ask a question
20
+ const result = await driver2.ask("What is 2 + 2?");
21
+ if (result.success) {
22
+ console.log("Answer:", result.answer);
23
+ console.log("Elapsed:", result.elapsedMs, "ms");
24
+ }
25
+ else {
26
+ console.error("Failed:", result.error?.message);
27
+ }
28
+ // Ask another question
29
+ const result2 = await driver2.ask("Explain briefly what TypeScript is.");
30
+ if (result2.success) {
31
+ console.log("Answer:", result2.answer);
32
+ }
33
+ }
34
+ catch (error) {
35
+ console.error("Error:", error);
36
+ }
37
+ finally {
38
+ // Clean up
39
+ driver2.kill();
40
+ }
41
+ }
42
+ main().catch(console.error);
43
+ //# sourceMappingURL=example.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.js","sourceRoot":"","sources":["../src/example.ts"],"names":[],"mappings":";;AAAA,yCAAwE;AAExE,KAAK,UAAU,IAAI;IACjB,0CAA0C;IAC1C,MAAM,OAAO,GAAG,IAAA,uBAAY,EAAC,aAAa,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IAE7D,oDAAoD;IACpD,MAAM,OAAO,GAAG,IAAI,oBAAS,CAAC;QAC5B,OAAO,EAAE,4BAAiB;QAC1B,KAAK,EAAE,IAAI;QACX,UAAU,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,EAAE;YAC9B,OAAO,CAAC,GAAG,CAAC,IAAI,KAAK,kBAAkB,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;QAC1D,CAAC;KACF,CAAC,CAAC;IAEH,IAAI,CAAC;QACH,eAAe;QACf,MAAM,OAAO,CAAC,IAAI,EAAE,CAAC;QACrB,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;QAEvC,iBAAiB;QACjB,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAEnD,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;YACnB,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;YACtC,OAAO,CAAC,GAAG,CAAC,UAAU,EAAE,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAClD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,KAAK,CAAC,SAAS,EAAE,MAAM,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;QAClD,CAAC;QAED,uBAAuB;QACvB,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,qCAAqC,CAAC,CAAC;QAEzE,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IACjC,CAAC;YAAS,CAAC;QACT,WAAW;QACX,OAAO,CAAC,IAAI,EAAE,CAAC;IACjB,CAAC;AACH,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC"}
@@ -0,0 +1,34 @@
1
+ import { HeadlessScreen } from "../term/HeadlessScreen.js";
2
+ import { ScreenSnapshot } from "../term/ScreenSnapshot.js";
3
+ import { MatcherFn } from "./Matchers.js";
4
+ export interface UntilOptions {
5
+ name: string;
6
+ match: MatcherFn;
7
+ stableMs?: number;
8
+ timeoutMs?: number;
9
+ pollIntervalMs?: number;
10
+ onTimeout?: () => void | Promise<void>;
11
+ }
12
+ export interface IdleOptions {
13
+ name: string;
14
+ idleMs: number;
15
+ timeoutMs?: number;
16
+ pollIntervalMs?: number;
17
+ onTimeout?: () => void | Promise<void>;
18
+ }
19
+ export interface ExpectResult {
20
+ success: boolean;
21
+ snapshot: ScreenSnapshot;
22
+ elapsedMs: number;
23
+ timedOut: boolean;
24
+ }
25
+ export declare class ExpectEngine {
26
+ private screen;
27
+ constructor(screen: HeadlessScreen);
28
+ until(options: UntilOptions): Promise<ExpectResult>;
29
+ untilIdle(options: IdleOptions): Promise<ExpectResult>;
30
+ waitForChange(timeoutMs?: number): Promise<ExpectResult>;
31
+ snapshot(): ScreenSnapshot;
32
+ private sleep;
33
+ }
34
+ //# sourceMappingURL=ExpectEngine.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ExpectEngine.d.ts","sourceRoot":"","sources":["../../src/expect/ExpectEngine.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAE1C,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,SAAS,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,SAAS,CAAC,EAAE,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;CACxC;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,SAAS,CAAC,EAAE,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;CACxC;AAED,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,cAAc,CAAC;IACzB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,OAAO,CAAC;CACnB;AAMD,qBAAa,YAAY;IACX,OAAO,CAAC,MAAM;gBAAN,MAAM,EAAE,cAAc;IAEpC,KAAK,CAAC,OAAO,EAAE,YAAY,GAAG,OAAO,CAAC,YAAY,CAAC;IAkDnD,SAAS,CAAC,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,YAAY,CAAC;IA+CtD,aAAa,CAAC,SAAS,GAAE,MAA2B,GAAG,OAAO,CAAC,YAAY,CAAC;IA+BlF,QAAQ,IAAI,cAAc;IAI1B,OAAO,CAAC,KAAK;CAGd"}
@@ -0,0 +1,121 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ExpectEngine = void 0;
4
+ const DEFAULT_TIMEOUT_MS = 30000;
5
+ const DEFAULT_POLL_INTERVAL_MS = 100;
6
+ const DEFAULT_STABLE_MS = 300;
7
+ class ExpectEngine {
8
+ screen;
9
+ constructor(screen) {
10
+ this.screen = screen;
11
+ }
12
+ async until(options) {
13
+ const { match, stableMs = DEFAULT_STABLE_MS, timeoutMs = DEFAULT_TIMEOUT_MS, pollIntervalMs = DEFAULT_POLL_INTERVAL_MS, onTimeout, } = options;
14
+ const startTime = Date.now();
15
+ let matchStartTime = null;
16
+ let lastSnapshot = this.screen.snapshot();
17
+ while (true) {
18
+ const elapsed = Date.now() - startTime;
19
+ if (elapsed >= timeoutMs) {
20
+ if (onTimeout) {
21
+ await onTimeout();
22
+ }
23
+ return {
24
+ success: false,
25
+ snapshot: lastSnapshot,
26
+ elapsedMs: elapsed,
27
+ timedOut: true,
28
+ };
29
+ }
30
+ lastSnapshot = this.screen.snapshot();
31
+ const matched = match(lastSnapshot);
32
+ if (matched) {
33
+ if (matchStartTime === null) {
34
+ matchStartTime = Date.now();
35
+ }
36
+ else if (Date.now() - matchStartTime >= stableMs) {
37
+ return {
38
+ success: true,
39
+ snapshot: lastSnapshot,
40
+ elapsedMs: Date.now() - startTime,
41
+ timedOut: false,
42
+ };
43
+ }
44
+ }
45
+ else {
46
+ matchStartTime = null;
47
+ }
48
+ await this.sleep(pollIntervalMs);
49
+ }
50
+ }
51
+ async untilIdle(options) {
52
+ const { idleMs, timeoutMs = DEFAULT_TIMEOUT_MS, pollIntervalMs = DEFAULT_POLL_INTERVAL_MS, onTimeout, } = options;
53
+ const startTime = Date.now();
54
+ let lastHash = this.screen.snapshot().hash;
55
+ let lastChangeTime = Date.now();
56
+ let lastSnapshot = this.screen.snapshot();
57
+ while (true) {
58
+ const elapsed = Date.now() - startTime;
59
+ if (elapsed >= timeoutMs) {
60
+ if (onTimeout) {
61
+ await onTimeout();
62
+ }
63
+ return {
64
+ success: false,
65
+ snapshot: lastSnapshot,
66
+ elapsedMs: elapsed,
67
+ timedOut: true,
68
+ };
69
+ }
70
+ lastSnapshot = this.screen.snapshot();
71
+ const currentHash = lastSnapshot.hash;
72
+ if (currentHash !== lastHash) {
73
+ lastHash = currentHash;
74
+ lastChangeTime = Date.now();
75
+ }
76
+ else if (Date.now() - lastChangeTime >= idleMs) {
77
+ return {
78
+ success: true,
79
+ snapshot: lastSnapshot,
80
+ elapsedMs: Date.now() - startTime,
81
+ timedOut: false,
82
+ };
83
+ }
84
+ await this.sleep(pollIntervalMs);
85
+ }
86
+ }
87
+ async waitForChange(timeoutMs = DEFAULT_TIMEOUT_MS) {
88
+ const startTime = Date.now();
89
+ const initialHash = this.screen.snapshot().hash;
90
+ let lastSnapshot = this.screen.snapshot();
91
+ while (true) {
92
+ const elapsed = Date.now() - startTime;
93
+ if (elapsed >= timeoutMs) {
94
+ return {
95
+ success: false,
96
+ snapshot: lastSnapshot,
97
+ elapsedMs: elapsed,
98
+ timedOut: true,
99
+ };
100
+ }
101
+ lastSnapshot = this.screen.snapshot();
102
+ if (lastSnapshot.hash !== initialHash) {
103
+ return {
104
+ success: true,
105
+ snapshot: lastSnapshot,
106
+ elapsedMs: Date.now() - startTime,
107
+ timedOut: false,
108
+ };
109
+ }
110
+ await this.sleep(DEFAULT_POLL_INTERVAL_MS);
111
+ }
112
+ }
113
+ snapshot() {
114
+ return this.screen.snapshot();
115
+ }
116
+ sleep(ms) {
117
+ return new Promise((resolve) => setTimeout(resolve, ms));
118
+ }
119
+ }
120
+ exports.ExpectEngine = ExpectEngine;
121
+ //# sourceMappingURL=ExpectEngine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ExpectEngine.js","sourceRoot":"","sources":["../../src/expect/ExpectEngine.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,24 @@
1
+ import { ScreenSnapshot } from "../term/ScreenSnapshot.js";
2
+ export type MatcherFn = (snapshot: ScreenSnapshot) => boolean;
3
+ export interface MatcherResult {
4
+ matched: boolean;
5
+ matchedPattern?: RegExp;
6
+ matchedText?: string;
7
+ }
8
+ export declare class Matchers {
9
+ static regex(pattern: RegExp, region?: "viewport" | "scrollback"): MatcherFn;
10
+ static anyOf(patterns: RegExp[], region?: "viewport" | "scrollback"): MatcherFn;
11
+ static allOf(patterns: RegExp[], region?: "viewport" | "scrollback"): MatcherFn;
12
+ static noneOf(patterns: RegExp[], region?: "viewport" | "scrollback"): MatcherFn;
13
+ static cursorAt(x: number, y: number): MatcherFn;
14
+ static cursorInRegion(minX: number, maxX: number, minY: number, maxY: number): MatcherFn;
15
+ static bottomRegion(pattern: RegExp, bottomLines?: number): MatcherFn;
16
+ static topRegion(pattern: RegExp, topLines?: number): MatcherFn;
17
+ static hashChanged(previousHash: string): MatcherFn;
18
+ static hashUnchanged(previousHash: string): MatcherFn;
19
+ static and(...matchers: MatcherFn[]): MatcherFn;
20
+ static or(...matchers: MatcherFn[]): MatcherFn;
21
+ static not(matcher: MatcherFn): MatcherFn;
22
+ static custom(fn: (snapshot: ScreenSnapshot) => boolean): MatcherFn;
23
+ }
24
+ //# sourceMappingURL=Matchers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Matchers.d.ts","sourceRoot":"","sources":["../../src/expect/Matchers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAE3D,MAAM,MAAM,SAAS,GAAG,CAAC,QAAQ,EAAE,cAAc,KAAK,OAAO,CAAC;AAE9D,MAAM,WAAW,aAAa;IAC5B,OAAO,EAAE,OAAO,CAAC;IACjB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,qBAAa,QAAQ;IACnB,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,GAAE,UAAU,GAAG,YAAyB,GAAG,SAAS;IAIxF,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,UAAU,GAAG,YAAyB,GAAG,SAAS;IAO3F,MAAM,CAAC,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,UAAU,GAAG,YAAyB,GAAG,SAAS;IAO3F,MAAM,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,UAAU,GAAG,YAAyB,GAAG,SAAS;IAO5F,MAAM,CAAC,QAAQ,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,MAAM,GAAG,SAAS;IAMhD,MAAM,CAAC,cAAc,CACnB,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAC1B,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GACzB,SAAS;IAOZ,MAAM,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,WAAW,GAAE,MAAU,GAAG,SAAS;IAQxE,MAAM,CAAC,SAAS,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,GAAE,MAAU,GAAG,SAAS;IAQlE,MAAM,CAAC,WAAW,CAAC,YAAY,EAAE,MAAM,GAAG,SAAS;IAInD,MAAM,CAAC,aAAa,CAAC,YAAY,EAAE,MAAM,GAAG,SAAS;IAIrD,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,SAAS,EAAE,GAAG,SAAS;IAI/C,MAAM,CAAC,EAAE,CAAC,GAAG,QAAQ,EAAE,SAAS,EAAE,GAAG,SAAS;IAI9C,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,SAAS,GAAG,SAAS;IAIzC,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,OAAO,GAAG,SAAS;CAGpE"}
@@ -0,0 +1,71 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Matchers = void 0;
4
+ class Matchers {
5
+ static regex(pattern, region = "viewport") {
6
+ return (snapshot) => snapshot.matchesPattern(pattern, region);
7
+ }
8
+ static anyOf(patterns, region = "viewport") {
9
+ return (snapshot) => {
10
+ const text = region === "viewport" ? snapshot.viewportText : snapshot.scrollbackText;
11
+ return patterns.some(p => p.test(text));
12
+ };
13
+ }
14
+ static allOf(patterns, region = "viewport") {
15
+ return (snapshot) => {
16
+ const text = region === "viewport" ? snapshot.viewportText : snapshot.scrollbackText;
17
+ return patterns.every(p => p.test(text));
18
+ };
19
+ }
20
+ static noneOf(patterns, region = "viewport") {
21
+ return (snapshot) => {
22
+ const text = region === "viewport" ? snapshot.viewportText : snapshot.scrollbackText;
23
+ return !patterns.some(p => p.test(text));
24
+ };
25
+ }
26
+ static cursorAt(x, y) {
27
+ return (snapshot) => {
28
+ return snapshot.cursor.x === x && snapshot.cursor.y === y;
29
+ };
30
+ }
31
+ static cursorInRegion(minX, maxX, minY, maxY) {
32
+ return (snapshot) => {
33
+ const { x, y } = snapshot.cursor;
34
+ return x >= minX && x <= maxX && y >= minY && y <= maxY;
35
+ };
36
+ }
37
+ static bottomRegion(pattern, bottomLines = 5) {
38
+ return (snapshot) => {
39
+ const lines = snapshot.getViewportLines();
40
+ const bottom = lines.slice(-bottomLines).join("\n");
41
+ return pattern.test(bottom);
42
+ };
43
+ }
44
+ static topRegion(pattern, topLines = 5) {
45
+ return (snapshot) => {
46
+ const lines = snapshot.getViewportLines();
47
+ const top = lines.slice(0, topLines).join("\n");
48
+ return pattern.test(top);
49
+ };
50
+ }
51
+ static hashChanged(previousHash) {
52
+ return (snapshot) => snapshot.hash !== previousHash;
53
+ }
54
+ static hashUnchanged(previousHash) {
55
+ return (snapshot) => snapshot.hash === previousHash;
56
+ }
57
+ static and(...matchers) {
58
+ return (snapshot) => matchers.every(m => m(snapshot));
59
+ }
60
+ static or(...matchers) {
61
+ return (snapshot) => matchers.some(m => m(snapshot));
62
+ }
63
+ static not(matcher) {
64
+ return (snapshot) => !matcher(snapshot);
65
+ }
66
+ static custom(fn) {
67
+ return fn;
68
+ }
69
+ }
70
+ exports.Matchers = Matchers;
71
+ //# sourceMappingURL=Matchers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Matchers.js","sourceRoot":"","sources":["../../src/expect/Matchers.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export { ExpectEngine, UntilOptions, IdleOptions, ExpectResult } from "./ExpectEngine.js";
2
+ export { Matchers, MatcherFn, MatcherResult } from "./Matchers.js";
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/expect/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,YAAY,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAC1F,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC"}
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Matchers = exports.ExpectEngine = void 0;
4
+ var ExpectEngine_js_1 = require("./ExpectEngine.js");
5
+ Object.defineProperty(exports, "ExpectEngine", { enumerable: true, get: function () { return ExpectEngine_js_1.ExpectEngine; } });
6
+ var Matchers_js_1 = require("./Matchers.js");
7
+ Object.defineProperty(exports, "Matchers", { enumerable: true, get: function () { return Matchers_js_1.Matchers; } });
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/expect/index.ts"],"names":[],"mappings":";;;AAAA,qDAA0F;AAAjF,+GAAA,YAAY,OAAA;AACrB,6CAAmE;AAA1D,uGAAA,QAAQ,OAAA"}
@@ -0,0 +1,10 @@
1
+ export interface DiffResult {
2
+ added: string[];
3
+ removed: string[];
4
+ unchanged: string[];
5
+ }
6
+ export declare function computeLineDiff(before: string, after: string): DiffResult;
7
+ export declare function computeCharDiff(before: string, after: string): Array<[number, string]>;
8
+ export declare function getAddedText(before: string, after: string): string;
9
+ export declare function getRemovedText(before: string, after: string): string;
10
+ //# sourceMappingURL=Diff.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Diff.d.ts","sourceRoot":"","sources":["../../src/extract/Diff.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,SAAS,EAAE,MAAM,EAAE,CAAC;CACrB;AAED,wBAAgB,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,UAAU,CAYzE;AAED,wBAAgB,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAEtF;AAED,wBAAgB,YAAY,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,MAAM,CAWlE;AAED,wBAAgB,cAAc,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,MAAM,CAWpE"}
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.computeLineDiff = computeLineDiff;
7
+ exports.computeCharDiff = computeCharDiff;
8
+ exports.getAddedText = getAddedText;
9
+ exports.getRemovedText = getRemovedText;
10
+ const fast_diff_1 = __importDefault(require("fast-diff"));
11
+ function computeLineDiff(before, after) {
12
+ const beforeLines = before.split("\n");
13
+ const afterLines = after.split("\n");
14
+ const beforeSet = new Set(beforeLines);
15
+ const afterSet = new Set(afterLines);
16
+ const added = afterLines.filter(line => !beforeSet.has(line));
17
+ const removed = beforeLines.filter(line => !afterSet.has(line));
18
+ const unchanged = afterLines.filter(line => beforeSet.has(line));
19
+ return { added, removed, unchanged };
20
+ }
21
+ function computeCharDiff(before, after) {
22
+ return (0, fast_diff_1.default)(before, after);
23
+ }
24
+ function getAddedText(before, after) {
25
+ const result = (0, fast_diff_1.default)(before, after);
26
+ const added = [];
27
+ for (const [type, text] of result) {
28
+ if (type === 1) {
29
+ added.push(text);
30
+ }
31
+ }
32
+ return added.join("");
33
+ }
34
+ function getRemovedText(before, after) {
35
+ const result = (0, fast_diff_1.default)(before, after);
36
+ const removed = [];
37
+ for (const [type, text] of result) {
38
+ if (type === -1) {
39
+ removed.push(text);
40
+ }
41
+ }
42
+ return removed.join("");
43
+ }
44
+ //# sourceMappingURL=Diff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Diff.js","sourceRoot":"","sources":["../../src/extract/Diff.ts"],"names":[],"mappings":";;;;;AAQA,0CAYC;AAED,0CAEC;AAED,oCAWC;AAED,wCAWC;AAlDD,0DAA6B;AAQ7B,SAAgB,eAAe,CAAC,MAAc,EAAE,KAAa;IAC3D,MAAM,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACvC,MAAM,UAAU,GAAG,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAErC,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC;IACvC,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,CAAC;IAErC,MAAM,KAAK,GAAG,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IAC9D,MAAM,OAAO,GAAG,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IAChE,MAAM,SAAS,GAAG,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IAEjE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;AACvC,CAAC;AAED,SAAgB,eAAe,CAAC,MAAc,EAAE,KAAa;IAC3D,OAAO,IAAA,mBAAI,EAAC,MAAM,EAAE,KAAK,CAAC,CAAC;AAC7B,CAAC;AAED,SAAgB,YAAY,CAAC,MAAc,EAAE,KAAa;IACxD,MAAM,MAAM,GAAG,IAAA,mBAAI,EAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IACnC,MAAM,KAAK,GAAa,EAAE,CAAC;IAE3B,KAAK,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,MAAM,EAAE,CAAC;QAClC,IAAI,IAAI,KAAK,CAAC,EAAE,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnB,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;AACxB,CAAC;AAED,SAAgB,cAAc,CAAC,MAAc,EAAE,KAAa;IAC1D,MAAM,MAAM,GAAG,IAAA,mBAAI,EAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IACnC,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,KAAK,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,MAAM,EAAE,CAAC;QAClC,IAAI,IAAI,KAAK,CAAC,CAAC,EAAE,CAAC;YAChB,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACrB,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;AAC1B,CAAC"}