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.
- checksums.yaml +4 -4
- data/README.md +113 -2
- data/app/controllers/action_mcp/messages_controller.rb +2 -14
- data/app/controllers/action_mcp/sse_controller.rb +113 -45
- data/app/models/action_mcp/session/message.rb +21 -16
- data/app/models/action_mcp/session.rb +3 -2
- data/config/routes.rb +1 -1
- data/db/migrate/20250324203409_remove_session_message_text.rb +7 -0
- data/lib/action_mcp/client/base.rb +12 -14
- data/lib/action_mcp/client/blueprint.rb +5 -71
- data/lib/action_mcp/client/catalog.rb +10 -74
- data/lib/action_mcp/client/collection.rb +93 -0
- data/lib/action_mcp/client/json_rpc_handler.rb +12 -7
- data/lib/action_mcp/client/logging.rb +1 -2
- data/lib/action_mcp/client/prompt_book.rb +5 -71
- data/lib/action_mcp/client/prompts.rb +9 -4
- data/lib/action_mcp/client/request_timeouts.rb +74 -0
- data/lib/action_mcp/client/resources.rb +23 -11
- data/lib/action_mcp/client/server.rb +3 -3
- data/lib/action_mcp/client/toolbox.rb +12 -54
- data/lib/action_mcp/client/tools.rb +9 -4
- data/lib/action_mcp/configuration.rb +134 -24
- data/lib/action_mcp/engine.rb +6 -0
- data/lib/action_mcp/json_rpc_handler_base.rb +1 -0
- data/lib/action_mcp/registry_base.rb +3 -1
- data/lib/action_mcp/server/capabilities.rb +1 -1
- data/lib/action_mcp/server/json_rpc_handler.rb +1 -1
- data/lib/action_mcp/server/messaging.rb +32 -9
- data/lib/action_mcp/version.rb +1 -1
- data/lib/generators/action_mcp/install/install_generator.rb +4 -0
- data/lib/generators/action_mcp/install/templates/mcp.yml +11 -0
- data/lib/tasks/action_mcp_tasks.rake +77 -6
- metadata +6 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 843ee149da81be07d55d13eb26fad244a21232a0d4f795c98429d556c67c2736
|
4
|
+
data.tar.gz: fbf8aa774b9afe3e72cc325a93d054f98950163e14bd618d851c10d70bf823dc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
-
|
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
|
-
|
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
|
-
|
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 =>
|
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 #
|
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
|
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
|
-
#
|
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
|
27
|
+
# Create SSE instance
|
21
28
|
sse = SSE.new(response.stream)
|
22
29
|
|
23
|
-
# Start
|
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
|
-
|
26
|
-
|
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
|
-
|
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
|
-
|
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:
|
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
|
-
|
57
|
-
|
58
|
-
|
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
|
-
|
132
|
+
mcp_session.id
|
81
133
|
end
|
82
134
|
|
83
135
|
def cache_key
|
84
|
-
|
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
|
-
|
105
|
-
@subscription_active
|
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
|
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
|
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
|
-
#
|
125
|
-
|
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
|
-
|
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
|
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 :
|
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
|
-
#
|
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
|
-
|
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
|
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
|
-
|
112
|
-
|
113
|
-
|
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
|
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
|
-
|
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
|
-
|
111
|
-
|
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
@@ -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
|
-
|
150
|
+
@server = if server.is_a?(Client::Server)
|
151
|
+
server
|
151
152
|
else
|
152
|
-
|
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
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
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
|