actionmcp 0.19.1 → 0.22.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/app/controllers/action_mcp/messages_controller.rb +2 -2
- data/app/models/action_mcp/session/message.rb +12 -1
- data/app/models/action_mcp/session.rb +8 -4
- data/lib/action_mcp/base_response.rb +86 -0
- data/lib/action_mcp/capability.rb +2 -3
- data/lib/action_mcp/client/base.rb +222 -0
- data/lib/action_mcp/client/blueprint.rb +227 -0
- data/lib/action_mcp/client/catalog.rb +226 -0
- data/lib/action_mcp/client/json_rpc_handler.rb +109 -0
- data/lib/action_mcp/client/logging.rb +20 -0
- data/lib/action_mcp/{transport → client}/messaging.rb +1 -1
- data/lib/action_mcp/client/prompt_book.rb +183 -0
- data/lib/action_mcp/client/prompts.rb +33 -0
- data/lib/action_mcp/client/resources.rb +70 -0
- data/lib/action_mcp/client/roots.rb +13 -0
- data/lib/action_mcp/client/server.rb +60 -0
- data/lib/action_mcp/{transport → client}/sse_client.rb +70 -111
- data/lib/action_mcp/{transport → client}/stdio_client.rb +38 -38
- data/lib/action_mcp/client/toolbox.rb +236 -0
- data/lib/action_mcp/client/tools.rb +33 -0
- data/lib/action_mcp/client.rb +20 -231
- data/lib/action_mcp/engine.rb +1 -3
- data/lib/action_mcp/instrumentation/controller_runtime.rb +1 -1
- data/lib/action_mcp/instrumentation/instrumentation.rb +2 -0
- data/lib/action_mcp/instrumentation/resource_instrumentation.rb +1 -0
- data/lib/action_mcp/json_rpc_handler_base.rb +106 -0
- data/lib/action_mcp/log_subscriber.rb +2 -0
- data/lib/action_mcp/logging.rb +1 -1
- data/lib/action_mcp/prompt.rb +4 -3
- data/lib/action_mcp/prompt_response.rb +14 -58
- data/lib/action_mcp/{transport → server}/capabilities.rb +2 -2
- data/lib/action_mcp/server/json_rpc_handler.rb +121 -0
- data/lib/action_mcp/server/messaging.rb +28 -0
- data/lib/action_mcp/{transport → server}/notifications.rb +1 -1
- data/lib/action_mcp/{transport → server}/prompts.rb +1 -1
- data/lib/action_mcp/{transport → server}/resources.rb +1 -18
- data/lib/action_mcp/{transport → server}/roots.rb +1 -1
- data/lib/action_mcp/{transport → server}/sampling.rb +1 -1
- data/lib/action_mcp/server/sampling_request.rb +115 -0
- data/lib/action_mcp/{transport → server}/tools.rb +1 -1
- data/lib/action_mcp/server/transport_handler.rb +41 -0
- data/lib/action_mcp/tool_response.rb +14 -59
- data/lib/action_mcp/uri_ambiguity_checker.rb +6 -10
- data/lib/action_mcp/version.rb +1 -1
- data/lib/action_mcp.rb +2 -1
- metadata +30 -33
- data/lib/action_mcp/base_json_rpc_handler.rb +0 -97
- data/lib/action_mcp/client_json_rpc_handler.rb +0 -69
- data/lib/action_mcp/json_rpc_handler.rb +0 -229
- data/lib/action_mcp/sampling_request.rb +0 -113
- data/lib/action_mcp/server_json_rpc_handler.rb +0 -90
- data/lib/action_mcp/transport/transport_base.rb +0 -126
- data/lib/action_mcp/transport_handler.rb +0 -39
@@ -1,97 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module ActionMCP
|
4
|
-
# Base handler for common functionality
|
5
|
-
class BaseJsonRpcHandler
|
6
|
-
delegate :initialize!, :initialized?, to: :transport
|
7
|
-
delegate :write, :read, to: :transport
|
8
|
-
attr_reader :transport
|
9
|
-
|
10
|
-
# @param transport [ActionMCP::TransportHandler]
|
11
|
-
def initialize(transport)
|
12
|
-
@transport = transport
|
13
|
-
end
|
14
|
-
|
15
|
-
# Process a single line of input.
|
16
|
-
# @param line [String, Hash]
|
17
|
-
def call(line)
|
18
|
-
request = parse_request(line)
|
19
|
-
return unless request
|
20
|
-
|
21
|
-
process_request(request)
|
22
|
-
end
|
23
|
-
|
24
|
-
protected
|
25
|
-
|
26
|
-
def parse_request(line)
|
27
|
-
if line.is_a?(String)
|
28
|
-
line.strip!
|
29
|
-
return if line.empty?
|
30
|
-
|
31
|
-
begin
|
32
|
-
MultiJson.load(line)
|
33
|
-
rescue MultiJson::ParseError => e
|
34
|
-
Rails.logger.error("Failed to parse JSON: #{e.message}")
|
35
|
-
nil
|
36
|
-
end
|
37
|
-
else
|
38
|
-
line
|
39
|
-
end
|
40
|
-
end
|
41
|
-
|
42
|
-
# @param request [Hash]
|
43
|
-
def process_request(request)
|
44
|
-
unless request["jsonrpc"] == "2.0"
|
45
|
-
puts "Invalid request: #{request}"
|
46
|
-
return
|
47
|
-
end
|
48
|
-
read(request)
|
49
|
-
return if request["error"]
|
50
|
-
return if request["result"] == {} # Probably a pong
|
51
|
-
|
52
|
-
rpc_method = request["method"]
|
53
|
-
id = request["id"]
|
54
|
-
params = request["params"]
|
55
|
-
|
56
|
-
# Common methods (both directions)
|
57
|
-
case rpc_method
|
58
|
-
when "ping" # [BOTH] Ping message
|
59
|
-
transport.send_pong(id)
|
60
|
-
when "initialize" # [BOTH] Initialization
|
61
|
-
handle_initialize(id, params)
|
62
|
-
when %r{^notifications/}
|
63
|
-
process_common_notifications(rpc_method, params)
|
64
|
-
else
|
65
|
-
handle_specific_method(rpc_method, id, params)
|
66
|
-
end
|
67
|
-
end
|
68
|
-
|
69
|
-
# Override in subclasses
|
70
|
-
def handle_initialize(id, params)
|
71
|
-
raise NotImplementedError, "Subclasses must implement #handle_initialize"
|
72
|
-
end
|
73
|
-
|
74
|
-
# Override in subclasses
|
75
|
-
def handle_specific_method(rpc_method, id, params)
|
76
|
-
raise NotImplementedError, "Subclasses must implement #handle_specific_method"
|
77
|
-
end
|
78
|
-
|
79
|
-
def process_common_notifications(rpc_method, params)
|
80
|
-
case rpc_method
|
81
|
-
when "notifications/initialized" # [BOTH] Initialization complete
|
82
|
-
puts "Initialized"
|
83
|
-
transport.initialize!
|
84
|
-
when "notifications/cancelled" # [BOTH] Request cancellation
|
85
|
-
puts "Request #{params['requestId']} cancelled: #{params['reason']}"
|
86
|
-
# Handle cancellation
|
87
|
-
else
|
88
|
-
handle_specific_notification(rpc_method, params)
|
89
|
-
end
|
90
|
-
end
|
91
|
-
|
92
|
-
# Override in subclasses
|
93
|
-
def handle_specific_notification(rpc_method, params)
|
94
|
-
raise NotImplementedError, "Subclasses must implement #handle_specific_notification"
|
95
|
-
end
|
96
|
-
end
|
97
|
-
end
|
@@ -1,69 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module ActionMCP
|
4
|
-
# Handler for client-side requests (server -> client)
|
5
|
-
class ClientJsonRpcHandler < BaseJsonRpcHandler
|
6
|
-
def handle_initialize(id, params)
|
7
|
-
# Client-specific initialization
|
8
|
-
transport.send_client_capabilities(id, params)
|
9
|
-
end
|
10
|
-
|
11
|
-
def handle_specific_method(rpc_method, id, params)
|
12
|
-
case rpc_method
|
13
|
-
when "client/setLoggingLevel" # [CLIENT] Server configuring client logging
|
14
|
-
transport.set_client_logging_level(id, params["level"])
|
15
|
-
when %r{^roots/} # [CLIENT] Roots management
|
16
|
-
process_roots(rpc_method, id, params)
|
17
|
-
when %r{^sampling/} # [CLIENT] Sampling requests
|
18
|
-
process_sampling(rpc_method, id, params)
|
19
|
-
else
|
20
|
-
Rails.logger.warn("Unknown client method: #{rpc_method}")
|
21
|
-
end
|
22
|
-
end
|
23
|
-
|
24
|
-
def handle_specific_notification(rpc_method, params)
|
25
|
-
case rpc_method
|
26
|
-
when "notifications/resources/updated" # [CLIENT] Resource update notification
|
27
|
-
puts "Resource #{params['uri']} was updated"
|
28
|
-
# Handle resource update notification
|
29
|
-
when "notifications/tools/list_changed" # [CLIENT] Tool list change notification
|
30
|
-
puts "Tool list has changed"
|
31
|
-
# Handle tool list change notification
|
32
|
-
when "notifications/prompts/list_changed" # [CLIENT] Prompt list change notification
|
33
|
-
puts "Prompt list has changed"
|
34
|
-
# Handle prompt list change notification
|
35
|
-
when "notifications/resources/list_changed" # [CLIENT] Resource list change notification
|
36
|
-
puts "Resource list has changed"
|
37
|
-
# Handle resource list change notification
|
38
|
-
else
|
39
|
-
Rails.logger.warn("Unknown client notification: #{rpc_method}")
|
40
|
-
end
|
41
|
-
end
|
42
|
-
|
43
|
-
private
|
44
|
-
|
45
|
-
# @param rpc_method [String]
|
46
|
-
# @param id [String]
|
47
|
-
# @param params [Hash]
|
48
|
-
def process_roots(rpc_method, id, params)
|
49
|
-
case rpc_method
|
50
|
-
when "roots/list" # [CLIENT] List available roots
|
51
|
-
transport.send_roots_list(id)
|
52
|
-
else
|
53
|
-
Rails.logger.warn("Unknown roots method: #{rpc_method}")
|
54
|
-
end
|
55
|
-
end
|
56
|
-
|
57
|
-
# @param rpc_method [String]
|
58
|
-
# @param id [String]
|
59
|
-
# @param params [Hash]
|
60
|
-
def process_sampling(rpc_method, id, params)
|
61
|
-
case rpc_method
|
62
|
-
when "sampling/createMessage" # [CLIENT] Create a message using AI
|
63
|
-
transport.send_sampling_create_message(id, params)
|
64
|
-
else
|
65
|
-
Rails.logger.warn("Unknown sampling method: #{rpc_method}")
|
66
|
-
end
|
67
|
-
end
|
68
|
-
end
|
69
|
-
end
|
@@ -1,229 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module ActionMCP
|
4
|
-
class JsonRpcHandler
|
5
|
-
delegate :initialize!, :initialized?, to: :transport
|
6
|
-
delegate :write, :read, to: :transport
|
7
|
-
attr_reader :transport
|
8
|
-
|
9
|
-
# @param transport [ActionMCP::TransportHandler]
|
10
|
-
def initialize(transport)
|
11
|
-
@transport = transport
|
12
|
-
end
|
13
|
-
|
14
|
-
# Process a single line of input.
|
15
|
-
# @param line [String, Hash]
|
16
|
-
def call(line)
|
17
|
-
request = if line.is_a?(String)
|
18
|
-
line.strip!
|
19
|
-
return if line.empty?
|
20
|
-
|
21
|
-
begin
|
22
|
-
MultiJson.load(line)
|
23
|
-
rescue MultiJson::ParseError => e
|
24
|
-
Rails.logger.error("Failed to parse JSON: #{e.message}")
|
25
|
-
return
|
26
|
-
end
|
27
|
-
else
|
28
|
-
line
|
29
|
-
end
|
30
|
-
process_request(request)
|
31
|
-
end
|
32
|
-
|
33
|
-
private
|
34
|
-
|
35
|
-
# @param request [Hash]
|
36
|
-
def process_request(request)
|
37
|
-
unless request["jsonrpc"] == "2.0"
|
38
|
-
puts "Invalid request: #{request}"
|
39
|
-
return
|
40
|
-
end
|
41
|
-
read(request)
|
42
|
-
return if request["error"]
|
43
|
-
return if request["result"] == {} # Probably a pong
|
44
|
-
|
45
|
-
rpc_method = request["method"]
|
46
|
-
id = request["id"]
|
47
|
-
params = request["params"]
|
48
|
-
|
49
|
-
# Common methods (both directions)
|
50
|
-
case rpc_method
|
51
|
-
when "initialize" # [SERVER] Client initializing the connection
|
52
|
-
transport.send_capabilities(id, params)
|
53
|
-
when "ping" # [BOTH] Client ping
|
54
|
-
transport.send_pong(id)
|
55
|
-
|
56
|
-
# Methods that servers must implement (client → server)
|
57
|
-
when %r{^prompts/} # [SERVER] Prompt-related requests
|
58
|
-
process_prompts(rpc_method, id, params)
|
59
|
-
when %r{^resources/} # [SERVER] Resource-related requests
|
60
|
-
process_resources(rpc_method, id, params)
|
61
|
-
when %r{^tools/} # [SERVER] Tool-related requests
|
62
|
-
process_tools(rpc_method, id, params)
|
63
|
-
when "completion/complete" # [SERVER] Completion requests
|
64
|
-
process_completion_complete(id, params)
|
65
|
-
|
66
|
-
# Methods that clients must implement (server → client)
|
67
|
-
when "client/setLoggingLevel" # [CLIENT] Server configuring client logging
|
68
|
-
process_client_logging(id, params)
|
69
|
-
when %r{^roots/} # [CLIENT] Roots management
|
70
|
-
process_roots(rpc_method, id, params)
|
71
|
-
when %r{^sampling/} # [CLIENT] Sampling requests
|
72
|
-
process_sampling(rpc_method, id, params)
|
73
|
-
|
74
|
-
# Notifications (can go both ways)
|
75
|
-
when %r{^notifications/}
|
76
|
-
puts "\e[31mProcessing notifications\e[0m"
|
77
|
-
process_notifications(rpc_method, params)
|
78
|
-
else
|
79
|
-
puts "\e[31mUnknown method: #{rpc_method} #{request}\e[0m"
|
80
|
-
end
|
81
|
-
end
|
82
|
-
|
83
|
-
# @param rpc_method [String]
|
84
|
-
def process_notifications(rpc_method, params)
|
85
|
-
case rpc_method
|
86
|
-
when "notifications/initialized" # [SERVER] Client initialization complete
|
87
|
-
puts "\e[31mInitialized\e[0m"
|
88
|
-
transport.initialize!
|
89
|
-
when "notifications/cancelled" # [BOTH] Request cancellation
|
90
|
-
puts "\e[31m Request #{params['requestId']} cancelled: #{params['reason']}\e[0m"
|
91
|
-
# we don't need to do anything here
|
92
|
-
when "notifications/resources/updated" # [CLIENT] Resource update notification
|
93
|
-
puts "\e[31m Resource #{params['uri']} was updated\e[0m"
|
94
|
-
# Handle resource update notification
|
95
|
-
when "notifications/tools/list_changed" # [CLIENT] Tool list change notification
|
96
|
-
puts "\e[31m Tool list has changed\e[0m"
|
97
|
-
# Handle tool list change notification
|
98
|
-
when "notifications/prompts/list_changed" # [CLIENT] Prompt list change notification
|
99
|
-
puts "\e[31m Prompt list has changed\e[0m"
|
100
|
-
# Handle prompt list change notification
|
101
|
-
when "notifications/resources/list_changed" # [CLIENT] Resource list change notification
|
102
|
-
puts "\e[31m Resource list has changed\e[0m"
|
103
|
-
# Handle resource list change notification
|
104
|
-
else
|
105
|
-
Rails.logger.warn("Unknown notifications method: #{rpc_method}")
|
106
|
-
end
|
107
|
-
end
|
108
|
-
|
109
|
-
# Server methods (client → server)
|
110
|
-
|
111
|
-
# @param id [String]
|
112
|
-
# @param params [Hash]
|
113
|
-
# @example {
|
114
|
-
# "ref": {
|
115
|
-
# "type": "ref/prompt",
|
116
|
-
# "name": "code_review"
|
117
|
-
# },
|
118
|
-
# "argument": {
|
119
|
-
# "name": "language",
|
120
|
-
# "value": "py"
|
121
|
-
# }
|
122
|
-
# }
|
123
|
-
# @return [Hash]
|
124
|
-
# @example {
|
125
|
-
# "completion": {
|
126
|
-
# "values": ["python", "pytorch", "pyside"],
|
127
|
-
# "total": 10,
|
128
|
-
# "hasMore": true
|
129
|
-
# }
|
130
|
-
# }
|
131
|
-
def process_completion_complete(id, params)
|
132
|
-
# TODO: Not Implemented, but to remove the error message in the inspector
|
133
|
-
transport.send_jsonrpc_response(id, result: { completion: { values: [], total: 0, hasMore: false } })
|
134
|
-
case params["ref"]["type"]
|
135
|
-
when "ref/prompt"
|
136
|
-
# TODO: Implement completion
|
137
|
-
when "ref/resource"
|
138
|
-
# TODO: Implement completion
|
139
|
-
end
|
140
|
-
end
|
141
|
-
|
142
|
-
# @param rpc_method [String]
|
143
|
-
# @param id [String]
|
144
|
-
# @param params [Hash]
|
145
|
-
def process_prompts(rpc_method, id, params)
|
146
|
-
case rpc_method
|
147
|
-
when "prompts/get" # [SERVER] Get specific prompt
|
148
|
-
transport.send_prompts_get(id, params["name"], params["arguments"])
|
149
|
-
when "prompts/list" # [SERVER] List available prompts
|
150
|
-
transport.send_prompts_list(id)
|
151
|
-
else
|
152
|
-
Rails.logger.warn("Unknown prompts method: #{rpc_method}")
|
153
|
-
end
|
154
|
-
end
|
155
|
-
|
156
|
-
# @param rpc_method [String]
|
157
|
-
# @param id [String]
|
158
|
-
# @param params [Hash]
|
159
|
-
def process_resources(rpc_method, id, params)
|
160
|
-
case rpc_method
|
161
|
-
when "resources/list" # [SERVER] List available resources
|
162
|
-
transport.send_resources_list(id)
|
163
|
-
when "resources/templates/list" # [SERVER] List resource templates
|
164
|
-
transport.send_resource_templates_list(id)
|
165
|
-
when "resources/read" # [SERVER] Read resource content
|
166
|
-
transport.send_resource_read(id, params)
|
167
|
-
when "resources/subscribe" # [SERVER] Subscribe to resource updates
|
168
|
-
transport.send_resource_subscribe(id, params["uri"])
|
169
|
-
when "resources/unsubscribe" # [SERVER] Unsubscribe from resource updates
|
170
|
-
transport.send_resource_unsubscribe(id, params["uri"])
|
171
|
-
else
|
172
|
-
Rails.logger.warn("Unknown resources method: #{rpc_method}")
|
173
|
-
end
|
174
|
-
end
|
175
|
-
|
176
|
-
# @param rpc_method [String]
|
177
|
-
# @param id [String]
|
178
|
-
# @param params [Hash]
|
179
|
-
def process_tools(rpc_method, id, params)
|
180
|
-
case rpc_method
|
181
|
-
when "tools/list" # [SERVER] List available tools
|
182
|
-
transport.send_tools_list(id)
|
183
|
-
when "tools/call" # [SERVER] Call a tool
|
184
|
-
transport.send_tools_call(id, params&.dig("name"), params&.dig("arguments"))
|
185
|
-
else
|
186
|
-
Rails.logger.warn("Unknown tools method: #{rpc_method}")
|
187
|
-
end
|
188
|
-
end
|
189
|
-
|
190
|
-
# Client methods (server → client)
|
191
|
-
|
192
|
-
# @param id [String]
|
193
|
-
# @param params [Hash]
|
194
|
-
def process_client_logging(id, params)
|
195
|
-
level = params["level"]
|
196
|
-
transport.set_client_logging_level(id, level)
|
197
|
-
end
|
198
|
-
|
199
|
-
# @param rpc_method [String]
|
200
|
-
# @param id [String]
|
201
|
-
# @param params [Hash]
|
202
|
-
def process_roots(rpc_method, id, params)
|
203
|
-
case rpc_method
|
204
|
-
when "roots/list" # [CLIENT] List available roots
|
205
|
-
transport.send_roots_list(id)
|
206
|
-
when "roots/add" # [CLIENT] Add a root
|
207
|
-
transport.send_roots_add(id, params["uri"], params["name"])
|
208
|
-
when "roots/remove" # [CLIENT] Remove a root
|
209
|
-
transport.send_roots_remove(id, params["uri"])
|
210
|
-
else
|
211
|
-
Rails.logger.warn("Unknown roots method: #{rpc_method}")
|
212
|
-
end
|
213
|
-
end
|
214
|
-
|
215
|
-
# @param rpc_method [String]
|
216
|
-
# @param id [String]
|
217
|
-
# @param params [Hash]
|
218
|
-
def process_sampling(rpc_method, id, params)
|
219
|
-
case rpc_method
|
220
|
-
when "sampling/createMessage" # [CLIENT] Create a message using AI
|
221
|
-
# @param id [String]
|
222
|
-
# @param params [SamplingRequest]
|
223
|
-
transport.send_sampling_create_message(id, params)
|
224
|
-
else
|
225
|
-
Rails.logger.warn("Unknown sampling method: #{rpc_method}")
|
226
|
-
end
|
227
|
-
end
|
228
|
-
end
|
229
|
-
end
|
@@ -1,113 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module ActionMCP
|
4
|
-
class SamplingRequest
|
5
|
-
class << self
|
6
|
-
attr_reader :default_messages, :default_system_prompt, :default_context,
|
7
|
-
:default_model_hints, :default_intelligence_priority,
|
8
|
-
:default_max_tokens, :default_temperature
|
9
|
-
|
10
|
-
def configure
|
11
|
-
yield self
|
12
|
-
end
|
13
|
-
|
14
|
-
def messages(messages = nil)
|
15
|
-
if messages
|
16
|
-
@default_messages = messages.map do |msg|
|
17
|
-
mutate_content(msg)
|
18
|
-
end
|
19
|
-
end
|
20
|
-
@default_messages ||= []
|
21
|
-
end
|
22
|
-
|
23
|
-
def system_prompt(prompt = nil)
|
24
|
-
@default_system_prompt = prompt if prompt
|
25
|
-
@default_system_prompt
|
26
|
-
end
|
27
|
-
|
28
|
-
def include_context(context = nil)
|
29
|
-
@default_context = context if context
|
30
|
-
@default_context
|
31
|
-
end
|
32
|
-
|
33
|
-
def model_hints(hints = nil)
|
34
|
-
@default_model_hints = hints if hints
|
35
|
-
@model_hints ||= []
|
36
|
-
end
|
37
|
-
|
38
|
-
def intelligence_priority(priority = nil)
|
39
|
-
@default_intelligence_priority = priority if priority
|
40
|
-
@intelligence_priority ||= 0.9
|
41
|
-
end
|
42
|
-
|
43
|
-
def max_tokens(tokens = nil)
|
44
|
-
@default_max_tokens = tokens if tokens
|
45
|
-
@max_tokens ||= 500
|
46
|
-
end
|
47
|
-
|
48
|
-
def temperature(temp = nil)
|
49
|
-
@default_temperature = temp if temp
|
50
|
-
@temperature ||= 0.7
|
51
|
-
end
|
52
|
-
|
53
|
-
private
|
54
|
-
|
55
|
-
def mutate_content(msg)
|
56
|
-
content = msg[:content]
|
57
|
-
if content.is_a?(ActionMCP::Content) || (content.respond_to?(:to_h) && !content.is_a?(Hash))
|
58
|
-
{ role: msg[:role], content: content.to_h }
|
59
|
-
else
|
60
|
-
msg
|
61
|
-
end
|
62
|
-
end
|
63
|
-
end
|
64
|
-
|
65
|
-
attr_accessor :system_prompt, :model_hints, :intelligence_priority, :max_tokens, :temperature
|
66
|
-
attr_reader :messages, :context
|
67
|
-
|
68
|
-
def initialize
|
69
|
-
@messages = self.class.default_messages.dup
|
70
|
-
@system_prompt = self.class.default_system_prompt
|
71
|
-
@context = self.class.default_context
|
72
|
-
@model_hints = self.class.default_model_hints.dup
|
73
|
-
@intelligence_priority = self.class.default_intelligence_priority
|
74
|
-
@max_tokens = self.class.default_max_tokens
|
75
|
-
@temperature = self.class.default_temperature
|
76
|
-
|
77
|
-
yield self if block_given?
|
78
|
-
end
|
79
|
-
|
80
|
-
def messages=(value)
|
81
|
-
@messages = value.map do |msg|
|
82
|
-
self.class.send(:mutate_content, msg)
|
83
|
-
end
|
84
|
-
end
|
85
|
-
|
86
|
-
def include_context=(value)
|
87
|
-
@context = value
|
88
|
-
end
|
89
|
-
|
90
|
-
def add_message(content, role: "user")
|
91
|
-
if content.is_a?(Content::Base) || (content.respond_to?(:to_h) && !content.is_a?(Hash))
|
92
|
-
@messages << { role: role, content: content.to_h }
|
93
|
-
else
|
94
|
-
content = Content::Text.new(content).to_h if content.is_a?(String)
|
95
|
-
@messages << { role: role, content: content }
|
96
|
-
end
|
97
|
-
end
|
98
|
-
|
99
|
-
def to_h
|
100
|
-
{
|
101
|
-
messages: messages.map { |msg| { role: msg[:role], content: msg[:content] } },
|
102
|
-
systemPrompt: system_prompt,
|
103
|
-
includeContext: context,
|
104
|
-
modelPreferences: {
|
105
|
-
hints: model_hints.map { |name| { name: name } },
|
106
|
-
intelligencePriority: intelligence_priority
|
107
|
-
},
|
108
|
-
maxTokens: max_tokens,
|
109
|
-
temperature: temperature
|
110
|
-
}.compact
|
111
|
-
end
|
112
|
-
end
|
113
|
-
end
|
@@ -1,90 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module ActionMCP
|
4
|
-
# Handler for server-side requests (client -> server)
|
5
|
-
class ServerJsonRpcHandler < BaseJsonRpcHandler
|
6
|
-
def handle_initialize(id, params)
|
7
|
-
# Server-specific initialization
|
8
|
-
transport.send_capabilities(id, params)
|
9
|
-
end
|
10
|
-
|
11
|
-
def handle_specific_method(rpc_method, id, params)
|
12
|
-
case rpc_method
|
13
|
-
when %r{^prompts/} # [SERVER] Prompt-related requests
|
14
|
-
process_prompts(rpc_method, id, params)
|
15
|
-
when %r{^resources/} # [SERVER] Resource-related requests
|
16
|
-
process_resources(rpc_method, id, params)
|
17
|
-
when %r{^tools/} # [SERVER] Tool-related requests
|
18
|
-
process_tools(rpc_method, id, params)
|
19
|
-
when "completion/complete" # [SERVER] Completion requests
|
20
|
-
process_completion_complete(id, params)
|
21
|
-
else
|
22
|
-
Rails.logger.warn("Unknown server method: #{rpc_method}")
|
23
|
-
end
|
24
|
-
end
|
25
|
-
def handle_specific_notification(rpc_method, _params)
|
26
|
-
# Server-specific notifications would go here
|
27
|
-
case rpc_method
|
28
|
-
when "notifications/initialized" # [SERVER] Initialization complete
|
29
|
-
puts "Initialized"
|
30
|
-
transport.initialize!
|
31
|
-
else
|
32
|
-
Rails.logger.warn("Unknown server notification: #{rpc_method}")
|
33
|
-
end
|
34
|
-
end
|
35
|
-
|
36
|
-
private
|
37
|
-
|
38
|
-
# All the server-specific methods below...
|
39
|
-
|
40
|
-
def process_completion_complete(id, params)
|
41
|
-
# Implementation as in original code
|
42
|
-
transport.send_jsonrpc_response(id, result: { completion: { values: [], total: 0, hasMore: false } })
|
43
|
-
case params["ref"]["type"]
|
44
|
-
when "ref/prompt"
|
45
|
-
# TODO: Implement completion
|
46
|
-
when "ref/resource"
|
47
|
-
# TODO: Implement completion
|
48
|
-
end
|
49
|
-
end
|
50
|
-
|
51
|
-
def process_prompts(rpc_method, id, params)
|
52
|
-
case rpc_method
|
53
|
-
when "prompts/get" # [SERVER] Get specific prompt
|
54
|
-
transport.send_prompts_get(id, params["name"], params["arguments"])
|
55
|
-
when "prompts/list" # [SERVER] List available prompts
|
56
|
-
transport.send_prompts_list(id)
|
57
|
-
else
|
58
|
-
Rails.logger.warn("Unknown prompts method: #{rpc_method}")
|
59
|
-
end
|
60
|
-
end
|
61
|
-
|
62
|
-
def process_resources(rpc_method, id, params)
|
63
|
-
case rpc_method
|
64
|
-
when "resources/list" # [SERVER] List available resources
|
65
|
-
transport.send_resources_list(id)
|
66
|
-
when "resources/templates/list" # [SERVER] List resource templates
|
67
|
-
transport.send_resource_templates_list(id)
|
68
|
-
when "resources/read" # [SERVER] Read resource content
|
69
|
-
transport.send_resource_read(id, params)
|
70
|
-
when "resources/subscribe" # [SERVER] Subscribe to resource updates
|
71
|
-
transport.send_resource_subscribe(id, params["uri"])
|
72
|
-
when "resources/unsubscribe" # [SERVER] Unsubscribe from resource updates
|
73
|
-
transport.send_resource_unsubscribe(id, params["uri"])
|
74
|
-
else
|
75
|
-
Rails.logger.warn("Unknown resources method: #{rpc_method}")
|
76
|
-
end
|
77
|
-
end
|
78
|
-
|
79
|
-
def process_tools(rpc_method, id, params)
|
80
|
-
case rpc_method
|
81
|
-
when "tools/list" # [SERVER] List available tools
|
82
|
-
transport.send_tools_list(id)
|
83
|
-
when "tools/call" # [SERVER] Call a tool
|
84
|
-
transport.send_tools_call(id, params&.dig("name"), params&.dig("arguments"))
|
85
|
-
else
|
86
|
-
Rails.logger.warn("Unknown tools method: #{rpc_method}")
|
87
|
-
end
|
88
|
-
end
|
89
|
-
end
|
90
|
-
end
|