actionmcp 0.9.0 → 0.11.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/lib/action_mcp/prompt.rb +4 -4
- data/lib/action_mcp/prompt_response.rb +50 -7
- data/lib/action_mcp/prompts_registry.rb +1 -14
- data/lib/action_mcp/tool.rb +2 -4
- data/lib/action_mcp/tool_response.rb +22 -19
- data/lib/action_mcp/tools_registry.rb +6 -6
- data/lib/action_mcp/transport/prompts.rb +6 -6
- data/lib/action_mcp/transport/tools.rb +5 -6
- data/lib/action_mcp/version.rb +1 -1
- metadata +1 -1
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: f825a63198153189d9970beec66012f8428dd625ec442559043597e4a938113e
|
4
|
+
data.tar.gz: 82891ffd29e22e8a36a80a002d264884756f6b48dfbe373eca963268ba82c4dc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: ff8d86f24b9f087f3dc0c81e6bda73045c554ce5f1bb1c90d16faa80411e99a79b4b64e905a552d9e68653759e7ca9380bee202586d52b5a13068626ba34c238
|
7
|
+
data.tar.gz: d872f57c2651d7a0c28db334aa88bb98862f969c5566cc65fc5beefc3fd0b83f46b6de7d9633eb1ba9571dc6e767ed9e4ec501e5858e7ef68b79870582e69e8e
|
data/lib/action_mcp/prompt.rb
CHANGED
@@ -57,7 +57,7 @@ module ActionMCP
|
|
57
57
|
validates arg_name, presence: true if required
|
58
58
|
|
59
59
|
if enum.present?
|
60
|
-
validates arg_name, inclusion: { in: enum }
|
60
|
+
validates arg_name, inclusion: { in: enum }, allow_blank: !required
|
61
61
|
end
|
62
62
|
end
|
63
63
|
|
@@ -109,13 +109,13 @@ module ActionMCP
|
|
109
109
|
if valid?
|
110
110
|
begin
|
111
111
|
perform # Invoke the subclass-specific logic if valid
|
112
|
-
rescue
|
112
|
+
rescue
|
113
113
|
# Handle exceptions during execution
|
114
|
-
|
114
|
+
@response.mark_as_error!(:internal_error, message: "Unhandled Error executing prompt")
|
115
115
|
end
|
116
116
|
else
|
117
117
|
# Handle validation failure
|
118
|
-
|
118
|
+
@response.mark_as_error!(:invalid_params, message: "Invalid input", data: errors.full_messages)
|
119
119
|
end
|
120
120
|
|
121
121
|
@response # Return the response with collected messages
|
@@ -2,10 +2,16 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
class PromptResponse
|
5
|
-
|
5
|
+
include Enumerable
|
6
|
+
|
7
|
+
attr_reader :messages
|
8
|
+
|
9
|
+
# Delegate methods to the underlying messages array
|
10
|
+
delegate :empty?, :size, :each, :find, :map, to: :messages
|
6
11
|
|
7
12
|
def initialize
|
8
13
|
@messages = []
|
14
|
+
@is_error = false
|
9
15
|
end
|
10
16
|
|
11
17
|
# Add a message to the response
|
@@ -20,24 +26,61 @@ module ActionMCP
|
|
20
26
|
self
|
21
27
|
end
|
22
28
|
|
29
|
+
def mark_as_error!(symbol = :invalid_request, message: nil, data: nil)
|
30
|
+
@is_error = true
|
31
|
+
@symbol = symbol
|
32
|
+
@error_message = message
|
33
|
+
@error_data = data
|
34
|
+
self
|
35
|
+
end
|
36
|
+
|
23
37
|
# Convert to hash format expected by MCP protocol
|
24
38
|
def to_h
|
25
|
-
|
26
|
-
|
27
|
-
|
39
|
+
if @is_error
|
40
|
+
JsonRpc::JsonRpcError.new(@symbol, message: @error_message, data: @error_data).to_h
|
41
|
+
else
|
42
|
+
{
|
43
|
+
messages: @messages
|
44
|
+
}
|
45
|
+
end
|
28
46
|
end
|
29
47
|
|
30
|
-
# Alias
|
31
|
-
alias_method :
|
48
|
+
# Alias as_json to to_h for consistency
|
49
|
+
alias_method :as_json, :to_h
|
32
50
|
|
33
51
|
# Handle to_json directly
|
34
52
|
def to_json(options = nil)
|
35
53
|
to_h.to_json(options)
|
36
54
|
end
|
37
55
|
|
56
|
+
# Compare with hash for easier testing
|
57
|
+
def ==(other)
|
58
|
+
case other
|
59
|
+
when Hash
|
60
|
+
# Convert both to normalized format for comparison
|
61
|
+
hash_self = to_h.deep_transform_keys { |key| key.to_s.underscore }
|
62
|
+
hash_other = other.deep_transform_keys { |key| key.to_s.underscore }
|
63
|
+
hash_self == hash_other
|
64
|
+
when PromptResponse
|
65
|
+
messages == other.messages
|
66
|
+
else
|
67
|
+
super
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
# Implement eql? for hash key comparison
|
72
|
+
def eql?(other)
|
73
|
+
self == other
|
74
|
+
end
|
75
|
+
|
76
|
+
# Implement hash method for hash key usage
|
77
|
+
def hash
|
78
|
+
[ messages ].hash
|
79
|
+
end
|
80
|
+
|
38
81
|
# Pretty print for better debugging
|
39
82
|
def inspect
|
40
|
-
"#<#{self.class.name} messages: #{messages.
|
83
|
+
"#<#{self.class.name} messages: #{messages.inspect}>"
|
41
84
|
end
|
42
85
|
end
|
43
86
|
end
|
@@ -17,20 +17,7 @@ module ActionMCP
|
|
17
17
|
def prompt_call(prompt_name, arguments)
|
18
18
|
prompt = find(prompt_name)
|
19
19
|
prompt = prompt.new(arguments)
|
20
|
-
prompt.
|
21
|
-
if prompt.valid?
|
22
|
-
{
|
23
|
-
messages: [ {
|
24
|
-
role: "user",
|
25
|
-
content: prompt.call
|
26
|
-
} ]
|
27
|
-
}
|
28
|
-
else
|
29
|
-
{
|
30
|
-
content: prompt.errors.full_messages.map { |msg| Content::Text.new(msg) },
|
31
|
-
isError: true
|
32
|
-
}
|
33
|
-
end
|
20
|
+
prompt.call
|
34
21
|
end
|
35
22
|
|
36
23
|
def item_klass
|
data/lib/action_mcp/tool.rb
CHANGED
@@ -134,13 +134,11 @@ module ActionMCP
|
|
134
134
|
perform # Invoke the subclass-specific logic if valid
|
135
135
|
rescue => e
|
136
136
|
# Handle exceptions during execution
|
137
|
-
@response.mark_as_error!
|
138
|
-
render text: "Error executing tool: #{e.message}"
|
137
|
+
@response.mark_as_error!(:internal_error, message: e.message)
|
139
138
|
end
|
140
139
|
else
|
141
140
|
# Handle validation failure
|
142
|
-
@response.mark_as_error!
|
143
|
-
render text: "Invalid input: #{errors.full_messages.join(', ')}"
|
141
|
+
@response.mark_as_error!(:invalid_request, message: "Invalid input", data: errors.full_messages)
|
144
142
|
end
|
145
143
|
|
146
144
|
@response # Return the response with collected content
|
@@ -4,14 +4,12 @@ module ActionMCP
|
|
4
4
|
# Manages the collection of content objects for tool results
|
5
5
|
class ToolResponse
|
6
6
|
include Enumerable
|
7
|
-
|
8
7
|
attr_reader :contents, :is_error
|
9
|
-
|
10
8
|
delegate :empty?, :size, :each, :find, :map, to: :contents
|
11
9
|
|
12
|
-
def initialize
|
10
|
+
def initialize
|
13
11
|
@contents = []
|
14
|
-
@is_error =
|
12
|
+
@is_error = false
|
15
13
|
end
|
16
14
|
|
17
15
|
# Add content to the response
|
@@ -21,37 +19,42 @@ module ActionMCP
|
|
21
19
|
end
|
22
20
|
|
23
21
|
# Mark response as error
|
24
|
-
def mark_as_error!
|
22
|
+
def mark_as_error!(symbol = :invalid_request, message: nil, data: nil)
|
25
23
|
@is_error = true
|
24
|
+
@symbol = symbol
|
25
|
+
@error_message = message
|
26
|
+
@error_data = data
|
26
27
|
self
|
27
28
|
end
|
28
29
|
|
29
30
|
# Convert to hash format expected by MCP protocol
|
30
|
-
def
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
31
|
+
def to_h
|
32
|
+
if @is_error
|
33
|
+
JsonRpc::JsonRpcError.new(@symbol, message: @error_message, data: @error_data).to_h
|
34
|
+
else
|
35
|
+
{
|
36
|
+
content: @contents.map { |c| c.to_h },
|
37
|
+
}
|
38
|
+
end
|
35
39
|
end
|
36
40
|
|
37
|
-
# Alias
|
38
|
-
alias_method :
|
41
|
+
# Alias as_json to to_h for consistency
|
42
|
+
alias_method :as_json, :to_h
|
39
43
|
|
40
44
|
# Handle to_json directly
|
41
45
|
def to_json(options = nil)
|
42
|
-
|
46
|
+
to_h.to_json(options)
|
43
47
|
end
|
44
48
|
|
45
|
-
# Compare with hash for easier testing
|
46
|
-
# This allows assertions like: assert_equal({content: [...], isError: false}, tool_response)
|
49
|
+
# Compare with hash for easier testing.
|
47
50
|
def ==(other)
|
48
51
|
case other
|
49
52
|
when Hash
|
50
|
-
#
|
51
|
-
|
52
|
-
|
53
|
+
# Convert both to normalized format for comparison
|
54
|
+
hash_self = to_h.deep_transform_keys { |key| key.to_s.underscore }
|
55
|
+
hash_other = other.deep_transform_keys { |key| key.to_s.underscore }
|
56
|
+
hash_self == hash_other
|
53
57
|
when ToolResponse
|
54
|
-
# Direct comparison with another ToolResponse
|
55
58
|
contents == other.contents && is_error == other.is_error
|
56
59
|
else
|
57
60
|
super
|
@@ -20,9 +20,11 @@ module ActionMCP
|
|
20
20
|
tool = tool_class.new(arguments)
|
21
21
|
|
22
22
|
tool.call.to_h
|
23
|
+
rescue RegistryBase::NotFound
|
24
|
+
error_response(:invalid_params, message: "Tool not found: #{tool_name}")
|
23
25
|
rescue StandardError => e
|
24
26
|
# FIXME, we should maybe not return the error message to the user
|
25
|
-
error_response(
|
27
|
+
error_response(:invalid_params, message: "Tool execution failed: #{e.message}")
|
26
28
|
end
|
27
29
|
|
28
30
|
def item_klass
|
@@ -31,11 +33,9 @@ module ActionMCP
|
|
31
33
|
|
32
34
|
private
|
33
35
|
|
34
|
-
def error_response(
|
35
|
-
|
36
|
-
|
37
|
-
isError: true
|
38
|
-
}
|
36
|
+
def error_response(symbol, message: nil, data: nil)
|
37
|
+
response = ToolResponse.new
|
38
|
+
response.mark_as_error!(symbol, message: message, data: data)
|
39
39
|
end
|
40
40
|
end
|
41
41
|
end
|
@@ -7,12 +7,12 @@ module ActionMCP
|
|
7
7
|
end
|
8
8
|
|
9
9
|
def send_prompts_get(request_id, prompt_name, params)
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
10
|
+
result = PromptsRegistry.prompt_call(prompt_name.to_s, params)
|
11
|
+
if result.is_error
|
12
|
+
send_jsonrpc_response(request_id, error: result)
|
13
|
+
else
|
14
|
+
send_jsonrpc_response(request_id, result:)
|
15
|
+
end
|
16
16
|
end
|
17
17
|
end
|
18
18
|
end
|
@@ -8,12 +8,11 @@ module ActionMCP
|
|
8
8
|
|
9
9
|
def send_tools_call(request_id, tool_name, arguments, _meta = {})
|
10
10
|
result = ToolsRegistry.tool_call(tool_name, arguments, _meta)
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
:
|
15
|
-
|
16
|
-
).as_json)
|
11
|
+
if result.is_error
|
12
|
+
send_jsonrpc_response(request_id, error: result)
|
13
|
+
else
|
14
|
+
send_jsonrpc_response(request_id, result:)
|
15
|
+
end
|
17
16
|
end
|
18
17
|
end
|
19
18
|
end
|
data/lib/action_mcp/version.rb
CHANGED