actionmcp 0.14.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.
Files changed (65) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +174 -144
  3. data/Rakefile +1 -1
  4. data/app/controllers/action_mcp/{application_controller.rb → mcp_controller.rb} +3 -1
  5. data/app/controllers/action_mcp/messages_controller.rb +7 -5
  6. data/app/controllers/action_mcp/sse_controller.rb +19 -13
  7. data/app/models/action_mcp/session/message.rb +95 -90
  8. data/app/models/action_mcp/session/resource.rb +10 -6
  9. data/app/models/action_mcp/session/subscription.rb +9 -5
  10. data/app/models/action_mcp/session.rb +22 -13
  11. data/app/models/action_mcp.rb +2 -0
  12. data/config/routes.rb +2 -0
  13. data/db/migrate/20250308122801_create_action_mcp_sessions.rb +12 -10
  14. data/db/migrate/20250314230152_add_is_ping_to_session_message.rb +2 -0
  15. data/db/migrate/20250316005021_create_action_mcp_session_subscriptions.rb +3 -1
  16. data/db/migrate/20250316005649_create_action_mcp_session_resources.rb +4 -2
  17. data/exe/actionmcp_cli +57 -55
  18. data/lib/action_mcp/base_json_rpc_handler.rb +97 -0
  19. data/lib/action_mcp/callbacks.rb +122 -0
  20. data/lib/action_mcp/capability.rb +6 -3
  21. data/lib/action_mcp/client.rb +20 -26
  22. data/lib/action_mcp/client_json_rpc_handler.rb +69 -0
  23. data/lib/action_mcp/configuration.rb +8 -8
  24. data/lib/action_mcp/content/resource.rb +1 -1
  25. data/lib/action_mcp/gem_version.rb +2 -0
  26. data/lib/action_mcp/instrumentation/controller_runtime.rb +37 -0
  27. data/lib/action_mcp/instrumentation/instrumentation.rb +26 -0
  28. data/lib/action_mcp/instrumentation/resource_instrumentation.rb +40 -0
  29. data/lib/action_mcp/json_rpc/response.rb +18 -2
  30. data/lib/action_mcp/json_rpc_handler.rb +93 -21
  31. data/lib/action_mcp/log_subscriber.rb +29 -0
  32. data/lib/action_mcp/logging.rb +1 -3
  33. data/lib/action_mcp/prompt.rb +15 -6
  34. data/lib/action_mcp/prompt_response.rb +1 -1
  35. data/lib/action_mcp/prompts_registry.rb +1 -0
  36. data/lib/action_mcp/registry_base.rb +1 -0
  37. data/lib/action_mcp/resource_callbacks.rb +156 -0
  38. data/lib/action_mcp/resource_template.rb +25 -19
  39. data/lib/action_mcp/resource_templates_registry.rb +19 -25
  40. data/lib/action_mcp/sampling_request.rb +113 -0
  41. data/lib/action_mcp/server.rb +4 -1
  42. data/lib/action_mcp/server_json_rpc_handler.rb +90 -0
  43. data/lib/action_mcp/test_helper.rb +6 -2
  44. data/lib/action_mcp/tool.rb +12 -3
  45. data/lib/action_mcp/tool_response.rb +3 -2
  46. data/lib/action_mcp/transport/capabilities.rb +5 -1
  47. data/lib/action_mcp/transport/messaging.rb +2 -0
  48. data/lib/action_mcp/transport/prompts.rb +2 -0
  49. data/lib/action_mcp/transport/resources.rb +23 -6
  50. data/lib/action_mcp/transport/roots.rb +11 -0
  51. data/lib/action_mcp/transport/sampling.rb +14 -0
  52. data/lib/action_mcp/transport/sse_client.rb +11 -15
  53. data/lib/action_mcp/transport/stdio_client.rb +12 -14
  54. data/lib/action_mcp/transport/tools.rb +2 -0
  55. data/lib/action_mcp/transport/transport_base.rb +16 -15
  56. data/lib/action_mcp/transport.rb +2 -0
  57. data/lib/action_mcp/transport_handler.rb +3 -0
  58. data/lib/action_mcp/version.rb +1 -1
  59. data/lib/action_mcp.rb +8 -2
  60. data/lib/generators/action_mcp/install/install_generator.rb +4 -1
  61. data/lib/generators/action_mcp/install/templates/application_mcp_res_template.rb +2 -0
  62. data/lib/generators/action_mcp/resource_template/resource_template_generator.rb +2 -0
  63. data/lib/generators/action_mcp/resource_template/templates/resource_template.rb.erb +1 -1
  64. data/lib/tasks/action_mcp_tasks.rake +11 -6
  65. metadata +26 -14
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a853f65451f67bb2da8d67e317ed0554d832ecb0660b8adbe9d002c949792d43
4
- data.tar.gz: 23212ca96b1538d71ed309a2e1f6b3e5143917ff117827b46306d4384ca0b0f0
3
+ metadata.gz: e33a00e3a56b0dba226465afd8ce6695158400bb5a26876fa92f72cea65a5418
4
+ data.tar.gz: 9243f9316686638c66f716d333803db0b8ad0e32d052728b1ca1e1dc5eda21cb
5
5
  SHA512:
