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
@@ -24,35 +24,16 @@ module ActionMCP
24
24
  # # Generate a concrete URI from a blueprint with parameters
25
25
  # uri = Blueprint.construct("file://{path}", { path: "/logs/app.log" })
26
26
  #
27
- class Blueprint
27
+ class Blueprint < Collection
28
28
  # Initialize a new Blueprints collection with URI template definitions
29
29
  #
30
30
  # @param templates [Array<Hash>] Array of URI template definition hashes, each containing
31
31
  # uriTemplate, name, description, and optionally mimeType keys
32
32
  # @param client [Object, nil] Optional client for lazy loading of templates
33
- attr_reader :client
34
-
35
33
  def initialize(templates, client)
36
- self.templates = templates
37
- @client = client
38
- @loaded = !templates.empty?
39
- end
40
-
41
- # Return all URI templates in the collection. If initialized with a client and templates
42
- # haven't been loaded yet, this will trigger lazy loading from the client.
43
- #
44
- # @return [Array<Blueprint>] All blueprint objects in the collection
45
- def all
46
- load_templates unless @loaded
47
- @templates
48
- end
49
-
50
- # Force reload all templates from the client and return them
51
- #
52
- # @return [Array<Blueprint>] All blueprint objects in the collection
53
- def all!
54
- load_templates(force: true)
55
- @templates
34
+ super(templates, client)
35
+ self.templates = @collection_data
36
+ @load_method = :list_resource_templates
56
37
  end
57
38
 
58
39
  # Find a blueprint by its URI pattern
@@ -84,23 +65,6 @@ module ActionMCP
84
65
  blueprint.construct(params)
85
66
  end
86
67
 
87
- # Filter blueprints based on a given block
88
- #
89
- # @yield [blueprint] Block that determines whether to include a blueprint
90
- # @yieldparam blueprint [Blueprint] A blueprint from the collection
91
- # @yieldreturn [Boolean] true to include the blueprint, false to exclude it
92
- # @return [Array<Blueprint>] Blueprints that match the filter criteria
93
- def filter(&block)
94
- all.select(&block)
95
- end
96
-
97
- # Number of blueprints in the collection
98
- #
99
- # @return [Integer] The number of blueprints
100
- def size
101
- all.size
102
- end
103
-
104
68
  # Check if the collection contains a blueprint with the given pattern
105
69
  #
106
70
  # @param pattern [String] The blueprint pattern to check for
@@ -116,41 +80,11 @@ module ActionMCP
116
80
  all.group_by(&:protocol)
117
81
  end
118
82
 
119
- # Implements enumerable functionality for the collection
120
- include Enumerable
121
-
122
- # Yield each blueprint in the collection to the given block
123
- #
124
- # @yield [blueprint] Block to execute for each blueprint
125
- # @yieldparam blueprint [Blueprint] A blueprint from the collection
126
- # @return [Enumerator] If no block is given
127
- def each(&block)
128
- all.each(&block)
129
- end
130
-
131
83
  # Convert raw template data into ResourceTemplate objects
132
84
  #
133
85
  # @param templates [Array<Hash>] Array of template definition hashes
134
86
  def templates=(templates)
135
- @templates = templates.map { |template_data| ResourceTemplate.new(template_data) }
136
- end
137
-
138
- private
139
-
140
- # Load or reload templates using the client
141
- #
142
- # @param force [Boolean] Whether to force reload even if templates are already loaded
143
- # @return [void]
144
- def load_templates(force: false)
145
- return if @loaded && !force
146
-
147
- begin
148
- @client.list_resource_templates
149
- @loaded = true
150
- rescue StandardError => e
151
- Rails.logger.error("Failed to load templates: #{e.message}")
152
- @loaded = true unless @templates.empty?
153
- end
87
+ @collection_data = templates.map { |template_data| ResourceTemplate.new(template_data) }
154
88
  end
155
89
 
156
90
  # Internal Blueprint class to represent individual URI templates
@@ -24,35 +24,16 @@ module ActionMCP
24
24
  # # Get all resources matching a criteria
25
25
  # rust_files = catalog.filter { |r| r.mime_type == "text/x-rust" }
26
26
  #
27
- class Catalog
27
+ class Catalog < Collection
28
28
  # Initialize a new Catalog with resource definitions
29
29
  #
30
30
  # @param resources [Array<Hash>] Array of resource definition hashes, each containing
