actionmcp 0.22.0 → 0.25.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 (33) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +113 -2
  3. data/app/controllers/action_mcp/messages_controller.rb +2 -14
  4. data/app/controllers/action_mcp/sse_controller.rb +113 -45
  5. data/app/models/action_mcp/session/message.rb +21 -16
  6. data/app/models/action_mcp/session.rb +3 -2
  7. data/config/routes.rb +1 -1
  8. data/db/migrate/20250324203409_remove_session_message_text.rb +7 -0
  9. data/lib/action_mcp/client/base.rb +12 -14
  10. data/lib/action_mcp/client/blueprint.rb +5 -71
  11. data/lib/action_mcp/client/catalog.rb +10 -74
  12. data/lib/action_mcp/client/collection.rb +93 -0
  13. data/lib/action_mcp/client/json_rpc_handler.rb +12 -7
  14. data/lib/action_mcp/client/logging.rb +1 -2
  15. data/lib/action_mcp/client/prompt_book.rb +5 -71
  16. data/lib/action_mcp/client/prompts.rb +9 -4
  17. data/lib/action_mcp/client/request_timeouts.rb +74 -0
  18. data/lib/action_mcp/client/resources.rb +23 -11
  19. data/lib/action_mcp/client/server.rb +3 -3
  20. data/lib/action_mcp/client/toolbox.rb +12 -54
  21. data/lib/action_mcp/client/tools.rb +9 -4
  22. data/lib/action_mcp/configuration.rb +134 -24
  23. data/lib/action_mcp/engine.rb +6 -0
  24. data/lib/action_mcp/json_rpc_handler_base.rb +1 -0
  25. data/lib/action_mcp/registry_base.rb +3 -1
  26. data/lib/action_mcp/server/capabilities.rb +1 -1
  27. data/lib/action_mcp/server/json_rpc_handler.rb +1 -1
  28. data/lib/action_mcp/server/messaging.rb +32 -9
  29. data/lib/action_mcp/version.rb +1 -1
  30. data/lib/generators/action_mcp/install/install_generator.rb +4 -0
  31. data/lib/generators/action_mcp/install/templates/mcp.yml +11 -0
  32. data/lib/tasks/action_mcp_tasks.rake +77 -6
  33. metadata +6 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 5c997c4c99f4ecf6409fb3749bb58efab0eb03622e70d9a447781125620e6c82
4
- data.tar.gz: dd9fc6be1b7364f306a6cbcfe0cd9e8db3b00530e46056c6ad49e6995ab7810b
3
+ metadata.gz: 843ee149da81be07d55d13eb26fad244a21232a0d4f795c98429d556c67c2736
4
+ data.tar.gz: fbf8aa774b9afe3e72cc325a93d054f98950163e14bd618d851c10d70bf823dc
5
5
  SHA512:
6
- metadata.gz: c6de9245cf20071a23b0f35181d1eb20db9a2fb96298c6afc165a05aa31f14c3f8035b2b3dbd9ac9b04319f0981ce26609f8afde1ab67ef99a055a070ce3018c
7
- data.tar.gz: e8b0006312a943b8228408ab0fe5b859d4a638bfbbe89ddcb6be0d38465476fdb9977f6d2b322676eb1ddc1608abcc7531c9352495c7bde11bb04868586edae1
6
+ metadata.gz: b32d0b8d10b4a7ed6d8c86a76db959d5a453a5d0004dc6ccf2cee643d57c347777a4581dc0a1cf34f9d41dd202c01b0a28ce24169d9b43fabce96b980d857b8e
7
+ data.tar.gz: a42653c1903e8e5c8078fd4fc3b3e206c1c92a58cde623b166b6bbd840be45cf1d6339f08d55cbe1018a2c448522901c30d7d6d72ba1d6ff97b22f0c1e4a76e1
data/README.md CHANGED
@@ -293,6 +293,117 @@ npx @modelcontextprotocol/inspector
293
293
 
294
294
  The default path will be http://localhost:3000/action_mcp
295
295
 
