actionmcp 0.2.0 → 0.2.3

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 (49) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +133 -30
  3. data/Rakefile +0 -2
  4. data/exe/actionmcp_cli +221 -0
  5. data/lib/action_mcp/capability.rb +52 -0
  6. data/lib/action_mcp/client.rb +243 -1
  7. data/lib/action_mcp/configuration.rb +50 -1
  8. data/lib/action_mcp/content/audio.rb +9 -0
  9. data/lib/action_mcp/content/image.rb +9 -0
  10. data/lib/action_mcp/content/resource.rb +13 -0
  11. data/lib/action_mcp/content/text.rb +7 -0
  12. data/lib/action_mcp/content.rb +11 -6
  13. data/lib/action_mcp/engine.rb +34 -0
  14. data/lib/action_mcp/gem_version.rb +2 -2
  15. data/lib/action_mcp/integer_array.rb +6 -0
  16. data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
  17. data/lib/action_mcp/json_rpc/notification.rb +8 -0
  18. data/lib/action_mcp/json_rpc/request.rb +14 -0
  19. data/lib/action_mcp/json_rpc/response.rb +32 -1
  20. data/lib/action_mcp/json_rpc.rb +1 -6
  21. data/lib/action_mcp/json_rpc_handler.rb +106 -0
  22. data/lib/action_mcp/logging.rb +19 -0
  23. data/lib/action_mcp/prompt.rb +30 -46
  24. data/lib/action_mcp/prompts_registry.rb +13 -1
  25. data/lib/action_mcp/registry_base.rb +47 -28
  26. data/lib/action_mcp/renderable.rb +26 -0
  27. data/lib/action_mcp/resource.rb +3 -1
  28. data/lib/action_mcp/server.rb +4 -1
  29. data/lib/action_mcp/string_array.rb +5 -0
  30. data/lib/action_mcp/tool.rb +16 -53
  31. data/lib/action_mcp/tools_registry.rb +14 -1
  32. data/lib/action_mcp/transport/capabilities.rb +21 -0
  33. data/lib/action_mcp/transport/messaging.rb +20 -0
  34. data/lib/action_mcp/transport/prompts.rb +19 -0
  35. data/lib/action_mcp/transport/sse_client.rb +309 -0
  36. data/lib/action_mcp/transport/stdio_client.rb +117 -0
  37. data/lib/action_mcp/transport/tools.rb +20 -0
  38. data/lib/action_mcp/transport/transport_base.rb +125 -0
  39. data/lib/action_mcp/transport.rb +1 -235
  40. data/lib/action_mcp/transport_handler.rb +54 -0
  41. data/lib/action_mcp/version.rb +4 -5
  42. data/lib/action_mcp.rb +36 -33
  43. data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
  44. data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
  45. data/lib/tasks/action_mcp_tasks.rake +28 -5
  46. metadata +62 -9
  47. data/exe/action_mcp_stdio +0 -0
  48. data/lib/action_mcp/railtie.rb +0 -27
  49. data/lib/action_mcp/resources_bank.rb +0 -94
