actionmcp 0.9.0 → 0.10.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: 39f50d723aa2a52bcc863de6cec116970a5595719f4f974e21a91d40e6279685
4
+ data.tar.gz: d5dd09cf26636b709e928c36b710a7319d4afa05e4538fe090e4a7d9f0571533
5
5
  SHA512:
6
- metadata.gz: 7e4caed2e47ffb767e8c6980a07542d093cc5bb7dd1d0438c342e0ec7a9c6af56e0409b108955b229f0fa4147d263a3060ce72a2864f7dc5405aab9f28e66bf6
7
- data.tar.gz: ddb5fa89098629fecd125c317d935a0d0f3e4d50d828f4cd5d3b1c4b435be648b973235961af62e566431d1f991d4f3f0e11aeb7812b6423e0e4186cb4785b73
6
+ metadata.gz: bec413dbeb4c8377f57f0d8a4942b0e04bc71412e8aea833730c521965e5b3e1f4803dd5f37553fa7d51d31baeefd8efde86bdb26309e698ae43546a2ad88957
7
+ data.tar.gz: 8790e5172337913ffc393bf1deaa4f09e689f85f9ba1bc611fcbbd261788cc625deed50fc361c3c1b8d131407777d39bf17461bf9f18893ed21597dd9edeefd1
@@ -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, 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
@@ -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
@@ -27,31 +25,30 @@ module ActionMCP
27
25
  end
28
26
 
29
27
  # Convert to hash format expected by MCP protocol
30
- def as_json(options = nil)
28
+ def to_h(options = {})
31
29
  {
32
- content: @contents.map { |c| c.as_json(options) },
30
+ content: @contents.map { |c| c.to_h },
33
31
  isError: @is_error
34
- }.compact
32
+ }
35
33
  end
36
34
 
37
- # Alias to_h to as_json for consistency
38
- alias_method :to_h, :as_json
35
+ # Alias as_json to to_h for consistency
36
+ alias_method :as_json, :to_h
39
37
 
40
38
  # Handle to_json directly
41
39
  def to_json(options = nil)
42
- as_json(options).to_json
40
+ to_h.to_json(options)
43
41
  end
44
42
 
45
- # Compare with hash for easier testing
46
- # This allows assertions like: assert_equal({content: [...], isError: false}, tool_response)
43
+ # Compare with hash for easier testing.
47
44
  def ==(other)
48
45
  case other
49
46
  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
47
+ # Convert both to normalized format for comparison
48
+ hash_self = to_h.deep_transform_keys { |key| key.to_s.underscore }
49
+ hash_other = other.deep_transform_keys { |key| key.to_s.underscore }
50
+ hash_self == hash_other
53
51
  when ToolResponse
54
- # Direct comparison with another ToolResponse
55
52
  contents == other.contents && is_error == other.is_error
56
53
  else
57
54
  super
@@ -2,7 +2,7 @@
2
2
 
3
3
  require_relative "gem_version"
4
4
  module ActionMCP
5
- VERSION = "0.9.0"
5
+ VERSION = "0.10.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.10.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Abdelkader Boudih