296
- ## Conclusion
296
+ Here's a section you can add to explain the profile system in ActionMCP:
297
+
298
+ ## Profiles
299
+
300
+ ActionMCP supports a flexible profile system that allows you to selectively expose tools, prompts, and resources based on different usage scenarios. This is particularly useful for applications that need different MCP capabilities for different contexts (e.g., public API vs. admin interface).
301
+
302
+ ### Understanding Profiles
303
+
304
+ Profiles are named configurations that define:
305
+
306
+ - Which tools are available
307
+ - Which prompts are accessible
308
+ - Which resources can be accessed
309
+ - Configuration options like logging level and change notifications
310
+
311
+ By default, ActionMCP includes two profiles:
312
+ - `primary`: Exposes all tools, prompts, and resources
313
+ - `minimal`: Exposes no tools, prompts, or resources by default
314
+
315
+ ### Configuring Profiles
316
+
317
+ Profiles are configured via a `config/mcp.yml` file in your Rails application. If this file doesn't exist, ActionMCP will use default settings from the gem.
318
+
319
+ **Example configuration:**
320
+
321
+ ```yaml
322
+ default:
323
+ tools:
324
+ - all # Include all tools
325
+ prompts:
326
+ - all # Include all prompts
327
+ resources:
328
+ - all # Include all resources
329
+ options:
330
+ list_changed: false
331
+ logging_enabled: true
332
+ logging_level: info
333
+ resources_subscribe: false
334
+
335
+ api_only:
336
+ tools:
337
+ - calculator
338
+ - weather
339
+ prompts: [] # No prompts for API
340
+ resources:
341
+ - user_profile
342
+ options:
343
+ list_changed: false
344
+ logging_level: warn
345
+
346
+ admin:
347
+ tools:
348
+ - all
349
+ options:
350
+ logging_level: debug
351
+ list_changed: true
352
+ resources_subscribe: true
353
+ ```
354
+
355
+ Each profile can specify:
356
+ - `tools`: Array of tool names to include (use `all` to include all tools)
357
+ - `prompts`: Array of prompt names to include (use `all` to include all prompts)
358
+ - `resources`: Array of resource names to include (use `all` to include all resources)
359
+ - `options`: Additional configuration options:
360
+ - `list_changed`: Whether to send change notifications
361
+ - `logging_enabled`: Whether to enable logging
362
+ - `logging_level`: The logging level to use
363
+ - `resources_subscribe`: Whether to enable resource subscriptions
364
+
365
+ ### Switching Profiles
366
+
367
+ You can switch between profiles programmatically in your code:
368
+
369
+ ```ruby
370
+ # Permanently switch to a different profile
371
+ ActionMCP.configuration.use_profile(:only_tools) # Switch to a profile named "only_tools"
372
+
373
+ # Temporarily use a profile for a specific operation
374
+ ActionMCP.with_profile(:minimal) do
375
+ # Code here uses the minimal profile
376
+ # After the block, reverts to the previous profile
377
+ end
378
+ ```
379
+
380
+ This makes it easy to control which MCP capabilities are available in different contexts of your application.
381
+
382
+ ### Inspecting Profiles
383
+
384
+ ActionMCP includes rake tasks to help you manage and inspect your profiles:
385
+
386
+ ```bash
387
+ # List all available profiles with their configurations
388
+ bin/rails action_mcp:list_profiles
389
+
390
+ # Show detailed information about a specific profile
391
+ bin/rails action_mcp:show_profile[admin]
392
+
393
+ # List all tools, prompts, resources, and profiles
394
+ bin/rails action_mcp:list
395
+ ```
396
+
397
+ The profile inspection tasks will highlight any issues, such as configured tools, prompts, or resources that don't actually exist in your application.
398
+
399
+ ### Use Cases
400
+
401
+ Profiles are particularly useful for:
402
+
403
+ 1. **Multi-tenant applications**: Use different profiles for different customer tiers with Dorp or other gems
404
+ 2. **Access control**: Create profiles for different user roles (admin, staff, public)
405
+ 3. **Performance optimization**: Use a minimal profile for high-traffic endpoints
406
+ 4. **Testing environments**: Use specific test profiles in your test environment
407
+ 5. **Progressive enhancement**: Start with a minimal profile and gradually add capabilities
297
408
 
298
- ActionMCP empowers developers to build MCP-compliant servers efficiently by handling the standardization and boilerplate associated with integrating with LLMs. With its intuitive abstractions for tools, prompts, and resource templates, you can quickly expose your application's capabilities to AI models while maintaining full control over how they interact with your system.
409
+ By leveraging profiles, you can maintain a single ActionMCP codebase while providing tailored MCP capabilities for different contexts.
@@ -6,11 +6,7 @@ module ActionMCP
6
6
 