@@ -0,0 +1,125 @@
1
+ module ActionMCP
2
+ module Transport
3
+ class TransportBase
4
+ attr_reader :logger, :client_capabilities, :server_capabilities
5
+
6
+ def initialize(logger: Logger.new(STDOUT))
7
+ @logger = logger
8
+ @on_message = nil
9
+ @on_error = nil
10
+ @client_capabilities = default_capabilities
11
+ @server_capabilities = nil
12
+ @initialize_request_id = SecureRandom.hex(6)
13
+ @initialization_sent = false
14
+ end
15
+
16
+ def on_message(&block)
17
+ @on_message = block
18
+ end
19
+
20
+ def on_error(&block)
21
+ @on_error = block
22
+ end
23
+
24
+ def send_initial_capabilities
25
+ return if @initialization_sent
26
+
27
+ log_info("Sending client capabilities: #{@client_capabilities}")
28
+
29
+ request = JsonRpc::Request.new(
30
+ id: @initialize_request_id,
31
+ method: "initialize",
32
+ params: {
33
+ protocolVersion: PROTOCOL_VERSION,
34
+ capabilities: @client_capabilities,
35
+ clientInfo: {
36
+ name: user_agent,
37
+ version: ActionMCP.gem_version.to_s
38
+ }
39
+ }
40
+ )
41
+ @initialization_sent = true
42
+ send_message(request.to_json)
43
+ end
44
+
45
+ def handle_initialize_response(response)
46
+ unless @server_capabilities
47
+
48
+ if response.result
49
+ @server_capabilities = response.result["capabilities"]
50
+ send_initialized_notification
51
+ else
52
+ log_error("Server initialization failed: #{response.error}")
53
+ end
54
+ end
55
+ end
56
+
57
+ protected
58
+
59
+ def handle_raw_message(raw)
60
+ # Debug - log all raw messages
61
+ log_debug("\e[31m<-- #{raw}\e[0m")
62
+
63
+ begin
64
+ msg_hash = MultiJson.load(raw)
65
+ response = nil
66
+
67
+ if msg_hash.key?("jsonrpc")
68
+ if msg_hash.key?("id")
69
+ response = JsonRpc::Response.new(**msg_hash.slice("id", "result", "error").symbolize_keys)
70
+ else
71
+ response = JsonRpc::Notification.new(**msg_hash.slice("method", "params").symbolize_keys)
72
+ end
73
+ end
74
+ # Check if this is a response to our initialize request
75
+ if response && @initialize_request_id && response.id == @initialize_request_id
76
+ handle_initialize_response(response)
77
+ else
78
+ @on_message&.call(response) if response
79
+ end
80
+ rescue MultiJson::ParseError => e
81
+ log_error("JSON parse error: #{e} (raw: #{raw})")
82
+ @on_error&.call(e) if @on_error
83
+ rescue StandardError => e
84
+ log_error("Error handling message: #{e} (raw: #{raw})")
85
+ @on_error&.call(e) if @on_error
86
+ end
87
+ end
88
+
89
+ # Send the initialized notification to the server
90
+ def send_initialized_notification
91
+ notification = JsonRpc::Notification.new(
92
+ method: "initialized"
93
+ )
94
+
95
+ logger.info("Sent initialized notification to server")
96
+ send_message(notification)
97
+ end
98
+
99
+ def default_capabilities
100
+ {
101
+ # Base client capabilities
102
+ # roots: {}, # Remove from now.
103
+ }
104
+ end
105
+
106
+ def log_debug(message)
107
+ @logger.debug("[#{log_prefix}] #{message}")
108
+ end
109
+
110
+ def log_info(message)
111
+ @logger.info("[#{log_prefix}] #{message}")
112
+ end
113
+
114
+ def log_error(message)
115
+ @logger.error("[#{log_prefix}] #{message}")
116
+ end
117
+
118
+ private
119
+
120
+ def log_prefix
121
+ self.class.name.split("::").last
122
+ end
123
+ end
124
+ end
125
+ end
@@ -1,238 +1,4 @@
1
- # frozen_string_literal: true
2
-
3
1
  module ActionMCP
