actionmcp 0.22.0 → 0.25.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 (33) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +113 -2
  3. data/app/controllers/action_mcp/messages_controller.rb +2 -14
  4. data/app/controllers/action_mcp/sse_controller.rb +113 -45
  5. data/app/models/action_mcp/session/message.rb +21 -16
  6. data/app/models/action_mcp/session.rb +3 -2
  7. data/config/routes.rb +1 -1
  8. data/db/migrate/20250324203409_remove_session_message_text.rb +7 -0
  9. data/lib/action_mcp/client/base.rb +12 -14
  10. data/lib/action_mcp/client/blueprint.rb +5 -71
  11. data/lib/action_mcp/client/catalog.rb +10 -74
  12. data/lib/action_mcp/client/collection.rb +93 -0
  13. data/lib/action_mcp/client/json_rpc_handler.rb +12 -7
  14. data/lib/action_mcp/client/logging.rb +1 -2
  15. data/lib/action_mcp/client/prompt_book.rb +5 -71
  16. data/lib/action_mcp/client/prompts.rb +9 -4
  17. data/lib/action_mcp/client/request_timeouts.rb +74 -0
  18. data/lib/action_mcp/client/resources.rb +23 -11
  19. data/lib/action_mcp/client/server.rb +3 -3
  20. data/lib/action_mcp/client/toolbox.rb +12 -54
  21. data/lib/action_mcp/client/tools.rb +9 -4
  22. data/lib/action_mcp/configuration.rb +134 -24
  23. data/lib/action_mcp/engine.rb +6 -0
  24. data/lib/action_mcp/json_rpc_handler_base.rb +1 -0
  25. data/lib/action_mcp/registry_base.rb +3 -1
  26. data/lib/action_mcp/server/capabilities.rb +1 -1
  27. data/lib/action_mcp/server/json_rpc_handler.rb +1 -1
  28. data/lib/action_mcp/server/messaging.rb +32 -9
  29. data/lib/action_mcp/version.rb +1 -1
  30. data/lib/generators/action_mcp/install/install_generator.rb +4 -0
  31. data/lib/generators/action_mcp/install/templates/mcp.yml +11 -0
  32. data/lib/tasks/action_mcp_tasks.rake +77 -6
  33. metadata +6 -2
@@ -4,40 +4,48 @@ module ActionMCP
4
4
  module Client
5
5
  module Resources
6
6
  # List all available resources from the server
7
- # @return [Array<Hash>] List of available resources with their metadata
7
+ # @return [String] Request ID for tracking the request
8
8
  def list_resources
9
9
  request_id = SecureRandom.uuid_v7
10
10
 
11
11
  # Send request
12
12
  send_jsonrpc_request("resources/list", id: request_id)
13
+
14
+ # Return request ID for tracking the request
15
+ request_id
13
16
  end
14
17
 
15
18
  # List resource templates from the server
16
- # @return [Array<Hash>] List of resource templates
19
+ # @return [String] Request ID for tracking the request
17
20
  def list_resource_templates
18
21
  request_id = SecureRandom.uuid_v7
19
22
 
20
23
  # Send request
21
24
  send_jsonrpc_request("resources/templates/list", id: request_id)
25
+
26
+ # Return request ID for tracking the request
27
+ request_id
22
28
  end
23
29
 
24
30
  # Read a specific resource
25
31
  # @param uri [String] URI of the resource to read
26
- # @return [Hash] Resource content
32
+ # @return [String] Request ID for tracking the request
27
33
  def read_resource(uri)
28
34
  request_id = SecureRandom.uuid_v7
29
35
 
30
36
  # Send request
31
37
  send_jsonrpc_request("resources/read",
32
38
  params: { uri: uri },
33
- id: request_id
34
- )
39
+ id: request_id)
40
+
41
+ # Return request ID for tracking the request
42
+ request_id
35
43
  end
36
44
 