7
7
  # @route POST / (sse_in)
8
8
  def create
9
- begin
10
- handle_post_message(clean_params, response)
11
- rescue StandardError
12
- head :internal_server_error
13
- end
9
+ handle_post_message(params, response)
14
10
  head response.status
15
11
  end
16
12
 
@@ -25,22 +21,14 @@ module ActionMCP
25
21
  end
26
22
 
27
23
  def handle_post_message(params, response)
28
- mcp_session.initialize! if params[:method] == "initialize"
29
24
  json_rpc_handler.call(params)
30
-
31
25
  response.status = :accepted
32
- rescue StandardError => e
33
- puts "Error: #{e.message}"
34
- puts e.backtrace.join("\n")
26
+ rescue StandardError => _e
35
27
  response.status = :bad_request
36
28
  end
37
29
 
38
30
  def mcp_session
39
31
  @mcp_session ||= Session.find_or_create_by(id: params[:session_id])
40
32
  end
41
-
42
- def clean_params
43
- params.slice(:id, :method, :jsonrpc, :params, :result, :error)
44
- end
45
33
  end
46
34
  end
@@ -2,70 +2,122 @@
2
2
 
3
3
  module ActionMCP
4
4
  class SSEController < MCPController
5
- HEARTBEAT_INTERVAL = 30 # TODO: The frequency of pings SHOULD be configurable
5
+ HEARTBEAT_INTERVAL = 30 # in seconds
6
+ INITIAL_CONNECTION_TIMEOUT = 5 # in seconds
6
7
  include ActionController::Live
7
8
 
8
9
  # @route GET /sse (sse_out)
9
10
  def events
10
- # Set headers first
11
+ # Set headers for SSE
11
12
  response.headers["X-Accel-Buffering"] = "no"
12
13
  response.headers["Content-Type"] = "text/event-stream"
13
14
  response.headers["Cache-Control"] = "no-cache"
14
15
  response.headers["Connection"] = "keep-alive"
15
16
 
16
- # Now start streaming - send endpoint
17
+ # Send the endpoint URL to the client
17
18
  send_endpoint_event(sse_in_url)
18
19
 
20
+ Rails.logger.info "SSE: Starting connection for session: #{session_id}"
21
+
22
+ # Use Concurrent primitives for state management
23
+ message_received = Concurrent::AtomicBoolean.new(false)
24
+ connection_active = Concurrent::AtomicBoolean.new(true)
25
+
19
26
  begin
20
- # Create SSE instance once outside the callback with retry parameter
27
+ # Create SSE instance
21
28
  sse = SSE.new(response.stream)
22
29
 
23
- # Start listener and process messages via the transport
30
+ # Start the connection monitor using a proper scheduled task
31
+ timeout_task = Concurrent::ScheduledTask.execute(INITIAL_CONNECTION_TIMEOUT) do
32
+ unless message_received.true?
33
+ Rails.logger.warn "No message received within #{INITIAL_CONNECTION_TIMEOUT} seconds, closing connection for session: #{session_id}"
34
+ error = build_timeout_error
35
+ # Safely write error and close the stream
36
+ Concurrent::Promise.execute do
37
+ sse.write(error) rescue nil
38
+ response.stream.close rescue nil
39
+ connection_active.make_false
40
+ end
41
+ end
42
+ end
43
+
44
+ # Initialize the listener
24
45
  listener = SSEListener.new(mcp_session)
25
- message_received = false
26
- if listener.start do |message|
27
- # Send with proper SSE formatting
46
+ listener_started = listener.start do |message|
47
+ message_received.make_true
28
48
  sse.write(message)
29
- message_received = true
30
49
  end
31
- sleep 1
32
- # Heartbeat loop
33
- unless message_received
34
- Rails.logger.warn "No message received within 1 second, closing connection for session: #{session_id}"
35
- error = JsonRpc::Response.new(id: SecureRandom.uuid_v7,
36
- error: JsonRpc::JsonRpcError.new(
37
- :server_error, message: "No message received within 1 second"
38
- ).to_h).to_h
39
- sse.write(error)
40
- return
41
- end
42
50
 
