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 +4 -4
- data/README.md +27 -1
- data/app/models/action_mcp/session/message.rb +2 -1
- data/lib/action_mcp/content/resource.rb +1 -1
- data/lib/action_mcp/instrumentation/controller_runtime.rb +3 -4
- data/lib/action_mcp/log_subscriber.rb +19 -18
- data/lib/action_mcp/resource_template.rb +9 -35
- data/lib/action_mcp/tool_response.rb +8 -0
- data/lib/action_mcp/transport/resources.rb +1 -1
- data/lib/action_mcp/uri_ambiguity_checker.rb +74 -0
- data/lib/action_mcp/version.rb +1 -1
- data/lib/tasks/action_mcp_tasks.rake +1 -1
- metadata +7 -4
- data/lib/action_mcp/instrumentation/log_subscriber.rb +0 -39
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: cb964a85c5ae8f64c05730e054c3e252c22e8fe3eeea6230496929948ea51f68
|
4
|
+
data.tar.gz: a7cfd82678922926ef8beb28cc0346ebb583f826d1cc9ff9d8d54f55cecd08d7
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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 =
|
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 =
|
15
|
+
mcp_rt_before_render = LogSubscriber.reset_runtime
|
16
16
|
runtime = super
|
17
|
-
mcp_rt_after_render =
|
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) +
|
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
|
5
|
-
#
|
6
|
-
|
7
|
-
|
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
|
-
#
|
10
|
-
Thread.current[:
|
11
|
-
Thread.current[:
|
12
|
-
end
|
9
|
+
# Reset both counters
|
10
|
+
Thread.current[:mcp_tool_runtime] = 0
|
11
|
+
Thread.current[:mcp_prompt_runtime] = 0
|
13
12
|
|
14
|
-
|
15
|
-
|
16
|
-
|
13
|
+
# Return the total runtime
|
14
|
+
total_rt
|
15
|
+
end
|
17
16
|
|
18
|
-
|
19
|
-
|
20
|
-
|
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
|
-
|
23
|
-
Thread.current[:
|
24
|
-
Thread.current[:
|
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,
|
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.
|
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
|
data/lib/action_mcp/version.rb
CHANGED
@@ -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.
|
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-
|
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.
|
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
|