31
31
  # uri, name, description, and mimeType keys
32
32
  # @param client [Object, nil] Optional client for lazy loading of resources
33
- attr_reader :client
34
-
35
33
  def initialize(resources, client)
36
- self.resources = resources
37
- @client = client
38
- @loaded = !resources.empty?
39
- end
40
-
41
- # Return all resources in the collection. If initialized with a client and resources
42
- # haven't been loaded yet, this will trigger lazy loading from the client.
43
- #
44
- # @return [Array<Resource>] All resource objects in the collection
45
- def all
46
- load_resources unless @loaded
47
- @resources
48
- end
49
-
50
- # Force reload all resources from the client and return them
51
- #
52
- # @return [Array<Resource>] All resource objects in the collection
53
- def all!
54
- load_resources(force: true)
55
- @resources
34
+ super(resources, client)
35
+ self.resources = @collection_data
36
+ @load_method = :list_resources
56
37
  end
57
38
 
58
39
  # Find a resource by URI
@@ -79,16 +60,6 @@ module ActionMCP
79
60
  all.select { |resource| resource.mime_type == mime_type }
80
61
  end
81
62
 
82
- # Filter resources based on a given block
83
- #
84
- # @yield [resource] Block that determines whether to include a resource
85
- # @yieldparam resource [Resource] A resource from the collection
86
- # @yieldreturn [Boolean] true to include the resource, false to exclude it
87
- # @return [Array<Resource>] Resources that match the filter criteria
88
- def filter(&block)
89
- all.select(&block)
90
- end
91
-
92
63
  # Get a list of all resource URIs
93
64
  #
94
65
  # @return [Array<String>] URIs of all resources in the collection
@@ -96,13 +67,6 @@ module ActionMCP
96
67
  all.map(&:uri)
97
68
  end
98
69
 
99
- # Number of resources in the collection
100
- #
101
- # @return [Integer] The number of resources
102
- def size
103
- all.size
104
- end
105
-
106
70
  # Check if the collection contains a resource with the given URI
107
71
  #
108
72
  # @param uri [String] The resource URI to check for
@@ -126,45 +90,15 @@ module ActionMCP
126
90
  keyword = keyword.downcase
127
91
  all.select do |resource|
128
92
  resource.name.downcase.include?(keyword) ||
129
- (resource.description && resource.description.downcase.include?(keyword))
93
+ resource.description&.downcase&.include?(keyword)
130
94
  end
131
95
  end
132
96
 
133
- # Implements enumerable functionality for the collection
134
- include Enumerable
135
-
136
- # Yield each resource in the collection to the given block
137
- #
138
- # @yield [resource] Block to execute for each resource
139
- # @yieldparam resource [Resource] A resource from the collection
140
- # @return [Enumerator] If no block is given
141
- def each(&block)
142
- all.each(&block)
143
- end
144
-
145
97
  # Convert raw resource data into Resource objects
146
98
  #
147
99
  # @param raw_resources [Array<Hash>] Array of resource definition hashes
148
100
  def resources=(raw_resources)
149
- @resources = raw_resources.map { |resource_data| Resource.new(resource_data) }
150
- end
151
-
152
- private
153
-
154
- # Load or reload resources using the client
155
- #
156
- # @param force [Boolean] Whether to force reload even if resources are already loaded
157
- # @return [void]
158
- def load_resources(force: false)
159
- return if @loaded && !force
160
-
161
- begin
162
- @client.list_resources
163
- @loaded = true
164
- rescue StandardError => e
165
- Rails.logger.error("Failed to load resources: #{e.message}")
166
- @loaded = true unless @resources.empty?
167
- end
101
+ @collection_data = raw_resources.map { |resource_data| Resource.new(resource_data) }
168
102
  end
169
103
 
170
104
  # Internal Resource class to represent individual resources
@@ -185,7 +119,7 @@ module ActionMCP
185
119
  #
186
120
  # @return [String, nil] The file extension or nil if no extension
187
121
  def extension
188
- File.extname(@name)[1..-1] if @name.include?(".")
122
+ File.extname(@name)[1..] if @name.include?(".")
189
123
  end
190
124
 
191
125
  # Check if this resource is a text file based on MIME type
@@ -206,7 +140,9 @@ module ActionMCP
206
140
  #
207
141
  # @return [String, nil] The path component of the URI
