@strands-agents/sdk 1.1.0 → 1.3.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 (199) hide show
  1. package/README.md +6 -0
  2. package/dist/src/__tests__/interrupt.test.js +13 -8
  3. package/dist/src/__tests__/interrupt.test.js.map +1 -1
  4. package/dist/src/__tests__/mcp.test.js +221 -7
  5. package/dist/src/__tests__/mcp.test.js.map +1 -1
  6. package/dist/src/agent/__tests__/agent.interrupt.test.js +54 -5
  7. package/dist/src/agent/__tests__/agent.interrupt.test.js.map +1 -1
  8. package/dist/src/agent/__tests__/agent.test.js +56 -0
  9. package/dist/src/agent/__tests__/agent.test.js.map +1 -1
  10. package/dist/src/agent/__tests__/printer.test.js +58 -18
  11. package/dist/src/agent/__tests__/printer.test.js.map +1 -1
  12. package/dist/src/agent/__tests__/snapshot.test.js +98 -0
  13. package/dist/src/agent/__tests__/snapshot.test.js.map +1 -1
  14. package/dist/src/agent/agent-as-tool.d.ts.map +1 -1
  15. package/dist/src/agent/agent-as-tool.js +2 -3
  16. package/dist/src/agent/agent-as-tool.js.map +1 -1
  17. package/dist/src/agent/agent.d.ts +65 -0
  18. package/dist/src/agent/agent.d.ts.map +1 -1
  19. package/dist/src/agent/agent.js +89 -10
  20. package/dist/src/agent/agent.js.map +1 -1
  21. package/dist/src/agent/printer.d.ts +14 -1
  22. package/dist/src/agent/printer.d.ts.map +1 -1
  23. package/dist/src/agent/printer.js +33 -5
  24. package/dist/src/agent/printer.js.map +1 -1
  25. package/dist/src/agent/snapshot.d.ts +9 -17
  26. package/dist/src/agent/snapshot.d.ts.map +1 -1
  27. package/dist/src/agent/snapshot.js +9 -17
  28. package/dist/src/agent/snapshot.js.map +1 -1
  29. package/dist/src/conversation-manager/__tests__/sliding-window-conversation-manager.test.js +371 -39
  30. package/dist/src/conversation-manager/__tests__/sliding-window-conversation-manager.test.js.map +1 -1
  31. package/dist/src/conversation-manager/sliding-window-conversation-manager.d.ts +26 -7
  32. package/dist/src/conversation-manager/sliding-window-conversation-manager.d.ts.map +1 -1
  33. package/dist/src/conversation-manager/sliding-window-conversation-manager.js +192 -41
  34. package/dist/src/conversation-manager/sliding-window-conversation-manager.js.map +1 -1
  35. package/dist/src/hooks/events.d.ts +24 -3
  36. package/dist/src/hooks/events.d.ts.map +1 -1
  37. package/dist/src/hooks/events.js +26 -4
  38. package/dist/src/hooks/events.js.map +1 -1
  39. package/dist/src/hooks/index.d.ts +1 -1
  40. package/dist/src/hooks/index.d.ts.map +1 -1
  41. package/dist/src/hooks/index.js +1 -1
  42. package/dist/src/hooks/index.js.map +1 -1
  43. package/dist/src/hooks/types.d.ts +2 -0
  44. package/dist/src/hooks/types.d.ts.map +1 -1
  45. package/dist/src/hooks/types.js +2 -0
  46. package/dist/src/hooks/types.js.map +1 -1
  47. package/dist/src/index.d.ts +8 -4
  48. package/dist/src/index.d.ts.map +1 -1
  49. package/dist/src/index.js +6 -2
  50. package/dist/src/index.js.map +1 -1
  51. package/dist/src/interrupt.d.ts +29 -2
  52. package/dist/src/interrupt.d.ts.map +1 -1
  53. package/dist/src/interrupt.js +45 -3
  54. package/dist/src/interrupt.js.map +1 -1
  55. package/dist/src/interventions/__tests__/handler.test.d.ts +2 -0
  56. package/dist/src/interventions/__tests__/handler.test.d.ts.map +1 -0
  57. package/dist/src/interventions/__tests__/handler.test.js +35 -0
  58. package/dist/src/interventions/__tests__/handler.test.js.map +1 -0
  59. package/dist/src/interventions/__tests__/registry.test.d.ts +2 -0
  60. package/dist/src/interventions/__tests__/registry.test.d.ts.map +1 -0
  61. package/dist/src/interventions/__tests__/registry.test.js +692 -0
  62. package/dist/src/interventions/__tests__/registry.test.js.map +1 -0
  63. package/dist/src/interventions/actions.d.ts +186 -0
  64. package/dist/src/interventions/actions.d.ts.map +1 -0
  65. package/dist/src/interventions/actions.js +56 -0
  66. package/dist/src/interventions/actions.js.map +1 -0
  67. package/dist/src/interventions/handler.d.ts +44 -0
  68. package/dist/src/interventions/handler.d.ts.map +1 -0
  69. package/dist/src/interventions/handler.js +41 -0
  70. package/dist/src/interventions/handler.js.map +1 -0
  71. package/dist/src/interventions/index.d.ts +12 -0
  72. package/dist/src/interventions/index.d.ts.map +1 -0
  73. package/dist/src/interventions/index.js +4 -0
  74. package/dist/src/interventions/index.js.map +1 -0
  75. package/dist/src/interventions/registry.d.ts +34 -0
  76. package/dist/src/interventions/registry.d.ts.map +1 -0
  77. package/dist/src/interventions/registry.js +248 -0
  78. package/dist/src/interventions/registry.js.map +1 -0
  79. package/dist/src/mcp.d.ts +38 -2
  80. package/dist/src/mcp.d.ts.map +1 -1
  81. package/dist/src/mcp.js +84 -7
  82. package/dist/src/mcp.js.map +1 -1
  83. package/dist/src/models/__tests__/anthropic.test.js +74 -10
  84. package/dist/src/models/__tests__/anthropic.test.js.map +1 -1
  85. package/dist/src/models/__tests__/bedrock.test.js +43 -20
  86. package/dist/src/models/__tests__/bedrock.test.js.map +1 -1
  87. package/dist/src/models/__tests__/google.test.js +14 -6
  88. package/dist/src/models/__tests__/google.test.js.map +1 -1
  89. package/dist/src/models/anthropic.d.ts +15 -3
  90. package/dist/src/models/anthropic.d.ts.map +1 -1
  91. package/dist/src/models/anthropic.js +30 -9
  92. package/dist/src/models/anthropic.js.map +1 -1
  93. package/dist/src/models/bedrock.d.ts +6 -7
  94. package/dist/src/models/bedrock.d.ts.map +1 -1
  95. package/dist/src/models/bedrock.js +31 -18
  96. package/dist/src/models/bedrock.js.map +1 -1
  97. package/dist/src/models/google/model.js +1 -1
  98. package/dist/src/models/google/model.js.map +1 -1
  99. package/dist/src/models/google/types.d.ts +5 -3
  100. package/dist/src/models/google/types.d.ts.map +1 -1
  101. package/dist/src/models/openai/__tests__/chat.test.js +10 -2
  102. package/dist/src/models/openai/__tests__/chat.test.js.map +1 -1
  103. package/dist/src/models/openai/__tests__/mantle.test.d.ts +2 -0
  104. package/dist/src/models/openai/__tests__/mantle.test.d.ts.map +1 -0
  105. package/dist/src/models/openai/__tests__/mantle.test.js +189 -0
  106. package/dist/src/models/openai/__tests__/mantle.test.js.map +1 -0
  107. package/dist/src/models/openai/__tests__/responses.test.js +19 -0
  108. package/dist/src/models/openai/__tests__/responses.test.js.map +1 -1
  109. package/dist/src/models/openai/errors.d.ts.map +1 -1
  110. package/dist/src/models/openai/errors.js +7 -4
  111. package/dist/src/models/openai/errors.js.map +1 -1
  112. package/dist/src/models/openai/index.d.ts +1 -0
  113. package/dist/src/models/openai/index.d.ts.map +1 -1
  114. package/dist/src/models/openai/mantle.d.ts +77 -0
  115. package/dist/src/models/openai/mantle.d.ts.map +1 -0
  116. package/dist/src/models/openai/mantle.js +83 -0
  117. package/dist/src/models/openai/mantle.js.map +1 -0
  118. package/dist/src/models/openai/model.d.ts.map +1 -1
  119. package/dist/src/models/openai/model.js +29 -1
  120. package/dist/src/models/openai/model.js.map +1 -1
  121. package/dist/src/models/openai/types.d.ts +11 -0
  122. package/dist/src/models/openai/types.d.ts.map +1 -1
  123. package/dist/src/multiagent/__tests__/graph.tracer.test.js +14 -0
  124. package/dist/src/multiagent/__tests__/graph.tracer.test.js.map +1 -1
  125. package/dist/src/multiagent/__tests__/interrupts.test.d.ts +2 -0
  126. package/dist/src/multiagent/__tests__/interrupts.test.d.ts.map +1 -0
  127. package/dist/src/multiagent/__tests__/interrupts.test.js +390 -0
  128. package/dist/src/multiagent/__tests__/interrupts.test.js.map +1 -0
  129. package/dist/src/multiagent/__tests__/state.test.js +139 -1
  130. package/dist/src/multiagent/__tests__/state.test.js.map +1 -1
  131. package/dist/src/multiagent/events.d.ts +15 -1
  132. package/dist/src/multiagent/events.d.ts.map +1 -1
  133. package/dist/src/multiagent/events.js +18 -0
  134. package/dist/src/multiagent/events.js.map +1 -1
  135. package/dist/src/multiagent/graph.d.ts +37 -1
  136. package/dist/src/multiagent/graph.d.ts.map +1 -1
  137. package/dist/src/multiagent/graph.js +171 -43
  138. package/dist/src/multiagent/graph.js.map +1 -1
  139. package/dist/src/multiagent/multiagent.d.ts +78 -6
  140. package/dist/src/multiagent/multiagent.d.ts.map +1 -1
  141. package/dist/src/multiagent/multiagent.js +115 -1
  142. package/dist/src/multiagent/multiagent.js.map +1 -1
  143. package/dist/src/multiagent/nodes.d.ts.map +1 -1
  144. package/dist/src/multiagent/nodes.js +55 -21
  145. package/dist/src/multiagent/nodes.js.map +1 -1
  146. package/dist/src/multiagent/state.d.ts +39 -3
  147. package/dist/src/multiagent/state.d.ts.map +1 -1
  148. package/dist/src/multiagent/state.js +80 -1
  149. package/dist/src/multiagent/state.js.map +1 -1
  150. package/dist/src/multiagent/swarm.d.ts +15 -0
  151. package/dist/src/multiagent/swarm.d.ts.map +1 -1
  152. package/dist/src/multiagent/swarm.js +127 -37
  153. package/dist/src/multiagent/swarm.js.map +1 -1
  154. package/dist/src/registry/__tests__/tool-registry.test.js +26 -0
  155. package/dist/src/registry/__tests__/tool-registry.test.js.map +1 -1
  156. package/dist/src/registry/tool-registry.d.ts +9 -7
  157. package/dist/src/registry/tool-registry.d.ts.map +1 -1
  158. package/dist/src/registry/tool-registry.js +29 -10
  159. package/dist/src/registry/tool-registry.js.map +1 -1
  160. package/dist/src/session/__tests__/session-manager.test.js +45 -3
  161. package/dist/src/session/__tests__/session-manager.test.js.map +1 -1
  162. package/dist/src/session/session-manager.d.ts +5 -2
  163. package/dist/src/session/session-manager.d.ts.map +1 -1
  164. package/dist/src/session/session-manager.js +9 -6
  165. package/dist/src/session/session-manager.js.map +1 -1
  166. package/dist/src/telemetry/__tests__/meter.test.js +5 -27
  167. package/dist/src/telemetry/__tests__/meter.test.js.map +1 -1
  168. package/dist/src/telemetry/meter.d.ts +12 -4
  169. package/dist/src/telemetry/meter.d.ts.map +1 -1
  170. package/dist/src/telemetry/meter.js +13 -8
  171. package/dist/src/telemetry/meter.js.map +1 -1
  172. package/dist/src/tools/mcp-tool.d.ts.map +1 -1
  173. package/dist/src/tools/mcp-tool.js +3 -2
  174. package/dist/src/tools/mcp-tool.js.map +1 -1
  175. package/dist/src/tsconfig.tsbuildinfo +1 -1
  176. package/dist/src/types/__tests__/agent.test.js +97 -0
  177. package/dist/src/types/__tests__/agent.test.js.map +1 -1
  178. package/dist/src/types/agent.d.ts +26 -5
  179. package/dist/src/types/agent.d.ts.map +1 -1
  180. package/dist/src/types/agent.js +20 -3
  181. package/dist/src/types/agent.js.map +1 -1
  182. package/dist/src/types/messages.d.ts +3 -1
  183. package/dist/src/types/messages.d.ts.map +1 -1
  184. package/dist/src/types/messages.js.map +1 -1
  185. package/dist/src/vended-plugins/context-offloader/__tests__/plugin.test.js +208 -0
  186. package/dist/src/vended-plugins/context-offloader/__tests__/plugin.test.js.map +1 -1
  187. package/dist/src/vended-plugins/context-offloader/__tests__/search.test.d.ts +2 -0
  188. package/dist/src/vended-plugins/context-offloader/__tests__/search.test.d.ts.map +1 -0
  189. package/dist/src/vended-plugins/context-offloader/__tests__/search.test.js +149 -0
  190. package/dist/src/vended-plugins/context-offloader/__tests__/search.test.js.map +1 -0
  191. package/dist/src/vended-plugins/context-offloader/plugin.d.ts.map +1 -1
  192. package/dist/src/vended-plugins/context-offloader/plugin.js +57 -10
  193. package/dist/src/vended-plugins/context-offloader/plugin.js.map +1 -1
  194. package/dist/src/vended-plugins/context-offloader/search.d.ts +25 -0
  195. package/dist/src/vended-plugins/context-offloader/search.d.ts.map +1 -0
  196. package/dist/src/vended-plugins/context-offloader/search.js +120 -0
  197. package/dist/src/vended-plugins/context-offloader/search.js.map +1 -0
  198. package/dist/src/vended-tools/notebook/notebook.d.ts +1 -1
  199. package/package.json +11 -1
