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.
Files changed (53) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +133 -30
  3. data/Rakefile +0 -2
  4. data/app/controllers/action_mcp/application_controller.rb +13 -0
  5. data/app/controllers/action_mcp/messages_controller.rb +51 -0
  6. data/app/controllers/action_mcp/sse_controller.rb +151 -0
  7. data/config/routes.rb +4 -0
  8. data/exe/actionmcp_cli +221 -0
  9. data/lib/action_mcp/capability.rb +52 -0
  10. data/lib/action_mcp/client.rb +243 -1
  11. data/lib/action_mcp/configuration.rb +50 -1
  12. data/lib/action_mcp/content/audio.rb +9 -0
  13. data/lib/action_mcp/content/image.rb +9 -0
  14. data/lib/action_mcp/content/resource.rb +13 -0
  15. data/lib/action_mcp/content/text.rb +7 -0
  16. data/lib/action_mcp/content.rb +11 -6
  17. data/lib/action_mcp/engine.rb +34 -0
  18. data/lib/action_mcp/gem_version.rb +2 -2
  19. data/lib/action_mcp/integer_array.rb +6 -0
  20. data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
  21. data/lib/action_mcp/json_rpc/notification.rb +8 -0
  22. data/lib/action_mcp/json_rpc/request.rb +14 -0
  23. data/lib/action_mcp/json_rpc/response.rb +32 -1
  24. data/lib/action_mcp/json_rpc.rb +1 -6
  25. data/lib/action_mcp/json_rpc_handler.rb +106 -0
  26. data/lib/action_mcp/logging.rb +19 -0
  27. data/lib/action_mcp/prompt.rb +30 -46
  28. data/lib/action_mcp/prompts_registry.rb +13 -1
  29. data/lib/action_mcp/registry_base.rb +47 -28
  30. data/lib/action_mcp/renderable.rb +26 -0
  31. data/lib/action_mcp/resource.rb +3 -1
  32. data/lib/action_mcp/server.rb +4 -1
  33. data/lib/action_mcp/string_array.rb +5 -0
  34. data/lib/action_mcp/tool.rb +16 -53
  35. data/lib/action_mcp/tools_registry.rb +14 -1
  36. data/lib/action_mcp/transport/capabilities.rb +21 -0
  37. data/lib/action_mcp/transport/messaging.rb +20 -0
  38. data/lib/action_mcp/transport/prompts.rb +19 -0
  39. data/lib/action_mcp/transport/sse_client.rb +309 -0
  40. data/lib/action_mcp/transport/stdio_client.rb +117 -0
  41. data/lib/action_mcp/transport/tools.rb +20 -0
  42. data/lib/action_mcp/transport/transport_base.rb +125 -0
  43. data/lib/action_mcp/transport.rb +1 -235
  44. data/lib/action_mcp/transport_handler.rb +54 -0
  45. data/lib/action_mcp/version.rb +4 -5
  46. data/lib/action_mcp.rb +36 -33
  47. data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
  48. data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
  49. data/lib/tasks/action_mcp_tasks.rake +28 -5
  50. metadata +66 -9
  51. data/exe/action_mcp_stdio +0 -0
  52. data/lib/action_mcp/railtie.rb +0 -27
  53. data/lib/action_mcp/resources_bank.rb +0 -94
@@ -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.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 "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
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.0
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-02-15 00:00:00.000000000 Z
10
+ date: 2025-03-11 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
- name: activemodel
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: activesupport
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
- - action_mcp_stdio
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
- - exe/action_mcp_stdio
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.2
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
@@ -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