actionmcp 0.16.0 → 0.19.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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 6794bfeea02de8502aa4b084548a93269149b9376117663d602ecfae54af8014
4
- data.tar.gz: 49bca81dc1f59d27fc8d27c1b8978a8def6d3414822f261ce1aee2a4ebfc45b0
3
+ metadata.gz: cb964a85c5ae8f64c05730e054c3e252c22e8fe3eeea6230496929948ea51f68
4
+ data.tar.gz: a7cfd82678922926ef8beb28cc0346ebb583f826d1cc9ff9d8d54f55cecd08d7
5
5
  SHA512:
6
- metadata.gz: 3f94f11e49df2549d1e43c4499f7fedfc9f7000279f6b08d2eb3ffb9e32add617a14277869fa3b34a31bde7aa2e6e66674f2e7b49cd57f5606daebbd060097f7
7
- data.tar.gz: 9ce4b731e107c76a1e057ea4ce33a90998d1d5700946f8e056eea3763535e12ca941ba875324914c910d375bbac04c12f4c0da25255d2d7cf5e746b9bd1fa19c
6
+ metadata.gz: f6c12f6088da3ca4e642fe9aac72d7caeacd9630619c61a1d2c93006e87ab2da7d35eb25467e086d6ecb5e3109546f0f61285e6f9c175183fb8b1365f7cbae96
7
+ data.tar.gz: 70cbe6bcbf594afcc7558f3c178771c5b2d9c06d00f7474288bfbe59d9da694ccef578c68a0c5cf9a6a571f4ef51f317ed2fa34d57600321364beffb5ba7fe77
data/README.md CHANGED
@@ -19,7 +19,7 @@ This means an AI (like an LLM) can request information or actions from your appl
19
19
  **ActionMCP** is targeted at developers building MCP-enabled applications.
20
20
  It simplifies the process of integrating Ruby and Rails apps with the MCP standard by providing a set of base classes and an easy-to-use server interface.
21
21
 
22
- Instead of implementing MCP support from scratch, you can subclass and configure the provided **Prompt**, **Tool**, and **ResourceTemplate** classes to expose your app's functionality to LLMs.
22
+ Instead of implementing MCP support from scratch, you can subclass and configure the provided **Prompt**, **Tool**, and **ResourceTemplate** classes to expose your app's functionality to LLMs.
23
23
 
24
24
  ActionMCP handles the underlying MCP message format and routing, so you can adhere to the open standard with minimal effort.
25
25
 
@@ -169,6 +169,32 @@ class ProductResourceTemplate < ApplicationMCPResTemplate
169
169
  )
170
170
  end
171
171
  end
172
+
173
+ # Example of callbacks:
174
+
175
+ ```ruby
176
+ before_resolve do |template|
177
+ logger.tagged("ProductsTemplate") { logger.info("Starting to resolve product: #{template.product_id}") }
178
+ end
179
+
180
+ after_resolve do |template|
181
+ logger.tagged("ProductsTemplate") { logger.info("Finished resolving product resource for product: #{template.product_id}") }
182
+ end
183
+
184
+ around_resolve do |template, block|
185
+ start_time = Time.current
186
+ logger.tagged("ProductsTemplate") { logger.info("Starting resolution for product: #{template.product_id}") }
187
+
188
+ resource = block.call
189
+
190
+ if resource
191
+ logger.tagged("ProductsTemplate") { logger.info("Product #{template.product_id} resolved successfully in #{Time.current - start_time}s") }
192
+ else
193
+ logger.tagged("ProductsTemplate") { logger.info("Product #{template.product_id} not found") }
194
+ end
195
+
196
+ resource
197
+ end
172
198
  ```
173
199
 
174
200
  Resource templates are automatically registered and used when LLMs request resources matching their patterns.
@@ -63,13 +63,14 @@ module ActionMCP
63
63
  begin
64
64
  parsed_json = MultiJson.load(payload)