@@ -0,0 +1,692 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
2
+ import { InterventionRegistry } from '../registry.js';
3
+ import { InterventionHandler } from '../handler.js';
4
+ import { HookRegistryImplementation } from '../../hooks/registry.js';
5
+ import { Agent } from '../../agent/agent.js';
6
+ import { BeforeInvocationEvent, BeforeToolCallEvent, BeforeModelCallEvent, AfterModelCallEvent, } from '../../hooks/events.js';
7
+ import { Message, TextBlock } from '../../types/messages.js';
8
+ import { deny } from '../actions.js';
9
+ import { Interrupt, InterruptState } from '../../interrupt.js';
10
+ class DenyHandler extends InterventionHandler {
11
+ name = 'deny-handler';
12
+ beforeToolCall() {
13
+ return { type: 'deny', reason: 'not authorized' };
14
+ }
15
+ }
16
+ class GuideHandler extends InterventionHandler {
17
+ name = 'guide-handler';
18
+ beforeToolCall() {
19
+ return { type: 'guide', feedback: 'add more context' };
20
+ }
21
+ }
22
+ class ConfirmHandler extends InterventionHandler {
23
+ name = 'confirm-handler';
24
+ beforeToolCall() {
25
+ return { type: 'confirm', prompt: 'approve this action?' };
26
+ }
27
+ }
28
+ class ProceedHandler extends InterventionHandler {
29
+ name = 'proceed-handler';
30
+ beforeToolCall() {
31
+ return { type: 'proceed', reason: 'all good' };
32
+ }
33
+ }
34
+ class ThrowingHandler extends InterventionHandler {
35
+ name = 'throwing-handler';
36
+ onError = 'throw';
37
+ beforeToolCall() {
38
+ throw new Error('handler crashed');
39
+ }
40
+ }
41
+ class ThrowingProceedHandler extends InterventionHandler {
42
+ name = 'throwing-proceed';
43
+ onError = 'proceed';
44
+ beforeToolCall() {
45
+ throw new Error('handler crashed');
46
+ }
47
+ }
48
+ class ThrowingDenyHandler extends InterventionHandler {
49
+ name = 'throwing-deny';
50
+ onError = 'deny';
51
+ beforeToolCall() {
52
+ throw new Error('handler crashed');
53
+ }
54
+ }
55
+ class AsyncDenyHandler extends InterventionHandler {
56
+ name = 'async-deny';
57
+ async beforeToolCall() {
58
+ return { type: 'deny', reason: 'async denial' };
59
+ }
60
+ }
61
+ class ModelGuideHandler extends InterventionHandler {
62
+ name = 'model-guide';
63
+ afterModelCall() {
64
+ return { type: 'guide', feedback: 'be more specific' };
65
+ }
66
+ }
67
+ describe('InterventionRegistry', () => {
68
+ let hookRegistry;
69
+ let agent;
70
+ const toolUse = { name: 'testTool', toolUseId: 'id-1', input: {} };
71
+ beforeEach(() => {
72
+ hookRegistry = new HookRegistryImplementation();
73
+ agent = new Agent();
74
+ });
75
+ function makeBeforeInvocationEvent() {
76
+ return new BeforeInvocationEvent({ agent, invocationState: {} });
77
+ }
78
+ function makeBeforeToolCallEvent() {
79
+ return new BeforeToolCallEvent({ agent, toolUse, tool: undefined, invocationState: {} });
80
+ }
81
+ function makeBeforeModelCallEvent() {
82
+ return new BeforeModelCallEvent({ agent, model: {}, invocationState: {} });
83
+ }
84
+ function makeAfterModelCallEvent() {
85
+ return new AfterModelCallEvent({
86
+ agent,
87
+ model: {},
88
+ invocationState: {},
89
+ attemptCount: 0,
90
+ stopData: {
91
+ message: new Message({ role: 'assistant', content: [new TextBlock('response')] }),
92
+ stopReason: 'endTurn',
93
+ },
94
+ });
95
+ }
96
+ describe('constructor', () => {
97
+ it('rejects duplicate handler names', () => {
98
+ expect(() => new InterventionRegistry([new DenyHandler(), new DenyHandler()], hookRegistry)).toThrow("Duplicate intervention handler name: 'deny-handler'");
99
+ });
100
+ it('accepts handlers with unique names', () => {
101
+ // No throw means success
102
+ new InterventionRegistry([new DenyHandler(), new GuideHandler()], hookRegistry);
103
+ });
104
+ });
105
+ describe('hook registration', () => {
106
+ it('only registers hooks for overridden methods', async () => {
107
+ new InterventionRegistry([new DenyHandler()], hookRegistry);
108
+ const beforeToolEvent = makeBeforeToolCallEvent();
109
+ await hookRegistry.invokeCallbacks(beforeToolEvent);
110
+ expect(beforeToolEvent.cancel).toBe('DENIED: not authorized');
111
+ // afterModelCall should not be registered — no handler overrides it
112
+ const afterModelEvent = makeAfterModelCallEvent();
113
+ await hookRegistry.invokeCallbacks(afterModelEvent);
114
+ expect(afterModelEvent.retry).toBeUndefined();
115
+ });
116
+ });
117
+ describe('dispatch ordering', () => {
118
+ it('calls handlers in registration order', async () => {
119
+ const callOrder = [];
120
+ class First extends InterventionHandler {
121
+ name = 'first';
122
+ beforeToolCall() {
123
+ callOrder.push('first');
124
+ return { type: 'proceed' };
125
+ }
126
+ }
127
+ class Second extends InterventionHandler {
128
+ name = 'second';
129
+ beforeToolCall() {
130
+ callOrder.push('second');
131
+ return { type: 'proceed' };
132
+ }
133
+ }
134
+ new InterventionRegistry([new First(), new Second()], hookRegistry);
135
+ await hookRegistry.invokeCallbacks(makeBeforeToolCallEvent());
136
+ expect(callOrder).toEqual(['first', 'second']);
137
+ });
138
+ it('skips handlers that do not override the method', async () => {
139
+ const callOrder = [];
140
+ class ToolHandler extends InterventionHandler {
141
+ name = 'tool';
142
+ beforeToolCall() {
143
+ callOrder.push('tool');
144
+ return { type: 'proceed' };
145
+ }
146
+ }
147
+ class ModelHandler extends InterventionHandler {
148
+ name = 'model';
149
+ afterModelCall() {
150
+ callOrder.push('model');
151
+ return { type: 'proceed' };
152
+ }
153
+ }
154
+ new InterventionRegistry([new ToolHandler(), new ModelHandler()], hookRegistry);
155
+ await hookRegistry.invokeCallbacks(makeBeforeToolCallEvent());
156
+ expect(callOrder).toEqual(['tool']);
157
+ });
158
+ });
159
+ describe('deny', () => {
160
+ it('sets cancel on BeforeToolCallEvent', async () => {
161
+ new InterventionRegistry([new DenyHandler()], hookRegistry);
162
+ const event = makeBeforeToolCallEvent();
163
+ await hookRegistry.invokeCallbacks(event);
164
+ expect(event.cancel).toBe('DENIED: not authorized');
165
+ });
166
+ it('short-circuits — later handlers do not run', async () => {
167
+ const laterCalled = vi.fn();
168
+ class LaterHandler extends InterventionHandler {
169
+ name = 'later';
170
+ beforeToolCall() {
171
+ laterCalled();
172
+ return { type: 'proceed' };
173
+ }
174
+ }
175
+ new InterventionRegistry([new DenyHandler(), new LaterHandler()], hookRegistry);
176
+ await hookRegistry.invokeCallbacks(makeBeforeToolCallEvent());
177
+ expect(laterCalled).not.toHaveBeenCalled();
178
+ });
179
+ it('sets cancel on BeforeInvocationEvent', async () => {
180
+ class InvocationDeny extends InterventionHandler {
181
+ name = 'invocation-deny';
182
+ beforeInvocation() {
183
+ return deny('unauthorized user');
184
+ }
185
+ }
186
+ new InterventionRegistry([new InvocationDeny()], hookRegistry);
187
+ const event = makeBeforeInvocationEvent();
188
+ await hookRegistry.invokeCallbacks(event);
189
+ expect(event.cancel).toBe('DENIED: unauthorized user');
190
+ });
191
+ it('sets cancel on BeforeModelCallEvent', async () => {
192
+ class ModelDeny extends InterventionHandler {
193
+ name = 'model-deny';
194
+ beforeModelCall() {
195
+ return deny('prompt injection detected');
196
+ }
197
+ }
198
+ new InterventionRegistry([new ModelDeny()], hookRegistry);
199
+ const event = makeBeforeModelCallEvent();
200
+ await hookRegistry.invokeCallbacks(event);
201
+ expect(event.cancel).toBe('DENIED: prompt injection detected');
202
+ });
203
+ });
204
+ describe('guide', () => {
205
+ it('sets cancel with guidance on BeforeToolCallEvent', async () => {
206
+ new InterventionRegistry([new GuideHandler()], hookRegistry);
207
+ const event = makeBeforeToolCallEvent();
208
+ await hookRegistry.invokeCallbacks(event);
209
+ expect(event.cancel).toBe('GUIDANCE: [guide-handler] add more context');
210
+ });
211
+ it('accumulates feedback from multiple handlers', async () => {
212
+ class SecondGuide extends InterventionHandler {
213
+ name = 'second-guide';
214
+ beforeToolCall() {
215
+ return { type: 'guide', feedback: 'also check permissions' };
216
+ }
217
+ }
218
+ new InterventionRegistry([new GuideHandler(), new SecondGuide()], hookRegistry);
219
+ const event = makeBeforeToolCallEvent();
220
+ await hookRegistry.invokeCallbacks(event);
221
+ expect(event.cancel).toBe('GUIDANCE: [guide-handler] add more context\n[second-guide] also check permissions');
222
+ });
223
+ it('sets retry=true and injects guidance message on AfterModelCallEvent', async () => {
224
+ new InterventionRegistry([new ModelGuideHandler()], hookRegistry);
225
+ const event = makeAfterModelCallEvent();
226
+ const messageCountBefore = event.agent.messages.length;
227
+ await hookRegistry.invokeCallbacks(event);
228
+ expect(event.retry).toBe(true);
229
+ expect(event.agent.messages).toHaveLength(messageCountBefore + 1);
230
+ const guidanceMessage = event.agent.messages[event.agent.messages.length - 1];
231
+ expect(guidanceMessage.role).toBe('user');
232
+ expect(guidanceMessage.content[0]).toMatchObject({ type: 'textBlock', text: '[model-guide] be more specific' });
233
+ });
234
+ });
235
+ describe('confirm', () => {
236
+ it('pauses agent when no response is provided', async () => {
237
+ new InterventionRegistry([new ConfirmHandler()], hookRegistry);
238
+ const event = makeBeforeToolCallEvent();
239
+ await expect(hookRegistry.invokeCallbacks(event)).rejects.toThrow('Interrupt raised');
240
+ });
241
+ it('short-circuits — later handlers do not run', async () => {
242
+ const laterCalled = vi.fn();
243
+ class LaterHandler extends InterventionHandler {
244
+ name = 'later';
245
+ beforeToolCall() {
246
+ laterCalled();
247
+ return { type: 'proceed' };
248
+ }
249
+ }
250
+ new InterventionRegistry([new ConfirmHandler(), new LaterHandler()], hookRegistry);
251
+ await expect(hookRegistry.invokeCallbacks(makeBeforeToolCallEvent())).rejects.toThrow();
252
+ expect(laterCalled).not.toHaveBeenCalled();
253
+ });
254
+ function preloadInterruptResponse(handlerName, response) {
255
+ const interruptId = `hook:beforeToolCall:${toolUse.toolUseId}:${handlerName}`;
256
+ const interruptState = agent._interruptState;
257
+ interruptState.interrupts[interruptId] = new Interrupt({
258
+ id: interruptId,
259
+ name: handlerName,
260
+ response: response,
261
+ source: 'hook',
262
+ });
263
+ }
264
+ describe('approve/deny on resume', () => {
265
+ const DENIED = 'CONFIRMATION_FAILED: approve this action?';
266
+ it.each([
267
+ [true, false],
268
+ ['yes', false],
269
+ ['y', false],
270
+ ['Y', false],
271
+ ['YES', false],
272
+ [' yes ', false],
273
+ ['no', DENIED],
274
+ [false, DENIED],
275
+ [null, DENIED],
276
+ ['', DENIED],
277
+ ])('response %j → cancel=%j', async (response, expectedCancel) => {
278
+ preloadInterruptResponse('confirm-handler', response);
279
+ new InterventionRegistry([new ConfirmHandler()], hookRegistry);
280
+ const event = makeBeforeToolCallEvent();
281
+ await hookRegistry.invokeCallbacks(event);
282
+ expect(event.cancel).toBe(expectedCancel);
283
+ });
284
+ });
285
+ it('uses custom evaluate when provided', async () => {
286
+ class CustomApprovalHandler extends InterventionHandler {
287
+ name = 'custom-approval';
288
+ beforeToolCall() {
289
+ return {
290
+ type: 'confirm',
291
+ prompt: 'approve?',
292
+ evaluate: (response) => response === 'custom-yes',
293
+ };
294
+ }
295
+ }
296
+ // 'yes' would pass default evaluate but fails custom
297
+ preloadInterruptResponse('custom-approval', 'yes');
298
+ new InterventionRegistry([new CustomApprovalHandler()], hookRegistry);
299
+ const event = makeBeforeToolCallEvent();
300
+ await hookRegistry.invokeCallbacks(event);
301
+ expect(event.cancel).toBe('CONFIRMATION_FAILED: approve?');
302
+ });
303
+ it('custom evaluate approves when its condition is met', async () => {
304
+ class CustomApprovalHandler extends InterventionHandler {
305
+ name = 'custom-approval';
306
+ beforeToolCall() {
307
+ return {
308
+ type: 'confirm',
309
+ prompt: 'approve?',
310
+ evaluate: (response) => response === 'custom-yes',
311
+ };
312
+ }
313
+ }
314
+ preloadInterruptResponse('custom-approval', 'custom-yes');
315
+ new InterventionRegistry([new CustomApprovalHandler()], hookRegistry);
316
+ const event = makeBeforeToolCallEvent();
317
+ await hookRegistry.invokeCallbacks(event);
318
+ expect(event.cancel).toBe(false);
319
+ });
320
+ it('approved confirm does not short-circuit later handlers', async () => {
321
+ preloadInterruptResponse('confirm-handler', 'yes');
322
+ const laterCalled = vi.fn();
323
+ class LaterHandler extends InterventionHandler {
324
+ name = 'later';
325
+ beforeToolCall() {
326
+ laterCalled();
327
+ return { type: 'proceed' };
328
+ }
329
+ }
330
+ new InterventionRegistry([new ConfirmHandler(), new LaterHandler()], hookRegistry);
331
+ const event = makeBeforeToolCallEvent();
332
+ await hookRegistry.invokeCallbacks(event);
333
+ expect(event.cancel).toBe(false);
334
+ expect(laterCalled).toHaveBeenCalled();
335
+ });
336
+ it('denied confirm short-circuits later handlers', async () => {
337
+ preloadInterruptResponse('confirm-handler', 'no');
338
+ const laterCalled = vi.fn();
339
+ class LaterHandler extends InterventionHandler {
340
+ name = 'later';
341
+ beforeToolCall() {
342
+ laterCalled();
343
+ return { type: 'proceed' };
344
+ }
345
+ }
346
+ new InterventionRegistry([new ConfirmHandler(), new LaterHandler()], hookRegistry);
347
+ const event = makeBeforeToolCallEvent();
348
+ await hookRegistry.invokeCallbacks(event);
349
+ expect(event.cancel).toBe('CONFIRMATION_FAILED: approve this action?');
350
+ expect(laterCalled).not.toHaveBeenCalled();
351
+ });
352
+ describe('preemptive response (inline mode)', () => {
353
+ it('approves when response is an approved value', async () => {
354
+ class InlineConfirmHandler extends InterventionHandler {
355
+ name = 'inline-confirm';
356
+ beforeToolCall() {
357
+ return { type: 'confirm', prompt: 'approve?', response: 'yes' };
358
+ }
359
+ }
360
+ new InterventionRegistry([new InlineConfirmHandler()], hookRegistry);
361
+ const event = makeBeforeToolCallEvent();
362
+ await hookRegistry.invokeCallbacks(event);
363
+ expect(event.cancel).toBe(false);
364
+ });
365
+ it('denies when response is a non-approved value', async () => {
366
+ class InlineConfirmHandler extends InterventionHandler {
367
+ name = 'inline-confirm';
368
+ beforeToolCall() {
369
+ return { type: 'confirm', prompt: 'approve?', response: 'no' };
370
+ }
371
+ }
372
+ new InterventionRegistry([new InlineConfirmHandler()], hookRegistry);
373
+ const event = makeBeforeToolCallEvent();
374
+ await hookRegistry.invokeCallbacks(event);
375
+ expect(event.cancel).toBe('CONFIRMATION_FAILED: approve?');
376
+ });
377
+ it('uses custom evaluate with preemptive response', async () => {
378
+ class OtpHandler extends InterventionHandler {
379
+ name = 'otp-handler';
380
+ beforeToolCall() {
381
+ return {
382
+ type: 'confirm',
383
+ prompt: 'Enter OTP:',
384
+ response: '123456',
385
+ evaluate: (r) => r === '123456',
386
+ };
387
+ }
388
+ }
389
+ new InterventionRegistry([new OtpHandler()], hookRegistry);
390
+ const event = makeBeforeToolCallEvent();
391
+ await hookRegistry.invokeCallbacks(event);
392
+ expect(event.cancel).toBe(false);
393
+ });
394
+ it('passes response as preemptive value so agent never pauses', async () => {
395
+ class InlineConfirmHandler extends InterventionHandler {
396
+ name = 'inline-confirm';
397
+ beforeToolCall() {
398
+ return { type: 'confirm', prompt: 'approve?', response: 'yes' };
399
+ }
400
+ }
401
+ new InterventionRegistry([new InlineConfirmHandler()], hookRegistry);
402
+ const event = makeBeforeToolCallEvent();
403
+ const interruptSpy = vi.spyOn(event, 'interrupt');
404
+ await hookRegistry.invokeCallbacks(event);
405
+ expect(interruptSpy).toHaveBeenCalledWith({ name: 'inline-confirm', reason: 'approve?', response: 'yes' });
406
+ });
407
+ it('denies when response is falsy but defined (false)', async () => {
408
+ class InlineConfirmHandler extends InterventionHandler {
409
+ name = 'inline-confirm';
410
+ beforeToolCall() {
411
+ return { type: 'confirm', prompt: 'approve?', response: false };
412
+ }
413
+ }
414
+ new InterventionRegistry([new InlineConfirmHandler()], hookRegistry);
415
+ const event = makeBeforeToolCallEvent();
416
+ await hookRegistry.invokeCallbacks(event);
417
+ expect(event.cancel).toBe('CONFIRMATION_FAILED: approve?');
418
+ });
419
+ });
420
+ it.each(['proceed', 'deny'])('InterruptError always propagates regardless of onError=%s', async (onError) => {
421
+ class ConfirmWithOnError extends InterventionHandler {
422
+ name = 'confirm-onerror';
423
+ onError = onError;
424
+ beforeToolCall() {
425
+ return { type: 'confirm', prompt: 'approve?' };
426
+ }
427
+ }
428
+ new InterventionRegistry([new ConfirmWithOnError()], hookRegistry);
429
+ const event = makeBeforeToolCallEvent();
430
+ await expect(hookRegistry.invokeCallbacks(event)).rejects.toThrow('Interrupt raised');
431
+ });
432
+ });
433
+ describe('transform', () => {
434
+ it('calls the apply function with the event', async () => {
435
+ const applyFn = vi.fn();
436
+ class TransformHandler extends InterventionHandler {
437
+ name = 'transform-handler';
438
+ beforeToolCall() {
439
+ return { type: 'transform', apply: applyFn, reason: 'sanitized input' };
440
+ }
441
+ }
442
+ new InterventionRegistry([new TransformHandler()], hookRegistry);
443
+ const event = makeBeforeToolCallEvent();
444
+ await hookRegistry.invokeCallbacks(event);
445
+ expect(applyFn).toHaveBeenCalledWith(event);
446
+ });
447
+ it('later handlers see the transformed state', async () => {
448
+ const observed = [];
449
+ class Transformer extends InterventionHandler {
450
+ name = 'transformer';
451
+ beforeToolCall() {
452
+ return {
453
+ type: 'transform',
454
+ apply: (e) => {
455
+ ;
456
+ e.cancel = 'transformed';
457
+ },
458
+ };
459
+ }
460
+ }
461
+ class Observer extends InterventionHandler {
462
+ name = 'observer';
463
+ beforeToolCall(event) {
464
+ observed.push(String(event.cancel));
465
+ return { type: 'proceed' };
466
+ }
467
+ }
468
+ new InterventionRegistry([new Transformer(), new Observer()], hookRegistry);
469
+ await hookRegistry.invokeCallbacks(makeBeforeToolCallEvent());
470
+ expect(observed).toEqual(['transformed']);
471
+ });
472
+ it('works on AfterModelCallEvent', async () => {
473
+ const applyFn = vi.fn();
474
+ class ModelTransform extends InterventionHandler {
475
+ name = 'model-transform';
476
+ afterModelCall() {
477
+ return { type: 'transform', apply: applyFn, reason: 'redacted output' };
478
+ }
479
+ }
480
+ new InterventionRegistry([new ModelTransform()], hookRegistry);
481
+ const event = makeAfterModelCallEvent();
482
+ await hookRegistry.invokeCallbacks(event);
483
+ expect(applyFn).toHaveBeenCalledWith(event);
484
+ });
485
+ it('is logged in the audit trail', async () => {
486
+ class TransformHandler extends InterventionHandler {
487
+ name = 'transform-handler';
488
+ beforeToolCall() {
489
+ return { type: 'transform', apply: () => { }, reason: 'sanitized' };
490
+ }
491
+ }
492
+ new InterventionRegistry([new TransformHandler()], hookRegistry);
493
+ await hookRegistry.invokeCallbacks(makeBeforeToolCallEvent());
494
+ // Transform was applied (verified by the apply fn mock tests above)
495
+ });
496
+ });
497
+ describe('proceed', () => {
498
+ it('does not mutate the event', async () => {
499
+ new InterventionRegistry([new ProceedHandler()], hookRegistry);
500
+ const event = makeBeforeToolCallEvent();
501
+ await hookRegistry.invokeCallbacks(event);
502
+ expect(event.cancel).toBe(false);
503
+ });
504
+ });
505
+ describe('error handling', () => {
506
+ it('onError=throw (default) rethrows the error', async () => {
507
+ new InterventionRegistry([new ThrowingHandler(), new ProceedHandler()], hookRegistry);
508
+ await expect(hookRegistry.invokeCallbacks(makeBeforeToolCallEvent())).rejects.toThrow('handler crashed');
509
+ });
510
+ it('onError=proceed skips the handler and continues to next', async () => {
511
+ new InterventionRegistry([new ThrowingProceedHandler(), new ProceedHandler()], hookRegistry);
512
+ const event = makeBeforeToolCallEvent();
513
+ await hookRegistry.invokeCallbacks(event);
514
+ expect(event.cancel).toBe(false);
515
+ });
516
+ it('onError=deny logs the error and applies deny', async () => {
517
+ const laterCalled = vi.fn();
518
+ class LaterHandler extends InterventionHandler {
519
+ name = 'later';
520
+ beforeToolCall() {
521
+ laterCalled();
522
+ return { type: 'proceed' };
523
+ }
524
+ }
525
+ new InterventionRegistry([new ThrowingDenyHandler(), new LaterHandler()], hookRegistry);
526
+ const event = makeBeforeToolCallEvent();
527
+ await hookRegistry.invokeCallbacks(event);
528
+ expect(event.cancel).toBe('DENIED: Handler threw: handler crashed');
529
+ expect(laterCalled).not.toHaveBeenCalled();
530
+ });
531
+ });
532
+ describe('async handlers', () => {
533
+ it('awaits async handler results', async () => {
534
+ new InterventionRegistry([new AsyncDenyHandler()], hookRegistry);
535
+ const event = makeBeforeToolCallEvent();
536
+ await hookRegistry.invokeCallbacks(event);
537
+ expect(event.cancel).toBe('DENIED: async denial');
538
+ });
539
+ });
540
+ describe('conflict resolution', () => {
541
+ it('deny wins over guide', async () => {
542
+ new InterventionRegistry([new GuideHandler(), new DenyHandler()], hookRegistry);
543
+ const event = makeBeforeToolCallEvent();
544
+ await hookRegistry.invokeCallbacks(event);
545
+ expect(event.cancel).toBe('DENIED: not authorized');
546
+ });
547
+ it('deny short-circuits before guide can accumulate', async () => {
548
+ new InterventionRegistry([new DenyHandler(), new GuideHandler()], hookRegistry);
549
+ const event = makeBeforeToolCallEvent();
550
+ await hookRegistry.invokeCallbacks(event);
551
+ expect(event.cancel).toBe('DENIED: not authorized');
552
+ });
553
+ it('confirm short-circuits before guide can accumulate', async () => {
554
+ new InterventionRegistry([new ConfirmHandler(), new GuideHandler()], hookRegistry);
555
+ await expect(hookRegistry.invokeCallbacks(makeBeforeToolCallEvent())).rejects.toThrow('Interrupt raised');
556
+ });
557
+ });
558
+ describe('agent integration', () => {
559
+ it('deny on beforeToolCall prevents tool execution', async () => {
560
+ const { MockMessageModel } = await import('../../__fixtures__/mock-message-model.js');
561
+ const { createMockTool } = await import('../../__fixtures__/tool-helpers.js');
562
+ let toolExecuted = false;
563
+ const tool = createMockTool('blockedTool', () => {
564
+ toolExecuted = true;
565
+ return 'should not reach here';
566
+ });
567
+ class BlockAllTools extends InterventionHandler {
568
+ name = 'block-all';
569
+ beforeToolCall() {
570
+ return { type: 'deny', reason: 'blocked by intervention' };
571
+ }
572
+ }
573
+ const model = new MockMessageModel()
574
+ .addTurn({ type: 'toolUseBlock', name: 'blockedTool', toolUseId: 'tool-1', input: {} })
575
+ .addTurn({ type: 'textBlock', text: 'Done' });
576
+ const agent = new Agent({
577
+ model,
578
+ tools: [tool],
579
+ interventions: [new BlockAllTools()],
580
+ });
581
+ const result = await agent.invoke('Test');
582
+ expect(result.stopReason).toBe('endTurn');
583
+ expect(toolExecuted).toBe(false);
584
+ });
585
+ it('interventions run before plugins (HookOrder.INTERVENTIONS < DEFAULT)', async () => {
586
+ const { MockMessageModel } = await import('../../__fixtures__/mock-message-model.js');
587
+ const { createMockTool } = await import('../../__fixtures__/tool-helpers.js');
588
+ const callOrder = [];
589
+ const tool = createMockTool('testTool', () => 'result');
590
+ class OrderTracker extends InterventionHandler {
591
+ name = 'order-tracker';
592
+ beforeToolCall() {
593
+ callOrder.push('intervention');
594
+ return { type: 'proceed' };
595
+ }
596
+ }
597
+ const model = new MockMessageModel()
598
+ .addTurn({ type: 'toolUseBlock', name: 'testTool', toolUseId: 'tool-1', input: {} })
599
+ .addTurn({ type: 'textBlock', text: 'Done' });
600
+ const agent = new Agent({
601
+ model,
602
+ tools: [tool],
603
+ interventions: [new OrderTracker()],
604
+ });
605
+ agent.addHook(BeforeToolCallEvent, () => {
606
+ callOrder.push('plugin');
607
+ });
608
+ await agent.invoke('Test');
609
+ // On Before*: plugins run first (DEFAULT:0), interventions last (INTERVENTIONS:90)
610
+ expect(callOrder[0]).toBe('plugin');
611
+ expect(callOrder[1]).toBe('intervention');
612
+ });
613
+ });
614
+ describe('edge cases', () => {
615
+ it('guide on beforeModelCall injects a user message', async () => {
616
+ class ModelGuide extends InterventionHandler {
617
+ name = 'model-guide';
618
+ beforeModelCall() {
619
+ return { type: 'guide', feedback: 'check your sources' };
620
+ }
621
+ }
622
+ new InterventionRegistry([new ModelGuide()], hookRegistry);
623
+ const event = makeBeforeModelCallEvent();
624
+ const messageCountBefore = event.agent.messages.length;
625
+ await hookRegistry.invokeCallbacks(event);
626
+ expect(event.cancel).toBe(false);
627
+ expect(event.agent.messages).toHaveLength(messageCountBefore + 1);
628
+ const injected = event.agent.messages[event.agent.messages.length - 1];
629
+ expect(injected.role).toBe('user');
630
+ expect(injected.content[0]).toMatchObject({ type: 'textBlock', text: '[model-guide] check your sources' });
631
+ });
632
+ it('transform apply() error is handled via onError policy', async () => {
633
+ class BadTransform extends InterventionHandler {
634
+ name = 'bad-transform';
635
+ onError = 'proceed';
636
+ beforeToolCall() {
637
+ return {
638
+ type: 'transform',
639
+ apply: () => {
640
+ throw new Error('apply boom');
641
+ },
642
+ };
643
+ }
644
+ }
645
+ class AfterTransform extends InterventionHandler {
646
+ name = 'after-transform';
647
+ beforeToolCall() {
648
+ return { type: 'proceed', reason: 'still running' };
649
+ }
650
+ }
651
+ new InterventionRegistry([new BadTransform(), new AfterTransform()], hookRegistry);
652
+ const event = makeBeforeToolCallEvent();
653
+ await hookRegistry.invokeCallbacks(event);
654
+ // onError=proceed means the error is swallowed and next handler runs
655
+ expect(event.cancel).toBe(false);
656
+ });
657
+ it('transform apply() error with onError=throw propagates', async () => {
658
+ class BadTransform extends InterventionHandler {
659
+ name = 'bad-transform';
660
+ onError = 'throw';
661
+ beforeToolCall() {
662
+ return {
663
+ type: 'transform',
664
+ apply: () => {
665
+ throw new Error('apply boom');
666
+ },
667
+ };
668
+ }
669
+ }
670
+ new InterventionRegistry([new BadTransform()], hookRegistry);
671
+ await expect(hookRegistry.invokeCallbacks(makeBeforeToolCallEvent())).rejects.toThrow('apply boom');
672
+ });
673
+ it('warns when action has no effect on event type', async () => {
674
+ const { logger } = await import('../../logging/logger.js');
675
+ const warnSpy = vi.spyOn(logger, 'warn');
676
+ // Force a confirm return on beforeInvocation (which doesn't support it)
677
+ // via cast to test the runtime warning path
678
+ class InterruptOnInvocation extends InterventionHandler {
679
+ name = 'confirm-invocation';
680
+ beforeInvocation() {
681
+ // Force a confirm return via any cast to test the runtime warning
682
+ return { type: 'confirm', prompt: 'test' };
683
+ }
684
+ }
685
+ new InterventionRegistry([new InterruptOnInvocation()], hookRegistry);
686
+ await hookRegistry.invokeCallbacks(makeBeforeInvocationEvent());
687
+ expect(warnSpy).toHaveBeenCalledWith(expect.stringContaining('has no effect'));
688
+ warnSpy.mockRestore();
689
+ });
690
+ });
691
+ });
692
+ //# sourceMappingURL=registry.test.js.map