actionmcp 0.14.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 +152 -148
- 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 +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 +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,193 +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
|
83
95
|
|
84
|
-
ActionMCP
|
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`.
|
85
97
|
|
86
|
-
|
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
|
87
103
|
|
88
|
-
|
89
|
-
bin/rails action_mcp:install:migrations # to copy the migrations
|
90
|
-
bin/rails generate action_mcp:install
|
91
|
-
```
|
104
|
+
**Example:**
|
92
105
|
|
93
|
-
|
94
|
-
|
95
|
-
|
106
|
+
```ruby
|
107
|
+
class CalculateSumTool < ApplicationMCPTool
|
108
|
+
tool_name "calculate_sum"
|
109
|
+
description "Calculate the sum of two numbers"
|
96
110
|
|
97
|
-
|
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
|
134
|
+
```
|
98
135
|
|
99
|
-
|
136
|
+
Tools can be executed by instantiating them and calling the `call` method:
|
100
137
|
|
101
|
-
```
|
102
|
-
|
138
|
+
```ruby
|
139
|
+
sum_tool = CalculateSumTool.new(a: 5, b: 10)
|
140
|
+
result = sum_tool.call
|
103
141
|
```
|
104
142
|
|
105
|
-
|
143
|
+
### ActionMCP::ResourceTemplate
|
106
144
|
|
107
|
-
|
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.
|
108
147
|
|
109
|
-
|
110
|
-
# Override the prompt_name (otherwise we'd get "analyze_code")
|
111
|
-
prompt_name "analyze-code"
|
148
|
+
**Example:**
|
112
149
|
|
113
|
-
|
114
|
-
description "Analyze code for potential improvements"
|
150
|
+
```ruby
|
115
151
|
|
116
|
-
|
117
|
-
|
118
|
-
|
152
|
+
class ProductResourceTemplate < ApplicationMCPResTemplate
|
153
|
+
uri_template "product/{id}"
|
154
|
+
description "Access product information by ID"
|
119
155
|
|
120
|
-
|
121
|
-
validates :language, inclusion: { in: %w[Ruby C Cobol FORTRAN] }
|
156
|
+
parameter :id, description: "Product identifier", required: true
|
122
157
|
|
123
|
-
|
124
|
-
|
125
|
-
|
158
|
+
validates :id, format: { with: /\A\d+\z/, message: "must be numeric" }
|
159
|
+
|
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
|
-
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!"}}]}
|
205
|
+
Rails.application.routes.draw do
|
206
|
+
mount ActionMCP::Engine => "/action_mcp"
|
207
|
+
end
|
181
208
|
```
|
182
209
|
|
183
|
-
|
210
|
+
## Generators
|
184
211
|
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
# Optionally update attributes later:
|
189
|
-
sum_tool.a = 15
|
190
|
-
sum_tool.b = 20
|
212
|
+
ActionMCP includes Rails generators to help you quickly set up your MCP server components.
|
213
|
+
|
214
|
+
You can generate the base classes for your MCP Prompt and Tool using the following command:
|
191
215
|
|
192
|
-
|
193
|
-
|
194
|
-
|
216
|
+
```bash
|
217
|
+
bin/rails action_mcp:install:migrations # to copy the migrations
|
218
|
+
bin/rails generate action_mcp:install
|
195
219
|
```
|
196
220
|
|
197
|
-
|
221
|
+
This will create the base application classes in your app directory.
|
198
222
|
|
199
|
-
|
223
|
+
### Generate a New Prompt
|
200
224
|
|
201
|
-
|
202
|
-
|
203
|
-
|
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
|
-
|
211
|
-
|
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).
|
225
|
+
```bash
|
226
|
+
bin/rails generate action_mcp:prompt AnalyzeCode
|
227
|
+
```
|
219
228
|
|
220
|
-
|
221
|
-
ActionMCP works with the ActiveCable Postgres notifier by default, but its architecture is flexible enough to support other notifier implementations.
|
229
|
+
### Generate a New Tool
|
222
230
|
|
223
|
-
|
224
|
-
|
231
|
+
```bash
|
232
|
+
bin/rails generate action_mcp:tool CalculateSum
|
233
|
+
```
|
225
234
|
|
226
|
-
## Testing with
|
235
|
+
## Testing with TestHelper
|
227
236
|
|
228
|
-
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:
|
229
238
|
|
230
239
|
```ruby
|
231
240
|
require "test_helper"
|
@@ -248,21 +257,16 @@ class ToolTest < ActiveSupport::TestCase
|
|
248
257
|
end
|
249
258
|
```
|
250
259
|
|
251
|
-
|
260
|
+
## Inspecting Your MCP Server
|
252
261
|
|
253
|
-
|
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.
|
262
|
+
You can use the MCP Inspector to test your server implementation:
|
259
263
|
|
260
|
-
|
261
|
-
|
262
|
-
```ruby
|
263
|
-
require "action_mcp/test_helper"
|
264
|
+
```bash
|
265
|
+
npx @modelcontextprotocol/inspector
|
264
266
|
```
|
265
267
|
|
268
|
+
The default path will be http://localhost:3000/action_mcp
|
269
|
+
|
266
270
|
## Conclusion
|
267
271
|
|
268
|
-
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
|