actionmcp 0.2.0 → 0.2.4

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 (53) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +133 -30
  3. data/Rakefile +0 -2
  4. data/app/controllers/action_mcp/application_controller.rb +13 -0
  5. data/app/controllers/action_mcp/messages_controller.rb +51 -0
  6. data/app/controllers/action_mcp/sse_controller.rb +151 -0
  7. data/config/routes.rb +4 -0
  8. data/exe/actionmcp_cli +221 -0
  9. data/lib/action_mcp/capability.rb +52 -0
  10. data/lib/action_mcp/client.rb +243 -1
  11. data/lib/action_mcp/configuration.rb +50 -1
  12. data/lib/action_mcp/content/audio.rb +9 -0
  13. data/lib/action_mcp/content/image.rb +9 -0
  14. data/lib/action_mcp/content/resource.rb +13 -0
  15. data/lib/action_mcp/content/text.rb +7 -0
  16. data/lib/action_mcp/content.rb +11 -6
  17. data/lib/action_mcp/engine.rb +34 -0
  18. data/lib/action_mcp/gem_version.rb +2 -2
  19. data/lib/action_mcp/integer_array.rb +6 -0
  20. data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
  21. data/lib/action_mcp/json_rpc/notification.rb +8 -0
  22. data/lib/action_mcp/json_rpc/request.rb +14 -0
  23. data/lib/action_mcp/json_rpc/response.rb +32 -1
  24. data/lib/action_mcp/json_rpc.rb +1 -6
  25. data/lib/action_mcp/json_rpc_handler.rb +106 -0
  26. data/lib/action_mcp/logging.rb +19 -0
  27. data/lib/action_mcp/prompt.rb +30 -46
  28. data/lib/action_mcp/prompts_registry.rb +13 -1
  29. data/lib/action_mcp/registry_base.rb +47 -28
  30. data/lib/action_mcp/renderable.rb +26 -0
  31. data/lib/action_mcp/resource.rb +3 -1
  32. data/lib/action_mcp/server.rb +4 -1
  33. data/lib/action_mcp/string_array.rb +5 -0
  34. data/lib/action_mcp/tool.rb +16 -53
  35. data/lib/action_mcp/tools_registry.rb +14 -1
  36. data/lib/action_mcp/transport/capabilities.rb +21 -0
  37. data/lib/action_mcp/transport/messaging.rb +20 -0
  38. data/lib/action_mcp/transport/prompts.rb +19 -0
  39. data/lib/action_mcp/transport/sse_client.rb +309 -0
  40. data/lib/action_mcp/transport/stdio_client.rb +117 -0
  41. data/lib/action_mcp/transport/tools.rb +20 -0
  42. data/lib/action_mcp/transport/transport_base.rb +125 -0
  43. data/lib/action_mcp/transport.rb +1 -235
  44. data/lib/action_mcp/transport_handler.rb +54 -0
  45. data/lib/action_mcp/version.rb +4 -5
  46. data/lib/action_mcp.rb +36 -33
  47. data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
  48. data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
  49. data/lib/tasks/action_mcp_tasks.rake +28 -5
  50. metadata +66 -9
  51. data/exe/action_mcp_stdio +0 -0
  52. data/lib/action_mcp/railtie.rb +0 -27
  53. data/lib/action_mcp/resources_bank.rb +0 -94
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 3e01fe9b15e57ab4450f9712dd2a38dbf355cbdc50d196a8026cbb205d3d8a07
4
- data.tar.gz: 8ce8ffd799f5b9487c0bb3b78c7d7242cff31742e8cc1a79a76116e50f6cb896
3
+ metadata.gz: 0a1cd6d70ca62ff8ecf4d25c00e29d9592bc9d80ca46db50330a2f818a66ce1f
4
+ data.tar.gz: 259e35b08f9627aa4165157b71436f4fdd4c0c9eaedc2ae41ddb6f0bb8c3fdde
5
5
  SHA512:
