actionmcp 0.22.0 → 0.25.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 +113 -2
- data/app/controllers/action_mcp/messages_controller.rb +2 -14
- data/app/controllers/action_mcp/sse_controller.rb +113 -45
- data/app/models/action_mcp/session/message.rb +21 -16
- data/app/models/action_mcp/session.rb +3 -2
- data/config/routes.rb +1 -1
- data/db/migrate/20250324203409_remove_session_message_text.rb +7 -0
- data/lib/action_mcp/client/base.rb +12 -14
- data/lib/action_mcp/client/blueprint.rb +5 -71
- data/lib/action_mcp/client/catalog.rb +10 -74
- data/lib/action_mcp/client/collection.rb +93 -0
- data/lib/action_mcp/client/json_rpc_handler.rb +12 -7
- data/lib/action_mcp/client/logging.rb +1 -2
- data/lib/action_mcp/client/prompt_book.rb +5 -71
- data/lib/action_mcp/client/prompts.rb +9 -4
- data/lib/action_mcp/client/request_timeouts.rb +74 -0
- data/lib/action_mcp/client/resources.rb +23 -11
- data/lib/action_mcp/client/server.rb +3 -3
- data/lib/action_mcp/client/toolbox.rb +12 -54
- data/lib/action_mcp/client/tools.rb +9 -4
- data/lib/action_mcp/configuration.rb +134 -24
- data/lib/action_mcp/engine.rb +6 -0
- data/lib/action_mcp/json_rpc_handler_base.rb +1 -0
- data/lib/action_mcp/registry_base.rb +3 -1
- data/lib/action_mcp/server/capabilities.rb +1 -1
- data/lib/action_mcp/server/json_rpc_handler.rb +1 -1
- data/lib/action_mcp/server/messaging.rb +32 -9
- data/lib/action_mcp/version.rb +1 -1
- data/lib/generators/action_mcp/install/install_generator.rb +4 -0
- data/lib/generators/action_mcp/install/templates/mcp.yml +11 -0
- data/lib/tasks/action_mcp_tasks.rake +77 -6
- metadata +6 -2
@@ -4,40 +4,48 @@ module ActionMCP
|
|
4
4
|
module Client
|
5
5
|
module Resources
|
6
6
|
# List all available resources from the server
|
7
|
-
# @return [
|
7
|
+
# @return [String] Request ID for tracking the request
|
8
8
|
def list_resources
|
9
9
|
request_id = SecureRandom.uuid_v7
|
10
10
|
|
11
11
|
# Send request
|
12
12
|
send_jsonrpc_request("resources/list", id: request_id)
|
13
|
+
|
14
|
+
# Return request ID for tracking the request
|
15
|
+
request_id
|
13
16
|
end
|
14
17
|
|
15
18
|
# List resource templates from the server
|
16
|
-
# @return [
|
19
|
+
# @return [String] Request ID for tracking the request
|
17
20
|
def list_resource_templates
|
18
21
|
request_id = SecureRandom.uuid_v7
|
19
22
|
|
20
23
|
# Send request
|
21
24
|
send_jsonrpc_request("resources/templates/list", id: request_id)
|
25
|
+
|
26
|
+
# Return request ID for tracking the request
|
27
|
+
request_id
|
22
28
|
end
|
23
29
|
|
24
30
|
# Read a specific resource
|
25
31
|
# @param uri [String] URI of the resource to read
|
26
|
-
# @return [
|
32
|
+
# @return [String] Request ID for tracking the request
|
27
33
|
def read_resource(uri)
|
28
34
|
request_id = SecureRandom.uuid_v7
|
29
35
|
|
30
36
|
# Send request
|
31
37
|
send_jsonrpc_request("resources/read",
|
32
38
|
params: { uri: uri },
|
33
|
-
id: request_id
|
34
|
-
|
39
|
+
id: request_id)
|
40
|
+
|
41
|
+
# Return request ID for tracking the request
|
42
|
+
request_id
|
35
43
|
end
|
36
44
|
|
37
45
|
# Subscribe to updates for a specific resource
|
38
46
|
# @param uri [String] URI of the resource to subscribe to
|
39
47
|
# @param update_callback [Proc] Callback for resource updates
|
40
|
-
# @return [
|
48
|
+
# @return [String] Request ID for tracking the request
|
41
49
|
def subscribe_resource(uri, update_callback)
|
42
50
|
@resource_subscriptions ||= {}
|
43
51
|
@resource_subscriptions[uri] = update_callback
|
@@ -47,13 +55,15 @@ module ActionMCP
|
|
47
55
|
# Send request
|
48
56
|
send_jsonrpc_request("resources/subscribe",
|
49
57
|
params: { uri: uri },
|
50
|
-
id: request_id
|
51
|
-
|
58
|
+
id: request_id)
|
59
|
+
|
60
|
+
# Return request ID for tracking the request
|
61
|
+
request_id
|
52
62
|
end
|
53
63
|
|
54
64
|
# Unsubscribe from updates for a specific resource
|
55
65
|
# @param uri [String] URI of the resource to unsubscribe from
|
56
|
-
# @return [
|
66
|
+
# @return [String] Request ID for tracking the request
|
57
67
|
def unsubscribe_resource(uri)
|
58
68
|
@resource_subscriptions&.delete(uri)
|
59
69
|
|
@@ -62,8 +72,10 @@ module ActionMCP
|
|
62
72
|
# Send request
|
63
73
|
send_jsonrpc_request("resources/unsubscribe",
|
64
74
|
params: { uri: uri },
|
65
|
-
id: request_id
|
66
|
-
|
75
|
+
id: request_id)
|
76
|
+
|
77
|
+
# Return request ID for tracking the request
|
78
|
+
request_id
|
67
79
|
end
|
68
80
|
end
|
69
81
|
end
|
@@ -1,9 +1,9 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActionMCP
|
2
4
|
module Client
|
3
5
|
class Server
|
4
|
-
attr_reader :name, :version
|
5
|
-
|
6
|
-
attr_reader :server_info, :capabilities
|
6
|
+
attr_reader :name, :version, :server_info, :capabilities
|
7
7
|
|
8
8
|
def initialize(data)
|
9
9
|
# Store protocol version if needed for later use
|
@@ -18,30 +18,15 @@ module ActionMCP
|
|
18
18
|
# # Get all tools matching a criteria
|
19
19
|
# calculation_tools = toolbox.filter { |t| t.name.include?("calculate") }
|
20
20
|
#
|
21
|
-
class Toolbox
|
22
|
-
attr_reader :client
|
23
|
-
|
21
|
+
class Toolbox < Collection
|
24
22
|
# Initialize a new Toolbox with tool definitions
|
25
23
|
#
|
26
24
|
# @param tools [Array<Hash>] Array of tool definition hashes, each containing
|
27
25
|
# name, description, and inputSchema keys
|
28
26
|
def initialize(tools, client)
|
29
|
-
|
30
|
-
|
31
|
-
@
|
32
|
-
end
|
33
|
-
|
34
|
-
# Return all tools in the collection
|
35
|
-
#
|
36
|
-
# @return [Array<Tool>] All tool objects in the collection
|
37
|
-
def all
|
38
|
-
load_tools unless @loaded
|
39
|
-
@tools
|
40
|
-
end
|
41
|
-
|
42
|
-
def all!
|
43
|
-
load_tools(force: true)
|
44
|
-
@tools
|
27
|
+
super(tools, client)
|
28
|
+
self.tools = @collection_data
|
29
|
+
@load_method = :list_tools
|
45
30
|
end
|
46
31
|
|
47
32
|
# Find a tool by name
|
@@ -89,8 +74,9 @@ module ActionMCP
|
|
89
74
|
# @param keyword [String] Keyword to search for in tool names and descriptions
|
90
75
|
# @return [Array<Tool>] Tools containing the keyword
|
91
76
|
def search(keyword)
|
92
|
-
|
93
|
-
tool.name.include?(keyword) ||
|
77
|
+
all.select do |tool|
|
78
|
+
tool.name.include?(keyword) ||
|
79
|
+
tool.description&.downcase&.include?(keyword.downcase)
|
94
80
|
end
|
95
81
|
end
|
96
82
|
|
@@ -102,46 +88,18 @@ module ActionMCP
|
|
102
88
|
case provider
|
103
89
|
when :claude
|
104
90
|
# Claude format
|
105
|
-
{ "tools" =>
|
91
|
+
{ "tools" => all.map(&:to_claude_h) }
|
106
92
|
when :openai
|
107
93
|
# OpenAI format
|
108
|
-
{ "tools" =>
|
94
|
+
{ "tools" => all.map(&:to_openai_h) }
|
109
95
|
else
|
110
96
|
# Default format (same as original)
|
111
|
-
{ "tools" =>
|
97
|
+
{ "tools" => all.map(&:to_h) }
|
112
98
|
end
|
113
99
|
end
|
114
100
|
|
115
|
-
# Implements enumerable functionality for the collection
|
116
|
-
include Enumerable
|
117
|
-
|
118
|
-
# Yield each tool in the collection to the given block
|
119
|
-
#
|
120
|
-
# @yield [tool] Block to execute for each tool
|
121
|
-
# @yieldparam tool [Tool] A tool from the collection
|
122
|
-
# @return [Enumerator] If no block is given
|
123
|
-
def each(&block)
|
124
|
-
@tools.each(&block)
|
125
|
-
end
|
126
|
-
|
127
101
|
def tools=(tools)
|
128
|
-
@
|
129
|
-
end
|
130
|
-
|
131
|
-
private
|
132
|
-
|
133
|
-
def load_tools(force: false)
|
134
|
-
return if @loaded && !force
|
135
|
-
|
136
|
-
begin
|
137
|
-
@client.list_tools
|
138
|
-
@loaded = true
|
139
|
-
rescue StandardError => e
|
140
|
-
# Handle error appropriately
|
141
|
-
Rails.logger.error("Failed to load tools: #{e.message}")
|
142
|
-
# Still mark as loaded but with empty list?
|
143
|
-
@loaded = true unless @tools.empty?
|
144
|
-
end
|
102
|
+
@collection_data = tools.map { |tool_data| Tool.new(tool_data) }
|
145
103
|
end
|
146
104
|
|
147
105
|
# Internal Tool class to represent individual tools
|
@@ -168,7 +126,7 @@ module ActionMCP
|
|
168
126
|
#
|
169
127
|
# @return [Hash] Hash of property definitions
|
170
128
|
def properties
|
171
|
-
@input_schema
|
129
|
+
@input_schema["properties"] || {}
|
172
130
|
end
|
173
131
|
|
174
132
|
# Check if the tool requires a specific property
|
@@ -4,18 +4,21 @@ module ActionMCP
|
|
4
4
|
module Client
|
5
5
|
module Tools
|
6
6
|
# List all available tools from the server
|
7
|
-
# @return [
|
7
|
+
# @return [String] Request ID for tracking the request
|
8
8
|
def list_tools
|
9
9
|
request_id = SecureRandom.uuid_v7
|
10
10
|
|
11
11
|
# Send request
|
12
12
|
send_jsonrpc_request("tools/list", id: request_id)
|
13
|
+
|
14
|
+
# Return request ID for timeout tracking
|
15
|
+
request_id
|
13
16
|
end
|
14
17
|
|
15
18
|
# Call a specific tool on the server
|
16
19
|
# @param name [String] Name of the tool to call
|
17
20
|
# @param arguments [Hash] Arguments to pass to the tool
|
18
|
-
# @return [
|
21
|
+
# @return [String] Request ID for tracking the request
|
19
22
|
def call_tool(name, arguments)
|
20
23
|
request_id = SecureRandom.uuid_v7
|
21
24
|
|
@@ -25,8 +28,10 @@ module ActionMCP
|
|
25
28
|
name: name,
|
26
29
|
arguments: arguments
|
27
30
|
},
|
28
|
-
id: request_id
|
29
|
-
|
31
|
+
id: request_id)
|
32
|
+
|
33
|
+
# Return request ID for tracking the request
|
34
|
+
request_id
|
30
35
|
end
|
31
36
|
end
|
32
37
|
end
|
@@ -14,21 +14,22 @@ module ActionMCP
|
|
14
14
|
# @!attribute resources_subscribe
|
15
15
|
# @return [Boolean] Whether to subscribe to resources.
|
16
16
|
# @!attribute logging_level
|
17
|
-
# @return [Symbol] The logging level.
|
17
|
+
# @return [Symbol] The logging level. attr_writer :name, :version
|
18
18
|
attr_writer :name, :version
|
19
|
-
attr_accessor :logging_enabled,
|
20
|
-
:list_changed,
|
21
|
-
:resources_subscribe,
|
22
|
-
:logging_level
|
23
|
-
|
24
|
-
|
25
|
-
#
|
26
|
-
# @return [void]
|
19
|
+
attr_accessor :logging_enabled,
|
20
|
+
:list_changed,
|
21
|
+
:resources_subscribe,
|
22
|
+
:logging_level,
|
23
|
+
:active_profile,
|
24
|
+
:profiles
|
27
25
|
|
28
26
|
def initialize
|
29
27
|
@logging_enabled = true
|
30
28
|
@list_changed = false
|
31
29
|
@logging_level = :info
|
30
|
+
@resources_subscribe = false
|
31
|
+
@active_profile = :primary
|
32
|
+
@profiles = default_profiles
|
32
33
|
end
|
33
34
|
|
34
35
|
def name
|
@@ -39,23 +40,127 @@ module ActionMCP
|
|
39
40
|
@version || (has_rails_version ? Rails.application.version.to_s : "0.0.1")
|
40
41
|
end
|
41
42
|
|
42
|
-
#
|
43
|
-
|
44
|
-
|
43
|
+
# Load custom profiles from Rails configuration
|
44
|
+
def load_profiles
|
45
|
+
# First load defaults from the gem
|
46
|
+
@profiles = default_profiles
|
47
|
+
|
48
|
+
# Then try to load from config/mcp.yml in the Rails app
|
49
|
+
config_path = Rails.root.join("config", "mcp.yml")
|
50
|
+
if File.exist?(config_path)
|
51
|
+
begin
|
52
|
+
yaml_content = YAML.safe_load(File.read(config_path), symbolize_names: true)
|
53
|
+
# Merge with defaults so user config overrides gem defaults
|
54
|
+
@profiles.deep_merge!(yaml_content) if yaml_content
|
55
|
+
rescue StandardError => e
|
56
|
+
Rails.logger.error "Failed to load MCP profiles from #{config_path}: #{e.message}"
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
# Apply the active profile
|
61
|
+
use_profile(@active_profile)
|
62
|
+
|
63
|
+
self
|
64
|
+
end
|
65
|
+
|
66
|
+
# Switch to a specific profile
|
67
|
+
def use_profile(profile_name)
|
68
|
+
profile_name = profile_name.to_sym
|
69
|
+
unless @profiles.key?(profile_name)
|
70
|
+
Rails.logger.warn "Profile '#{profile_name}' not found, using default"
|
71
|
+
profile_name = :default
|
72
|
+
end
|
73
|
+
|
74
|
+
@active_profile = profile_name
|
75
|
+
apply_profile_options
|
76
|
+
|
77
|
+
self
|
78
|
+
end
|
79
|
+
|
80
|
+
# Filter tools based on active profile
|
81
|
+
def filtered_tools
|
82
|
+
return ToolsRegistry.non_abstract if should_include_all?(:tools)
|
83
|
+
|
84
|
+
tool_names = @profiles[@active_profile][:tools] || []
|
85
|
+
# Convert tool names to underscored format
|
86
|
+
tool_names = tool_names.map { |name| name.to_s.underscore }
|
87
|
+
ToolsRegistry.non_abstract.select { |tool| tool_names.include?(tool.name.underscore) }
|
88
|
+
end
|
89
|
+
|
90
|
+
# Filter prompts based on active profile
|
91
|
+
def filtered_prompts
|
92
|
+
return PromptsRegistry.non_abstract if should_include_all?(:prompts)
|
93
|
+
|
94
|
+
prompt_names = @profiles[@active_profile][:prompts] || []
|
95
|
+
PromptsRegistry.non_abstract.select { |prompt| prompt_names.include?(prompt.name) }
|
96
|
+
end
|
97
|
+
|
98
|
+
# Filter resources based on active profile
|
99
|
+
def filtered_resources
|
100
|
+
return ResourceTemplatesRegistry.non_abstract if should_include_all?(:resources)
|
101
|
+
|
102
|
+
resource_names = @profiles[@active_profile][:resources] || []
|
103
|
+
ResourceTemplatesRegistry.non_abstract.select { |resource| resource_names.include?(resource.name) }
|
104
|
+
end
|
105
|
+
|
106
|
+
# Returns capabilities based on active profile
|
45
107
|
def capabilities
|
46
108
|
capabilities = {}
|
47
|
-
# Only include each capability if the corresponding registry is non-empty
|
48
|
-
capabilities[:tools] = { listChanged:
|
49
|
-
capabilities[:prompts] = { listChanged:
|
109
|
+
# Only include each capability if the corresponding filtered registry is non-empty
|
110
|
+
capabilities[:tools] = { listChanged: @list_changed } if filtered_tools.any?
|
111
|
+
capabilities[:prompts] = { listChanged: @list_changed } if filtered_prompts.any?
|
50
112
|
capabilities[:logging] = {} if @logging_enabled
|
51
|
-
|
52
|
-
# For Resources, we need to think about how to pass the list to the session.
|
53
|
-
capabilities[:resources] = {} if ResourceTemplatesRegistry.non_abstract.any?
|
113
|
+
capabilities[:resources] = { subscribe: @resources_subscribe } if filtered_resources.any?
|
54
114
|
capabilities
|
55
115
|
end
|
56
116
|
|
57
117
|
private
|
58
118
|
|
119
|
+
def default_profiles
|
120
|
+
{
|
121
|
+
primary: {
|
122
|
+
tools: [ "all" ],
|
123
|
+
prompts: [ "all" ],
|
124
|
+
resources: [ "all" ],
|
125
|
+
options: {
|
126
|
+
list_changed: false,
|
127
|
+
logging_enabled: true,
|
128
|
+
logging_level: :info,
|
129
|
+
resources_subscribe: false
|
130
|
+
}
|
131
|
+
},
|
132
|
+
minimal: {
|
133
|
+
tools: [],
|
134
|
+
prompts: [],
|
135
|
+
resources: [],
|
136
|
+
options: {
|
137
|
+
list_changed: false,
|
138
|
+
logging_enabled: false,
|
139
|
+
logging_level: :warn,
|
140
|
+
resources_subscribe: false
|
141
|
+
}
|
142
|
+
}
|
143
|
+
}
|
144
|
+
end
|
145
|
+
|
146
|
+
def apply_profile_options
|
147
|
+
profile = @profiles[@active_profile]
|
148
|
+
return unless profile && profile[:options]
|
149
|
+
|
150
|
+
options = profile[:options]
|
151
|
+
@list_changed = options[:list_changed] unless options[:list_changed].nil?
|
152
|
+
@logging_enabled = options[:logging_enabled] unless options[:logging_enabled].nil?
|
153
|
+
@logging_level = options[:logging_level] unless options[:logging_level].nil?
|
154
|
+
@resources_subscribe = options[:resources_subscribe] unless options[:resources_subscribe].nil?
|
155
|
+
end
|
156
|
+
|
157
|
+
def should_include_all?(type)
|
158
|
+
return true unless @profiles[@active_profile]
|
159
|
+
|
160
|
+
items = @profiles[@active_profile][type]
|
161
|
+
items.nil? || items.include?("all")
|
162
|
+
end
|
163
|
+
|
59
164
|
def has_rails_version
|
60
165
|
gem "rails_app_version"
|
61
166
|
require "rails_app_version/railtie"
|
@@ -66,21 +171,26 @@ module ActionMCP
|
|
66
171
|
end
|
67
172
|
|
68
173
|
class << self
|
69
|
-
attr_accessor :server
|
174
|
+
attr_accessor :server, :logger
|
70
175
|
|
71
176
|
# Returns the configuration instance.
|
72
|
-
#
|
73
|
-
# @return [Configuration] the configuration instance
|
74
177
|
def configuration
|
75
178
|
@configuration ||= Configuration.new
|
76
179
|
end
|
77
180
|
|
78
181
|
# Configures the ActionMCP module.
|
79
|
-
#
|
80
|
-
# @yield [configuration] the configuration instance
|
81
|
-
# @return [void]
|
82
182
|
def configure
|
83
183
|
yield(configuration)
|
84
184
|
end
|
185
|
+
|
186
|
+
# Temporarily use a different profile
|
187
|
+
def with_profile(profile_name)
|
188
|
+
previous_profile = configuration.active_profile
|
189
|
+
configuration.use_profile(profile_name)
|
190
|
+
|
191
|
+
yield if block_given?
|
192
|
+
ensure
|
193
|
+
configuration.use_profile(previous_profile) if block_given?
|
194
|
+
end
|
85
195
|
end
|
86
196
|
end
|
data/lib/action_mcp/engine.rb
CHANGED
@@ -12,6 +12,7 @@ module ActionMCP
|
|
12
12
|
inflect.acronym "SSE"
|
13
13
|
inflect.acronym "MCP"
|
14
14
|
end
|
15
|
+
|
15
16
|
# Provide a configuration namespace for ActionMCP
|
16
17
|
config.action_mcp = ActionMCP.configuration
|
17
18
|
|
@@ -19,6 +20,11 @@ module ActionMCP
|
|
19
20
|
ActionMCP::ResourceTemplate.registered_templates.clear
|
20
21
|
end
|
21
22
|
|
23
|
+
# Load MCP profiles during initialization
|
24
|
+
initializer "action_mcp.load_profiles" do
|
25
|
+
ActionMCP.configuration.load_profiles
|
26
|
+
end
|
27
|
+
|
22
28
|
# Configure autoloading for the mcp/tools directory
|
23
29
|
initializer "action_mcp.autoloading", before: :set_autoload_paths do |app|
|
24
30
|
mcp_path = app.root.join("app/mcp")
|
@@ -12,7 +12,7 @@ module ActionMCP
|
|
12
12
|
session.store_client_info(@client_info)
|
13
13
|
session.store_client_capabilities(@client_capabilities)
|
14
14
|
session.set_protocol_version(@protocol_version)
|
15
|
-
session.
|
15
|
+
session.initialize!
|
16
16
|
# TODO: , if the server don't support the protocol version, send a response with error
|
17
17
|
send_jsonrpc_response(request_id, result: session.server_capabilities_payload)
|
18
18
|
end
|
@@ -22,7 +22,7 @@ module ActionMCP
|
|
22
22
|
when "completion/complete" # Completion requests
|
23
23
|
process_completion_complete(id, params)
|
24
24
|
else
|
25
|
-
|
25
|
+
transport.send_jsonrpc_error(id, :method_not_found, "Method not found")
|
26
26
|
end
|
27
27
|
end
|
28
28
|
|
@@ -4,24 +4,47 @@ module ActionMCP
|
|
4
4
|
module Server
|
5
5
|
module Messaging
|
6
6
|
def send_jsonrpc_request(method, params: nil, id: SecureRandom.uuid_v7)
|
7
|
-
|
8
|
-
write_message(request)
|
7
|
+
send_message(:request, method: method, params: params, id: id)
|
9
8
|
end
|
10
9
|
|
11
10
|
def send_jsonrpc_response(request_id, result: nil, error: nil)
|
12
|
-
response
|
13
|
-
write_message(response)
|
11
|
+
send_message(:response, id: request_id, result: result, error: error)
|
14
12
|
end
|
15
13
|
|
16
14
|
def send_jsonrpc_notification(method, params = nil)
|
17
|
-
notification
|
18
|
-
write_message(notification)
|
15
|
+
send_message(:notification, method: method, params: params)
|
19
16
|
end
|
20
17
|
|
21
18
|
def send_jsonrpc_error(request_id, symbol, message, data = nil)
|
22
|
-
error = JsonRpc::JsonRpcError.new(symbol, message
|
23
|
-
|
24
|
-
|
19
|
+
error = JsonRpc::JsonRpcError.new(symbol, message: message, data: data)
|
20
|
+
send_jsonrpc_response(request_id, error: error)
|
21
|
+
end
|
22
|
+
|
23
|
+
private
|
24
|
+
|
25
|
+
# Factory method to create and send appropriate JSON-RPC message
|
26
|
+
def send_message(type, **args)
|
27
|
+
message = case type
|
28
|
+
when :request
|
29
|
+
JsonRpc::Request.new(
|
30
|
+
id: args[:id],
|
31
|
+
method: args[:method],
|
32
|
+
params: args[:params]
|
33
|
+
)
|
34
|
+
when :response
|
35
|
+
JsonRpc::Response.new(
|
36
|
+
id: args[:id],
|
37
|
+
result: args[:result],
|
38
|
+
error: args[:error]
|
39
|
+
)
|
40
|
+
when :notification
|
41
|
+
JsonRpc::Notification.new(
|
42
|
+
method: args[:method],
|
43
|
+
params: args[:params]
|
44
|
+
)
|
45
|
+
end
|
46
|
+
|
47
|
+
write_message(message)
|
25
48
|
end
|
26
49
|
end
|
27
50
|
end
|
data/lib/action_mcp/version.rb
CHANGED
@@ -19,6 +19,10 @@ module ActionMcp
|
|
19
19
|
template "application_mcp_res_template.rb",
|
20
20
|
File.join("app/mcp/resource_templates", "application_mcp_res_template.rb")
|
21
21
|
end
|
22
|
+
|
23
|
+
def create_mcp_profile_file
|
24
|
+
template "mcp.yml", File.join("config", "mcp.yml")
|
25
|
+
end
|
22
26
|
end
|
23
27
|
end
|
24
28
|
end
|