65
65
  self.message_json = parsed_json
66
+ self.message_text = nil
66
67
  process_json_content(parsed_json)
67
68
  rescue MultiJson::ParseError
68
69
  self.message_type = "text"
69
70
  end
70
71
  else
71
72
  self.message_json = payload
72
- self.message_text = MultiJson.dump(payload)
73
+ self.message_text = nil
73
74
  process_json_content(payload)
74
75
  end
75
76
  end
@@ -17,7 +17,7 @@ module ActionMCP
17
17
  # @param mime_type [String] The MIME type of the resource.
18
18
  # @param text [String, nil] The text content of the resource (optional).
19
19
  # @param blob [String, nil] The base64-encoded blob of the resource (optional).
20
- def initialize(uri, mime_type, text: nil, blob: nil)
20
+ def initialize(uri, mime_type = "text/plain", text: nil, blob: nil)
21
21
  super("resource")
22
22
  @uri = uri
23
23
  @mime_type = mime_type
@@ -12,22 +12,21 @@ module ActionMCP
12
12
  attr_internal :mcp_runtime
13
13
 
14
14
  def cleanup_view_runtime
15
- mcp_rt_before_render = Instrumentation::LogSubscriber.reset_runtime
15
+ mcp_rt_before_render = LogSubscriber.reset_runtime
16
16
  runtime = super
17
- mcp_rt_after_render = Instrumentation::LogSubscriber.reset_runtime
17
+ mcp_rt_after_render = LogSubscriber.reset_runtime
18
18
  self.mcp_runtime = mcp_rt_before_render + mcp_rt_after_render
19
19
  runtime - mcp_rt_after_render
20
20
  end
21
21
 
22
22
  def append_info_to_payload(payload)
23
23
  super
24
- payload[:mcp_runtime] = (mcp_runtime || 0) + Instrumentation::LogSubscriber.reset_runtime
24
+ payload[:mcp_runtime] = (mcp_runtime || 0) + LogSubscriber.reset_runtime
25
25
  end
26
26
 
27
27
  class_methods do
28
28
  def log_process_action(payload)
29
29
  messages = super
30
- binding.irb
31
30
  mcp_runtime = payload[:mcp_runtime]
32
31
  messages << ("mcp: %.1fms" % mcp_runtime.to_f) if mcp_runtime
33
32
  messages
@@ -1,28 +1,29 @@
1
- # In log_subscriber.rb
2
1
  module ActionMCP
3
2
  class LogSubscriber < ActiveSupport::LogSubscriber
4
- def tool_call(event)
5
- # Try both debug and info to ensure output regardless of logger level
6
- debug "Tool: #{event.payload[:tool_name]} (#{event.duration.round(2)}ms)"
7
- info "Tool: #{event.payload[:tool_name]} (#{event.duration.round(2)}ms)"
3
+ def self.reset_runtime
4
+ # Get the combined runtime from both tool and prompt operations
5
+ tool_rt = Thread.current[:mcp_tool_runtime] || 0
6
+ prompt_rt = Thread.current[:mcp_prompt_runtime] || 0
7
+ total_rt = tool_rt + prompt_rt
8
8
 
9
- # Track total tool time for summary
10
- Thread.current[:tool_runtime] ||= 0
11
- Thread.current[:tool_runtime] += event.duration
12
- end
9
+ # Reset both counters
10
+ Thread.current[:mcp_tool_runtime] = 0
11
+ Thread.current[:mcp_prompt_runtime] = 0
13
12
 
14
- def prompt_call(event)
15
- # Add debug output to confirm method is called
16
- puts "LogSubscriber#prompt_call called with: #{event.name}"
13
+ # Return the total runtime
14
+ total_rt
15
+ end
17
16
 