43
- until response.stream.closed?
44
- sleep HEARTBEAT_INTERVAL
45
- # mcp_session.send_ping!
46
- end
47
- else
51
+ unless listener_started
48
52
  Rails.logger.error "Listener failed to activate for session: #{session_id}"
49
- raise "Failed to establish subscription"
53
+ error = build_listener_error
54
+ sse.write(error)
55
+ connection_active.make_false
56
+ return
57
+ end
58
+
59
+ # Schedule heartbeats using a proper timer
60
+ heartbeat = Concurrent::TimerTask.new(
61
+ execution_interval: HEARTBEAT_INTERVAL,
62
+ timeout_interval: 5 # Timeout for heartbeat operation
63
+ ) do
64
+ if connection_active.true? && !response.stream.closed?
65
+ begin
66
+ sse.write({ ping: true })
67
+ rescue StandardError => e
68
+ Rails.logger.debug "SSE: Heartbeat error: #{e.message}"
69
+ connection_active.make_false
70
+ end
71
+ else
72
+ raise Concurrent::CancelledOperationError
73
+ end
74
+ end
75
+ heartbeat.execute
76
+
77
+ # Wait for connection to be closed or cancelled
78
+ while connection_active.true? && !response.stream.closed?
79
+ sleep 0.1
50
80
  end
51
81
  rescue ActionController::Live::ClientDisconnected, IOError => e
52
- Rails.logger.debug "SSE: Expected disconnection: #{e.message}"
82
+ Rails.logger.debug "SSE: Client disconnected: #{e.message}"
53
83
  rescue StandardError => e
54
84
  Rails.logger.error "SSE: Unexpected error: #{e.class} - #{e.message}\n#{e.backtrace.join("\n")}"
55
85
  ensure
56
- response.stream.close
57
- listener.stop
58
- Rails.logger.debug "SSE: Connection closed for session: #{session_id}"
86
+ # Clean up resources
87
+ timeout_task&.cancel
88
+ heartbeat&.shutdown
89
+ listener&.stop
90
+ response.stream.close rescue nil
91
+
92
+ Rails.logger.debug "SSE: Connection cleaned up for session: #{session_id}"
59
93
  end
60
94
  end
61
95
 
62
96
  private
63
97
 
98
+ def build_timeout_error
99
+ JsonRpc::Response.new(
100
+ id: SecureRandom.uuid_v7,
101
+ error: JsonRpc::JsonRpcError.new(
102
+ :server_error,
103
+ message: "No message received within initial connection timeout"
104
+ ).to_h
105
+ ).to_h
106
+ end
107
+
108
+ def build_listener_error
109
+ JsonRpc::Response.new(
110
+ id: SecureRandom.uuid_v7,
111
+ error: JsonRpc::JsonRpcError.new(
112
+ :server_error,
113
+ message: "Failed to establish server connection"
114
+ ).to_h
115
+ ).to_h
116
+ end
117
+
64
118
  def send_endpoint_event(messages_url)
65
119
  endpoint = "#{messages_url}?session_id=#{session_id}"
66
- SSE.new(response.stream,
67
- event: "endpoint")
68
- .write(endpoint)
120
+ SSE.new(response.stream, event: "endpoint").write(endpoint)
69
121
  end
70
122
 
71
123
  def default_url_options
@@ -77,11 +129,11 @@ module ActionMCP
77
129
  end
78
130
 
79
131
  def session_id
80
- @session_id ||= mcp_session.id
132
+ mcp_session.id
81
133
  end
82
134
 
83
135
  def cache_key
84
- "action_mcp:session:#{session_id}"
136
+ mcp_session.session_key
85
137
  end
86
138
  end
87
139
 
@@ -93,7 +145,8 @@ module ActionMCP
93
145
  # @param session [ActionMCP::Session]
94
146
  def initialize(session)
95
147
  @session = session
96
- @stopped = false
148
+ @stopped = Concurrent::AtomicBoolean.new(false)
149
+ @subscription_active = Concurrent::AtomicBoolean.new(false)
97
150
  end
98
151
 
99
152
  # Start listening using ActionCable's adapter
@@ -101,34 +154,49 @@ module ActionMCP
101
154
  Rails.logger.debug "Starting listener for channel: #{session_key}"
102
155
 
