actionmcp 0.16.0 → 0.17.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/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 +8 -1
- data/lib/action_mcp/transport/resources.rb +1 -1
- data/lib/action_mcp/version.rb +1 -1
- metadata +2 -3
- 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: e33a00e3a56b0dba226465afd8ce6695158400bb5a26876fa92f72cea65a5418
|
4
|
+
data.tar.gz: 9243f9316686638c66f716d333803db0b8ad0e32d052728b1ca1e1dc5eda21cb
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d0bdf008e20cbe4bc36d4a295201eb16570fd5e5dc83c24fef759daac1c622c6b58ce16f671db06d326714587e7ecca1b22234829d9c59ea40759b9c78b8e976
|
7
|
+
data.tar.gz: 948bf31356b50d4004093890ca8a336a1e3eb41a85ce95d46e01a41f1a1823f0a2ff20fd849ab2eccda1b7569db9573ab0c84911ea263c2db0b85f2f20c07cdc
|
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.
|
@@ -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
|
@@ -14,7 +14,8 @@ module ActionMCP
|
|
14
14
|
@registered_templates = []
|
15
15
|
|
16
16
|
class << self
|
17
|
-
attr_reader :registered_templates, :description, :uri_template,
|
17
|
+
attr_reader :registered_templates, :description, :uri_template,
|
18
|
+
:mime_type, :template_name, :parameters
|
18
19
|
|
19
20
|
def abstract?
|
20
21
|
@abstract ||= false
|
@@ -249,5 +250,11 @@ module ActionMCP
|
|
249
250
|
end
|
250
251
|
|
251
252
|
attr_reader :description, :uri_template, :mime_type
|
253
|
+
|
254
|
+
def call
|
255
|
+
run_callbacks :resolve do
|
256
|
+
resolve
|
257
|
+
end
|
258
|
+
end
|
252
259
|
end
|
253
260
|
end
|
@@ -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)
|
data/lib/action_mcp/version.rb
CHANGED
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: actionmcp
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.17.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Abdelkader Boudih
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-03-
|
10
|
+
date: 2025-03-21 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: actioncable
|
@@ -136,7 +136,6 @@ files:
|
|
136
136
|
- lib/action_mcp/gem_version.rb
|
137
137
|
- lib/action_mcp/instrumentation/controller_runtime.rb
|
138
138
|
- lib/action_mcp/instrumentation/instrumentation.rb
|
139
|
-
- lib/action_mcp/instrumentation/log_subscriber.rb
|
140
139
|
- lib/action_mcp/instrumentation/resource_instrumentation.rb
|
141
140
|
- lib/action_mcp/integer_array.rb
|
142
141
|
- lib/action_mcp/json_rpc.rb
|
@@ -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
|