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.
Files changed (50) 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/capability.rb +2 -3
  6. data/lib/action_mcp/client/base.rb +222 -0
  7. data/lib/action_mcp/client/blueprint.rb +227 -0
  8. data/lib/action_mcp/client/catalog.rb +226 -0
  9. data/lib/action_mcp/client/json_rpc_handler.rb +109 -0
  10. data/lib/action_mcp/client/logging.rb +20 -0
  11. data/lib/action_mcp/{transport → client}/messaging.rb +1 -1
  12. data/lib/action_mcp/client/prompt_book.rb +183 -0
  13. data/lib/action_mcp/client/prompts.rb +33 -0
  14. data/lib/action_mcp/client/resources.rb +70 -0
  15. data/lib/action_mcp/client/roots.rb +13 -0
  16. data/lib/action_mcp/client/server.rb +60 -0
  17. data/lib/action_mcp/{transport → client}/sse_client.rb +70 -111
  18. data/lib/action_mcp/{transport → client}/stdio_client.rb +38 -38
  19. data/lib/action_mcp/client/toolbox.rb +236 -0
  20. data/lib/action_mcp/client/tools.rb +33 -0
  21. data/lib/action_mcp/client.rb +20 -231
  22. data/lib/action_mcp/engine.rb +1 -3
  23. data/lib/action_mcp/instrumentation/controller_runtime.rb +1 -1
  24. data/lib/action_mcp/instrumentation/instrumentation.rb +2 -0
  25. data/lib/action_mcp/instrumentation/resource_instrumentation.rb +1 -0
  26. data/lib/action_mcp/json_rpc_handler_base.rb +106 -0
  27. data/lib/action_mcp/log_subscriber.rb +2 -0
  28. data/lib/action_mcp/logging.rb +1 -1
  29. data/lib/action_mcp/{transport → server}/capabilities.rb +2 -2
  30. data/lib/action_mcp/server/json_rpc_handler.rb +121 -0
  31. data/lib/action_mcp/server/messaging.rb +28 -0
  32. data/lib/action_mcp/{transport → server}/notifications.rb +1 -1
  33. data/lib/action_mcp/{transport → server}/prompts.rb +1 -1
  34. data/lib/action_mcp/{transport → server}/resources.rb +1 -18
  35. data/lib/action_mcp/{transport → server}/roots.rb +1 -1
  36. data/lib/action_mcp/{transport → server}/sampling.rb +1 -1
  37. data/lib/action_mcp/server/sampling_request.rb +115 -0
  38. data/lib/action_mcp/{transport → server}/tools.rb +1 -1
  39. data/lib/action_mcp/server/transport_handler.rb +41 -0
  40. data/lib/action_mcp/uri_ambiguity_checker.rb +6 -10
  41. data/lib/action_mcp/version.rb +1 -1
  42. data/lib/action_mcp.rb +2 -1
  43. metadata +29 -33
  44. data/lib/action_mcp/base_json_rpc_handler.rb +0 -97
  45. data/lib/action_mcp/client_json_rpc_handler.rb +0 -69
  46. data/lib/action_mcp/json_rpc_handler.rb +0 -229
  47. data/lib/action_mcp/sampling_request.rb +0 -113
  48. data/lib/action_mcp/server_json_rpc_handler.rb +0 -90
  49. data/lib/action_mcp/transport/transport_base.rb +0 -126
  50. 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
@@ -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"]
@@ -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 Transport
4
+ module Server
5
5
  module Notifications
6
6
  # Notify client that the resources list has changed
7
7
  def send_resources_list_changed_notification
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
- module Transport
4
+ module Server
5
5
  module Prompts
6
6
  def send_prompts_list(request_id)
7
7
  prompts = format_registry_items(PromptsRegistry.non_abstract)
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
- module Transport
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
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
- module Transport
4
+ module Server
5
5
  module Roots
6
6
  def send_roots_list(id)
7
7
  send_jsonrpc_response(id, result: { roots: [] })
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
- module Transport
4
+ module Server
5
5
  module Sampling
6
6
  # @param [String] id
7
7
  # @param [SamplingRequest] request