18
- # Try both debug and info to ensure output regardless of logger level
19
- debug "Prompt: #{event.payload[:prompt_name]} (#{event.duration.round(2)}ms)"
20
- info "Prompt: #{event.payload[:prompt_name]} (#{event.duration.round(2)}ms)"
17
+ def tool_call(event)
18
+ Thread.current[:mcp_tool_runtime] ||= 0
19
+ Thread.current[:mcp_tool_runtime] += event.duration
20
+ end
21
21
 
22
- # Track total prompt time for summary
23
- Thread.current[:prompt_runtime] ||= 0
24
- Thread.current[:prompt_runtime] += event.duration
22
+ def prompt_call(event)
23
+ Thread.current[:mcp_prompt_runtime] ||= 0
24
+ Thread.current[:mcp_prompt_runtime] += event.duration
25
25
  end
26
+
26
27
  attach_to :action_mcp
27
28
  end
28
29
  end
@@ -9,12 +9,14 @@ module ActionMCP
9
9
  include ActiveModel::Validations
10
10
  include ResourceCallbacks
11
11
  include Logging
12
+ include UriAmbiguityChecker
12
13
 
13
14
  # Track all registered templates
14
15
  @registered_templates = []
15
16
 
16
17
  class << self
17
- attr_reader :registered_templates, :description, :uri_template, :mime_type, :template_name, :parameters
18
+ attr_reader :registered_templates, :description, :uri_template,
19
+ :mime_type, :template_name, :parameters
18
20
 
19
21
  def abstract?
20
22
  @abstract ||= false
@@ -194,40 +196,6 @@ module ActionMCP
194
196
  "URI template conflict detected: '#{new_template}' conflicts with existing template '#{registered_class.uri_template}' registered by #{registered_class.name}"
195
197
  end
196
198
  end
197
-
198
- # Determine if two normalized patterns could be ambiguous
199
- def are_potentially_ambiguous?(pattern1, pattern2)
200
- # If the patterns are exactly the same, they're definitely ambiguous
201
- return true if pattern1 == pattern2
202
-
203
- # Split into segments to compare structure
204
- segments1 = pattern1.split("/")
205
- segments2 = pattern2.split("/")
206
-
207
- # If different number of segments, they can't be ambiguous
208
- return false if segments1.size != segments2.size
209
-
210
- # Count parameter segments
211
- param_segments1 = segments1.count { |s| s.include?("{param}") }
212
- param_segments2 = segments2.count { |s| s.include?("{param}") }
213
-
214
- # If they have different number of parameter segments, they're not ambiguous
215
- return false if param_segments1 != param_segments2
216
-
217
- # If we have the same number of segments and same number of parameters,
218
- # but the patterns aren't identical, they could be ambiguous
219
- # due to parameter position swapping
220
- if param_segments1.positive? && param_segments1 == param_segments2
221
- # Create pattern maps (P for param, S for static)
222
- pattern_map1 = segments1.map { |s| s.include?("{param}") ? "P" : "S" }
223
- pattern_map2 = segments2.map { |s| s.include?("{param}") ? "P" : "S" }
224
-
225
- # If pattern maps are different but have same param count, potentially ambiguous
226
- return pattern_map1 != pattern_map2
227
- end
228
-
229
- false
230
- end
231
199
  end
232
200
 
233
201
  # Initialize with attribute values
@@ -249,5 +217,11 @@ module ActionMCP
249
217
  end
250
218
 
251
219
  attr_reader :description, :uri_template, :mime_type
220
+
221
+ def call
222
+ run_callbacks :resolve do
223
+ resolve
224
+ end
225
+ end
252
226
  end
253
227
  end
@@ -72,6 +72,14 @@ module ActionMCP
72
72
  [ contents, is_error ].hash
73
73
  end
74
74
 
75
+ def success?
76
+ !is_error
77
+ end
78
+
79
+ def error?
80
+ is_error
81
+ end
82
+
75
83
  # Pretty print for better debugging
76
84
  def inspect
77
85
  "#<#{self.class.name} content: #{contents.inspect}, isError: #{is_error}>"