4
- class Transport
5
- HEARTBEAT_INTERVAL = 15 # seconds
6
- attr_reader :initialized
7
-
8
- # Initializes a new Transport.
9
- #
10
- # @param output_io [IO] An IO-like object where events will be written.
11
- def initialize(output_io)
12
- # output_io can be any IO-like object where we write events.
13
- @output = output_io
14
- @output.sync = true
15
- @initialized = false
16
- @client_capabilities = {}
17
- @client_info = {}
18
- @protocol_version = ""
19
- end
20
-
21
- # Sends the capabilities JSON-RPC notification.
22
- #
23
- # @param request_id [String, Integer] The request identifier.
24
- def send_capabilities(request_id, params = {})
25
- @protocol_version = params["protocolVersion"]
26
- @client_info = params["clientInfo"]
27
- @client_capabilities = params["capabilities"]
28
- Rails.logger.info("Client capabilities stored: #{@client_capabilities}")
29
- capabilities = {}
30
-
31
- # Only include each capability if the corresponding registry is non-empty.
32
- capabilities[:tools] = { listChanged: ActionMCP.configuration.list_changed } if ToolsRegistry.available_tools.any?
33
- if PromptsRegistry.available_prompts.any?
34
- capabilities[:prompts] =
35
- { listChanged: ActionMCP.configuration.list_changed }
36
- end
37
- capabilities[:resources] = { subscribe: ActionMCP.configuration.list_changed } if ResourcesBank.all_resources.any?
38
- capabilities[:logging] = {} if ActionMCP.configuration.logging_enabled
39
-
40
- payload = {
41
- protocolVersion: "2024-11-05",
42
- capabilities: capabilities,
43
- serverInfo: {
44
- name: ActionMCP.configuration.name,
45
- version: ActionMCP.configuration.version
46
- }
47
- }
48
- send_jsonrpc_response(request_id, result: payload)
49
- end
50
-
51
- def initialized!
52
- @initialized = true
53
- Rails.logger.info("Transport initialized.")
54
- end
55
-
56
- # Sends the resources list JSON-RPC response.
57
- #
58
- # @param request_id [String, Integer] The request identifier.
59
- def send_resources_list(request_id)
60
- resources = ResourcesBank.all_resources # fetch all resources
61
- result_data = { "resources" => resources }
62
- send_jsonrpc_response(request_id, result: result_data)
63
- Rails.logger.info("resources/list: Returned #{resources.size} resources.")
64
- rescue StandardError => e
65
- Rails.logger.error("resources/list failed: #{e.message}")
66
- error_obj = JsonRpc::JsonRpcError.new(
67
- :internal_error,
68
- message: "Failed to list resources: #{e.message}"
69
- ).as_json
70
- send_jsonrpc_response(request_id, error: error_obj)
71
- end
72
-
73
- # Sends the resource templates list JSON-RPC response.
74
- #
75
- # @param request_id [String, Integer] The request identifier.
76
- def send_resource_templates_list(request_id)
77
- templates = ResourcesBank.all_templates # get all resource templates
78
- result_data = { "resourceTemplates" => templates }
79
- send_jsonrpc_response(request_id, result: result_data)
80
- Rails.logger.info("resources/templates/list: Returned #{templates.size} resource templates.")
81
- rescue StandardError => e
82
- Rails.logger.error("resources/templates/list failed: #{e.message}")
83
- error_obj = JsonRpc::JsonRpcError.new(
84
- :internal_error,
85
- message: "Failed to list resource templates: #{e.message}"
86
- ).as_json
87
- send_jsonrpc_response(request_id, error: error_obj)
88
- end
89
-
90
- # Sends the resource read JSON-RPC response.
91
- #
92
- # @param request_id [String, Integer] The request identifier.
93
- # @param params [Hash] The parameters including the 'uri' for the resource.
94
- def send_resource_read(request_id, params)
95
- uri = params&.fetch("uri", nil)
96
- if uri.nil? || uri.empty?
97
- Rails.logger.error("resources/read: 'uri' parameter is missing")
98
- error_obj = JsonRpc::JsonRpcError.new(
99
- :invalid_params,
100
- message: "Missing 'uri' parameter for resources/read"
101
- ).as_json
102
- return send_jsonrpc_response(request_id, error: error_obj)
103
- end
104
-
105
- begin
106
- content = ResourcesBank.read(uri) # Expecting an instance of an ActionMCP::Content subclass
107
- if content.nil?
108
- Rails.logger.error("resources/read: Resource not found for URI #{uri}")
109
- error_obj = JsonRpc::JsonRpcError.new(
110
- :invalid_params,
111
- message: "Resource not found: #{uri}"
112
- ).as_json
113
- return send_jsonrpc_response(request_id, error: error_obj)
114
- end
115
-
116
- # Use the content object's `to_h` to build the JSON-RPC result.
117
- result_data = { "contents" => [ content.to_h ] }
118
- send_jsonrpc_response(request_id, result: result_data)
119
-
120
- log_msg = "resources/read: Successfully read content of #{uri}"
121
- log_msg += " (#{content.text.size} bytes)" if content.respond_to?(:text) && content.text
122
- Rails.logger.info(log_msg)
123
- rescue StandardError => e
124
- Rails.logger.error("resources/read: Error reading #{uri} - #{e.message}")
125
- error_obj = JsonRpc::JsonRpcError.new(
126
- :internal_error,
127
- message: "Failed to read resource: #{e.message}"
128
- ).as_json
129
- send_jsonrpc_response(request_id, error: error_obj)
130
- end
131
- end
132
-
133
- # Sends the tools list JSON-RPC notification.
134
- #
135
- # @param request_id [String, Integer] The request identifier.
136
- def send_tools_list(request_id)
137
- tools = format_registry_items(ToolsRegistry.available_tools)
138
- send_jsonrpc_response(request_id, result: { tools: tools })
139
- end
140
-
141
- # Sends a call to a tool.
142
- #
143
- # @param request_id [String, Integer] The request identifier.
144
- # @param tool_name [String] The name of the tool.
145
- # @param arguments [Hash] The arguments for the tool.
146
- # @param _meta [Hash] Additional metadata.
147
- def send_tools_call(request_id, tool_name, arguments, _meta = {})
148
- result = ToolsRegistry.tool_call(tool_name, arguments, _meta)
149
- send_jsonrpc_response(request_id, result:)
150
- rescue RegistryBase::NotFound
151
- send_jsonrpc_response(request_id, error: JsonRpc::JsonRpcError.new(:method_not_found,
152
- message: "Tool not found: #{tool_name}").as_json)
153
- end
154
-
155
- # Sends the prompts list JSON-RPC notification.
156
- #
157
- # @param request_id [String, Integer] The request identifier.
158
- def send_prompts_list(request_id)
159
- prompts = format_registry_items(PromptsRegistry.available_prompts)
160
- send_jsonrpc_response(request_id, result: { prompts: prompts })
161
- end
162
-
163
- def send_prompts_get(request_id, prompt_name, params)
164
- send_jsonrpc_response(request_id, result: PromptsRegistry.prompt_call(prompt_name.to_s, params))
165
- rescue RegistryBase::NotFound
166
- send_jsonrpc_response(request_id, error: JsonRpc::JsonRpcError.new(:method_not_found,
167
- message: "Prompt not found: #{prompt_name}").as_json)
168
- end
169
-
170
- # Sends the roots list JSON-RPC request.
171
- # TODO: test it
172
- def send_roots_list
173
- send_jsonrpc_request("roots/list")
174
- end
175
-
176
- # Sends a JSON-RPC pong response.
177
- # We don't actually to send any data back because the spec are not fun anymore.
178
- #
179
- # @param request_id [String, Integer] The request identifier.
180
- def send_pong(request_id)
181
- send_jsonrpc_response(request_id, result: {})
182
- end
183
-
184
- def send_ping
185
- send_jsonrpc_request("ping")
186
- end
187
-
188
- # Sends a JSON-RPC request.
189
- # @param method [String] The JSON-RPC method.
190
- # @param params [Hash] The parameters for the method.
191
- # @param id [String] The request identifier.
192
- def send_jsonrpc_request(method, params: nil, id: SecureRandom.uuid_v7)
193
- request = JsonRpc::Request.new(id: id, method: method, params: params)
194
- write_message(request.to_json)
195
- end
196
-
197
- # Sends a JSON-RPC response.
198
- #
199
- # @param request_id [String, Integer] The request identifier.
200
- # @param result [Object] The result data.
201
- # @param error [Object, nil] The error data, if any.
202
- def send_jsonrpc_response(request_id, result: nil, error: nil)
203
- response = JsonRpc::Response.new(id: request_id, result: result, error: error)
204
- write_message(response.to_json)
205
- end
206
-
207
- # Sends a generic JSON-RPC notification (no response expected).
208
- #
209
- # @param method [String] The JSON-RPC method.
210
- # @param params [Hash] The parameters for the method.
211
- def send_jsonrpc_notification(method, params = nil)
212
- notification = JsonRpc::Notification.new(method: method, params: params)
213
- write_message(notification.to_json)
214
- end
215
-
216
- private
217
-
218
- # Formats registry items to a hash representation.
219
- #
220
- # @param registry [Hash] The registry containing tool or prompt definitions.
221
- # @return [Array<Hash>] The formatted registry items.
222
- def format_registry_items(registry)
223
- registry.map { |item| item.klass.to_h }
224
- end
225
-
226
- # Writes a message to the output IO.
227
- #
228
- # @param data [String] The data to write.
229
- def write_message(data)
230
- Timeout.timeout(5) do # 5 second timeout
231
- @output.write("#{data}\n")
232
- end
233
- rescue Timeout::Error
234
- Rails.logger.error("Write operation timed out")
235
- # Handle timeout appropriately
236
- end
2
+ module Transport
237
3
  end