6
- metadata.gz: e4701665f0d8912fe33213d17e0e3d69ff01cc3adac1d69d9d461963c366fca879d09b9be06a2bdfb6579e8bdbcb842eb349bcde45f53316162a58be382f62cb
7
- data.tar.gz: 59dfe42ba51f7bae693bbcee84833961eb3f1ea1312e046d105760328a2459842968d85e67289a411c57ddbc9d0bad54aef59ea54e37053fa5bb26360b3c4384
6
+ metadata.gz: d0bdf008e20cbe4bc36d4a295201eb16570fd5e5dc83c24fef759daac1c622c6b58ce16f671db06d326714587e7ecca1b22234829d9c59ea40759b9c78b8e976
7
+ data.tar.gz: 948bf31356b50d4004093890ca8a336a1e3eb41a85ce95d46e01a41f1a1823f0a2ff20fd849ab2eccda1b7569db9573ab0c84911ea263c2db0b85f2f20c07cdc
data/README.md CHANGED
@@ -1,27 +1,27 @@
1
1
  # ActionMCP
2
2
 
3
- **ActionMCP** is a Ruby gem that provides essential tooling for building Model Context Protocol (MCP) capable servers.
3
+ **ActionMCP** is a Ruby gem that provides essential tooling for building Model Context Protocol (MCP) capable servers in Ruby on Rails applications.
4
4
 
5
- It offers base classes and helpers for creating MCP applications, making it easier to integrate your Ruby/Rails application with the MCP standard.
5
+ It offers base classes and helpers for creating MCP applications, making it easier to integrate your Ruby/Rails application with the MCP standard.
6
6
 
7
7
  With ActionMCP, you can focus on your app's logic while it handles the boilerplate for MCP compliance.
8
8
 
9
9
  ## Introduction
10
10
 