37
45
  # Subscribe to updates for a specific resource
38
46
  # @param uri [String] URI of the resource to subscribe to
39
47
  # @param update_callback [Proc] Callback for resource updates
40
- # @return [Boolean] Success status
48
+ # @return [String] Request ID for tracking the request
41
49
  def subscribe_resource(uri, update_callback)
42
50
  @resource_subscriptions ||= {}
43
51
  @resource_subscriptions[uri] = update_callback
@@ -47,13 +55,15 @@ module ActionMCP
47
55
  # Send request
48
56
  send_jsonrpc_request("resources/subscribe",
49
57
  params: { uri: uri },
50
- id: request_id
51
- )
58
+ id: request_id)
59
+
60
+ # Return request ID for tracking the request
61
+ request_id
52
62
  end
53
63
 
54
64
  # Unsubscribe from updates for a specific resource
55
65
  # @param uri [String] URI of the resource to unsubscribe from
56
- # @return [Boolean] Success status
66
+ # @return [String] Request ID for tracking the request
57
67
  def unsubscribe_resource(uri)
58
68
  @resource_subscriptions&.delete(uri)
59
69
 
@@ -62,8 +72,10 @@ module ActionMCP
62
72
  # Send request
63
73
  send_jsonrpc_request("resources/unsubscribe",
64
74
  params: { uri: uri },
65
- id: request_id
66
- )
75
+ id: request_id)
76
+
77
+ # Return request ID for tracking the request
78
+ request_id
67
79
  end
68
80
  end
69
81
  end
@@ -1,9 +1,9 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
4
  module Client
3
5
  class Server
4
- attr_reader :name, :version
5
-
6
- attr_reader :server_info, :capabilities
6
+ attr_reader :name, :version, :server_info, :capabilities
7
7
 
8
8
  def initialize(data)
9
9
  # Store protocol version if needed for later use
@@ -18,30 +18,15 @@ module ActionMCP
18
18
  # # Get all tools matching a criteria
19
19
  # calculation_tools = toolbox.filter { |t| t.name.include?("calculate") }
20
20
  #
21
- class Toolbox
22
- attr_reader :client
23
-
21
+ class Toolbox < Collection
24
22
  # Initialize a new Toolbox with tool definitions
25
23
  #
26
24
  # @param tools [Array<Hash>] Array of tool definition hashes, each containing
27
25
  # name, description, and inputSchema keys
28
26
  def initialize(tools, client)
29
- self.tools = tools
30
- @client = client
31
- @loaded = !tools.empty?
32
- end
33
-
34
- # Return all tools in the collection
35
- #
36
- # @return [Array<Tool>] All tool objects in the collection
37
- def all
38
- load_tools unless @loaded
39
- @tools
40
- end
41
-
42
- def all!
43
- load_tools(force: true)
44
- @tools
27
+ super(tools, client)
28
+ self.tools = @collection_data
29
+ @load_method = :list_tools
45
30
  end
46
31
 
47
32
  # Find a tool by name
@@ -89,8 +74,9 @@ module ActionMCP
89
74
  # @param keyword [String] Keyword to search for in tool names and descriptions
90
75
  # @return [Array<Tool>] Tools containing the keyword
91
76
  def search(keyword)
92
- @tools.select do |tool|
93
- tool.name.include?(keyword) || tool.description.downcase.include?(keyword.downcase)
77
+ all.select do |tool|
78
+ tool.name.include?(keyword) ||
79
+ tool.description&.downcase&.include?(keyword.downcase)
94
80
  end
95
81
  end
96
82
 
@@ -102,46 +88,18 @@ module ActionMCP
102
88
  case provider
103
89
  when :claude
104
90
  # Claude format
105
- { "tools" => @tools.map(&:to_claude_h) }
91
+ { "tools" => all.map(&:to_claude_h) }
106
92
  when :openai
107
93
  # OpenAI format