238
4
  end
@@ -0,0 +1,54 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActionMCP
4
+ class TransportHandler
5
+ include Logging
6
+
7
+ include Transport::Capabilities
8
+ include Transport::Tools
9
+ include Transport::Prompts
10
+ include Transport::Messaging
11
+
12
+ HEARTBEAT_INTERVAL = 15 # seconds
13
+ attr_reader :initialized
14
+
15
+ def initialize(output_io)
16
+ @output = output_io
17
+ @output.sync = true if @output.respond_to?(:sync=)
18
+ @initialized = false
19
+ @client_capabilities = {}
20
+ @client_info = {}
21
+ @protocol_version = ""
22
+ end
23
+
24
+ def send_ping
25
+ send_jsonrpc_request("ping")
26
+ end
27
+
28
+ def send_pong(request_id)
29
+ send_jsonrpc_response(request_id, result: {})
30
+ end
31
+
32
+ def initialized?
33
+ @initialized
34
+ end
35
+
36
+ def initialized!
37
+ @initialized = true
38
+ end
39
+
40
+ private
41
+
42
+ def write_message(data)
43
+ Timeout.timeout(5) do
44
+ @output.write("#{data}\n")
45
+ end
46
+ rescue Timeout::Error
47
+ # ActionMCP.logger.error("Write operation timed out")
48
+ end
49
+
50
+ def format_registry_items(registry)
51
+ registry.map { |item| item.klass.to_h }
52
+ end
53
+ end
54
+ end
@@ -1,11 +1,10 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require_relative "gem_version"
4
-
5
4
  module ActionMCP
