actionmcp 0.7.0 → 0.7.2
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
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: bea2e23118051ba1a05eb00dfb44395841068fe6358b341e01a1951f7064a259
|
4
|
+
data.tar.gz: e1ff02d641a4684c0369fa252139edeb8bf1dae41193deeca122832bcb0ac318
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: '097b59b77ef2028887ef159f211a314a9a9fa83edb7ff26f50d62fc56a834bf5df96e7781c5cd715d511ee957736f242971f76baac3d8b1fffd213f6373afe5c'
|
7
|
+
data.tar.gz: 13582b839164819da648df7e319df9e8aac565f2792bee75c44d23d1fd7af531e01d6c7a8e82f63d46fc9cd68f7d42fe10622748202a6ca10dc99ff332d8d7a5
|
@@ -1,7 +1,6 @@
|
|
1
1
|
module ActionMCP
|
2
2
|
class SSEController < ApplicationController
|
3
3
|
HEARTBEAT_INTERVAL = 30 # TODO: The frequency of pings SHOULD be configurable
|
4
|
-
INITIALIZATION_TIMEOUT = 2
|
5
4
|
include ActionController::Live
|
6
5
|
|
7
6
|
# @route GET /sse (sse_out)
|
@@ -16,11 +15,13 @@ module ActionMCP
|
|
16
15
|
send_endpoint_event(sse_in_url)
|
17
16
|
|
18
17
|
begin
|
18
|
+
# Create SSE instance once outside the callback with retry parameter
|
19
|
+
sse = SSE.new(response.stream)
|
20
|
+
|
19
21
|
# Start listener and process messages via the transport
|
20
22
|
listener = SSEListener.new(mcp_session)
|
21
23
|
if listener.start do |message|
|
22
24
|
# Send with proper SSE formatting
|
23
|
-
sse = SSE.new(response.stream)
|
24
25
|
sse.write(message)
|
25
26
|
end
|
26
27
|
|
@@ -74,25 +75,22 @@ module ActionMCP
|
|
74
75
|
def initialize(session)
|
75
76
|
@session = session
|
76
77
|
@stopped = false
|
77
|
-
@subscription_active = false
|
78
78
|
end
|
79
79
|
|
80
80
|
# Start listening using ActionCable's adapter
|
81
81
|
def start(&callback)
|
82
82
|
Rails.logger.debug "Starting listener for channel: #{session_key}"
|
83
83
|
|
84
|
-
# Set up success callback
|
85
84
|
success_callback = -> {
|
86
|
-
|
85
|
+
puts "Successfully subscribed to channel: #{session_key}"
|
87
86
|
@subscription_active = true
|
88
87
|
}
|
89
88
|
|
90
|
-
# Set up message callback
|
89
|
+
# Set up message callback
|
91
90
|
message_callback = ->(raw_message) {
|
92
91
|
begin
|
93
92
|
# Try to parse the message if it's JSON
|
94
93
|
message = MultiJson.load(raw_message)
|
95
|
-
|
96
94
|
# Send the message to the callback
|
97
95
|
callback.call(message) if callback && !@stopped
|
98
96
|
rescue => e
|
@@ -102,7 +100,7 @@ module ActionMCP
|
|
102
100
|
}
|
103
101
|
|
104
102
|
# Subscribe using the ActionCable adapter
|
105
|
-
adapter.subscribe(session_key, message_callback, success_callback)
|
103
|
+
success = adapter.subscribe(session_key, message_callback, success_callback)
|
106
104
|
|
107
105
|
# Give some time for the subscription to be established
|
108
106
|
sleep 0.5
|
@@ -111,21 +109,10 @@ module ActionMCP
|
|
111
109
|
end
|
112
110
|
|
113
111
|
def stop
|
114
|
-
Rails.logger.debug "Stopping listener for: #{session_key}"
|
115
112
|
@stopped = true
|
116
113
|
|
117
|
-
|
118
|
-
|
119
|
-
# Create a dummy callback that matches the one we provided in start
|
120
|
-
@session.close!
|
121
|
-
Rails.logger.debug "Unsubscribed from: #{session_key}"
|
122
|
-
rescue => e
|
123
|
-
Rails.logger.error "Error unsubscribing from #{session_key}: #{e.message}"
|
124
|
-
end
|
125
|
-
end
|
126
|
-
|
127
|
-
def active?
|
128
|
-
@subscription_active
|
114
|
+
@session.close!
|
115
|
+
Rails.logger.debug "Unsubscribed from: #{session_key}"
|
129
116
|
end
|
130
117
|
end
|
131
118
|
end
|
data/lib/action_mcp/engine.rb
CHANGED
@@ -15,6 +15,10 @@ module ActionMCP
|
|
15
15
|
# Provide a configuration namespace for ActionMCP
|
16
16
|
config.action_mcp = ActionMCP.configuration
|
17
17
|
|
18
|
+
config.to_prepare do
|
19
|
+
ActionMCP::ResourceTemplate.registered_templates.clear
|
20
|
+
end
|
21
|
+
|
18
22
|
# Configure autoloading for the mcp/tools directory
|
19
23
|
initializer "action_mcp.autoloading", before: :set_autoload_paths do |app|
|
20
24
|
mcp_path = app.root.join("app/mcp")
|
@@ -179,6 +179,8 @@ module ActionMCP
|
|
179
179
|
ResourceTemplate.registered_templates.each do |registered_class|
|
180
180
|
next if registered_class == self || registered_class.abstract?
|
181
181
|
next unless registered_class.uri_template
|
182
|
+
# Ignore conflicts with resource templates that have the same name
|
183
|
+
next if registered_class.name == self.name
|
182
184
|
|
183
185
|
existing_template_data = parse_uri_template(registered_class.uri_template)
|
184
186
|
|
data/lib/action_mcp/tool.rb
CHANGED
@@ -74,7 +74,7 @@ module ActionMCP
|
|
74
74
|
|
75
75
|
return unless %w[number integer].include?(type)
|
76
76
|
|
77
|
-
validates prop_name, numericality: true
|
77
|
+
validates prop_name, numericality: true, allow_nil: true
|
78
78
|
end
|
79
79
|
|
80
80
|
# --------------------------------------------------------------------------
|
@@ -56,7 +56,7 @@ module ActionMCP
|
|
56
56
|
content = resource.map(&:to_h)
|
57
57
|
send_jsonrpc_response(id, result: { contents: content })
|
58
58
|
else
|
59
|
-
send_jsonrpc_error(id, :
|
59
|
+
send_jsonrpc_error(id, :invalid_params, "Resource not found")
|
60
60
|
end
|
61
61
|
else
|
62
62
|
send_jsonrpc_error(id, :invalid_params, "Invalid resource URI")
|
data/lib/action_mcp/version.rb
CHANGED