actionmcp 0.13.0 → 0.16.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 +153 -158
- data/Rakefile +1 -1
- data/app/controllers/action_mcp/{application_controller.rb → mcp_controller.rb} +3 -1
- data/app/controllers/action_mcp/messages_controller.rb +7 -5
- data/app/controllers/action_mcp/sse_controller.rb +19 -13
- data/app/models/action_mcp/session/message.rb +95 -90
- data/app/models/action_mcp/session/resource.rb +10 -6
- data/app/models/action_mcp/session/subscription.rb +9 -5
- data/app/models/action_mcp/session.rb +22 -13
- data/app/models/action_mcp.rb +2 -0
- data/config/routes.rb +2 -0
- data/db/migrate/20250308122801_create_action_mcp_sessions.rb +12 -10
- data/db/migrate/20250314230152_add_is_ping_to_session_message.rb +2 -0
- data/db/migrate/20250316005021_create_action_mcp_session_subscriptions.rb +3 -1
- data/db/migrate/20250316005649_create_action_mcp_session_resources.rb +4 -2
- data/exe/actionmcp_cli +57 -55
- data/lib/action_mcp/base_json_rpc_handler.rb +97 -0
- data/lib/action_mcp/callbacks.rb +122 -0
- data/lib/action_mcp/capability.rb +6 -3
- data/lib/action_mcp/client.rb +20 -26
- data/lib/action_mcp/client_json_rpc_handler.rb +69 -0
- data/lib/action_mcp/configuration.rb +8 -8
- data/lib/action_mcp/gem_version.rb +2 -0
- data/lib/action_mcp/instrumentation/controller_runtime.rb +38 -0
- data/lib/action_mcp/instrumentation/instrumentation.rb +26 -0
- data/lib/action_mcp/instrumentation/log_subscriber.rb +39 -0
- data/lib/action_mcp/instrumentation/resource_instrumentation.rb +40 -0
- data/lib/action_mcp/json_rpc/response.rb +18 -2
- data/lib/action_mcp/json_rpc_handler.rb +93 -21
- data/lib/action_mcp/log_subscriber.rb +28 -0
- data/lib/action_mcp/logging.rb +1 -3
- data/lib/action_mcp/prompt.rb +15 -6
- data/lib/action_mcp/prompt_response.rb +1 -1
- data/lib/action_mcp/prompts_registry.rb +1 -0
- data/lib/action_mcp/registry_base.rb +1 -0
- data/lib/action_mcp/resource_callbacks.rb +156 -0
- data/lib/action_mcp/resource_template.rb +18 -19
- data/lib/action_mcp/resource_templates_registry.rb +19 -25
- data/lib/action_mcp/sampling_request.rb +113 -0
- data/lib/action_mcp/server.rb +4 -1
- data/lib/action_mcp/server_json_rpc_handler.rb +90 -0
- data/lib/action_mcp/test_helper.rb +26 -9
- data/lib/action_mcp/tool.rb +12 -3
- data/lib/action_mcp/tool_response.rb +3 -2
- data/lib/action_mcp/tools_registry.rb +1 -1
- data/lib/action_mcp/transport/capabilities.rb +5 -1
- data/lib/action_mcp/transport/messaging.rb +2 -0
- data/lib/action_mcp/transport/prompts.rb +2 -0
- data/lib/action_mcp/transport/resources.rb +23 -6
- data/lib/action_mcp/transport/roots.rb +11 -0
- data/lib/action_mcp/transport/sampling.rb +14 -0
- data/lib/action_mcp/transport/sse_client.rb +11 -15
- data/lib/action_mcp/transport/stdio_client.rb +12 -14
- data/lib/action_mcp/transport/tools.rb +2 -0
- data/lib/action_mcp/transport/transport_base.rb +16 -15
- data/lib/action_mcp/transport.rb +2 -0
- data/lib/action_mcp/transport_handler.rb +3 -0
- data/lib/action_mcp/version.rb +1 -1
- data/lib/action_mcp.rb +8 -2
- data/lib/generators/action_mcp/install/install_generator.rb +4 -1
- data/lib/generators/action_mcp/install/templates/application_mcp_res_template.rb +2 -0
- data/lib/generators/action_mcp/resource_template/resource_template_generator.rb +2 -0
- data/lib/generators/action_mcp/resource_template/templates/resource_template.rb.erb +1 -1
- data/lib/tasks/action_mcp_tasks.rake +11 -6
- metadata +27 -14
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 6794bfeea02de8502aa4b084548a93269149b9376117663d602ecfae54af8014
|
4
|
+
data.tar.gz: 49bca81dc1f59d27fc8d27c1b8978a8def6d3414822f261ce1aee2a4ebfc45b0
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 3f94f11e49df2549d1e43c4499f7fedfc9f7000279f6b08d2eb3ffb9e32add617a14277869fa3b34a31bde7aa2e6e66674f2e7b49cd57f5606daebbd060097f7
|
7
|
+
data.tar.gz: 9ce4b731e107c76a1e057ea4ce33a90998d1d5700946f8e056eea3763535e12ca941ba875324914c910d375bbac04c12f4c0da25255d2d7cf5e746b9bd1fa19c
|
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)
|
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
|
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 **
|
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,202 +39,202 @@ 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:
|
42
|
+
ActionMCP provides three core abstractions to streamline MCP server development:
|
43
43
|
|
44
|
-
|
44
|
+
### ActionMCP::Prompt
|
45
45
|
|
46
|
-
|
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
|
-
|
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
|
-
|
54
|
+
**Example:**
|
51
55
|
|
52
|
-
|
56
|
+
```ruby
|
57
|
+
class AnalyzeCodePrompt < ApplicationMCPPrompt
|
58
|
+
prompt_name "analyze_code"
|
59
|
+
description "Analyze code for potential improvements"
|
53
60
|
|
54
|
-
|
61
|
+
argument :language, description: "Programming language", default: "Ruby"
|
62
|
+
argument :code, description: "Code to explain", required: true
|
55
63
|
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
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
|
-
|
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
|
-
|
78
|
-
|
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
|
-
###
|
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
|
-
|
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
|
-
|
104
|
+
**Example:**
|
87
105
|
|
88
|
-
```
|
89
|
-
|
90
|
-
|
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
|
-
|
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
|
-
|
138
|
+
```ruby
|
139
|
+
sum_tool = CalculateSumTool.new(a: 5, b: 10)
|
140
|
+
result = sum_tool.call
|
141
|
+
```
|
98
142
|
|
99
|
-
|
143
|
+
### ActionMCP::ResourceTemplate
|
100
144
|
|
101
|
-
|
102
|
-
|
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
|
-
|
148
|
+
**Example:**
|
106
149
|
|
107
150
|
```ruby
|
108
151
|
|
109
|
-
class
|
110
|
-
|
111
|
-
|
152
|
+
class ProductResourceTemplate < ApplicationMCPResTemplate
|
153
|
+
uri_template "product/{id}"
|
154
|
+
description "Access product information by ID"
|
112
155
|
|
113
|
-
|
114
|
-
description "Analyze code for potential improvements"
|
156
|
+
parameter :id, description: "Product identifier", required: true
|
115
157
|
|
116
|
-
|
117
|
-
argument :language, description: "Programming language", default: "Ruby"
|
118
|
-
argument :code, description: "Code to explain", required: true
|
119
|
-
|
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
|
124
|
-
|
125
|
-
|
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
172
|
```
|
129
173
|
|
130
|
-
|
174
|
+
Resource templates are automatically registered and used when LLMs request resources matching their patterns.
|
131
175
|
|
132
|
-
|
176
|
+
## Configuration
|
133
177
|
|
134
|
-
|
135
|
-
bin/rails generate action_mcp:tool CalculateSum
|
136
|
-
```
|
178
|
+
ActionMCP is configured via `config.action_mcp` in your Rails application.
|
137
179
|
|
138
|
-
|
180
|
+
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.
|
139
181
|
|
140
|
-
|
141
|
-
class CalculateSumTool < ApplicationMCPTool
|
142
|
-
tool_name "calculate_sum"
|
143
|
-
description "Calculate the sum of two numbers"
|
182
|
+
You can override these settings in your configuration (e.g., in `config/application.rb`):
|
144
183
|
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
184
|
+
```ruby
|
185
|
+
module Tron
|
186
|
+
class Application < Rails::Application
|
187
|
+
config.action_mcp.name = "Friendly MCP (Master Control Program)" # defaults to Rails.application.name
|
188
|
+
config.action_mcp.version = "1.2.3" # defaults to "0.0.1"
|
189
|
+
config.action_mcp.logging_enabled = true # defaults to true
|
190
|
+
config.action_mcp.logging_level = :info # defaults to :info, can be :debug, :info, :warn, :error, :fatal
|
150
191
|
end
|
151
192
|
end
|
152
193
|
```
|
153
194
|
|
154
|
-
|
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.
|
157
|
-
|
158
|
-
### ActionMCP::Tool
|
159
|
-
|
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.
|
161
|
-
|
162
|
-
### ActionMCP::Resource
|
163
|
-
|
164
|
-
*I don't need this for now.*
|
195
|
+
For dynamic versioning, consider adding the `rails_app_version` gem.
|
165
196
|
|
166
|
-
##
|
197
|
+
## Engine and Mounting
|
167
198
|
|
168
|
-
|
199
|
+
ActionMCP is implemented as a Rails engine, which means it can be mounted in your application's routes.
|
200
|
+
The engine provides no authentication or authorization by default, so you'll need to handle that in your application for now.
|
169
201
|
|
170
|
-
|
202
|
+
To mount the ActionMCP engine in your routes, add the following line to your `config/routes.rb`:
|
171
203
|
|
172
204
|
```ruby
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
# Optionally update attributes later:
|
177
|
-
analyze_prompt.code = "def goodbye; puts 'Goodbye!'; end"
|
178
|
-
|
179
|
-
# Validate the prompt before calling it
|
180
|
-
if analyze_prompt.valid?
|
181
|
-
result = analyze_prompt.call # => #<ActionMCP::Content::Text:0x00000001239398c8 @text="The code you provided is written in Ruby and looks great!", @type="text">
|
182
|
-
puts result.to_h # => {type: "text", text: "The code you provided is written in Ruby and looks great!"}
|
183
|
-
else
|
184
|
-
puts analyze_prompt.errors.full_messages
|
205
|
+
Rails.application.routes.draw do
|
206
|
+
mount ActionMCP::Engine => "/action_mcp"
|
185
207
|
end
|
186
208
|
```
|
187
209
|
|
188
|
-
|
210
|
+
## Generators
|
189
211
|
|
190
|
-
|
191
|
-
# Instantiate the tool with initial values
|
192
|
-
sum_tool = CalculateSumTool.new(a: 5, b: 10)
|
193
|
-
# Optionally update attributes later:
|
194
|
-
sum_tool.a = 15
|
195
|
-
sum_tool.b = 20
|
196
|
-
|
197
|
-
# Validate the tool before calling it
|
198
|
-
if sum_tool.valid?
|
199
|
-
result = sum_tool.call # => #<ActionMCP::Content::Text:0x0000000124cfaba0 @text="35.0", @type="text">
|
200
|
-
puts result.to_h # => {type: "text", text: "35.0"}
|
201
|
-
else
|
202
|
-
puts sum_tool.errors.full_messages
|
203
|
-
end
|
204
|
-
```
|
212
|
+
ActionMCP includes Rails generators to help you quickly set up your MCP server components.
|
205
213
|
|
206
|
-
|
214
|
+
You can generate the base classes for your MCP Prompt and Tool using the following command:
|
207
215
|
|
208
|
-
|
216
|
+
```bash
|
217
|
+
bin/rails action_mcp:install:migrations # to copy the migrations
|
218
|
+
bin/rails generate action_mcp:install
|
219
|
+
```
|
209
220
|
|
210
|
-
|
211
|
-
After creating the database with `bin/rails db:prepare`, you can run the dummy application using:
|
212
|
-
```bash
|
213
|
-
bin/rails s
|
214
|
-
```
|
215
|
-
This allows you to test and interact with the MCP server from the dummy environment.
|
216
|
-
- **Inspecting the App:**
|
217
|
-
You can use the mcp inspector to test your app ```npx @modelcontextprotocol/inspector```
|
218
|
-
the path by default will be http://localhost:3000/action_mcp
|
219
|
-
|
221
|
+
This will create the base application classes in your app directory.
|
220
222
|
|
221
|
-
|
222
|
-
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:
|
223
|
-
```bash
|
224
|
-
export PGGSSENCMODE=disable
|
225
|
-
export OBJC_DISABLE_INITIALIZE_FORK_SAFETY=YES
|
226
|
-
```
|
227
|
-
More details can be found in [Rails Issue #38560](https://github.com/rails/rails/issues/38560).
|
223
|
+
### Generate a New Prompt
|
228
224
|
|
229
|
-
|
230
|
-
|
225
|
+
```bash
|
226
|
+
bin/rails generate action_mcp:prompt AnalyzeCode
|
227
|
+
```
|
231
228
|
|
232
|
-
|
233
|
-
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.
|
229
|
+
### Generate a New Tool
|
234
230
|
|
235
|
-
|
231
|
+
```bash
|
232
|
+
bin/rails generate action_mcp:tool CalculateSum
|
233
|
+
```
|
234
|
+
|
235
|
+
## Testing with TestHelper
|
236
236
|
|
237
|
-
ActionMCP provides a `TestHelper` module to simplify testing of tools and prompts
|
237
|
+
ActionMCP provides a `TestHelper` module to simplify testing of tools and prompts:
|
238
238
|
|
239
239
|
```ruby
|
240
240
|
require "test_helper"
|
@@ -251,27 +251,22 @@ class ToolTest < ActiveSupport::TestCase
|
|
251
251
|
|
252
252
|
test "AnalyzeCodePrompt returns the correct analysis" do
|
253
253
|
assert_prompt_findable("analyze_code")
|
254
|
-
result =
|
254
|
+
result = execute_prompt("analyze_code", language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
|
255
255
|
assert_equal "Analyzing Ruby code: def hello; puts 'Hello, world!'; end", assert_prompt_output(result)
|
256
256
|
end
|
257
257
|
end
|
258
258
|
```
|
259
259
|
|
260
|
-
|
260
|
+
## Inspecting Your MCP Server
|
261
261
|
|
262
|
-
|
263
|
-
* `assert_prompt_findable(prompt_name)`: Asserts that a prompt is findable in the `PromptsRegistry`.
|
264
|
-
* `execute_tool(tool_name, args = {})`: Executes a tool with the given name and arguments.
|
265
|
-
* `execute_prompt(prompt_name, args = {})`: Executes a prompt with the given name and arguments.
|
266
|
-
* `assert_tool_output(result, expected_output)`: Asserts that the output of a tool is equal to the expected output.
|
267
|
-
* `assert_prompt_output(result)`: Asserts that the output of a prompt is equal to the expected output.
|
262
|
+
You can use the MCP Inspector to test your server implementation:
|
268
263
|
|
269
|
-
|
270
|
-
|
271
|
-
```ruby
|
272
|
-
require "action_mcp/test_helper"
|
264
|
+
```bash
|
265
|
+
npx @modelcontextprotocol/inspector
|
273
266
|
```
|
274
267
|
|
268
|
+
The default path will be http://localhost:3000/action_mcp
|
269
|
+
|
275
270
|
## Conclusion
|
276
271
|
|
277
|
-
ActionMCP empowers developers to build MCP-compliant servers efficiently by handling the standardization and boilerplate associated with integrating with LLMs. With
|
272
|
+
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
@@ -1,5 +1,7 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActionMCP
|
2
|
-
class
|
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 <
|
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
|
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 <
|
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
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
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 =
|
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
|
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
|