6
- VERSION = "0.2.0"
7
- # Returns the currently loaded version of Active MCP as a +Gem::Version+.
8
- def self.version
9
- gem_version
5
+ VERSION = "0.2.3"
6
+
7
+ class << self
8
+ alias version gem_version
10
9
  end
11
10
  end
data/lib/action_mcp.rb CHANGED
@@ -2,58 +2,61 @@
2
2
 
3
3
  require "rails"
4
4
  require "active_support"
5
- require "active_model"
6
- require "action_mcp/version"
5
+ require "active_support/rails"
7
6
  require "multi_json"
8
- require "action_mcp/railtie" if defined?(Rails)
9
- require_relative "action_mcp/integer_array"
10
- require_relative "action_mcp/string_array"
7
+ require "concurrent"
8
+ require "active_record/railtie"
9
+ require "action_controller/railtie"
10
+ require "action_cable/engine"
11
+ require "action_mcp/engine"
12
+ require "zeitwerk"
11
13
 
12
- ActiveSupport::Inflector.inflections(:en) do |inflect|
13
- inflect.acronym "MCP"
14
- end
15
- module ActionMCP
16
- extend ActiveSupport::Autoload
17
-
18
- autoload :RegistryBase
19
- autoload :Resource
20
- autoload :ToolsRegistry
21
- autoload :PromptsRegistry
22
- autoload :ResourcesBank
23
- autoload :Tool
24
- autoload :Prompt
25
- autoload :JsonRpc
26
- autoload :Transport
27
- autoload :Content
28
- autoload :Renderable
29
-
30
- eager_autoload do
31
- autoload :Configuration
32
- end
14
+ lib = File.dirname(__FILE__)
33
15
 
34
- # Accessor for the configuration instance.
35
- def self.configuration
36
- @configuration ||= Configuration.new
37
- end
16
+ Zeitwerk::Loader.for_gem.tap do |loader|
17
+ loader.ignore(
18
+ "#{lib}/generators",
19
+ "#{lib}/action_mcp/version.rb",
20
+ "#{lib}/action_mcp/gem_version.rb",
21
+ "#{lib}/actionmcp.rb"
22
+ )
38
23
 