103
156
  success_callback = lambda {
104
- puts "Successfully subscribed to channel: #{session_key}"
105
- @subscription_active = true
157
+ Rails.logger.info "Successfully subscribed to channel: #{session_key}"
158
+ @subscription_active.make_true
106
159
  }
107
160
 
108
161
  # Set up message callback
109
162
  message_callback = lambda { |raw_message|
163
+ return if @stopped.true?
164
+
110
165
  begin
111
166
  # Try to parse the message if it's JSON
112
167
  message = MultiJson.load(raw_message)
113
168
  # Send the message to the callback
114
- callback.call(message) if callback && !@stopped
115
- rescue StandardError
169
+ callback.call(message) if callback
170
+ rescue StandardError => e
171
+ Rails.logger.error "Error processing message: #{e.message}"
116
172
  # Still try to send the raw message as a fallback
117
- callback.call(raw_message) if callback && !@stopped
173
+ callback.call(raw_message) if callback
118
174
  end
119
175
  }
120
176
 
121
177
  # Subscribe using the ActionCable adapter
122
178
  adapter.subscribe(session_key, message_callback, success_callback)
123
179
 
124
- # Give some time for the subscription to be established
125
- sleep 0.5
180
+ # Use a future with timeout to check subscription status
181
+ subscription_future = Concurrent::Promises.future do
182
+ while !@subscription_active.true? && !@stopped.true?
183
+ sleep 0.1
184
+ end
185
+ @subscription_active.true?
186
+ end
126
187
 
127
- @subscription_active
188
+ # Wait up to 1 second for subscription to be established
189
+ begin
190
+ subscription_result = subscription_future.value(1)
191
+ subscription_result || @subscription_active.true?
192
+ rescue Concurrent::TimeoutError
193
+ Rails.logger.warn "Timed out waiting for subscription activation"
194
+ false
195
+ end
128
196
  end
129
197
 
130
198
  def stop
131
- @stopped = true
199
+ @stopped.make_true
132
200
  if (mcp_session = Session.find_by(id: session_key))
133
201
  mcp_session.close
134
202
  end
@@ -8,7 +8,6 @@
8
8
  # direction(The message recipient) :string default("client"), not null
9
9
  # is_ping(Whether the message is a ping) :boolean default(FALSE), not null
10
10
  # message_json :jsonb
11
- # message_text :string
12
11
  # message_type(The type of the message) :string not null
13
12
  # request_acknowledged :boolean default(FALSE), not null
14
13
  # request_cancelled :boolean default(FALSE), not null
@@ -48,7 +47,7 @@ module ActionMCP
48
47
 
49
48
  after_create_commit :broadcast_message, if: :outgoing_message?
50
49
  # Set is_ping on responses if the original request was a ping
51
- after_create :handle_ping_response, if: -> { %w[response error].include?(message_type) }
50
+ after_create :acknowledge_request, if: -> { %w[response error].include?(message_type) }
52
51
 
53
52
  # Scope to exclude both "ping" requests and their responses
54
53
  scope :without_pings, -> { where(is_ping: false) }
@@ -61,26 +60,26 @@ module ActionMCP
61
60
  def data=(payload)
62
61
  @data = payload
63
62
 
64
- # Store original version and attempt to determine type
63
+ # Convert string payloads to JSON
65
64
  if payload.is_a?(String)
66
- self.message_text = payload
67
65
  begin
68
66
  parsed_json = MultiJson.load(payload)
69
67
  self.message_json = parsed_json
70
- self.message_text = nil
71
- process_json_content(parsed_json)
72
68
  rescue MultiJson::ParseError
73
- self.message_type = "text"
69
+ # Handle invalid JSON by creating an error object
70
+ self.message_json = { "error" => "Invalid JSON", "raw" => payload }
71
+ self.message_type = "invalid_json"
72
+ return
74
73
  end
75
74
  else
75
+ # Handle direct hash assignment
76
76
  self.message_json = payload
77
- self.message_text = nil
78
- process_json_content(payload)
79
77
  end
78
+ process_json_content(payload)
80
79
  end
81
80
 
82
81
  def data
83
- message_json.presence || message_text
82
+ message_json
84
83
  end
85
84
 
86
85
  # Helper methods
@@ -98,6 +97,7 @@ module ActionMCP
98
97
 
99
98
  def rpc_method
100
99
  return false unless request?
100
+
101
101
  data["method"]
102
102
  end
103
103
 
