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.
Files changed (54) hide show
  1. checksums.yaml +4 -4
  2. data/app/controllers/action_mcp/messages_controller.rb +2 -2
  3. data/app/models/action_mcp/session/message.rb +12 -1
  4. data/app/models/action_mcp/session.rb +8 -4
  5. data/lib/action_mcp/base_response.rb +86 -0
  6. data/lib/action_mcp/capability.rb +2 -3
  7. data/lib/action_mcp/client/base.rb +222 -0
  8. data/lib/action_mcp/client/blueprint.rb +227 -0
  9. data/lib/action_mcp/client/catalog.rb +226 -0
  10. data/lib/action_mcp/client/json_rpc_handler.rb +109 -0
  11. data/lib/action_mcp/client/logging.rb +20 -0
  12. data/lib/action_mcp/{transport → client}/messaging.rb +1 -1
  13. data/lib/action_mcp/client/prompt_book.rb +183 -0
  14. data/lib/action_mcp/client/prompts.rb +33 -0
  15. data/lib/action_mcp/client/resources.rb +70 -0
  16. data/lib/action_mcp/client/roots.rb +13 -0
  17. data/lib/action_mcp/client/server.rb +60 -0
  18. data/lib/action_mcp/{transport → client}/sse_client.rb +70 -111
  19. data/lib/action_mcp/{transport → client}/stdio_client.rb +38 -38
  20. data/lib/action_mcp/client/toolbox.rb +236 -0
  21. data/lib/action_mcp/client/tools.rb +33 -0
  22. data/lib/action_mcp/client.rb +20 -231
  23. data/lib/action_mcp/engine.rb +1 -3
  24. data/lib/action_mcp/instrumentation/controller_runtime.rb +1 -1
  25. data/lib/action_mcp/instrumentation/instrumentation.rb +2 -0
  26. data/lib/action_mcp/instrumentation/resource_instrumentation.rb +1 -0
  27. data/lib/action_mcp/json_rpc_handler_base.rb +106 -0
  28. data/lib/action_mcp/log_subscriber.rb +2 -0
  29. data/lib/action_mcp/logging.rb +1 -1
  30. data/lib/action_mcp/prompt.rb +4 -3
  31. data/lib/action_mcp/prompt_response.rb +14 -58
  32. data/lib/action_mcp/{transport → server}/capabilities.rb +2 -2
  33. data/lib/action_mcp/server/json_rpc_handler.rb +121 -0
  34. data/lib/action_mcp/server/messaging.rb +28 -0
  35. data/lib/action_mcp/{transport → server}/notifications.rb +1 -1
  36. data/lib/action_mcp/{transport → server}/prompts.rb +1 -1
  37. data/lib/action_mcp/{transport → server}/resources.rb +1 -18
  38. data/lib/action_mcp/{transport → server}/roots.rb +1 -1
  39. data/lib/action_mcp/{transport → server}/sampling.rb +1 -1
  40. data/lib/action_mcp/server/sampling_request.rb +115 -0
  41. data/lib/action_mcp/{transport → server}/tools.rb +1 -1
  42. data/lib/action_mcp/server/transport_handler.rb +41 -0
  43. data/lib/action_mcp/tool_response.rb +14 -59
  44. data/lib/action_mcp/uri_ambiguity_checker.rb +6 -10
  45. data/lib/action_mcp/version.rb +1 -1
  46. data/lib/action_mcp.rb +2 -1
  47. metadata +30 -33
  48. data/lib/action_mcp/base_json_rpc_handler.rb +0 -97
  49. data/lib/action_mcp/client_json_rpc_handler.rb +0 -69
  50. data/lib/action_mcp/json_rpc_handler.rb +0 -229
  51. data/lib/action_mcp/sampling_request.rb +0 -113
  52. data/lib/action_mcp/server_json_rpc_handler.rb +0 -90
  53. data/lib/action_mcp/transport/transport_base.rb +0 -126
  54. 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
@@ -1,243 +1,32 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
- # Create a client appropriate for the given endpoint
5
- # @param endpoint [String] The endpoint to connect to (URL or command)
6
- # @param logger [Logger] The logger to use
7
- # @return [Client] An SSEClient or StdioClient depending on the endpoint
8
- def self.create_client(endpoint, logger: Logger.new($stdout))
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
- # Base client class for MCP protocol
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
@@ -36,9 +36,7 @@ module ActionMCP
36
36
 
37
37
  # Initialize the ActionMCP logger.
38
38
  initializer "action_mcp.logger" do
39
- ActiveSupport.on_load(:action_mcp) do
40
- self.logger = ::Rails.logger
41
- end
39
+ ActionMCP.logger = ::Rails.logger
42
40
  end
43
41
  end
44
42
  end
@@ -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 << ("mcp: %.1fms" % mcp_runtime.to_f) if mcp_runtime
31
+ messages << (format("MCP: %.1fms", mcp_runtime.to_f)) if mcp_runtime
32
32
  messages
33
33
  end
34
34
  end
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
4
  module Instrumentation
3
5
  module Instrumentation # :nodoc:
@@ -12,6 +12,7 @@ module ActionMCP
12
12
  end
13
13
 
14
14
  private
15
+
15
16
  def instrument(operation, payload = {}, &block)
16
17
  payload[:resource_template] = self
17
18
  payload[:uri_template] = uri_template if respond_to?(:uri_template)
@@ -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
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
4
  class LogSubscriber < ActiveSupport::LogSubscriber
3
5
  def self.reset_runtime
@@ -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(STDOUT))
14
+ cattr_accessor :logger, default: ActiveSupport::TaggedLogging.new(ActiveSupport::Logger.new($stdout))
15
15
  end
16
16
  end
17
17
  end
@@ -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, :string, default: default
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
- include Enumerable
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
- def mark_as_error!(symbol = :invalid_request, message: nil, data: nil)
29
- @is_error = true
30
- @symbol = symbol
31
- @error_message = message
32
- @error_data = data
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
- def success?
81
- !is_error
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
- def error?
85
- is_error
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 Transport
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"]