@@ -47,7 +47,7 @@ module ActionMCP
47
47
  def send_resource_read(id, params)
48
48
  if (template = ResourceTemplatesRegistry.find_template_for_uri(params[:uri]))
49
49
  record = template.process(params[:uri])
50
- if (resource = record.resolve)
50
+ if (resource = record.call)
51
51
  # if resource is a array or a collection, return each item then it ok
52
52
  # else wrap it in a array
53
53
  resource = [ resource ] unless resource.respond_to?(:each)
@@ -0,0 +1,74 @@
1
+ module ActionMCP
2
+ module UriAmbiguityChecker
3
+ extend ActiveSupport::Concern
4
+
5
+ class_methods do
6
+ # Determines if a segment is a parameter
7
+ def parameter?(segment)
8
+ segment =~ /\A\{[a-z0-9_]+\}\z/
9
+ end
10
+
11
+ # Checks if two URI patterns could potentially match the same URI
12
+ def are_potentially_ambiguous?(pattern1, pattern2)
13
+ # If the patterns are exactly the same, they're definitely ambiguous
14
+ return true if pattern1 == pattern2
15
+
16
+ segments1 = pattern1.split("/")
17
+ segments2 = pattern2.split("/")
18
+
19
+ # If different number of segments, they can't be ambiguous
20
+ if segments1.size != segments2.size
21
+ return false
22
+ end
23
+
24
+ # Extract literals (non-parameters) from each pattern
25
+ literals1 = []
26
+ literals2 = []
27
+
28
+ segments1.each_with_index do |seg, i|
29
+ literals1 << [ seg, i ] unless parameter?(seg)
30
+ end
31
+
32
+ segments2.each_with_index do |seg, i|
33
+ literals2 << [ seg, i ] unless parameter?(seg)
34
+ end
35
+
36
+ # Check each segment for direct literal mismatches
37
+ segments1.zip(segments2).each_with_index do |(seg1, seg2), index|
38
+ param1 = parameter?(seg1)
39
+ param2 = parameter?(seg2)
40
+
41
+ # When both segments are literals, they must match exactly
42
+ if !param1 && !param2 && seg1 != seg2
43
+ return false
44
+ end
45
+ end
46
+
47
+ # Check for structural incompatibility in the literals
48
+ # If the same literals appear in different relative order, the patterns are structurally different
49
+ if literals1.size >= 2 && literals2.size >= 2
50
+ # Create arrays of just the literals (without positions)
51
+ lit_values1 = literals1.map(&:first)
52
+ lit_values2 = literals2.map(&:first)
53
+
54
+ # Find common literals
55
+ common_literals = lit_values1 & lit_values2
56
+
57
+ if common_literals.size >= 2
58
+ # Check if the relative ordering of common literals differs
59
+ common_literal_indices1 = common_literals.map { |lit| lit_values1.index(lit) }
60
+ common_literal_indices2 = common_literals.map { |lit| lit_values2.index(lit) }
61
+
62
+ # If the relative ordering is different, patterns are not ambiguous
63
+ if common_literal_indices1 != common_literal_indices2
64
+ return false
65
+ end
66
+ end
67
+ end
68
+
69
+ # If we got here, the patterns are potentially ambiguous
70
+ true
71
+ end
72
+ end
73
+ end
74
+ end
@@ -2,7 +2,7 @@
2
2
 
3
3
  require_relative "gem_version"
4
4
  module ActionMCP
5
- VERSION = "0.16.0"
5
+ VERSION = "0.19.0"
6
6
 
7
7
  class << self
8
8
  alias version gem_version
@@ -44,7 +44,7 @@ namespace :action_mcp do
44
44
  ActionMCP::ResourceTemplate.descendants.each do |resource|
45
45
  next if resource.abstract?
46
46
 