108
- { "tools" => @tools.map(&:to_openai_h) }
94
+ { "tools" => all.map(&:to_openai_h) }
109
95
  else
110
96
  # Default format (same as original)
111
- { "tools" => @tools.map(&:to_h) }
97
+ { "tools" => all.map(&:to_h) }
112
98
  end
113
99
  end
114
100
 
115
- # Implements enumerable functionality for the collection
116
- include Enumerable
117
-
118
- # Yield each tool in the collection to the given block
119
- #
120
- # @yield [tool] Block to execute for each tool
121
- # @yieldparam tool [Tool] A tool from the collection
122
- # @return [Enumerator] If no block is given
123
- def each(&block)
124
- @tools.each(&block)
125
- end
126
-
127
101
  def tools=(tools)
128
- @tools = tools.map { |tool_data| Tool.new(tool_data) }
129
- end
130
-
131
- private
132
-
133
- def load_tools(force: false)
134
- return if @loaded && !force
135
-
136
- begin
137
- @client.list_tools
138
- @loaded = true
139
- rescue StandardError => e
140
- # Handle error appropriately
141
- Rails.logger.error("Failed to load tools: #{e.message}")
142
- # Still mark as loaded but with empty list?
143
- @loaded = true unless @tools.empty?
144
- end
102
+ @collection_data = tools.map { |tool_data| Tool.new(tool_data) }
145
103
  end
146
104
 
147
105
  # Internal Tool class to represent individual tools
@@ -168,7 +126,7 @@ module ActionMCP
168
126
  #
169
127
  # @return [Hash] Hash of property definitions
170
128
  def properties
171
- @input_schema.dig("properties") || {}
129
+ @input_schema["properties"] || {}
172
130
  end
173
131
 
174
132
  # Check if the tool requires a specific property
@@ -4,18 +4,21 @@ module ActionMCP
4
4
  module Client
5
5
  module Tools
6
6
  # List all available tools from the server
7
- # @return [Array<Hash>] List of available tools with their metadata
7
+ # @return [String] Request ID for tracking the request
8
8
  def list_tools
9
9
  request_id = SecureRandom.uuid_v7
10
10
 
11
11
  # Send request
12
12
  send_jsonrpc_request("tools/list", id: request_id)
13
+
14
+ # Return request ID for timeout tracking
15
+ request_id
13
16
  end
14
17
 
15
18
  # Call a specific tool on the server
16
19
  # @param name [String] Name of the tool to call
17
20
  # @param arguments [Hash] Arguments to pass to the tool
18
- # @return [Hash] The result of the tool execution
21
+ # @return [String] Request ID for tracking the request
19
22
  def call_tool(name, arguments)
20
23
  request_id = SecureRandom.uuid_v7
21
24
 
@@ -25,8 +28,10 @@ module ActionMCP
25
28
  name: name,
26
29
  arguments: arguments
27
30
  },
28
- id: request_id
29
- )
31
+ id: request_id)
32
+
33
+ # Return request ID for tracking the request
34
+ request_id
30
35
  end
31
36
  end
32
37
  end
@@ -14,21 +14,22 @@ module ActionMCP
14
14
  # @!attribute resources_subscribe
15
15
  # @return [Boolean] Whether to subscribe to resources.
16
16
  # @!attribute logging_level
17
- # @return [Symbol] The logging level.
17
+ # @return [Symbol] The logging level. attr_writer :name, :version
18
18
  attr_writer :name, :version
19
- attr_accessor :logging_enabled, # This is not working yet
20
- :list_changed, # This is not working yet
21
- :resources_subscribe, # This is not working yet
22
- :logging_level # This is not working yet
23
-
24
- # Initializes a new Configuration instance.
25
- #
26
- # @return [void]
19
+ attr_accessor :logging_enabled,
20
+ :list_changed,
21
+ :resources_subscribe,
22
+ :logging_level,
23
+ :active_profile,
24
+ :profiles
27
25
 