208
142
  def path
209
- URI(@uri).path rescue nil
143
+ URI(@uri).path
144
+ rescue StandardError
145
+ nil
210
146
  end
211
147
 
212
148
  # Generate a hash representation of the resource
@@ -0,0 +1,93 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActionMCP
4
+ module Client
5
+ # Base collection class for MCP client collections
6
+ class Collection
7
+ include RequestTimeouts
8
+
9
+ attr_reader :client, :loaded
10
+
11
+ def initialize(items, client, silence_sql: true)
12
+ @collection_data = items || []
13
+ @client = client
14
+ @loaded = !@collection_data.empty?
15
+ @silence_sql = silence_sql
16
+ end
17
+
18
+ def all
19
+ silence_logs { load_items unless @loaded }
20
+ @collection_data
21
+ end
22
+
23
+ def all!(timeout: DEFAULT_TIMEOUT)
24
+ silence_logs { load_items(force: true, timeout: timeout) }
25
+ @collection_data
26
+ end
27
+
28
+ # Filter items based on a given block
29
+ #
30
+ # @yield [item] Block that determines whether to include an item
31
+ # @yieldparam item [Object] An item from the collection
32
+ # @yieldreturn [Boolean] true to include the item, false to exclude it
33
+ # @return [Array<Object>] Items that match the filter criteria
34
+ def filter(&block)
35
+ all.select(&block)
36
+ end
37
+
38
+ # Number of items in the collection
39
+ #
40
+ # @return [Integer] The number of items
41
+ def size
42
+ all.size
43
+ end
44
+
45
+ # Implements enumerable functionality
46
+ include Enumerable
47
+
48
+ def each(&block)
49
+ all.each(&block)
50
+ end
51
+
52
+ alias loaded? loaded
53
+
54
+ protected
55
+
56
+ def load_items(force: false, timeout: DEFAULT_TIMEOUT)
57
+ return if @loaded && !force
58
+
59
+ # Make sure @load_method is defined in the subclass
60
+ raise NotImplementedError, "Subclass must define @load_method" unless defined?(@load_method)
61
+
62
+ # Use the RequestTimeouts module to handle the request
63
+ load_with_timeout(@load_method, force: force, timeout: timeout)
64
+ end
65
+
66
+ private
67
+
68
+ def silence_logs
69
+ return yield unless @silence_sql
70
+
71
+ original_log_level = Session.logger&.level
72
+ begin
73
+ # Temporarily increase log level to suppress SQL queries
74
+ Session.logger.level = Logger::WARN if Session.logger
75
+ yield
76
+ ensure
77
+ # Restore original log level
78
+ Session.logger.level = original_log_level if Session.logger
79
+ end
80
+ end
81
+
82
+ def log_error(message)
83
+ # Safely handle logging - don't assume Rails.logger exists
84
+ if defined?(Rails) && Rails.respond_to?(:logger) && Rails.logger
85
+ Rails.logger.error("[#{self.class.name}] #{message}")
86
+ else
87
+ # Fall back to puts if Rails.logger is not available
88
+ puts "[ERROR] [#{self.class.name}] #{message}"
89
+ end
90
+ end
91
+ end
92
+ end
93
+ end
@@ -4,6 +4,7 @@ module ActionMCP
4
4
  module Client
5
5
  class JsonRpcHandler < JsonRpcHandlerBase
6
6
  attr_reader :client
7
+
7
8
  def initialize(transport, client)
8
9
  super(transport)
9
10
  @client = client
@@ -70,25 +71,30 @@ module ActionMCP
70
71
 
71
72
  def process_response(id, result)
72
73
  if transport.id == id
73
- ## This initialize the transport
74
+ ## This initializes the transport
74
75
  client.server = Client::Server.new(result)
75
76
  return send_initialized_notification
76
77
  end
78
+
77
79
  request = transport.messages.requests.find_by(jsonrpc_id: id)
78
80
  return unless request
81
+
82
+ # Mark the request as acknowledged
83
+ request.update(request_acknowledged: true)
84
+
79
85
  case request.rpc_method
80
86
  when "tools/list"
81
- client.toolbox.tools = result["tools"]
82
- return client.toolbox.all
87
+ client.toolbox.tools = result["tools"]
88
+ return true
83
89
  when "prompts/list"
84
90
  client.prompt_book.prompts = result["prompts"]