@@ -108,9 +108,9 @@ module ActionMCP
108
108
  end
109
109
 
110
110
  def broadcast_message
111
- if adapter.present?
112
- adapter.broadcast(session_key, data.to_json)
113
- end
111
+ return unless adapter.present?
112
+
113
+ adapter.broadcast(session_key, data.to_json)
114
114
  end
115
115
 
116
116
  def process_json_content(content)
@@ -136,17 +136,22 @@ module ActionMCP
136
136
  end
137
137
  end
138
138
 
139
- def handle_ping_response
139
+ def acknowledge_request
140
140
  return unless jsonrpc_id.present?
141
141
 
142
142
  request_message = session.messages.find_by(
143
143
  jsonrpc_id: jsonrpc_id,
144
144
  message_type: "request"
145
145
  )
146
- return unless request_message&.is_ping
147
146
 
148
- self.is_ping = true
147
+ return unless request_message
148
+
149
+ # Set is_ping based on the request
150
+ self.is_ping = request_message.is_ping
151
+
152
+ # Mark the request as acknowledged for all responses
149
153
  request_message.update(request_acknowledged: true)
154
+
150
155
  save! if changed?
151
156
  end
152
157
  end
@@ -107,8 +107,9 @@ module ActionMCP
107
107
  # update the session initialized to true
108
108
  return false if initialized?
109
109
 
110
- update!(initialized: true,
111
- status: "initialized")
110
+ self.initialized = true
111
+ self.status = "initialized"
112
+ save
112
113
  end
113
114
 
114
115
  def message_flow
data/config/routes.rb CHANGED
@@ -2,5 +2,5 @@
2
2
 
3
3
  ActionMCP::Engine.routes.draw do
4
4
  get "/", to: "sse#events", as: :sse_out
5
- post "/", to: "messages#create", as: :sse_in
5
+ post "/", to: "messages#create", as: :sse_in, defaults: { format: "json" }
6
6
  end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ class RemoveSessionMessageText < ActiveRecord::Migration[8.0]
4
+ def up
5
+ remove_column :action_mcp_session_messages, :message_text
6
+ end
7
+ end
@@ -17,6 +17,7 @@ module ActionMCP
17
17
  :server_capabilities, :session,
18
18
  :catalog, :blueprint,
19
19
  :prompt_book, :toolbox
20
+
20
21
  delegate :initialized?, to: :session
21
22
 
22
23
  def initialize(logger: ActionMCP.logger)
@@ -146,15 +147,14 @@ module ActionMCP
146
147
  end
147
148
 
148
149
  def server=(server)
149
- if server.is_a?(Client::Server)
150
- @server = server
150
+ @server = if server.is_a?(Client::Server)
151
+ server
151
152
  else
152
- @server = Client::Server.new(server)
153
+ Client::Server.new(server)
153
154
  end
154
155
  session.server_capabilities = server.capabilities
155
156
  session.server_info = server.server_info
156
157
  session.save
157
- server
158
158
  end
159
159
 
160
160
  def inspect
@@ -164,15 +164,13 @@ module ActionMCP
164
164
  protected
165
165
 
166
166
  def handle_raw_message(raw)
167
- begin
168
- @message_callback&.call(raw)
169
- rescue MultiJson::ParseError => e
170
- log_error("JSON parse error: #{e} (raw: #{raw})")
171
- @error_callback&.call(e)
172
- rescue StandardError => e
173
- log_error("Error handling message: #{e} (raw: #{raw})")
174
- @error_callback&.call(e)
175
- end
167
+ @message_callback&.call(raw)
168
+ rescue MultiJson::ParseError => e
169
+ log_error("JSON parse error: #{e} (raw: #{raw})")
170
+ @error_callback&.call(e)
171
+ rescue StandardError => e
172
+ log_error("Error handling message: #{e} (raw: #{raw})")
173
+ @error_callback&.call(e)
176
174
  end
177
175
 
178
176
  def send_initial_capabilities
@@ -180,7 +178,7 @@ module ActionMCP
180
178
  # We have contact! Let's send our CV to the recruiter.
181
179
  # We persist the session object to the database
182
180
  session.save
183
- params= {
181
+ params = {
184
182
  protocolVersion: session.protocol_version,
185
183
  capabilities: session.client_capabilities,
186
184
  clientInfo: session.client_info