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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 6794bfeea02de8502aa4b084548a93269149b9376117663d602ecfae54af8014
4
- data.tar.gz: 49bca81dc1f59d27fc8d27c1b8978a8def6d3414822f261ce1aee2a4ebfc45b0
3
+ metadata.gz: e33a00e3a56b0dba226465afd8ce6695158400bb5a26876fa92f72cea65a5418
4
+ data.tar.gz: 9243f9316686638c66f716d333803db0b8ad0e32d052728b1ca1e1dc5eda21cb
5
5
  SHA512:
6
- metadata.gz: 3f94f11e49df2549d1e43c4499f7fedfc9f7000279f6b08d2eb3ffb9e32add617a14277869fa3b34a31bde7aa2e6e66674f2e7b49cd57f5606daebbd060097f7
7
- data.tar.gz: 9ce4b731e107c76a1e057ea4ce33a90998d1d5700946f8e056eea3763535e12ca941ba875324914c910d375bbac04c12f4c0da25255d2d7cf5e746b9bd1fa19c
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 = 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
@@ -14,7 +14,8 @@ module ActionMCP
14
14
  @registered_templates = []
15
15
 
16
16
  class << self
17
- attr_reader :registered_templates, :description, :uri_template, :mime_type, :template_name, :parameters
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.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)
@@ -2,7 +2,7 @@
2
2
 
3
3
  require_relative "gem_version"
4
4
  module ActionMCP
5
- VERSION = "0.16.0"
5
+ VERSION = "0.17.0"
6
6
 
7
7
  class << self
8
8
  alias version gem_version
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.16.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-20 00:00:00.000000000 Z
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