actionmcp 0.19.1 → 0.22.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/app/controllers/action_mcp/messages_controller.rb +2 -2
- data/app/models/action_mcp/session/message.rb +12 -1
- data/app/models/action_mcp/session.rb +8 -4
- data/lib/action_mcp/base_response.rb +86 -0
- data/lib/action_mcp/capability.rb +2 -3
- data/lib/action_mcp/client/base.rb +222 -0
- data/lib/action_mcp/client/blueprint.rb +227 -0
- data/lib/action_mcp/client/catalog.rb +226 -0
- data/lib/action_mcp/client/json_rpc_handler.rb +109 -0
- data/lib/action_mcp/client/logging.rb +20 -0
- data/lib/action_mcp/{transport → client}/messaging.rb +1 -1
- data/lib/action_mcp/client/prompt_book.rb +183 -0
- data/lib/action_mcp/client/prompts.rb +33 -0
- data/lib/action_mcp/client/resources.rb +70 -0
- data/lib/action_mcp/client/roots.rb +13 -0
- data/lib/action_mcp/client/server.rb +60 -0
- data/lib/action_mcp/{transport → client}/sse_client.rb +70 -111
- data/lib/action_mcp/{transport → client}/stdio_client.rb +38 -38
- data/lib/action_mcp/client/toolbox.rb +236 -0
- data/lib/action_mcp/client/tools.rb +33 -0
- data/lib/action_mcp/client.rb +20 -231
- data/lib/action_mcp/engine.rb +1 -3
- data/lib/action_mcp/instrumentation/controller_runtime.rb +1 -1
- data/lib/action_mcp/instrumentation/instrumentation.rb +2 -0
- data/lib/action_mcp/instrumentation/resource_instrumentation.rb +1 -0
- data/lib/action_mcp/json_rpc_handler_base.rb +106 -0
- data/lib/action_mcp/log_subscriber.rb +2 -0
- data/lib/action_mcp/logging.rb +1 -1
- data/lib/action_mcp/prompt.rb +4 -3
- data/lib/action_mcp/prompt_response.rb +14 -58
- data/lib/action_mcp/{transport → server}/capabilities.rb +2 -2
- data/lib/action_mcp/server/json_rpc_handler.rb +121 -0
- data/lib/action_mcp/server/messaging.rb +28 -0
- data/lib/action_mcp/{transport → server}/notifications.rb +1 -1
- data/lib/action_mcp/{transport → server}/prompts.rb +1 -1
- data/lib/action_mcp/{transport → server}/resources.rb +1 -18
- data/lib/action_mcp/{transport → server}/roots.rb +1 -1
- data/lib/action_mcp/{transport → server}/sampling.rb +1 -1
- data/lib/action_mcp/server/sampling_request.rb +115 -0
- data/lib/action_mcp/{transport → server}/tools.rb +1 -1
- data/lib/action_mcp/server/transport_handler.rb +41 -0
- data/lib/action_mcp/tool_response.rb +14 -59
- data/lib/action_mcp/uri_ambiguity_checker.rb +6 -10
- data/lib/action_mcp/version.rb +1 -1
- data/lib/action_mcp.rb +2 -1
- metadata +30 -33
- data/lib/action_mcp/base_json_rpc_handler.rb +0 -97
- data/lib/action_mcp/client_json_rpc_handler.rb +0 -69
- data/lib/action_mcp/json_rpc_handler.rb +0 -229
- data/lib/action_mcp/sampling_request.rb +0 -113
- data/lib/action_mcp/server_json_rpc_handler.rb +0 -90
- data/lib/action_mcp/transport/transport_base.rb +0 -126
- data/lib/action_mcp/transport_handler.rb +0 -39
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
module Client
|
5
|
+
module Tools
|
6
|
+
# List all available tools from the server
|
7
|
+
# @return [Array<Hash>] List of available tools with their metadata
|
8
|
+
def list_tools
|
9
|
+
request_id = SecureRandom.uuid_v7
|
10
|
+
|
11
|
+
# Send request
|
12
|
+
send_jsonrpc_request("tools/list", id: request_id)
|
13
|
+
end
|
14
|
+
|
15
|
+
# Call a specific tool on the server
|
16
|
+
# @param name [String] Name of the tool to call
|
17
|
+
# @param arguments [Hash] Arguments to pass to the tool
|
18
|
+
# @return [Hash] The result of the tool execution
|
19
|
+
def call_tool(name, arguments)
|
20
|
+
request_id = SecureRandom.uuid_v7
|
21
|
+
|
22
|
+
# Send request
|
23
|
+
send_jsonrpc_request("tools/call",
|
24
|
+
params: {
|
25
|
+
name: name,
|
26
|
+
arguments: arguments
|
27
|
+
},
|
28
|
+
id: request_id
|
29
|
+
)
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
data/lib/action_mcp/client.rb
CHANGED
@@ -1,243 +1,32 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
-
#
|
5
|
-
#
|
6
|
-
# @param
|
7
|
-
# @
|
8
|
-
|
4
|
+
# Creates a client appropriate for the given endpoint.
|
5
|
+
#
|
6
|
+
# @param endpoint [String] The endpoint to connect to (URL or command).
|
7
|
+
# @param logger [Logger] The logger to use. Default is Logger.new($stdout).
|
8
|
+
# @param options [Hash] Additional options to pass to the client constructor.
|
9
|
+
#
|
10
|
+
# @return [Client::SSEClient, Client::StdioClient] An instance of either SSEClient or StdioClient
|
11
|
+
# depending on the format of the endpoint.
|
12
|
+
#
|
13
|
+
# @example
|
14
|
+
# client = ActionMCP.create_client("http://127.0.0.1:3001/action_mcp")
|
15
|
+
# client.connect
|
16
|
+
#
|
17
|
+
# @example
|
18
|
+
# client = ActionMCP.create_client("some_command")
|
19
|
+
# client.execute
|
20
|
+
def self.create_client(endpoint, logger: Logger.new($stdout), **options)
|
9
21
|
if endpoint =~ %r{\Ahttps?://}
|
10
22
|
logger.info("Creating SSE client for endpoint: #{endpoint}")
|
11
|
-
SSEClient.new(endpoint, logger: logger)
|
23
|
+
Client::SSEClient.new(endpoint, logger: logger, **options)
|
12
24
|
else
|
13
25
|
logger.info("Creating STDIO client for command: #{endpoint}")
|
14
|
-
StdioClient.new(endpoint, logger: logger)
|
26
|
+
Client::StdioClient.new(endpoint, logger: logger, **options)
|
15
27
|
end
|
16
28
|
end
|
17
29
|
|
18
|
-
|
19
|
-
class Client
|
20
|
-
attr_reader :logger, :capabilities, :type, :connection_error
|
21
|
-
|
22
|
-
def initialize(logger: Logger.new($stdout))
|
23
|
-
@logger = logger
|
24
|
-
@connected = false
|
25
|
-
@initialize_request_id = SecureRandom.uuid_v7
|
26
|
-
@server_capabilities = nil
|
27
|
-
@message_callback = nil
|
28
|
-
@error_callback = nil
|
29
|
-
@connection_error = nil
|
30
|
-
end
|
31
|
-
|
32
|
-
def connect
|
33
|
-
return true if @connected
|
34
|
-
|
35
|
-
begin
|
36
|
-
logger.info("Connecting to MCP server...")
|
37
|
-
@connection_error = nil
|
38
|
-
|
39
|
-
# Start transport with proper error handling
|
40
|
-
success = start_transport
|
41
|
-
|
42
|
-
unless success
|
43
|
-
logger.error("Failed to establish connection to MCP server")
|
44
|
-
return false
|
45
|
-
end
|
46
|
-
|
47
|
-
@connected = true
|
48
|
-
logger.info("Connected to MCP server")
|
49
|
-
true
|
50
|
-
rescue StandardError => e
|
51
|
-
@connection_error = e.message
|
52
|
-
logger.error("Failed to connect to MCP server: #{e.message}")
|
53
|
-
false
|
54
|
-
end
|
55
|
-
end
|
56
|
-
|
57
|
-
# Disconnect from the MCP server
|
58
|
-
# @return [Boolean] true if disconnection was successful
|
59
|
-
def disconnect
|
60
|
-
return true unless @connected
|
61
|
-
|
62
|
-
begin
|
63
|
-
stop_transport
|
64
|
-
@connected = false
|
65
|
-
logger.info("Disconnected from MCP server")
|
66
|
-
true
|
67
|
-
rescue StandardError => e
|
68
|
-
logger.error("Error disconnecting from MCP server: #{e.message}")
|
69
|
-
false
|
70
|
-
end
|
71
|
-
end
|
72
|
-
|
73
|
-
# Send a request to the MCP server
|
74
|
-
# @param payload [Hash, String] The request payload
|
75
|
-
# @return [Boolean] true if the request was sent successfully
|
76
|
-
def send_request(payload)
|
77
|
-
unless @connected
|
78
|
-
logger.error("Cannot send request - not connected")
|
79
|
-
return false
|
80
|
-
end
|
81
|
-
|
82
|
-
begin
|
83
|
-
json = prepare_payload(payload)
|
84
|
-
send_message(json)
|
85
|
-
true
|
86
|
-
rescue StandardError => e
|
87
|
-
logger.error("Failed to send request: #{e.message}")
|
88
|
-
false
|
89
|
-
end
|
90
|
-
end
|
91
|
-
|
92
|
-
# Check if the client is ready to send requests
|
93
|
-
# @return [Boolean] true if the client is connected and ready
|
94
|
-
def ready?
|
95
|
-
@connected && transport_ready?
|
96
|
-
end
|
97
|
-
|
98
|
-
# Set a callback for incoming messages
|
99
|
-
# @yield [message] Called when a message is received
|
100
|
-
# @yieldparam message The received message
|
101
|
-
def on_message(&block)
|
102
|
-
@message_callback = block
|
103
|
-
end
|
104
|
-
|
105
|
-
# Set a callback for errors
|
106
|
-
# @yield [error] Called when an error occurs
|
107
|
-
# @yieldparam error The error that occurred
|
108
|
-
def on_error(&block)
|
109
|
-
@error_callback = block
|
110
|
-
end
|
111
|
-
|
112
|
-
# Get the server capabilities
|
113
|
-
# @return [Hash, nil] The server capabilities, or nil if not connected
|
114
|
-
attr_reader :server_capabilities
|
115
|
-
|
116
|
-
protected
|
117
|
-
|
118
|
-
# Start the transport - implemented by subclasses
|
119
|
-
def start_transport
|
120
|
-
raise NotImplementedError, "Subclasses must implement start_transport"
|
121
|
-
end
|
122
|
-
|
123
|
-
# Stop the transport
|
124
|
-
def stop_transport
|
125
|
-
@transport.stop
|
126
|
-
end
|
127
|
-
|
128
|
-
# Send a message through the transport
|
129
|
-
def send_message(json)
|
130
|
-
@transport.send_message(json)
|
131
|
-
end
|
132
|
-
|
133
|
-
# Check if the transport is ready
|
134
|
-
def transport_ready?
|
135
|
-
@transport.ready?
|
136
|
-
end
|
137
|
-
|
138
|
-
private
|
139
|
-
|
140
|
-
# Prepare a payload for sending
|
141
|
-
# @param payload [Hash, String] The payload to prepare
|
142
|
-
# @return [String] The JSON-encoded payload
|
143
|
-
def prepare_payload(payload)
|
144
|
-
case payload
|
145
|
-
when String
|
146
|
-
# Assume it's already JSON
|
147
|
-
payload
|
148
|
-
else
|
149
|
-
# Try to convert to JSON
|
150
|
-
MultiJson.dump(payload)
|
151
|
-
end
|
152
|
-
end
|
153
|
-
end
|
154
|
-
|
155
|
-
# MCP client using Server-Sent Events (SSE) transport
|
156
|
-
class SSEClient < Client
|
157
|
-
# Initialize an SSE client
|
158
|
-
# @param endpoint [String] The SSE endpoint URL
|
159
|
-
# @param logger [Logger] The logger to use
|
160
|
-
def initialize(endpoint, logger: Logger.new($stdout))
|
161
|
-
super(logger: logger)
|
162
|
-
@endpoint = endpoint
|
163
|
-
@transport = Transport::SSEClient.new(endpoint, logger: logger)
|
164
|
-
@type = :sse
|
165
|
-
|
166
|
-
# Set up callbacks after transport is initialized
|
167
|
-
setup_callbacks
|
168
|
-
end
|
169
|
-
|
170
|
-
protected
|
171
|
-
|
172
|
-
def start_transport
|
173
|
-
@transport.start(@initialize_request_id)
|
174
|
-
true
|
175
|
-
rescue Transport::SSEClient::ConnectionError => e
|
176
|
-
@connection_error = e.message
|
177
|
-
@error_callback&.call(e)
|
178
|
-
false
|
179
|
-
rescue StandardError => e
|
180
|
-
@connection_error = e.message
|
181
|
-
@error_callback&.call(e)
|
182
|
-
false
|
183
|
-
end
|
184
|
-
|
185
|
-
private
|
186
|
-
|
187
|
-
def setup_callbacks
|
188
|
-
@transport.on_message do |message|
|
189
|
-
# Check if this is a response to our initialize request
|
190
|
-
puts @initialize_request_id
|
191
|
-
if message&.id == @initialize_request_id
|
192
|
-
@transport.handle_initialize_response(message)
|
193
|
-
else
|
194
|
-
puts "\e[32mCalling message callback\e[0m"
|
195
|
-
@message_callback&.call(message)
|
196
|
-
end
|
197
|
-
end
|
198
|
-
|
199
|
-
@transport.on_error do |error|
|
200
|
-
@error_callback&.call(error)
|
201
|
-
end
|
202
|
-
end
|
203
|
-
end
|
204
|
-
|
205
|
-
# MCP client using Standard I/O (STDIO) transport
|
206
|
-
class StdioClient < Client
|
207
|
-
# Initialize a STDIO client
|
208
|
-
# @param command [String] The command to execute
|
209
|
-
# @param logger [Logger] The logger to use
|
210
|
-
def initialize(command, logger: Logger.new($stdout))
|
211
|
-
super(logger: logger)
|
212
|
-
@command = command
|
213
|
-
@transport = Transport::StdioClient.new(command, logger: logger)
|
214
|
-
@type = :stdio
|
215
|
-
|
216
|
-
# Set up callbacks after transport is initialized
|
217
|
-
setup_callbacks
|
218
|
-
end
|
219
|
-
|
220
|
-
protected
|
221
|
-
|
222
|
-
def start_transport
|
223
|
-
@transport.start
|
224
|
-
# For STDIO, we'll send the capabilities from the connect method
|
225
|
-
# after this method completes and @connected is set to true
|
226
|
-
end
|
227
|
-
|
228
|
-
private
|
229
|
-
|
230
|
-
def setup_callbacks
|
231
|
-
@transport.on_message do |message|
|
232
|
-
# Check if this is a response to our initialize request
|
233
|
-
@transport.handle_initialize_response(message) if message&.id && message.id == @initialize_request_id
|
234
|
-
|
235
|
-
@message_callback&.call(message)
|
236
|
-
end
|
237
|
-
|
238
|
-
@transport.on_error do |error|
|
239
|
-
@error_callback&.call(error)
|
240
|
-
end
|
241
|
-
end
|
30
|
+
module Client
|
242
31
|
end
|
243
32
|
end
|
data/lib/action_mcp/engine.rb
CHANGED
@@ -28,7 +28,7 @@ module ActionMCP
|
|
28
28
|
def log_process_action(payload)
|
29
29
|
messages = super
|
30
30
|
mcp_runtime = payload[:mcp_runtime]
|
31
|
-
messages << ("
|
31
|
+
messages << (format("MCP: %.1fms", mcp_runtime.to_f)) if mcp_runtime
|
32
32
|
messages
|
33
33
|
end
|
34
34
|
end
|
@@ -0,0 +1,106 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
class JsonRpcHandlerBase
|
5
|
+
delegate :initialize!, :initialized?, to: :transport
|
6
|
+
delegate :write, :read, to: :transport
|
7
|
+
attr_reader :transport
|
8
|
+
|
9
|
+
# @param transport [ActionMCP::TransportHandler]
|
10
|
+
def initialize(transport)
|
11
|
+
@transport = transport
|
12
|
+
end
|
13
|
+
|
14
|
+
# Process a single line of input.
|
15
|
+
# @param line [String, Hash]
|
16
|
+
def call(line)
|
17
|
+
request = if line.is_a?(String)
|
18
|
+
line.strip!
|
19
|
+
return if line.empty?
|
20
|
+
|
21
|
+
begin
|
22
|
+
MultiJson.load(line)
|
23
|
+
rescue MultiJson::ParseError => e
|
24
|
+
Rails.logger.error("Failed to parse JSON: #{e.message}")
|
25
|
+
return
|
26
|
+
end
|
27
|
+
else
|
28
|
+
line
|
29
|
+
end
|
30
|
+
process_request(request)
|
31
|
+
end
|
32
|
+
|
33
|
+
protected
|
34
|
+
|
35
|
+
# Validate if the request follows JSON-RPC 2.0 specification
|
36
|
+
# @param request [Hash]
|
37
|
+
# @return [Boolean]
|
38
|
+
def valid_request?(request)
|
39
|
+
if request["jsonrpc"] != "2.0"
|
40
|
+
puts "Invalid request: #{request}"
|
41
|
+
return false
|
42
|
+
end
|
43
|
+
true
|
44
|
+
end
|
45
|
+
|
46
|
+
# Handle common methods for both client and server
|
47
|
+
# @param rpc_method [String]
|
48
|
+
# @param id [String, Integer]
|
49
|
+
# @param params [Hash]
|
50
|
+
# @return [Boolean] true if handled, false otherwise
|
51
|
+
def handle_common_methods(rpc_method, id, params)
|
52
|
+
case rpc_method
|
53
|
+
when "ping"
|
54
|
+
transport.send_pong(id)
|
55
|
+
true
|
56
|
+
when %r{^notifications/}
|
57
|
+
puts "\e[31mProcessing notifications\e[0m"
|
58
|
+
process_notifications(rpc_method, params)
|
59
|
+
true
|
60
|
+
else
|
61
|
+
false
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
# Method to be overridden by subclasses to handle specific RPC methods
|
66
|
+
# @param rpc_method [String]
|
67
|
+
# @param id [String, Integer]
|
68
|
+
# @param params [Hash]
|
69
|
+
def handle_method(rpc_method, id, params)
|
70
|
+
raise NotImplementedError, "Subclasses must implement handle_method"
|
71
|
+
end
|
72
|
+
|
73
|
+
private
|
74
|
+
|
75
|
+
# @param request [Hash]
|
76
|
+
def process_request(request)
|
77
|
+
return unless valid_request?(request)
|
78
|
+
read(request)
|
79
|
+
|
80
|
+
id = request["id"]
|
81
|
+
|
82
|
+
unless (rpc_method = request["method"])
|
83
|
+
# this is a response or a bobo
|
84
|
+
return process_response(id, request["result"]) if request["result"]
|
85
|
+
return process_error(id, request["error"]) if request["error"]
|
86
|
+
end
|
87
|
+
|
88
|
+
params = request["params"]
|
89
|
+
# Try to handle common methods first
|
90
|
+
return if handle_common_methods(rpc_method, id, params)
|
91
|
+
|
92
|
+
# Delegate to subclass-specific handling
|
93
|
+
handle_method(rpc_method, id, params)
|
94
|
+
end
|
95
|
+
|
96
|
+
def process_notifications(rpc_method, params)
|
97
|
+
case rpc_method
|
98
|
+
when "notifications/cancelled" # [BOTH] Request cancellation
|
99
|
+
puts "\e[31m Request #{params['requestId']} cancelled: #{params['reason']}\e[0m"
|
100
|
+
# we don't need to do anything here
|
101
|
+
else
|
102
|
+
Rails.logger.warn("Unknown notifications method: #{rpc_method}")
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
106
|
+
end
|
data/lib/action_mcp/logging.rb
CHANGED
@@ -11,7 +11,7 @@ module ActionMCP
|
|
11
11
|
|
12
12
|
# Included hook to configure the logger.
|
13
13
|
included do
|
14
|
-
cattr_accessor :logger, default: ActiveSupport::TaggedLogging.new(ActiveSupport::Logger.new(
|
14
|
+
cattr_accessor :logger, default: ActiveSupport::TaggedLogging.new(ActiveSupport::Logger.new($stdout))
|
15
15
|
end
|
16
16
|
end
|
17
17
|
end
|
data/lib/action_mcp/prompt.rb
CHANGED
@@ -46,8 +46,9 @@ module ActionMCP
|
|
46
46
|
# @param required [Boolean] Whether the argument is required.
|
47
47
|
# @param default [Object] The default value of the argument.
|
48
48
|
# @param enum [Array<String>] The list of allowed values for the argument.
|
49
|
+
# @param type [Symbol] The type of the argument (e.g., :string, :integer, :boolean). Defaults to :string.
|
49
50
|
# @return [void]
|
50
|
-
def self.argument(arg_name, description: "", required: false, default: nil, enum: nil)
|
51
|
+
def self.argument(arg_name, description: "", required: false, default: nil, enum: nil, type: :string)
|
51
52
|
arg_def = {
|
52
53
|
name: arg_name.to_s,
|
53
54
|
description: description,
|
@@ -58,7 +59,7 @@ module ActionMCP
|
|
58
59
|
self._argument_definitions += [ arg_def ]
|
59
60
|
|
60
61
|
# Register the attribute so it's recognized by ActiveModel
|
61
|
-
attribute arg_name,
|
62
|
+
attribute arg_name, type, default: default
|
62
63
|
validates arg_name, presence: true if required
|
63
64
|
|
64
65
|
return unless enum.present?
|
@@ -81,7 +82,7 @@ module ActionMCP
|
|
81
82
|
{
|
82
83
|
name: prompt_name,
|
83
84
|
description: description.presence,
|
84
|
-
arguments: arguments.map { |arg| arg.slice(:name, :description, :required) }
|
85
|
+
arguments: arguments.map { |arg| arg.slice(:name, :description, :required, :type) }
|
85
86
|
}.compact
|
86
87
|
end
|
87
88
|
|
@@ -1,16 +1,15 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
-
class PromptResponse
|
5
|
-
|
6
|
-
attr_reader :messages, :is_error
|
4
|
+
class PromptResponse < BaseResponse
|
5
|
+
attr_reader :messages
|
7
6
|
|
8
7
|
# Delegate methods to the underlying messages array
|
9
8
|
delegate :empty?, :size, :each, :find, :map, to: :messages
|
10
9
|
|
11
10
|
def initialize
|
11
|
+
super
|
12
12
|
@messages = []
|
13
|
-
@is_error = false
|
14
13
|
end
|
15
14
|
|
16
15
|
# Add a message to the response
|
@@ -25,64 +24,21 @@ module ActionMCP
|
|
25
24
|
self
|
26
25
|
end
|
27
26
|
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
self
|
34
|
-
end
|
35
|
-
|
36
|
-
# Convert to hash format expected by MCP protocol
|
37
|
-
def to_h
|
38
|
-
if @is_error
|
39
|
-
JsonRpc::JsonRpcError.new(@symbol, message: @error_message, data: @error_data).to_h
|
40
|
-
else
|
41
|
-
{
|
42
|
-
messages: @messages
|
43
|
-
}
|
44
|
-
end
|
45
|
-
end
|
46
|
-
|
47
|
-
# Alias as_json to to_h for consistency
|
48
|
-
alias as_json to_h
|
49
|
-
|
50
|
-
# Handle to_json directly
|
51
|
-
def to_json(options = nil)
|
52
|
-
to_h.to_json(options)
|
53
|
-
end
|
54
|
-
|
55
|
-
# Compare with hash for easier testing
|
56
|
-
def ==(other)
|
57
|
-
case other
|
58
|
-
when Hash
|
59
|
-
# Convert both to normalized format for comparison
|
60
|
-
hash_self = to_h.deep_transform_keys { |key| key.to_s.underscore }
|
61
|
-
hash_other = other.deep_transform_keys { |key| key.to_s.underscore }
|
62
|
-
hash_self == hash_other
|
63
|
-
when PromptResponse
|
64
|
-
messages == other.messages
|
65
|
-
else
|
66
|
-
super
|
67
|
-
end
|
68
|
-
end
|
69
|
-
|
70
|
-
# Implement eql? for hash key comparison
|
71
|
-
def eql?(other)
|
72
|
-
self == other
|
73
|
-
end
|
74
|
-
|
75
|
-
# Implement hash method for hash key usage
|
76
|
-
def hash
|
77
|
-
[ messages ].hash
|
27
|
+
# Implementation of build_success_hash for PromptResponse
|
28
|
+
def build_success_hash
|
29
|
+
{
|
30
|
+
messages: @messages
|
31
|
+
}
|
78
32
|
end
|
79
33
|
|
80
|
-
|
81
|
-
|
34
|
+
# Implementation of compare_with_same_class for PromptResponse
|
35
|
+
def compare_with_same_class(other)
|
36
|
+
messages == other.messages
|
82
37
|
end
|
83
38
|
|
84
|
-
|
85
|
-
|
39
|
+
# Implementation of hash_components for PromptResponse
|
40
|
+
def hash_components
|
41
|
+
[ messages ]
|
86
42
|
end
|
87
43
|
|
88
44
|
# Pretty print for better debugging
|
@@ -1,10 +1,10 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
-
module
|
4
|
+
module Server
|
5
5
|
module Capabilities
|
6
6
|
def send_capabilities(request_id, params = {})
|
7
|
-
# TODO fix this if client send incorrect params
|
7
|
+
# TODO: fix this if client send incorrect params
|
8
8
|
# TODO refuse connection if protocol version is not supported
|
9
9
|
@protocol_version = params["protocolVersion"]
|
10
10
|
@client_info = params["clientInfo"]
|