85
- return client.prompt_book.all
91
+ return true
86
92
  when "resources/list"
87
93
  client.catalog.resources = result["resources"]
88
- return client.catalog.all
94
+ return true
89
95
  when "resources/templates/list"
90
96
  client.blueprint.templates = result["resourceTemplates"]
91
- return client.blueprint.all
97
+ return true
92
98
  end
93
99
 
94
100
  puts "\e[31mUnknown response: #{id} #{result}\e[0m"
@@ -99,7 +105,6 @@ module ActionMCP
99
105
  puts "\e[31mUnknown error: #{id} #{error}\e[0m"
100
106
  end
101
107
 
102
-
103
108
  def send_initialized_notification
104
109
  transport.initialize!
105
110
  client.send_jsonrpc_notification("notifications/initialized")
@@ -12,8 +12,7 @@ module ActionMCP
12
12
  # Send request
13
13
  send_jsonrpc_request("client/setLoggingLevel",
14
14
  params: { level: level },
15
- id: request_id
16
- )
15
+ id: request_id)
17
16
  end
18
17
  end
19
18
  end
@@ -24,27 +24,16 @@ module ActionMCP
24
24
  # # Get all prompts matching a criteria
25
25
  # text_prompts = book.filter { |p| p.name.include?("text") }
26
26
  #
27
- class PromptBook
27
+ class PromptBook < Collection
28
28
  # Initialize a new PromptBook with prompt definitions
29
29
  #
30
30
  # @param prompts [Array<Hash>] Array of prompt definition hashes, each containing
31
31
  # name, description, and arguments keys
32
32
  # @param client [Object, nil] Optional client for lazy loading of prompts
33
- attr_reader :client
34
-
35
33
  def initialize(prompts, client)
36
- self.prompts = prompts
37
- @client = client
38
- @loaded = !prompts.empty?
39
- end
40
-
41
- # Return all prompts in the collection. If initialized with a client and prompts
42
- # haven't been loaded yet, this will trigger lazy loading from the client.
43
- #
44
- # @return [Array<Prompt>] All prompt objects in the collection
45
- def all
46
- load_prompts unless @loaded
47
- @prompts
34
+ super(prompts, client)
35
+ self.prompts = @collection_data
36
+ @load_method = :list_prompts
48
37
  end
49
38
 
50
39
  # Find a prompt by name
@@ -55,16 +44,6 @@ module ActionMCP
55
44
  all.find { |prompt| prompt.name == name }
56
45
  end
57
46
 
58
- # Filter prompts based on a given block
59
- #
60
- # @yield [prompt] Block that determines whether to include a prompt
61
- # @yieldparam prompt [Prompt] A prompt from the collection
62
- # @yieldreturn [Boolean] true to include the prompt, false to exclude it
63
- # @return [Array<Prompt>] Prompts that match the filter criteria
64
- def filter(&block)
65
- all.select(&block)
66
- end
67
-
68
47
  # Get a list of all prompt names
69
48
  #
70
49
  # @return [Array<String>] Names of all prompts in the collection
@@ -72,13 +51,6 @@ module ActionMCP
72
51
  all.map(&:name)
73
52
  end
74
53
 
75
- # Number of prompts in the collection
76
- #
77
- # @return [Integer] The number of prompts
78
- def size
79
- all.size
80
- end
81
-
82
54
  # Check if the collection contains a prompt with the given name
83
55
  #
84
56
  # @param name [String] The prompt name to check for
@@ -87,49 +59,11 @@ module ActionMCP
87
59
  all.any? { |prompt| prompt.name == name }
88
60
  end
89
61
 
90
- # Implements enumerable functionality for the collection
91
- include Enumerable
92
-
93
- # Yield each prompt in the collection to the given block
94
- #
95
- # @yield [prompt] Block to execute for each prompt
96
- # @yieldparam prompt [Prompt] A prompt from the collection
97
- # @return [Enumerator] If no block is given
98
- def each(&block)
99
- all.each(&block)
100
- end
101
-
102
- # Force reload all prompts from the client and return them
103
- #
104
- # @return [Array<Prompt>] All prompt objects in the collection
105
- def all!
106
- load_prompts(force: true)
107
- all
108
- end
109
-
110
62
  # Convert raw prompt data into Prompt objects
111
63
  #
112
64
  # @param prompts [Array<Hash>] Array of prompt definition hashes