39
- def self.configure
40
- yield(configuration)
41
- end
24
+ loader.inflector.inflect("action_mcp" => "ActionMCP")
25
+ loader.inflector.inflect("sse_client" => "SSEClient")
26
+ loader.inflector.inflect("sse_server" => "SSEServer")
27
+ end.setup
28
+
29
+ module ActionMCP
30
+ require_relative "action_mcp/version"
31
+ require_relative "action_mcp/configuration"
32
+ PROTOCOL_VERSION = "2024-11-05"
42
33
 
43
34
  module_function
44
35
 
36
+ # Returns the tools registry.
37
+ #
38
+ # @return [Hash] the tools registry
45
39
  def tools
46
40
  ToolsRegistry.tools
47
41
  end
48
42
 
43
+ # Returns the prompts registry.
44
+ #
45
+ # @return [Hash] the prompts registry
49
46
  def prompts
50
47
  PromptsRegistry.prompts
51
48
  end
52
49
 
50
+ # Returns the available tools.
51
+ #
52
+ # @return [ActionMCP::RegistryBase::RegistryScope] the available tools
53
53
  def available_tools
54
54
  ToolsRegistry.available_tools
55
55
  end
56
56
 
57
+ # Returns the available prompts.
58
+ #
59
+ # @return [ActionMCP::RegistryBase::RegistryScope] the available prompts
57
60
  def available_prompts
58
61
  PromptsRegistry.available_prompts
59
62
  end
@@ -1,6 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ # Template for generating new prompts.
3
4
  class <%= class_name %> < ApplicationPrompt
5
+ # Set the prompt name.
4
6
  prompt_name "<%= prompt_name %>"
5
7
 
6
8
  # Provide a user-facing description for your prompt.
@@ -13,7 +15,7 @@ class <%= class_name %> < ApplicationPrompt
13
15
  # Add validations (note: "Ruby" is not allowed per the validation)
14
16
  validates :language, inclusion: { in: %w[Ruby C Cobol FORTRAN] }
15
17
 
18
+ # Implement your prompt's behavior here
16
19
  def call
17
- # Implement your prompt's behavior here
18
20
  end
19
21
  end
@@ -1,13 +1,17 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ # Template for generating new tools.
3
4
  class <%= class_name %> < ApplicationTool
5
+ # Set the tool name.
4
6
  tool_name "<%= tool_name %>"
5
7
  description "Calculate the sum of two numbers"
6
8
 
9
+ # Define input properties.
7
10
  property :a, type: "number", description: "First number", required: true
8
11
  property :b, type: "number", description: "Second number", required: true
9
12
 
13
+ # Implement the tool's logic here.
10
14
  def call
11
- a + b
15
+ render_text(a + b)
12
16
  end
13
17
  end
@@ -1,6 +1,29 @@
1
- # frozen_string_literal: true
1
+ namespace :action_mcp do
2
+ desc "List all tools with their names and descriptions"
3
+ task list_tools: :environment do
4
+ # Ensure Rails eager loads all classes
5
+ Rails.application.eager_load!
2
6
 
3
- # desc "Explaining what the task does"
4
- # task :action_mcp do
5
- # # Task goes here
6
- # end
7
+ puts "\e[34mACTION MCP TOOLS\e[0m" # Blue
8
+ puts "\e[34m---------------\e[0m" # Blue
9
+ ActionMCP::Tool.descendants.each do |tool|
10
+ puts "\e[34m#{tool.capability_name}:\e[0m #{tool.description}" # Blue name
11
+ end
12
+ end
13
+
14
+ desc "List all prompts with their names and descriptions"
15
+ task list_prompts: :environment do
16
+ # Ensure Rails eager loads all classes
17
+ Rails.application.eager_load!
18
+
19
+ puts "\e[32mACTION MCP PROMPTS\e[0m" # Red
20
+ puts "\e[32m-----------------\e[0m" # Red
21
+ ActionMCP::Prompt.descendants.each do |prompt|
22
+ puts "\e[32m#{prompt.capability_name}:\e[0m #{prompt.description}" # Red name
23
+ end
24
+ end
25
+
26
+ desc "List all tools and prompts with their names and descriptions"
27
+ task list: [ :list_tools, :list_prompts ] do
28
+ end
29
+ end