11
- **Model Context Protocol (MCP)** is an open protocol that standardizes how applications provide context to large language models (LLMs) ([Introduction - Model Context Protocol](https://modelcontextprotocol.io/introduction#:~:text=MCP%20is%20an%20open%20protocol,different%20data%20sources%20and%20tools)).
11
+ **Model Context Protocol (MCP)** is an open protocol that standardizes how applications provide context to large language models (LLMs).
12
12
 
13
- Think of it as a universal interface for connecting AI assistants to external data sources and tools.
13
+ Think of it as a universal interface for connecting AI assistants to external data sources and tools.
14
14
 
15
- MCP allows AI systems to plug into various resources in a consistent, secure way, enabling two-way integration between your data and AI-powered applications ([Introducing the Model Context Protocol \ Anthropic](https://www.anthropic.com/news/model-context-protocol#:~:text=The%20Model%20Context%20Protocol%20is,that%20connect%20to%20these%20servers)).
15
+ MCP allows AI systems to plug into various resources in a consistent, secure way, enabling two-way integration between your data and AI-powered applications.
16
16
 
17
17
  This means an AI (like an LLM) can request information or actions from your application through a well-defined protocol, and your app can provide context or perform tasks for the AI in return.
18
18
 
19
- **ActionMCP** is targeted at developers building MCP-enabled applications.
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.
19
+ **ActionMCP** is targeted at developers building MCP-enabled applications.
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 **Resource** classes to expose your apps 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
- ActionMCP handles the underlying MCP message format and routing, so you can adhere to the open standard with minimal effort.
24
+ ActionMCP handles the underlying MCP message format and routing, so you can adhere to the open standard with minimal effort.
25
25
 
26
26
  In short, ActionMCP helps you build an MCP server (the component that exposes capabilities to AI) more quickly and with fewer mistakes.
27
27
 
@@ -39,193 +39,228 @@ This will load the ActionMCP library so you can start defining MCP prompts, tool
39
39
 
40
40
  ## Core Components
41
41
 
42
- ActionMCP provides three core abstractions to streamline MCP server development: **Prompt**, **Tool**, and **Resource**.
42
+ ActionMCP provides three core abstractions to streamline MCP server development:
43
43
 
44
- These correspond to key MCP concepts and let you define what context or capabilities your server exposes to LLMs.
44
+ ### ActionMCP::Prompt
45
45
 
46
- Note that ActionMCP requires a Rails application; it is not meant for standalone Ruby apps.
46
+ `ActionMCP::Prompt` enables you to create reusable prompt templates that can be discovered and used by LLMs. Each prompt is defined as a Ruby class that inherits from `ApplicationMCPPrompt`.
47
47
 
48
- ### Configuration
48
+ Key features:
49
+ - Define expected arguments with descriptions and validation rules
50
+ - Build multi-step conversations with mixed content types
51
+ - Support for text, images, audio, and resource attachments
52
+ - Add messages with different roles (user/assistant)
49
53
 
50
- ActionMCP is configured via `config.action_mcp` in your Rails application.
54
+ **Example:**
51
55
 
52
- By default, the name is set to your application's name and the version defaults to "0.0.1" unless your app has a version file.
56
+ ```ruby
57
+ class AnalyzeCodePrompt < ApplicationMCPPrompt
58
+ prompt_name "analyze_code"
59
+ description "Analyze code for potential improvements"
53
60
 
54
- You can override these settings in your configuration (e.g., in `config/application.rb`):
61
+ argument :language, description: "Programming language", default: "Ruby"
62
+ argument :code, description: "Code to explain", required: true
55
63
 
56
- ```ruby
57
- module Tron
58
- class Application < Rails::Application
59
- config.action_mcp.name = "Friendly MCP (Master Control Program)" # defaults to Rails.application.name
60
- config.action_mcp.version = "1.2.3" # defaults to "0.0.1"
61
- config.action_mcp.logging_enabled = true # defaults to true
62
- config.action_mcp.logging_level = :info # defaults to :info, can be :debug, :info, :warn, :error, :fatal
64
+ validates :language, inclusion: { in: %w[Ruby Python JavaScript] }
65
+
66
+ def perform
67
+ render(text: "Please analyze this #{language} code for improvements:")
68
+ render(text: code)
69
+
70
+ # You can add assistant messages too
71
+ render(text: "Here are some things to focus on in your analysis:", role: :assistant)
72
+
73
+ # Even add resources if needed
74
+ render(resource: "file://documentation/#{language.downcase}_style_guide.pdf",
75
+ mime_type: "application/pdf",
76
+ blob: get_style_guide_pdf(language))
77
+ end
78
+
79
+ private
80
+
81
+ def get_style_guide_pdf(language)
82
+ # Implementation to retrieve style guide as base64
63
83
  end
64
84
  end
65
85
  ```
66
86
 
67
- For dynamic versioning, consider adding the `rails_app_version` gem.
68
-
69
- ### Engine
70
-
71
- ActionMCP is implemented as a Rails engine, which means it can be mounted in your application's routes.
72
- The engine provides no authentication or authorization by default, so you'll need to handle that in your application for now.
73
-
74
- To mount the ActionMCP engine in your routes, add the following line to your `config/routes.rb`:
87
+ Prompts can be executed by instantiating them and calling the `call` method:
75
88
 
76
89
  ```ruby
77
- Rails.application.routes.draw do
78
- mount ActionMCP::Engine => "/action_mcp"
79
- end
90
+ analyze_prompt = AnalyzeCodePrompt.new(language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
91
+ result = analyze_prompt.call
80
92
  ```
81
93
 
82
- ### Generators
94
+ ### ActionMCP::Tool
95
+
96
+ `ActionMCP::Tool` allows you to create interactive functions that LLMs can call with arguments to perform specific tasks. Each tool is a Ruby class that inherits from `ApplicationMCPTool`.
83
97
 
84
- ActionMCP includes Rails generators to help you quickly set up your MCP server components.
98
+ Key features:
99
+ - Define input properties with types, descriptions, and validation
100
+ - Return multiple response types (text, images, errors)
101
+ - Progressive responses with multiple render calls
102
+ - Automatic input validation based on property definitions
85
103
 
86
- You can generate the base classes for your MCP Prompt and Tool using the following command:
104
+ **Example:**
87
105
 
88
- ```bash
89
- bin/rails action_mcp:install:migrations # to copy the migrations
90
- bin/rails generate action_mcp:install
106
+ ```ruby
107
+ class CalculateSumTool < ApplicationMCPTool
108
+ tool_name "calculate_sum"
109
+ description "Calculate the sum of two numbers"
110
+
111
+ property :a, type: "number", description: "First number", required: true
112
+ property :b, type: "number", description: "Second number", required: true
113
+
114
+ def perform
115
+ sum = a + b
116
+ render(text: "Calculating #{a} + #{b}...")
117
+ render(text: "The sum is #{sum}")
118
+
119
+ # You can render errors if needed
120
+ if sum > 1000
121
+ render(error: ["Warning: Sum exceeds recommended limit"])
122
+ end
123
+
124
+ # Or even images
125
+ render(image: generate_visualization(a, b), mime_type: "image/png")
126
+ end
127
+
128
+ private
129
+
130
+ def generate_visualization(a, b)
131
+ # Implementation to create a visualization as base64
132
+ end
133
+ end
91
134
  ```
92
135
 
93
- This command will create:
94
- - `app/mcp/prompts/application_prompt.rb`
95
- - `app/mcp/tools/application_tool.rb`
136
+ Tools can be executed by instantiating them and calling the `call` method:
96
137
 
97
- #### Generate a New Prompt
138
+ ```ruby
139
+ sum_tool = CalculateSumTool.new(a: 5, b: 10)
140
+ result = sum_tool.call
141
+ ```
98
142
 
99
- Run the following command to generate a new prompt class:
143
+ ### ActionMCP::ResourceTemplate
100
144
 
101
- ```bash
102
- bin/rails generate action_mcp:prompt AnalyzeCode
103
- ```
145
+ `ActionMCP::ResourceTemplate` facilitates the creation of URI templates for dynamic resources that LLMs can access.
146
+ This allows models to request specific data using parameterized URIs.
104
147
 
105
- This command will create a file at `app/mcp/prompts/analyze_code_prompt.rb` with content similar to:
148
+ **Example:**
106
149
 
107
150
  ```ruby
108
151
 
109
- class AnalyzeCodePrompt < ApplicationMCPPrompt
110
- # Override the prompt_name (otherwise we'd get "analyze_code")
111
- prompt_name "analyze-code"
112
-
113
- # Provide a user-facing description for your prompt.
114
- description "Analyze code for potential improvements"
152
+ class ProductResourceTemplate < ApplicationMCPResTemplate
153
+ uri_template "product/{id}"
154
+ description "Access product information by ID"
115
155
 
116
- # Configure arguments via the new DSL
117
- argument :language, description: "Programming language", default: "Ruby"
118
- argument :code, description: "Code to explain", required: true
156
+ parameter :id, description: "Product identifier", required: true
119
157
 
120
- # Add validations
121
- validates :language, inclusion: { in: %w[Ruby C Cobol FORTRAN] }
158
+ validates :id, format: { with: /\A\d+\z/, message: "must be numeric" }
122
159
 
123
- def perform
124
- # Implement your prompt logic here
125
- render(text: "Analyzing #{language} code: #{code}")
160
+ def resolve
161
+ product = Product.find_by(id: id)
162
+ return unless product
163
+ ActionMCP::Resource.new(
164
+ uri: "ecommerce://products/#{product_id}",
165
+ name: "Product #{product_id}",
166
+ description: "Product information for product #{product_id}",
167
+ mime_type: "application/json",
168
+ size: product.to_json.length
169
+ )
126
170
  end
127
171
  end
128
- ```
129
172
 
130
- #### Generate a New Tool
173
+ # Example of callbacks:
131
174
 
132
- Similarly, run the following command to generate a new tool class:
175
+ ```ruby
176
+ before_resolve do |template|
177
+ logger.tagged("ProductsTemplate") { logger.info("Starting to resolve product: #{template.product_id}") }
178
+ end
133
179
 
134
- ```bash
135
- bin/rails generate action_mcp:tool CalculateSum
136
- ```
180
+ after_resolve do |template|
181
+ logger.tagged("ProductsTemplate") { logger.info("Finished resolving product resource for product: #{template.product_id}") }
182
+ end
137
183
 
138
- This command will create a file at `app/mcp/tools/calculate_sum_tool.rb` with content similar to:
184
+ around_resolve do |template, block|
185
+ start_time = Time.current
186
+ logger.tagged("ProductsTemplate") { logger.info("Starting resolution for product: #{template.product_id}") }
139
187
 
140
- ```ruby
141
- class CalculateSumTool < ApplicationMCPTool
142
- tool_name "calculate_sum"
143
- description "Calculate the sum of two numbers"
188
+ resource = block.call
144
189
 
145
- property :a, type: "number", description: "First number", required: true
146
- property :b, type: "number", description: "Second number", required: true
147
-
148
- def perform
149
- render(text: a + b)
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") }
150
194
  end
195
+
196
+ resource
151
197
  end
152
198
  ```
153
199
 
154
- ### ActionMCP::Prompt
155
-
156
- A **Prompt** defines a question or request that an LLM can make to your application. It encapsulates the input parameters required for the request and any validations that need to be performed. For example, you might define a prompt called "analyze-code" that takes a code snippet as input and returns an analysis of the code.
200
+ Resource templates are automatically registered and used when LLMs request resources matching their patterns.
157
201
 
158
- ### ActionMCP::Tool
202
+ ## Configuration
159
203
 
160
- A **Tool** defines an action that your application can perform on behalf of an LLM. It encapsulates the input parameters required for the action and any logic that needs to be executed. For example, you might define a tool called "execute-command" that takes a shell command as input and executes it on the server, returning the output. This could be used to retrieve system information, run scripts, or perform other administrative tasks.
204
+ ActionMCP is configured via `config.action_mcp` in your Rails application.
161
205
 
162
- ### ActionMCP::Resource
206
+ By default, the name is set to your application's name and the version defaults to "0.0.1" unless your app has a version file.
163
207
 
164
- *I don't need this for now.*
208
+ You can override these settings in your configuration (e.g., in `config/application.rb`):
165
209
 
166
- ## Usage Example
210
+ ```ruby
211
+ module Tron
212
+ class Application < Rails::Application
213
+ config.action_mcp.name = "Friendly MCP (Master Control Program)" # defaults to Rails.application.name
214
+ config.action_mcp.version = "1.2.3" # defaults to "0.0.1"
215
+ config.action_mcp.logging_enabled = true # defaults to true
216
+ config.action_mcp.logging_level = :info # defaults to :info, can be :debug, :info, :warn, :error, :fatal
217
+ end
218
+ end
219
+ ```
167
220
 
168
- Both Tool and Prompt classes are based on ActiveModel, which means they share the same initialization and validation behavior. You can instantiate them with initial values, update their attributes later if necessary, and then call the `call` method to execute the logic defined in your class.
221
+ For dynamic versioning, consider adding the `rails_app_version` gem.
169
222
 
170
- ### Example for a Prompt
223
+ ## Engine and Mounting
171
224
 
172
- ```ruby
173
- # Instantiate the prompt with initial values
174
- analyze_prompt = AnalyzeCodePrompt.new(language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
225
+ ActionMCP is implemented as a Rails engine, which means it can be mounted in your application's routes.
226
+ The engine provides no authentication or authorization by default, so you'll need to handle that in your application for now.
175
227
 
176
- # Optionally update attributes later:
177
- analyze_prompt.code = "def goodbye; puts 'Goodbye!'; end"
228
+ To mount the ActionMCP engine in your routes, add the following line to your `config/routes.rb`:
178
229
 
179
- result = analyze_prompt.call #=> #<ActionMCP::PromptResponse messages: [{role: "user", content: {type: "text", text: "The code you provided is written in Ruby and looks great!"}}]>
180
- puts result.to_h #=> {messages: [{role: "user", content: {type: "text", text: "The code you provided is written in Ruby and looks great!"}}]}
230
+ ```ruby
231
+ Rails.application.routes.draw do
232
+ mount ActionMCP::Engine => "/action_mcp"
233
+ end
181
234
  ```
182
235
 
183
- ### Example for a Tool
236
+ ## Generators
184
237
 
185
- ```ruby
186
- # Instantiate the tool with initial values
187
- sum_tool = CalculateSumTool.new(a: 5, b: 10)
188
- # Optionally update attributes later:
189
- sum_tool.a = 15
190
- sum_tool.b = 20
238
+ ActionMCP includes Rails generators to help you quickly set up your MCP server components.
191
239
 
192
- # Validate the tool before calling it
193
- result = sum_tool.call # => #<ActionMCP::ToolResponse content: [#<ActionMCP::Content::Text:0x000000012bb50f78 @type="text", @text="35.0">], isError: false>
194
- puts result.to_h # => {content: [{type: "text", text: "35.0"}]}
195
- ```
240
+ You can generate the base classes for your MCP Prompt and Tool using the following command:
196
241
 
197
- These examples show that both prompts and tools follow a consistent pattern for initialization, validation, and execution, making it easy to integrate them into your application logic.
242
+ ```bash
243
+ bin/rails action_mcp:install:migrations # to copy the migrations
244
+ bin/rails generate action_mcp:install
245
+ ```
198
246
 
199
- ## Examples & Important Notes
247
+ This will create the base application classes in your app directory.
200
248
 
201
- - **Running the Dummy App:**
202
- After creating the database with `bin/rails db:prepare`, you can run the dummy application using:
203
- ```bash
204
- bin/rails s
205
- ```
206
- This allows you to test and interact with the MCP server from the dummy environment.
207
- - **Inspecting the App:**
208
- You can use the mcp inspector to test your app ```npx @modelcontextprotocol/inspector```
209
- the path by default will be http://localhost:3000/action_mcp
210
-
249
+ ### Generate a New Prompt
211
250
 
212
- - **Postgres on macOS:**
213
- If you are using Postgres on macOS, you may encounter issues due to a bug in Puma and the `pg` gem. To work around this, set the following environment variables:
214
- ```bash
215
- export PGGSSENCMODE=disable
216
- export OBJC_DISABLE_INITIALIZE_FORK_SAFETY=YES
217
- ```
218
- More details can be found in [Rails Issue #38560](https://github.com/rails/rails/issues/38560).
251
+ ```bash
252
+ bin/rails generate action_mcp:prompt AnalyzeCode
253
+ ```
219
254
 
220
- - **Notifiers:**
221
- ActionMCP works with the ActiveCable Postgres notifier by default, but its architecture is flexible enough to support other notifier implementations.
255
+ ### Generate a New Tool
222
256
 
223
- - **API Stability:**
224
- The ActionMCP API is stable, though it is acceptable for improvements and changes to be introduced as we move forward. This approach ensures the gem stays modern and adaptable to evolving requirements.
257
+ ```bash
258
+ bin/rails generate action_mcp:tool CalculateSum
259
+ ```
225
260
 
226
- ## Testing with the TestHelper
261
+ ## Testing with TestHelper
227
262
 
228
- ActionMCP provides a `TestHelper` module to simplify testing of tools and prompts. To use the `TestHelper`, include it in your test class:
263
+ ActionMCP provides a `TestHelper` module to simplify testing of tools and prompts:
229
264
 
230
265
  ```ruby
231
266
  require "test_helper"
@@ -248,21 +283,16 @@ class ToolTest < ActiveSupport::TestCase
248
283
  end
249
284
  ```
250
285
 
251
- The `TestHelper` module provides the following methods:
252
-
253
- * `assert_tool_findable(tool_name)`: Asserts that a tool is findable in the `ToolsRegistry`.
254
- * `assert_prompt_findable(prompt_name)`: Asserts that a prompt is findable in the `PromptsRegistry`.
255
- * `execute_tool(tool_name, args = {})`: Executes a tool with the given name and arguments.
256
- * `execute_prompt(prompt_name, args = {})`: Executes a prompt with the given name and arguments.
257
- * `assert_tool_output(expected_output, result)`: Asserts that the output of a tool is equal to the expected output.
258
- * `assert_prompt_output(expected_output, result)`: Asserts that the output of a prompt is equal to the expected output.
286
+ ## Inspecting Your MCP Server
259
287
 
260
- To use the `TestHelper`, you need to require it in your `test_helper.rb` file:
288
+ You can use the MCP Inspector to test your server implementation:
261
289
 
262
- ```ruby
263
- require "action_mcp/test_helper"
290
+ ```bash
291
+ npx @modelcontextprotocol/inspector
264
292
  ```
265
293
 
294
+ The default path will be http://localhost:3000/action_mcp
295
+
266
296
  ## Conclusion
267
297
 
268
- ActionMCP empowers developers to build MCP-compliant servers efficiently by handling the standardization and boilerplate associated with integrating with LLMs. With built-in generators, clear configuration options, robust usage examples, important deployment considerations, and a helpful testing module, it is designed to accelerate development and integration work while remaining flexible for future enhancements.
298
+ ActionMCP empowers developers to build MCP-compliant servers efficiently by handling the standardization and boilerplate associated with integrating with LLMs. With its intuitive abstractions for tools, prompts, and resource templates, you can quickly expose your application's capabilities to AI models while maintaining full control over how they interact with your system.
data/Rakefile CHANGED
@@ -3,6 +3,6 @@
3
3
  require 'bundler/setup'
4
4
 
5
5
  APP_RAKEFILE = File.expand_path('test/dummy/Rakefile', __dir__)
6
- load "rails/tasks/engine.rake"
6
+ load 'rails/tasks/engine.rake'
7
7
 
8
8
  require 'bundler/gem_tasks'
@@ -1,5 +1,7 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
- class ApplicationController < ActionController::Metal
4
+ class MCPController < ActionController::Metal
3
5
  abstract!
4
6
  ActionController::API.without_modules(:StrongParameters, :ParamsWrapper).each do |left|
5
7
  include left
@@ -1,10 +1,14 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
- class MessagesController < ApplicationController
4
+ class MessagesController < MCPController
5
+ include Instrumentation::ControllerRuntime
6
+
3
7
  # @route POST / (sse_in)
4
8
  def create
5
9
  begin
6
10
  handle_post_message(clean_params, response)
7
- rescue => e
11
+ rescue StandardError
8
12
  head :internal_server_error
9
13
  end
10
14
  head response.status
@@ -21,9 +25,7 @@ module ActionMCP
21
25
  end
22
26
 
23
27
  def handle_post_message(params, response)
24
- if params[:method] == "initialize"
25
- mcp_session.initialize!
26
- end
28
+ mcp_session.initialize! if params[:method] == "initialize"
27
29
  json_rpc_handler.call(params)
28
30
 
29
31
  response.status = :accepted
@@ -1,5 +1,7 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
- class SSEController < ApplicationController
4
+ class SSEController < MCPController
3
5
  HEARTBEAT_INTERVAL = 30 # TODO: The frequency of pings SHOULD be configurable
4
6
  include ActionController::Live
5
7
 
@@ -26,14 +28,17 @@ module ActionMCP
26
28
  sse.write(message)
27
29
  message_received = true
28
30
  end
29
- sleep 1
30
- # Heartbeat loop
31
- unless message_received
32
- Rails.logger.warn "No message received within 1 second, closing connection for session: #{session_id}"
33
- error = JsonRpc::Response.new(id: SecureRandom.uuid_v7, error: JsonRpc::JsonRpcError.new(:server_error, message: "No message received within 1 second").to_h).to_h
34
- sse.write(error)
35
- return
36
- end
31
+ sleep 1
32
+ # Heartbeat loop
33
+ unless message_received
34
+ Rails.logger.warn "No message received within 1 second, closing connection for session: #{session_id}"
35
+ error = JsonRpc::Response.new(id: SecureRandom.uuid_v7,
36
+ error: JsonRpc::JsonRpcError.new(
37
+ :server_error, message: "No message received within 1 second"
38
+ ).to_h).to_h
39
+ sse.write(error)
40
+ return
41
+ end
37
42
 
38
43
  until response.stream.closed?
39
44
  sleep HEARTBEAT_INTERVAL
@@ -45,7 +50,7 @@ module ActionMCP
45
50
  end
46
51
  rescue ActionController::Live::ClientDisconnected, IOError => e
47
52
  Rails.logger.debug "SSE: Expected disconnection: #{e.message}"
48
- rescue => e
53
+ rescue StandardError => e
49
54
  Rails.logger.error "SSE: Unexpected error: #{e.class} - #{e.message}\n#{e.backtrace.join("\n")}"
50
55
  ensure
51
56
  response.stream.close
@@ -82,6 +87,7 @@ module ActionMCP
82
87
 
83
88
  class SSEListener
84
89
  attr_reader :session_key, :adapter
90
+
85
91
  delegate :session_key, :adapter, to: :@session
86
92
 
87
93
  # @param session [ActionMCP::Session]
@@ -94,19 +100,19 @@ module ActionMCP
94
100
  def start(&callback)
95
101
  Rails.logger.debug "Starting listener for channel: #{session_key}"
96
102
 
97
- success_callback = -> {
103
+ success_callback = lambda {
98
104
  puts "Successfully subscribed to channel: #{session_key}"
99
105
  @subscription_active = true
100
106
  }
101
107
 
102
108
  # Set up message callback
103
- message_callback = ->(raw_message) {
109
+ message_callback = lambda { |raw_message|
104
110
  begin
105
111
  # Try to parse the message if it's JSON
106
112
  message = MultiJson.load(raw_message)
107
113
  # Send the message to the callback
108
114
  callback.call(message) if callback && !@stopped
109
- rescue => e
115
+ rescue StandardError
110
116
  # Still try to send the raw message as a fallback
111
117
  callback.call(raw_message) if callback && !@stopped
112
118
  end