113
65
  def prompts=(prompts)
114
- @prompts = prompts.map { |data| Prompt.new(data) }
115
- end
116
-
117
- private
118
-
119
- # Load or reload prompts using the client
120
- #
121
- # @param force [Boolean] Whether to force reload even if prompts are already loaded
122
- # @return [void]
123
- def load_prompts(force: false)
124
- return if @loaded && !force
125
-
126
- begin
127
- @client.list_prompts
128
- @loaded = true
129
- rescue StandardError => e
130
- Rails.logger.error("Failed to load prompts: #{e.message}")
131
- @loaded = true unless @prompts.empty?
132
- end
66
+ @collection_data = prompts.map { |data| Prompt.new(data) }
133
67
  end
134
68
 
135
69
  # Internal Prompt class to represent individual prompts
@@ -4,18 +4,21 @@ module ActionMCP
4
4
  module Client
5
5
  module Prompts
6
6
  # List all available prompts from the server
7
- # @return [Array<Hash>] List of available prompts with their metadata
7
+ # @return [String] Request ID for tracking the request
8
8
  def list_prompts
9
9
  request_id = SecureRandom.uuid_v7
10
10
 
11
11
  # Send request
12
12
  send_jsonrpc_request("prompts/list", id: request_id)
13
+
14
+ # Return request ID for tracking the request
15
+ request_id
13
16
  end
14
17
 
15
18
  # Get a specific prompt with arguments
16
19
  # @param name [String] Name of the prompt to get
17
20
  # @param arguments [Hash] Arguments to pass to the prompt
18
- # @return [Hash] Prompt content with messages
21
+ # @return [String] Request ID for tracking the request
19
22
  def get_prompt(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
@@ -0,0 +1,74 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActionMCP
4
+ module Client
5
+ module RequestTimeouts
6
+ # Default timeout in seconds
7
+ DEFAULT_TIMEOUT = 1.0
8
+
9
+ # Load resources with timeout support - blocking until response or timeout
10
+ # @param method_name [Symbol] The method to call for loading (e.g., :list_resources)
11
+ # @param force [Boolean] Whether to force reload even if already loaded
12
+ # @param timeout [Float] Timeout in seconds
13
+ # @return [Boolean] Success status
14
+ def load_with_timeout(method_name, force: false, timeout: DEFAULT_TIMEOUT)
15
+ return true if @loaded && !force
16
+
17
+ # Make the request and store its ID
18
+ request_id = client.send(method_name)
19
+
20
+ start_time = Time.now
21
+
22
+ # Wait until either:
23
+ # 1. The collection is loaded (@loaded becomes true from JsonRpcHandler)
24
+ # 2. The timeout is reached
25
+ sleep(0.1) while !@loaded && (Time.now - start_time) < timeout
26
+
27
+ # If we timed out
28
+ unless @loaded
29
+ request = client.session.messages.requests.find_by(jsonrpc_id: request_id)
30
+
31
+ if request && !request.request_acknowledged?
32
+ # Send cancel notification
33
+ client.send_jsonrpc_notification("notifications/cancelled", {
34
+ requestId: request_id,
35
+ reason: "Request timed out after #{timeout} seconds"
36
+ })
37
+
38
+ # Mark as cancelled in the database
39
+ request.update(request_cancelled: true)
40
+
41
+ log_error("Request #{method_name} timed out after #{timeout} seconds")
42
+ end
43
+
44
+ # Mark as loaded even though we timed out
45
+ @loaded = true
46
+ return false
47
+ end
48
+
49
+ # Collection was successfully loaded
50
+ true
51
+ end
52
+
53
+ private
54
+
55
+ def handle_timeout(request_id, method_name, timeout)
56
+ # Find the request
57
+ request = client.session.messages.requests.find_by(jsonrpc_id: request_id)
58
+
59
+ return unless request && !request.request_acknowledged?
60
+
61
+ # Send cancel notification
62
+ client.send_jsonrpc_notification("notifications/cancelled", {
63
+ requestId: request_id,
64
+ reason: "Request timed out after #{timeout} seconds"
65
+ })
66
+
67
+ # Mark as cancelled in the database
68
+ request.update(request_cancelled: true)
69
+
70
+ log_error("Request #{method_name} timed out after #{timeout} seconds")
71
+ end
72
+ end
73
+ end
74
+ end