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.
Files changed (66) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +153 -158
  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/gem_version.rb +2 -0
  25. data/lib/action_mcp/instrumentation/controller_runtime.rb +38 -0
  26. data/lib/action_mcp/instrumentation/instrumentation.rb +26 -0
  27. data/lib/action_mcp/instrumentation/log_subscriber.rb +39 -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 +28 -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 +18 -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 +26 -9
  44. data/lib/action_mcp/tool.rb +12 -3
  45. data/lib/action_mcp/tool_response.rb +3 -2
  46. data/lib/action_mcp/tools_registry.rb +1 -1
  47. data/lib/action_mcp/transport/capabilities.rb +5 -1
  48. data/lib/action_mcp/transport/messaging.rb +2 -0
  49. data/lib/action_mcp/transport/prompts.rb +2 -0
  50. data/lib/action_mcp/transport/resources.rb +23 -6
  51. data/lib/action_mcp/transport/roots.rb +11 -0
  52. data/lib/action_mcp/transport/sampling.rb +14 -0
  53. data/lib/action_mcp/transport/sse_client.rb +11 -15
  54. data/lib/action_mcp/transport/stdio_client.rb +12 -14
  55. data/lib/action_mcp/transport/tools.rb +2 -0
  56. data/lib/action_mcp/transport/transport_base.rb +16 -15
  57. data/lib/action_mcp/transport.rb +2 -0
  58. data/lib/action_mcp/transport_handler.rb +3 -0
  59. data/lib/action_mcp/version.rb +1 -1
  60. data/lib/action_mcp.rb +8 -2
  61. data/lib/generators/action_mcp/install/install_generator.rb +4 -1
  62. data/lib/generators/action_mcp/install/templates/application_mcp_res_template.rb +2 -0
  63. data/lib/generators/action_mcp/resource_template/resource_template_generator.rb +2 -0
  64. data/lib/generators/action_mcp/resource_template/templates/resource_template.rb.erb +1 -1
  65. data/lib/tasks/action_mcp_tasks.rake +11 -6
  66. metadata +27 -14
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: db5dc901e94ecc707a9182889ac8604af07912b8e7361d69758a8122f4572142
4
- data.tar.gz: 7363fea49767f889941711fc7da630015774e09b96540bcd3249a0ced0ebcccf
3
+ metadata.gz: 6794bfeea02de8502aa4b084548a93269149b9376117663d602ecfae54af8014
4
+ data.tar.gz: 49bca81dc1f59d27fc8d27c1b8978a8def6d3414822f261ce1aee2a4ebfc45b0
5
5
  SHA512:
6
- metadata.gz: 6b58d65e571daf89731fcc4f82cd5b7d413390fcd34889a01249c29cb799feb34d7bb49d94f69ced069f097f399728e933c6b3fa0a0184a329422b20610fd844
7
- data.tar.gz: e32dedcb932a8699461ff16244ebd97ec30bec2246bd88a8aefcea0bacf4661a49776b9195d97dc062e68c7d5b08ab66046e6866e36f6a4f3723bca98f10f553
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) ([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,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: **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"
152
+ class ProductResourceTemplate < ApplicationMCPResTemplate
153
+ uri_template "product/{id}"
154
+ description "Access product information by ID"
112
155
 
113
- # Provide a user-facing description for your prompt.
114
- description "Analyze code for potential improvements"
156
+ parameter :id, description: "Product identifier", required: true
115
157
 
116
- # Configure arguments via the new DSL
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 call
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
172
  ```
129
173
 
130
- #### Generate a New Tool
174
+ Resource templates are automatically registered and used when LLMs request resources matching their patterns.
131
175
 
132
- Similarly, run the following command to generate a new tool class:
176
+ ## Configuration
133
177
 
134
- ```bash
135
- bin/rails generate action_mcp:tool CalculateSum
136
- ```
178
+ ActionMCP is configured via `config.action_mcp` in your Rails application.
137
179
 
138
- This command will create a file at `app/mcp/tools/calculate_sum_tool.rb` with content similar to:
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
- ```ruby
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
- property :a, type: "number", description: "First number", required: true
146
- property :b, type: "number", description: "Second number", required: true
147
-
148
- def call
149
- render(text: a + b)
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
- ### 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.
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
- ## Usage Example
197
+ ## Engine and Mounting
167
198
 
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.
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
- ### Example for a Prompt
202
+ To mount the ActionMCP engine in your routes, add the following line to your `config/routes.rb`:
171
203
 
172
204
  ```ruby
173
- # Instantiate the prompt with initial values
174
- analyze_prompt = AnalyzeCodePrompt.new(language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
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
- ### Example for a Tool
210
+ ## Generators
189
211
 
190
- ```ruby
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
- 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.
214
+ You can generate the base classes for your MCP Prompt and Tool using the following command:
207
215
 
208
- ## Examples & Important Notes
216
+ ```bash
217
+ bin/rails action_mcp:install:migrations # to copy the migrations
218
+ bin/rails generate action_mcp:install
219
+ ```
209
220
 
210
- - **Running the Dummy App:**
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
- - **Postgres on macOS:**
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
- - **Notifiers:**
230
- ActionMCP works with the ActiveCable Postgres notifier by default, but its architecture is flexible enough to support other notifier implementations.
225
+ ```bash
226
+ bin/rails generate action_mcp:prompt AnalyzeCode
227
+ ```
231
228
 
232
- - **API Stability:**
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
- ## Testing with the TestHelper
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. To use the `TestHelper`, include it in your test class:
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 = execute_tool("analyze_code", language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
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
- The `TestHelper` module provides the following methods:
260
+ ## Inspecting Your MCP Server
261
261
 
262
- * `assert_tool_findable(tool_name)`: Asserts that a tool is findable in the `ToolsRegistry`.
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
- To use the `TestHelper`, you need to require it in your `test_helper.rb` file:
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 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.
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
@@ -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