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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a310c4bc57df9c1cd4d0fceb7b80fd9628414609bb6131e5d420584f5605f59c
4
- data.tar.gz: 881fb9d39845277adcb24d42e73475dfc9aca5e25afb2b049c5ee55446aca474
3
+ metadata.gz: f825a63198153189d9970beec66012f8428dd625ec442559043597e4a938113e
4
+ data.tar.gz: 82891ffd29e22e8a36a80a002d264884756f6b48dfbe373eca963268ba82c4dc
5
5
  SHA512:
6
- metadata.gz: 7e4caed2e47ffb767e8c6980a07542d093cc5bb7dd1d0438c342e0ec7a9c6af56e0409b108955b229f0fa4147d263a3060ce72a2864f7dc5405aab9f28e66bf6
7
- data.tar.gz: ddb5fa89098629fecd125c317d935a0d0f3e4d50d828f4cd5d3b1c4b435be648b973235961af62e566431d1f991d4f3f0e11aeb7812b6423e0e4186cb4785b73
6
+ metadata.gz: ff8d86f24b9f087f3dc0c81e6bda73045c554ce5f1bb1c90d16faa80411e99a79b4b64e905a552d9e68653759e7ca9380bee202586d52b5a13068626ba34c238
7
+ data.tar.gz: d872f57c2651d7a0c28db334aa88bb98862f969c5566cc65fc5beefc3fd0b83f46b6de7d9633eb1ba9571dc6e767ed9e4ec501e5858e7ef68b79870582e69e8e
@@ -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 => e
112
+ rescue
113
113
  # Handle exceptions during execution
114
- render text: "Error executing prompt: #{e.message}"
114
+ @response.mark_as_error!(:internal_error, message: "Unhandled Error executing prompt")
115
115
  end
116
116
  else
117
117
  # Handle validation failure
118
- render text: "Invalid input: #{errors.full_messages.join(', ')}"
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
- attr_reader :messages, :description
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
- messages: @messages
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 to_h to as_json for consistency
31
- alias_method :to_h, :as_json
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.size}"
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.valid?
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
@@ -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(is_error: false)
10
+ def initialize
13
11
  @contents = []
14
- @is_error = 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 as_json(options = nil)
31
- {
32
- content: @contents.map { |c| c.as_json(options) },
33
- isError: @is_error
34
- }.compact
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 to_h to as_json for consistency
38
- alias_method :to_h, :as_json
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
- as_json(options).to_json
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
- # Compare our hash representation with the other hash
51
- # Use deep symbolization to handle both string and symbol keys
52
- to_h.deep_symbolize_keys == other.deep_symbolize_keys
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([ "Tool execution failed: #{e.message}" ])
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(messages)
35
- {
36
- content: messages.map { |msg| Content::Text.new(msg) },
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
- send_jsonrpc_response(request_id, result: PromptsRegistry.prompt_call(prompt_name.to_s, params))
11
- rescue RegistryBase::NotFound
12
- send_jsonrpc_response(request_id, error: JsonRpc::JsonRpcError.new(
13
- :method_not_found,
14
- message: "Prompt not found: #{prompt_name}"
15
- ).as_json)
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
- send_jsonrpc_response(request_id, result: result)
12
- rescue RegistryBase::NotFound
13
- send_jsonrpc_response(request_id, error: JsonRpc::JsonRpcError.new(
14
- :method_not_found,
15
- message: "Tool not found: #{tool_name}"
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
@@ -2,7 +2,7 @@
2
2
 
3
3
  require_relative "gem_version"
4
4
  module ActionMCP
5
- VERSION = "0.9.0"
5
+ VERSION = "0.11.0"
6
6
 
7
7
  class << self
8
8
  alias version gem_version
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: actionmcp
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.9.0
4
+ version: 0.11.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Abdelkader Boudih