28
26
  def initialize
29
27
  @logging_enabled = true
30
28
  @list_changed = false
31
29
  @logging_level = :info
30
+ @resources_subscribe = false
31
+ @active_profile = :primary
32
+ @profiles = default_profiles
32
33
  end
33
34
 
34
35
  def name
@@ -39,23 +40,127 @@ module ActionMCP
39
40
  @version || (has_rails_version ? Rails.application.version.to_s : "0.0.1")
40
41
  end
41
42
 
42
- # Returns a hash of capabilities.
43
- #
44
- # @return [Hash] A hash containing the resources capabilities.
43
+ # Load custom profiles from Rails configuration
44
+ def load_profiles
45
+ # First load defaults from the gem
46
+ @profiles = default_profiles
47
+
48
+ # Then try to load from config/mcp.yml in the Rails app
49
+ config_path = Rails.root.join("config", "mcp.yml")
50
+ if File.exist?(config_path)
51
+ begin
52
+ yaml_content = YAML.safe_load(File.read(config_path), symbolize_names: true)
53
+ # Merge with defaults so user config overrides gem defaults
54
+ @profiles.deep_merge!(yaml_content) if yaml_content
55
+ rescue StandardError => e
56
+ Rails.logger.error "Failed to load MCP profiles from #{config_path}: #{e.message}"
57
+ end
58
+ end
59
+
60
+ # Apply the active profile
61
+ use_profile(@active_profile)
62
+
63
+ self
64
+ end
65
+
66
+ # Switch to a specific profile
67
+ def use_profile(profile_name)
68
+ profile_name = profile_name.to_sym
69
+ unless @profiles.key?(profile_name)
70
+ Rails.logger.warn "Profile '#{profile_name}' not found, using default"
71
+ profile_name = :default
72
+ end
73
+
74
+ @active_profile = profile_name
75
+ apply_profile_options
76
+
77
+ self
78
+ end
79
+
80
+ # Filter tools based on active profile
81
+ def filtered_tools
82
+ return ToolsRegistry.non_abstract if should_include_all?(:tools)
83
+
84
+ tool_names = @profiles[@active_profile][:tools] || []
85
+ # Convert tool names to underscored format
86
+ tool_names = tool_names.map { |name| name.to_s.underscore }
87
+ ToolsRegistry.non_abstract.select { |tool| tool_names.include?(tool.name.underscore) }
88
+ end
89
+
90
+ # Filter prompts based on active profile
91
+ def filtered_prompts
92
+ return PromptsRegistry.non_abstract if should_include_all?(:prompts)
93
+
94
+ prompt_names = @profiles[@active_profile][:prompts] || []
95
+ PromptsRegistry.non_abstract.select { |prompt| prompt_names.include?(prompt.name) }
96
+ end
97
+
98
+ # Filter resources based on active profile
99
+ def filtered_resources
100
+ return ResourceTemplatesRegistry.non_abstract if should_include_all?(:resources)
101
+
102
+ resource_names = @profiles[@active_profile][:resources] || []
103
+ ResourceTemplatesRegistry.non_abstract.select { |resource| resource_names.include?(resource.name) }
104
+ end
105
+
106
+ # Returns capabilities based on active profile
45
107
  def capabilities
46
108
  capabilities = {}
47
- # Only include each capability if the corresponding registry is non-empty.
48
- capabilities[:tools] = { listChanged: false } if ToolsRegistry.non_abstract.any?
49
- capabilities[:prompts] = { listChanged: false } if PromptsRegistry.non_abstract.any?
109
+ # Only include each capability if the corresponding filtered registry is non-empty
110
+ capabilities[:tools] = { listChanged: @list_changed } if filtered_tools.any?
111
+ capabilities[:prompts] = { listChanged: @list_changed } if filtered_prompts.any?
50
112
  capabilities[:logging] = {} if @logging_enabled
