actionmcp 0.20.0 → 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/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/{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/uri_ambiguity_checker.rb +6 -10
- data/lib/action_mcp/version.rb +1 -1
- data/lib/action_mcp.rb +2 -1
- metadata +29 -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
|
@@ -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"]
|
@@ -0,0 +1,121 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
module Server
|
5
|
+
class JsonRpcHandler < JsonRpcHandlerBase
|
6
|
+
protected
|
7
|
+
|
8
|
+
# Handle server-specific methods
|
9
|
+
# @param rpc_method [String]
|
10
|
+
# @param id [String, Integer]
|
11
|
+
# @param params [Hash]
|
12
|
+
def handle_method(rpc_method, id, params)
|
13
|
+
case rpc_method
|
14
|
+
when "initialize" # [SERVER] Client initializing the connection
|
15
|
+
transport.send_capabilities(id, params)
|
16
|
+
when %r{^prompts/} # Prompt-related requests
|
17
|
+
process_prompts(rpc_method, id, params)
|
18
|
+
when %r{^resources/} # Resource-related requests
|
19
|
+
process_resources(rpc_method, id, params)
|
20
|
+
when %r{^tools/} # Tool-related requests
|
21
|
+
process_tools(rpc_method, id, params)
|
22
|
+
when "completion/complete" # Completion requests
|
23
|
+
process_completion_complete(id, params)
|
24
|
+
else
|
25
|
+
puts "\e[31mUnknown client method: #{rpc_method}\e[0m"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# Server methods (client → server)
|
30
|
+
|
31
|
+
# @param id [String]
|
32
|
+
# @param params [Hash]
|
33
|
+
# @example {
|
34
|
+
# "ref": {
|
35
|
+
# "type": "ref/prompt",
|
36
|
+
# "name": "code_review"
|
37
|
+
# },
|
38
|
+
# "argument": {
|
39
|
+
# "name": "language",
|
40
|
+
# "value": "py"
|
41
|
+
# }
|
42
|
+
# }
|
43
|
+
# @return [Hash]
|
44
|
+
# @example {
|
45
|
+
# "completion": {
|
46
|
+
# "values": ["python", "pytorch", "pyside"],
|
47
|
+
# "total": 10,
|
48
|
+
# "hasMore": true
|
49
|
+
# }
|
50
|
+
# }
|
51
|
+
def process_completion_complete(id, params)
|
52
|
+
# TODO: Not Implemented, but to remove the error message in the inspector
|
53
|
+
transport.send_jsonrpc_response(id, result: { completion: { values: [], total: 0, hasMore: false } })
|
54
|
+
case params["ref"]["type"]
|
55
|
+
when "ref/prompt"
|
56
|
+
# TODO: Implement completion
|
57
|
+
when "ref/resource"
|
58
|
+
# TODO: Implement completion
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
# @param rpc_method [String]
|
63
|
+
# @param id [String]
|
64
|
+
# @param params [Hash]
|
65
|
+
def process_prompts(rpc_method, id, params)
|
66
|
+
case rpc_method
|
67
|
+
when "prompts/get" # Get specific prompt
|
68
|
+
transport.send_prompts_get(id, params["name"], params["arguments"])
|
69
|
+
when "prompts/list" # List available prompts
|
70
|
+
transport.send_prompts_list(id)
|
71
|
+
else
|
72
|
+
Rails.logger.warn("Unknown prompts method: #{rpc_method}")
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
# @param rpc_method [String]
|
77
|
+
# @param id [String]
|
78
|
+
# @param params [Hash]
|
79
|
+
def process_tools(rpc_method, id, params)
|
80
|
+
case rpc_method
|
81
|
+
when "tools/list" # List available tools
|
82
|
+
transport.send_tools_list(id)
|
83
|
+
when "tools/call" # Call a tool
|
84
|
+
transport.send_tools_call(id, params["name"], params["arguments"])
|
85
|
+
else
|
86
|
+
Rails.logger.warn("Unknown tools method: #{rpc_method}")
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
# @param rpc_method [String]
|
91
|
+
# @param id [String]
|
92
|
+
# @param params [Hash]
|
93
|
+
def process_resources(rpc_method, id, params)
|
94
|
+
case rpc_method
|
95
|
+
when "resources/list" # List available resources
|
96
|
+
transport.send_resources_list(id)
|
97
|
+
when "resources/templates/list" # List resource templates
|
98
|
+
transport.send_resource_templates_list(id)
|
99
|
+
when "resources/read" # Read resource content
|
100
|
+
transport.send_resource_read(id, params)
|
101
|
+
when "resources/subscribe" # Subscribe to resource updates
|
102
|
+
transport.send_resource_subscribe(id, params["uri"])
|
103
|
+
when "resources/unsubscribe" # Unsubscribe from resource updates
|
104
|
+
transport.send_resource_unsubscribe(id, params["uri"])
|
105
|
+
else
|
106
|
+
Rails.logger.warn("Unknown resources method: #{rpc_method}")
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
def process_notifications(rpc_method, params)
|
111
|
+
case rpc_method
|
112
|
+
when "notifications/initialized" # Client initialization complete
|
113
|
+
puts "\e[31mInitialized\e[0m"
|
114
|
+
transport.initialize!
|
115
|
+
else
|
116
|
+
super
|
117
|
+
end
|
118
|
+
end
|
119
|
+
end
|
120
|
+
end
|
121
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
module Server
|
5
|
+
module Messaging
|
6
|
+
def send_jsonrpc_request(method, params: nil, id: SecureRandom.uuid_v7)
|
7
|
+
request = JsonRpc::Request.new(id: id, method: method, params: params)
|
8
|
+
write_message(request)
|
9
|
+
end
|
10
|
+
|
11
|
+
def send_jsonrpc_response(request_id, result: nil, error: nil)
|
12
|
+
response = JsonRpc::Response.new(id: request_id, result: result, error: error)
|
13
|
+
write_message(response)
|
14
|
+
end
|
15
|
+
|
16
|
+
def send_jsonrpc_notification(method, params = nil)
|
17
|
+
notification = JsonRpc::Notification.new(method: method, params: params)
|
18
|
+
write_message(notification)
|
19
|
+
end
|
20
|
+
|
21
|
+
def send_jsonrpc_error(request_id, symbol, message, data = nil)
|
22
|
+
error = JsonRpc::JsonRpcError.new(symbol, message:, data:)
|
23
|
+
response = JsonRpc::Response.new(id: request_id, error:)
|
24
|
+
write_message(response)
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -1,7 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
-
module
|
4
|
+
module Server
|
5
5
|
module Resources
|
6
6
|
# Send list of available resources to the client
|
7
7
|
#
|
@@ -61,23 +61,6 @@ module ActionMCP
|
|
61
61
|
end
|
62
62
|
end
|
63
63
|
|
64
|
-
def send_resource_subscribe(id, uri)
|
65
|
-
session.resource_subscribe(uri)
|
66
|
-
send_jsonrpc_response(id, result: {})
|
67
|
-
end
|
68
|
-
|
69
|
-
def send_resource_unsubscribe(id, uri)
|
70
|
-
session.resource_unsubscribe(uri)
|
71
|
-
send_jsonrpc_response(id, result: {})
|
72
|
-
end
|
73
|
-
|
74
|
-
# Client logging
|
75
|
-
def set_client_logging_level(id, level)
|
76
|
-
# Store the client's preferred log level
|
77
|
-
@client_log_level = level
|
78
|
-
send_jsonrpc_response(id, result: {})
|
79
|
-
end
|
80
|
-
|
81
64
|
private
|
82
65
|
|
83
66
|
# Log all registered resource templates
|