47
- puts "\e[33m#{resource.capability_name}:\e[0m #{resource.description}" # Yellow name
47
+ puts "\e[33m#{resource.capability_name}:\e[0m #{resource.description} : #{resource.uri_template}" # Yellow name
48
48
  end
49
49
  puts "\n"
50
50
  end
metadata CHANGED
@@ -1,13 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: actionmcp
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.16.0
4
+ version: 0.19.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Abdelkader Boudih
8
+ autorequire:
8
9
  bindir: exe
9
10
  cert_chain: []
10
- date: 2025-03-20 00:00:00.000000000 Z
11
+ date: 2025-03-23 00:00:00.000000000 Z
11
12
  dependencies:
12
13
  - !ruby/object:Gem::Dependency
13
14
  name: actioncable
@@ -136,7 +137,6 @@ files:
136
137
  - lib/action_mcp/gem_version.rb
137
138
  - lib/action_mcp/instrumentation/controller_runtime.rb
138
139
  - lib/action_mcp/instrumentation/instrumentation.rb
139
- - lib/action_mcp/instrumentation/log_subscriber.rb
140
140
  - lib/action_mcp/instrumentation/resource_instrumentation.rb
141
141
  - lib/action_mcp/integer_array.rb
142
142
  - lib/action_mcp/json_rpc.rb
@@ -177,6 +177,7 @@ files:
177
177
  - lib/action_mcp/transport/tools.rb
178
178
  - lib/action_mcp/transport/transport_base.rb
179
179
  - lib/action_mcp/transport_handler.rb
180
+ - lib/action_mcp/uri_ambiguity_checker.rb
180
181
  - lib/action_mcp/version.rb
181
182
  - lib/actionmcp.rb
182
183
  - lib/generators/action_mcp/install/install_generator.rb
@@ -198,6 +199,7 @@ metadata:
198
199
  source_code_uri: https://github.com/seuros/action_mcp
199
200
  changelog_uri: https://github.com/seuros/action_mcp/blob/master/CHANGELOG.md
200
201
  rubygems_mfa_required: 'true'
202
+ post_install_message:
201
203
  rdoc_options: []
202
204
  require_paths:
203
205
  - lib
@@ -212,7 +214,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
212
214
  - !ruby/object:Gem::Version
213
215
  version: '0'
214
216
  requirements: []
215
- rubygems_version: 3.6.5
217
+ rubygems_version: 3.5.22
218
+ signing_key:
216
219
  specification_version: 4
217
220
  summary: Provides essential tooling for building Model Context Protocol (MCP) capable
218
221
  servers
@@ -1,39 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module ActionMCP
4
- module Instrumentation
5
- # A log subscriber to attach to Elasticsearch related events
6
- #
7
- # @see https://github.com/rails/rails/blob/master/activerecord/lib/active_record/log_subscriber.rb
8
- #
9
- class LogSubscriber < ActiveSupport::LogSubscriber
10
- def self.runtime=(value)
11
- Thread.current["elasticsearch_runtime"] = value
12
- end
13
-
14
- def self.runtime
15
- Thread.current["elasticsearch_runtime"] ||= 0
16
- end
17
-
18
- def self.reset_runtime
19
- rt = runtime
20
- self.runtime = 0
21
- rt
22
- end
23
-
24
- # Intercept `search.elasticsearch` events, and display them in the Rails log
25
- #
26
- def search(event)
27
- self.class.runtime += event.duration
28
- return unless logger.debug?
29
-
30
- payload = event.payload
31
- name = "#{payload[:klass]} #{payload[:name]} (#{event.duration.round(1)}ms)"
32
- search = payload[:search].inspect.gsub(/:(\w+)=>/, '\1: ')
33
-
34
- debug %( #{color(name, GREEN, bold: true)} #{colorize_logging ? "\e[2m#{search}\e[0m" : search})
35
- end
36
- end
37
- end
38
- Instrumentation::LogSubscriber.attach_to :elasticsearch
39
- end