51
- # For now, we only have one type of resource, ResourceTemplate
52
- # For Resources, we need to think about how to pass the list to the session.
53
- capabilities[:resources] = {} if ResourceTemplatesRegistry.non_abstract.any?
113
+ capabilities[:resources] = { subscribe: @resources_subscribe } if filtered_resources.any?
54
114
  capabilities
55
115
  end
56
116
 
57
117
  private
58
118
 
119
+ def default_profiles
120
+ {
121
+ primary: {
122
+ tools: [ "all" ],
123
+ prompts: [ "all" ],
124
+ resources: [ "all" ],
125
+ options: {
126
+ list_changed: false,
127
+ logging_enabled: true,
128
+ logging_level: :info,
129
+ resources_subscribe: false
130
+ }
131
+ },
132
+ minimal: {
133
+ tools: [],
134
+ prompts: [],
135
+ resources: [],
136
+ options: {
137
+ list_changed: false,
138
+ logging_enabled: false,
139
+ logging_level: :warn,
140
+ resources_subscribe: false
141
+ }
142
+ }
143
+ }
144
+ end
145
+
146
+ def apply_profile_options
147
+ profile = @profiles[@active_profile]
148
+ return unless profile && profile[:options]
149
+
150
+ options = profile[:options]
151
+ @list_changed = options[:list_changed] unless options[:list_changed].nil?
152
+ @logging_enabled = options[:logging_enabled] unless options[:logging_enabled].nil?
153
+ @logging_level = options[:logging_level] unless options[:logging_level].nil?
154
+ @resources_subscribe = options[:resources_subscribe] unless options[:resources_subscribe].nil?
155
+ end
156
+
157
+ def should_include_all?(type)
158
+ return true unless @profiles[@active_profile]
159
+
160
+ items = @profiles[@active_profile][type]
161
+ items.nil? || items.include?("all")
162
+ end
163
+
59
164
  def has_rails_version
60
165
  gem "rails_app_version"
61
166
  require "rails_app_version/railtie"
@@ -66,21 +171,26 @@ module ActionMCP
66
171
  end
67
172
 
68
173
  class << self
69
- attr_accessor :server
174
+ attr_accessor :server, :logger
70
175
 
71
176
  # Returns the configuration instance.
72
- #
73
- # @return [Configuration] the configuration instance
74
177
  def configuration
75
178
  @configuration ||= Configuration.new
76
179
  end
77
180
 
78
181
  # Configures the ActionMCP module.
79
- #
80
- # @yield [configuration] the configuration instance
81
- # @return [void]
82
182
  def configure
83
183
  yield(configuration)
84
184
  end
185
+
186
+ # Temporarily use a different profile
187
+ def with_profile(profile_name)
188
+ previous_profile = configuration.active_profile
189
+ configuration.use_profile(profile_name)
190
+
191
+ yield if block_given?
192
+ ensure
193
+ configuration.use_profile(previous_profile) if block_given?
194
+ end
85
195
  end
86
196
  end
@@ -12,6 +12,7 @@ module ActionMCP
12
12
  inflect.acronym "SSE"
13
13
  inflect.acronym "MCP"
14
14
  end
15
+
15
16
  # Provide a configuration namespace for ActionMCP
16
17
  config.action_mcp = ActionMCP.configuration
17
18
 
@@ -19,6 +20,11 @@ module ActionMCP
19
20
  ActionMCP::ResourceTemplate.registered_templates.clear
20
21
  end
21
22
 
23
+ # Load MCP profiles during initialization
24
+ initializer "action_mcp.load_profiles" do
25
+ ActionMCP.configuration.load_profiles
26
+ end
27
+
22
28
  # Configure autoloading for the mcp/tools directory
23
29
  initializer "action_mcp.autoloading", before: :set_autoload_paths do |app|
24
30
  mcp_path = app.root.join("app/mcp")
@@ -75,6 +75,7 @@ module ActionMCP
75
75
  # @param request [Hash]