6
- metadata.gz: b044250b38fc6680fc5afa90f33dfc1082cb4ca7ec5916339e8e2a1f0b5ff64a32229968c24eeec19ec706a4341c413cc0d3546f7edee8dd9dd4ffd807427056
7
- data.tar.gz: 8b98f315be3dd31d80f77d85eeddd66ab434af8077d04358928c7be3793e7ad0565d37c9042666a93810694688a45e7147b9a540ae0505b061b34b77a58cf991
6
+ metadata.gz: c213f50edbec5dc96af78b3090f562dae9fe479a405a44d573927ef65c1c042d6a43ef174297bca3bc92280a372d5a3a74c61aec5a8dfc3e1381550c56b27397
7
+ data.tar.gz: 343bee907d44e4e7869372b5ffa250c6d628c8789ef51a4690b311c494d6b5b20199837dbbc5f39959e36b8a71d3ce2942b82685b4076897ddc393e59e73d8e3
data/README.md CHANGED
@@ -1,6 +1,6 @@
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.
4
4
 
5
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
 
@@ -12,7 +12,7 @@ With ActionMCP, you can focus on your app's logic while it handles the boilerpla
12
12
 
13
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 ([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)).
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
 
@@ -30,63 +30,82 @@ In short, ActionMCP helps you build an MCP server (the component that exposes ca
30
30
  To start using ActionMCP, add it to your project:
31
31
 
32
32
  - **Using Bundler (Rails or Ruby projects):** Add the gem to your Gemfile and run bundle install:
33
-
34
- execute:
35
- ```
33
+
34
+ ```bash
36
35
  $ bundle add actionmcp
37
36
  ```
38
37
 
39
- After installing, include the gem in your code by requiring it:
40
-
41
38
  This will load the ActionMCP library so you can start defining MCP prompts, tools, and resources in your application.
42
39
 
43
40
  ## Core Components
44
41
 
45
42
  ActionMCP provides three core abstractions to streamline MCP server development: **Prompt**, **Tool**, and **Resource**.
43
+
46
44
  These correspond to key MCP concepts and let you define what context or capabilities your server exposes to LLMs.
47
- Below is an overview of each component and how you might use it:
45
+
46
+ Note that ActionMCP requires a Rails application; it is not meant for standalone Ruby apps.
48
47
 
49
48
  ### Configuration
50
- ActionMCP is configured via config.action_mcp in your Rails application.
51
- By default, the name is set to your application's name and the version defaults to "0.0.1".
52
- You can override these settings in your configuration (e.g., in config/application.rb):
49
+
50
+ ActionMCP is configured via `config.action_mcp` in your Rails application.
51
+
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.
53
+
54
+ You can override these settings in your configuration (e.g., in `config/application.rb`):
55
+
53
56
  ```ruby
54
57
  module Tron
55
58
  class Application < Rails::Application
56
59
  config.action_mcp.name = "Friendly MCP (Master Control Program)" # defaults to Rails.application.name
57
- config.action_mcp.version = "1.2.3" # defaults to "0.0.1"
58
- config.action_mcp.logging_enabled = true # defaults to true
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
59
63
  end
60
64
  end
61
65
  ```
62
- For dynamic versioning, consider adding the rails_app_version gem.
63
66
 
64
- ## Generators
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`:
75
+
76
+ ```ruby
77
+ Rails.application.routes.draw do
78
+ mount ActionMCP::Engine => "/action_mcp"
79
+ end
80
+ ```
81
+
82
+ ### Generators
65
83
 
66
- ActionMCP includes Rails generators to help you quickly set up your MCP server components. You can generate the base classes for your MCP Prompt and Tool using the following commands.
84
+ ActionMCP includes Rails generators to help you quickly set up your MCP server components.
67
85
 
68
- To generate both the ApplicationPrompt and ApplicationTool files in your application, run:
86
+ You can generate the base classes for your MCP Prompt and Tool using the following command:
69
87
 
70
88
  ```bash
71
89
  bin/rails generate action_mcp:install
72
90
  ```
73
91
 
74
92
  This command will create:
75
- app/prompts/application_prompt.rb
76
- app/tools/application_tool.rb
93
+ - `app/prompts/application_prompt.rb`
94
+ - `app/tools/application_tool.rb`
77
95
 
78
- ### Generate a New Prompt
96
+ #### Generate a New Prompt
79
97
 
80
98
  Run the following command to generate a new prompt class:
81
99
 
82
100
  ```bash
83
101
  bin/rails generate action_mcp:prompt AnalyzeCode
84
102
  ```
85
- This command will create a file at app/prompts/analyze_code_prompt.rb with content similar to:
103
+
104
+ This command will create a file at `app/prompts/analyze_code_prompt.rb` with content similar to:
86
105
 
87
106
  ```ruby
88
107
  class AnalyzeCodePrompt < ApplicationPrompt
89
- # Override the prompt_name (otherwise we'd get "analyze-code")
108
+ # Override the prompt_name (otherwise we'd get "analyze_code")
90
109
  prompt_name "analyze-code"
91
110
 
92
111
  # Provide a user-facing description for your prompt.
@@ -96,38 +115,122 @@ class AnalyzeCodePrompt < ApplicationPrompt
96
115
  argument :language, description: "Programming language", default: "Ruby"
97
116
  argument :code, description: "Code to explain", required: true
98
117
 
99
- # Add validations (note: "Ruby" is not allowed per the validation)
100
- validates :language, inclusion: { in: %w[C Cobol FORTRAN] }
118
+ # Add validations
119
+ validates :language, inclusion: { in: %w[Ruby C Cobol FORTRAN] }
120
+
121
+ def call
122
+ # Implement your prompt logic here
123
+ render_text("Analyzing #{language} code: #{code}")
124
+ end
101
125
  end
102
126
  ```
103
127
 
104
- ## Generate a New Tool
128
+ #### Generate a New Tool
129
+
105
130
  Similarly, run the following command to generate a new tool class:
106
131
 
107
132
  ```bash
108
133
  bin/rails generate action_mcp:tool CalculateSum
109
134
  ```
110
135
 
111
- This command will create a file at app/tools/calculate_sum_tool.rb with content similar to:
136
+ This command will create a file at `app/tools/calculate_sum_tool.rb` with content similar to:
112
137
 
113
138
  ```ruby
114
139
  class CalculateSumTool < ApplicationTool
115
- tool_name "calculate-sum"
140
+ tool_name "calculate_sum"
116
141
  description "Calculate the sum of two numbers"
117
142
 
118
143
  property :a, type: "number", description: "First number", required: true
119
144
  property :b, type: "number", description: "Second number", required: true
145
+
146
+ def call
147
+ render_text(a + b)
148
+ end
120
149
  end
121
150
  ```
122
151
 
123
152
  ### ActionMCP::Prompt
124
153
 
125
- Make Rails Say Sexy stuff
154
+ 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.
126
155
 
127
156
  ### ActionMCP::Tool
128
157
 
129
- Make Rails Do Sexy stuff and serve beer to Clients.
158
+ 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.
130
159
 
131
160
  ### ActionMCP::Resource
132
161
 
133
- I dont need this for now
162
+ *I don't need this for now.*
163
+
164
+ ## Usage Example
165
+
166
+ 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.
167
+
168
+ ### Example for a Prompt
169
+
170
+ ```ruby
171
+ # Instantiate the prompt with initial values
172
+ analyze_prompt = AnalyzeCodePrompt.new(language: "Ruby", code: "def hello; puts 'Hello, world!'; end")
173
+
174
+ # Optionally update attributes later:
175
+ analyze_prompt.code = "def goodbye; puts 'Goodbye!'; end"
176
+
177
+ # Validate the prompt before calling it
178
+ if analyze_prompt.valid?
179
+ result = analyze_prompt.call # => #<ActionMCP::Content::Text:0x00000001239398c8 @text="The code you provided is written in Ruby and looks great!", @type="text">
180
+ puts result.to_h # => {type: "text", text: "The code you provided is written in Ruby and looks great!"}
181
+ else
182
+ puts analyze_prompt.errors.full_messages
183
+ end
184
+ ```
185
+
186
+ ### Example for a Tool
187
+
188
+ ```ruby
189
+ # Instantiate the tool with initial values
190
+ sum_tool = CalculateSumTool.new(a: 5, b: 10)
191
+
192
+ # Optionally update attributes later:
193
+ sum_tool.a = 15
194
+ sum_tool.b = 20
195
+
196
+ # Validate the tool before calling it
197
+ if sum_tool.valid?
198
+ result = sum_tool.call # => #<ActionMCP::Content::Text:0x0000000124cfaba0 @text="35.0", @type="text">
199
+ puts result.to_h # => {type: "text", text: "35.0"}
200
+ else
201
+ puts sum_tool.errors.full_messages
202
+ end
203
+ ```
204
+
205
+ 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.
206
+
207
+ ## Examples & Important Notes
208
+
209
+ - **Running the Dummy App:**
210
+ After creating the database with `bin/rails db:prepare`, you can run the dummy application using:
211
+ ```bash
212
+ bin/rails s
213
+ ```
214
+ This allows you to test and interact with the MCP server from the dummy environment.
215
+ - **Inspecting the App:**
216
+ You can use the mcp inspector to test your app ```npx @modelcontextprotocol/inspector```
217
+ the path by default will be http://localhost:3000/action_mcp
218
+
219
+
220
+ - **Postgres on macOS:**
221
+ 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:
222
+ ```bash
223
+ export PGGSSENCMODE=disable
224
+ export OBJC_DISABLE_INITIALIZE_FORK_SAFETY=YES
225
+ ```
226
+ More details can be found in [Rails Issue #38560](https://github.com/rails/rails/issues/38560).
227
+
228
+ - **Notifiers:**
229
+ ActionMCP works with the ActiveCable Postgres notifier by default, but its architecture is flexible enough to support other notifier implementations.
230
+
231
+ - **API Stability:**
232
+ 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.
233
+
234
+ ## Conclusion
235
+
236
+ 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, and important deployment considerations, it is designed to accelerate development and integration work while remaining flexible for future enhancements.
data/Rakefile CHANGED
@@ -4,6 +4,4 @@ require 'bundler/setup'
4
4
 
5
5
  APP_RAKEFILE = File.expand_path('test/dummy/Rakefile', __dir__)
6
6
 
7
- load 'rails/tasks/statistics.rake'
8
-
9
7
  require 'bundler/gem_tasks'
@@ -0,0 +1,13 @@
1
+ module ActionMCP
2
+ class ApplicationController < ActionController::Metal
3
+ abstract!
4
+ ActionController::API.without_modules(:StrongParameters, :ParamsWrapper).each do |left|
5
+ include left
6
+ end
7
+ include Engine.routes.url_helpers
8
+
9
+ def session_key
10
+ @session_key = "action_mcp-sessions-#{session_id}"
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,51 @@
1
+ module ActionMCP
2
+ class MessagesController < ApplicationController
3
+ # @route POST / (sse_in)
4
+ def create
5
+ begin
6
+ handle_post_message(params, response)
7
+ rescue => e
8
+ head :internal_server_error
9
+ end
10
+ head response.status
11
+ end
12
+
13
+ private
14
+
15
+ def transport
16
+ @transport ||= Transport.new(session_key)
17
+ end
18
+
19
+ def transport_handler
20
+ TransportHandler.new(transport)
21
+ end
22
+
23
+ def json_rpc_handler
24
+ @json_rpc_handler ||= ActionMCP::JsonRpcHandler.new(transport_handler)
25
+ end
26
+
27
+ def handle_post_message(params, response)
28
+ json_rpc_handler.call(params)
29
+
30
+ response.status = :accepted
31
+ rescue StandardError => e
32
+ response.status = :bad_request
33
+ end
34
+
35
+ def session_id
36
+ params[:session_id]
37
+ end
38
+
39
+ class Transport
40
+ attr_reader :session_key, :adapter
41
+ def initialize(session_key)
42
+ @session_key = session_key
43
+ @adapter = ActionMCP::Server.server.pubsub
44
+ end
45
+
46
+ def write(data)
47
+ adapter.broadcast(session_key, data.to_json)
48
+ end
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,151 @@
1
+ module ActionMCP
2
+ class SSEController < ApplicationController
3
+ HEARTBEAT_INTERVAL = 10
4
+ INITIALIZATION_TIMEOUT = 2
5
+ include ActionController::Live
6
+
7
+ # @route GET /sse (sse_out)
8
+ def events
9
+ # Set headers first
10
+ response.headers["X-Accel-Buffering"] = "no"
11
+ response.headers["Content-Type"] = "text/event-stream"
12
+ response.headers["Cache-Control"] = "no-cache"
13
+ response.headers["Connection"] = "keep-alive"
14
+
15
+ listener = nil
16
+ begin
17
+ # Now start streaming - send endpoint
18
+ send_endpoint_event(sse_in_url)
19
+
20
+ # Start listener and process messages via the transport
21
+ listener = SseListener.new(session_key)
22
+ if listener.start do |message|
23
+ begin
24
+ Rails.logger.debug "Processing message in controller: #{message.inspect} (#{message.class})"
25
+
26
+ # Send with proper SSE formatting
27
+ sse = SSE.new(response.stream)
28
+ sse.write(message)
29
+ rescue => e
30
+ Rails.logger.error "Error sending SSE message: #{e.class} - #{e.message}\n#{e.backtrace.join("\n")}"
31
+ end
32
+ end
33
+
34
+ # Heartbeat loop
35
+ until response.stream.closed?
36
+ sleep HEARTBEAT_INTERVAL
37
+ send_ping!
38
+ end
39
+ else
40
+ Rails.logger.error "Listener failed to activate for session: #{session_id}"
41
+ raise "Failed to establish subscription"
42
+ end
43
+ rescue ActionController::Live::ClientDisconnected, IOError => e
44
+ Rails.logger.debug "SSE: Expected disconnection: #{e.message}"
45
+ rescue => e
46
+ Rails.logger.error "SSE: Unexpected error: #{e.class} - #{e.message}\n#{e.backtrace.join("\n")}"
47
+ ensure
48
+ listener&.stop
49
+ response.stream.close
50
+ Rails.logger.debug "SSE: Connection closed for session: #{session_id}"
51
+ end
52
+ end
53
+
54
+ private
55
+
56
+ def send_endpoint_event(messages_url)
57
+ endpoint = "#{messages_url}?session_id=#{session_id}"
58
+ SSE.new(response.stream,
59
+ event: "endpoint")
60
+ .write(endpoint)
61
+ end
62
+
63
+ def send_ping!
64
+ SSE.new(response.stream,
65
+ event: "ping")
66
+ .write(Time.now.to_i)
67
+ end
68
+
69
+ def default_url_options
70
+ { host: request.host, port: request.port }
71
+ end
72
+
73
+ def session_id
74
+ @session_id ||= SecureRandom.hex(6)
75
+ end
76
+ end
77
+
78
+ class SseListener
79
+ attr_reader :session_key, :adapter
80
+
81
+ def initialize(session_key)
82
+ @session_key = session_key
83
+ @adapter = ActionMCP::Server.server.pubsub
84
+ @stopped = false
85
+ @subscription_active = false
86
+ end
87
+
88
+ # Start listening using ActionCable's PostgreSQL adapter
89
+ def start(&callback)
90
+ Rails.logger.debug "Starting listener for channel: #{session_key}"
91
+
92
+ # Set up success callback
93
+ success_callback = -> {
94
+ Rails.logger.debug "Successfully subscribed to channel: #{session_key}"
95
+ @subscription_active = true
96
+ }
97
+
98
+ # Set up message callback with detailed debugging
99
+ message_callback = ->(raw_message) {
100
+ Rails.logger.debug "Received raw message via adapter: #{raw_message.inspect} (#{raw_message.class})"
101
+
102
+ begin
103
+ # Try to parse the message if it's JSON
104
+ message = raw_message.is_a?(String) ? JSON.parse(raw_message) : raw_message
105
+ Rails.logger.debug "Processed message: #{message.inspect}"
106
+
107
+ # Send the message to the callback
108
+ callback.call(message) if callback && !@stopped
109
+ rescue => e
110
+ Rails.logger.error "Error processing message: #{e.class} - #{e.message}"
111
+ # Still try to send the raw message as a fallback
112
+ callback.call(raw_message) if callback && !@stopped
113
+ end
114
+ }
115
+
116
+ # Subscribe using the ActionCable adapter
117
+ adapter.subscribe(session_key, message_callback, success_callback)
118
+
119
+ # Give some time for the subscription to be established
120
+ sleep 1.5
121
+
122
+ # Check if subscription was successful
123
+ if @subscription_active
124
+ Rails.logger.debug "Subscription confirmed active for: #{session_key}"
125
+ true
126
+ else
127
+ Rails.logger.error "Failed to activate subscription for: #{session_key}"
128
+ false
129
+ end
130
+ end
131
+
132
+ def stop
133
+ Rails.logger.debug "Stopping listener for: #{session_key}"
134
+ @stopped = true
135
+
136
+ # Unsubscribe using the correct method signature
137
+ begin
138
+ # Create a dummy callback that matches the one we provided in start
139
+ dummy_callback = ->(_) { }
140
+ adapter.unsubscribe(session_key, dummy_callback)
141
+ Rails.logger.debug "Unsubscribed from: #{session_key}"
142
+ rescue => e
143
+ Rails.logger.error "Error unsubscribing from #{session_key}: #{e.message}"
144
+ end
145
+ end
146
+
147
+ def active?
148
+ @subscription_active
149
+ end
150
+ end
151
+ end
data/config/routes.rb ADDED
@@ -0,0 +1,4 @@
1
+ ActionMCP::Engine.routes.draw do
2
+ get "/", to: "sse#events", as: :sse_out
3
+ post "/", to: "messages#create", as: :sse_in
4
+ end