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.
- checksums.yaml +4 -4
- data/README.md +174 -144
- 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/content/resource.rb +1 -1
- data/lib/action_mcp/gem_version.rb +2 -0
- data/lib/action_mcp/instrumentation/controller_runtime.rb +37 -0
- data/lib/action_mcp/instrumentation/instrumentation.rb +26 -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 +29 -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 +25 -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 +6 -2
- data/lib/action_mcp/tool.rb +12 -3
- data/lib/action_mcp/tool_response.rb +3 -2
- 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 +26 -14
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
@@ -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,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:
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
```
|
129
172
|
|
130
|
-
|
173
|
+
# Example of callbacks:
|
131
174
|
|
132
|
-
|
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
|
-
|
135
|
-
|
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
|
-
|
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
|
-
|
141
|
-
class CalculateSumTool < ApplicationMCPTool
|
142
|
-
tool_name "calculate_sum"
|
143
|
-
description "Calculate the sum of two numbers"
|
188
|
+
resource = block.call
|
144
189
|
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
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
|
-
|
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
|
-
|
202
|
+
## Configuration
|
159
203
|
|
160
|
-
|
204
|
+
ActionMCP is configured via `config.action_mcp` in your Rails application.
|
161
205
|
|
162
|
-
|
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
|
-
|
208
|
+
You can override these settings in your configuration (e.g., in `config/application.rb`):
|
165
209
|
|
166
|
-
|
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
|
-
|
221
|
+
For dynamic versioning, consider adding the `rails_app_version` gem.
|
169
222
|
|
170
|
-
|
223
|
+
## Engine and Mounting
|
171
224
|
|
172
|
-
|
173
|
-
|
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
|
-
|
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
|
-
|
180
|
-
|
230
|
+
```ruby
|
231
|
+
Rails.application.routes.draw do
|
232
|
+
mount ActionMCP::Engine => "/action_mcp"
|
233
|
+
end
|
181
234
|
```
|
182
235
|
|
183
|
-
|
236
|
+
## Generators
|
184
237
|
|
185
|
-
|
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
|
-
|
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
|
-
|
242
|
+
```bash
|
243
|
+
bin/rails action_mcp:install:migrations # to copy the migrations
|
244
|
+
bin/rails generate action_mcp:install
|
245
|
+
```
|
198
246
|
|
199
|
-
|
247
|
+
This will create the base application classes in your app directory.
|
200
248
|
|
201
|
-
|
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
|
-
|
213
|
-
|
214
|
-
|
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
|
-
|
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
|
-
|
224
|
-
|
257
|
+
```bash
|
258
|
+
bin/rails generate action_mcp:tool CalculateSum
|
259
|
+
```
|
225
260
|
|
226
|
-
## Testing with
|
261
|
+
## Testing with TestHelper
|
227
262
|
|
228
|
-
ActionMCP provides a `TestHelper` module to simplify testing of tools and prompts
|
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
|
-
|
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
|
-
|
288
|
+
You can use the MCP Inspector to test your server implementation:
|
261
289
|
|
262
|
-
```
|
263
|
-
|
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
|
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
@@ -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
|