76
76
  def process_request(request)
77
77
  return unless valid_request?(request)
78
+
78
79
  read(request)
79
80
 
80
81
  id = request["id"]
@@ -64,7 +64,9 @@ module ActionMCP
64
64
  include Enumerable
65
65
 
66
66
  # Using a Data type for items.
67
- Item = Data.define(:name, :klass)
67
+ Item = Data.define(:name, :klass) do
68
+ delegate :description, to: :klass
69
+ end
68
70
 
69
71
  # Initializes a new RegistryScope instance.
70
72
  #
@@ -12,7 +12,7 @@ module ActionMCP
12
12
  session.store_client_info(@client_info)
13
13
  session.store_client_capabilities(@client_capabilities)
14
14
  session.set_protocol_version(@protocol_version)
15
- session.save
15
+ session.initialize!
16
16
  # TODO: , if the server don't support the protocol version, send a response with error
17
17
  send_jsonrpc_response(request_id, result: session.server_capabilities_payload)
18
18
  end
@@ -22,7 +22,7 @@ module ActionMCP
22
22
  when "completion/complete" # Completion requests
23
23
  process_completion_complete(id, params)
24
24
  else
25
- puts "\e[31mUnknown client method: #{rpc_method}\e[0m"
25
+ transport.send_jsonrpc_error(id, :method_not_found, "Method not found")
26
26
  end
27
27
  end
28
28
 
@@ -4,24 +4,47 @@ module ActionMCP
4
4
  module Server
5
5
  module Messaging
6
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)
7
+ send_message(:request, method: method, params: params, id: id)
9
8
  end
10
9
 
11
10
  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)
11
+ send_message(:response, id: request_id, result: result, error: error)
14
12
  end
15
13
 
16
14
  def send_jsonrpc_notification(method, params = nil)
17
- notification = JsonRpc::Notification.new(method: method, params: params)
18
- write_message(notification)
15
+ send_message(:notification, method: method, params: params)
19
16
  end
20
17
 
21
18
  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)
19
+ error = JsonRpc::JsonRpcError.new(symbol, message: message, data: data)
20
+ send_jsonrpc_response(request_id, error: error)
21
+ end
22
+
23
+ private
24
+
25
+ # Factory method to create and send appropriate JSON-RPC message
26
+ def send_message(type, **args)
27
+ message = case type
28
+ when :request
29
+ JsonRpc::Request.new(
30
+ id: args[:id],
31
+ method: args[:method],
32
+ params: args[:params]
33
+ )
34
+ when :response
35
+ JsonRpc::Response.new(
36
+ id: args[:id],
37
+ result: args[:result],
38
+ error: args[:error]
39
+ )
40
+ when :notification
41
+ JsonRpc::Notification.new(
42
+ method: args[:method],
43
+ params: args[:params]
44
+ )
45
+ end
46
+
47
+ write_message(message)
25
48
  end
26
49
  end
27
50
  end
@@ -2,7 +2,7 @@
2
2
 
3
3
  require_relative "gem_version"
4
4
  module ActionMCP
5
- VERSION = "0.22.0"
5
+ VERSION = "0.25.0"
6
6
 
7
7
  class << self
8
8
  alias version gem_version
@@ -19,6 +19,10 @@ module ActionMcp
19
19
  template "application_mcp_res_template.rb",
20
20
  File.join("app/mcp/resource_templates", "application_mcp_res_template.rb")
21
21
  end
22
+
23
+ def create_mcp_profile_file
24
+ template "mcp.yml", File.join("config", "mcp.yml")
25
+ end
22
26
  end
23
27
  end
24
28
  end
@@ -0,0 +1,11 @@
1
+ primary:
2
+ tools:
3
+ - all
4
+ prompts:
5
+ - all
6
+ resources:
7
+ - all
8
+ options:
9
+ list_changed: false
10
+ resources_subscribe: false
11
+