actionmcp 0.2.0 → 0.2.4
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/README.md +133 -30
- data/Rakefile +0 -2
- data/app/controllers/action_mcp/application_controller.rb +13 -0
- data/app/controllers/action_mcp/messages_controller.rb +51 -0
- data/app/controllers/action_mcp/sse_controller.rb +151 -0
- data/config/routes.rb +4 -0
- data/exe/actionmcp_cli +221 -0
- data/lib/action_mcp/capability.rb +52 -0
- data/lib/action_mcp/client.rb +243 -1
- data/lib/action_mcp/configuration.rb +50 -1
- data/lib/action_mcp/content/audio.rb +9 -0
- data/lib/action_mcp/content/image.rb +9 -0
- data/lib/action_mcp/content/resource.rb +13 -0
- data/lib/action_mcp/content/text.rb +7 -0
- data/lib/action_mcp/content.rb +11 -6
- data/lib/action_mcp/engine.rb +34 -0
- data/lib/action_mcp/gem_version.rb +2 -2
- data/lib/action_mcp/integer_array.rb +6 -0
- data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
- data/lib/action_mcp/json_rpc/notification.rb +8 -0
- data/lib/action_mcp/json_rpc/request.rb +14 -0
- data/lib/action_mcp/json_rpc/response.rb +32 -1
- data/lib/action_mcp/json_rpc.rb +1 -6
- data/lib/action_mcp/json_rpc_handler.rb +106 -0
- data/lib/action_mcp/logging.rb +19 -0
- data/lib/action_mcp/prompt.rb +30 -46
- data/lib/action_mcp/prompts_registry.rb +13 -1
- data/lib/action_mcp/registry_base.rb +47 -28
- data/lib/action_mcp/renderable.rb +26 -0
- data/lib/action_mcp/resource.rb +3 -1
- data/lib/action_mcp/server.rb +4 -1
- data/lib/action_mcp/string_array.rb +5 -0
- data/lib/action_mcp/tool.rb +16 -53
- data/lib/action_mcp/tools_registry.rb +14 -1
- data/lib/action_mcp/transport/capabilities.rb +21 -0
- data/lib/action_mcp/transport/messaging.rb +20 -0
- data/lib/action_mcp/transport/prompts.rb +19 -0
- data/lib/action_mcp/transport/sse_client.rb +309 -0
- data/lib/action_mcp/transport/stdio_client.rb +117 -0
- data/lib/action_mcp/transport/tools.rb +20 -0
- data/lib/action_mcp/transport/transport_base.rb +125 -0
- data/lib/action_mcp/transport.rb +1 -235
- data/lib/action_mcp/transport_handler.rb +54 -0
- data/lib/action_mcp/version.rb +4 -5
- data/lib/action_mcp.rb +36 -33
- data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
- data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
- data/lib/tasks/action_mcp_tasks.rake +28 -5
- metadata +66 -9
- data/exe/action_mcp_stdio +0 -0
- data/lib/action_mcp/railtie.rb +0 -27
- data/lib/action_mcp/resources_bank.rb +0 -94
data/lib/action_mcp/transport.rb
CHANGED
@@ -1,238 +1,4 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
1
|
module ActionMCP
|
4
|
-
|
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
|
data/lib/action_mcp/version.rb
CHANGED
@@ -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.
|
7
|
-
|
8
|
-
|
9
|
-
gem_version
|
5
|
+
VERSION = "0.2.4"
|
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 "
|
6
|
-
require "action_mcp/version"
|
5
|
+
require "active_support/rails"
|
7
6
|
require "multi_json"
|
8
|
-
require "
|
9
|
-
|
10
|
-
|
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
|
-
|
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
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
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
|
-
|
40
|
-
|
41
|
-
|
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
|
-
|
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
|
-
|
4
|
-
#
|
5
|
-
|
6
|
-
#
|
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
|
metadata
CHANGED
@@ -1,16 +1,16 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: actionmcp
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.2.
|
4
|
+
version: 0.2.4
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Abdelkader Boudih
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-03-11 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
|
-
name:
|
13
|
+
name: railties
|
14
14
|
requirement: !ruby/object:Gem::Requirement
|
15
15
|
requirements:
|
16
16
|
- - ">="
|
@@ -24,7 +24,21 @@ dependencies:
|
|
24
24
|
- !ruby/object:Gem::Version
|
25
25
|
version: 8.0.1
|
26
26
|
- !ruby/object:Gem::Dependency
|
27
|
-
name:
|
27
|
+
name: activerecord
|
28
|
+
requirement: !ruby/object:Gem::Requirement
|
29
|
+
requirements:
|
30
|
+
- - ">="
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: 8.0.1
|
33
|
+
type: :runtime
|
34
|
+
prerelease: false
|
35
|
+
version_requirements: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - ">="
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: 8.0.1
|
40
|
+
- !ruby/object:Gem::Dependency
|
41
|
+
name: actioncable
|
28
42
|
requirement: !ruby/object:Gem::Requirement
|
29
43
|
requirements:
|
30
44
|
- - ">="
|
@@ -51,20 +65,53 @@ dependencies:
|
|
51
65
|
- - ">="
|
52
66
|
- !ruby/object:Gem::Version
|
53
67
|
version: '0'
|
68
|
+
- !ruby/object:Gem::Dependency
|
69
|
+
name: faraday
|
70
|
+
requirement: !ruby/object:Gem::Requirement
|
71
|
+
requirements:
|
72
|
+
- - "~>"
|
73
|
+
- !ruby/object:Gem::Version
|
74
|
+
version: '2.0'
|
75
|
+
type: :runtime
|
76
|
+
prerelease: false
|
77
|
+
version_requirements: !ruby/object:Gem::Requirement
|
78
|
+
requirements:
|
79
|
+
- - "~>"
|
80
|
+
- !ruby/object:Gem::Version
|
81
|
+
version: '2.0'
|
82
|
+
- !ruby/object:Gem::Dependency
|
83
|
+
name: zeitwerk
|
84
|
+
requirement: !ruby/object:Gem::Requirement
|
85
|
+
requirements:
|
86
|
+
- - "~>"
|
87
|
+
- !ruby/object:Gem::Version
|
88
|
+
version: '2.6'
|
89
|
+
type: :runtime
|
90
|
+
prerelease: false
|
91
|
+
version_requirements: !ruby/object:Gem::Requirement
|
92
|
+
requirements:
|
93
|
+
- - "~>"
|
94
|
+
- !ruby/object:Gem::Version
|
95
|
+
version: '2.6'
|
54
96
|
description: It offers base classes and helpers for creating MCP applications, making
|
55
97
|
it easier to integrate your Ruby/Rails application with the MCP standard
|
56
98
|
email:
|
57
99
|
- terminale@gmail.com
|
58
100
|
executables:
|
59
|
-
-
|
101
|
+
- actionmcp_cli
|
60
102
|
extensions: []
|
61
103
|
extra_rdoc_files: []
|
62
104
|
files:
|
63
105
|
- MIT-LICENSE
|
64
106
|
- README.md
|
65
107
|
- Rakefile
|
66
|
-
-
|
108
|
+
- app/controllers/action_mcp/application_controller.rb
|
109
|
+
- app/controllers/action_mcp/messages_controller.rb
|
110
|
+
- app/controllers/action_mcp/sse_controller.rb
|
111
|
+
- config/routes.rb
|
112
|
+
- exe/actionmcp_cli
|
67
113
|
- lib/action_mcp.rb
|
114
|
+
- lib/action_mcp/capability.rb
|
68
115
|
- lib/action_mcp/client.rb
|
69
116
|
- lib/action_mcp/configuration.rb
|
70
117
|
- lib/action_mcp/content.rb
|
@@ -72,6 +119,7 @@ files:
|
|
72
119
|
- lib/action_mcp/content/image.rb
|
73
120
|
- lib/action_mcp/content/resource.rb
|
74
121
|
- lib/action_mcp/content/text.rb
|
122
|
+
- lib/action_mcp/engine.rb
|
75
123
|
- lib/action_mcp/gem_version.rb
|
76
124
|
- lib/action_mcp/integer_array.rb
|
77
125
|
- lib/action_mcp/json_rpc.rb
|
@@ -79,18 +127,26 @@ files:
|
|
79
127
|
- lib/action_mcp/json_rpc/notification.rb
|
80
128
|
- lib/action_mcp/json_rpc/request.rb
|
81
129
|
- lib/action_mcp/json_rpc/response.rb
|
130
|
+
- lib/action_mcp/json_rpc_handler.rb
|
131
|
+
- lib/action_mcp/logging.rb
|
82
132
|
- lib/action_mcp/prompt.rb
|
83
133
|
- lib/action_mcp/prompts_registry.rb
|
84
|
-
- lib/action_mcp/railtie.rb
|
85
134
|
- lib/action_mcp/registry_base.rb
|
86
135
|
- lib/action_mcp/renderable.rb
|
87
136
|
- lib/action_mcp/resource.rb
|
88
|
-
- lib/action_mcp/resources_bank.rb
|
89
137
|
- lib/action_mcp/server.rb
|
90
138
|
- lib/action_mcp/string_array.rb
|
91
139
|
- lib/action_mcp/tool.rb
|
92
140
|
- lib/action_mcp/tools_registry.rb
|
93
141
|
- lib/action_mcp/transport.rb
|
142
|
+
- lib/action_mcp/transport/capabilities.rb
|
143
|
+
- lib/action_mcp/transport/messaging.rb
|
144
|
+
- lib/action_mcp/transport/prompts.rb
|
145
|
+
- lib/action_mcp/transport/sse_client.rb
|
146
|
+
- lib/action_mcp/transport/stdio_client.rb
|
147
|
+
- lib/action_mcp/transport/tools.rb
|
148
|
+
- lib/action_mcp/transport/transport_base.rb
|
149
|
+
- lib/action_mcp/transport_handler.rb
|
94
150
|
- lib/action_mcp/version.rb
|
95
151
|
- lib/actionmcp.rb
|
96
152
|
- lib/generators/action_mcp/install/install_generator.rb
|
@@ -107,6 +163,7 @@ licenses:
|
|
107
163
|
metadata:
|
108
164
|
homepage_uri: https://github.com/seuros/action_mcp
|
109
165
|
source_code_uri: https://github.com/seuros/action_mcp
|
166
|
+
changelog_uri: https://github.com/seuros/action_mcp/blob/master/CHANGELOG.md
|
110
167
|
rubygems_mfa_required: 'true'
|
111
168
|
rdoc_options: []
|
112
169
|
require_paths:
|
@@ -122,7 +179,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
122
179
|
- !ruby/object:Gem::Version
|
123
180
|
version: '0'
|
124
181
|
requirements: []
|
125
|
-
rubygems_version: 3.6.
|
182
|
+
rubygems_version: 3.6.5
|
126
183
|
specification_version: 4
|
127
184
|
summary: Provides essential tooling for building Model Context Protocol (MCP) capable
|
128
185
|
servers
|
data/exe/action_mcp_stdio
DELETED
File without changes
|
data/lib/action_mcp/railtie.rb
DELETED
@@ -1,27 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require "rails"
|
4
|
-
require "active_model/railtie"
|
5
|
-
|
6
|
-
module ActionMCP
|
7
|
-
class Railtie < Rails::Railtie # :nodoc:
|
8
|
-
# Provide a configuration namespace for ActionMCP
|
9
|
-
config.action_mcp = ActiveSupport::OrderedOptions.new
|
10
|
-
|
11
|
-
config.after_initialize do |app|
|
12
|
-
options = app.config.action_mcp.to_h.symbolize_keys
|
13
|
-
|
14
|
-
# Override the default configuration if specified in the Rails app.
|
15
|
-
ActionMCP.configuration.name = options[:name] if options.key?(:name)
|
16
|
-
ActionMCP.configuration.version = options[:version] if options.key?(:version)
|
17
|
-
ActionMCP.configuration.logging_enabled = options.fetch(:logging_enabled, true)
|
18
|
-
end
|
19
|
-
|
20
|
-
initializer "action_mcp.clear_registry" do |app|
|
21
|
-
app.config.to_prepare do
|
22
|
-
ActionMCP::ToolsRegistry.clear!
|
23
|
-
ActionMCP::PromptsRegistry.clear!
|
24
|
-
end
|
25
|
-
end
|
26
|
-
end
|
27
|
-
end
|