actionmcp 0.2.0 → 0.2.4

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.
Files changed (53) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +133 -30
  3. data/Rakefile +0 -2
  4. data/app/controllers/action_mcp/application_controller.rb +13 -0
  5. data/app/controllers/action_mcp/messages_controller.rb +51 -0
  6. data/app/controllers/action_mcp/sse_controller.rb +151 -0
  7. data/config/routes.rb +4 -0
  8. data/exe/actionmcp_cli +221 -0
  9. data/lib/action_mcp/capability.rb +52 -0
  10. data/lib/action_mcp/client.rb +243 -1
  11. data/lib/action_mcp/configuration.rb +50 -1
  12. data/lib/action_mcp/content/audio.rb +9 -0
  13. data/lib/action_mcp/content/image.rb +9 -0
  14. data/lib/action_mcp/content/resource.rb +13 -0
  15. data/lib/action_mcp/content/text.rb +7 -0
  16. data/lib/action_mcp/content.rb +11 -6
  17. data/lib/action_mcp/engine.rb +34 -0
  18. data/lib/action_mcp/gem_version.rb +2 -2
  19. data/lib/action_mcp/integer_array.rb +6 -0
  20. data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
  21. data/lib/action_mcp/json_rpc/notification.rb +8 -0
  22. data/lib/action_mcp/json_rpc/request.rb +14 -0
  23. data/lib/action_mcp/json_rpc/response.rb +32 -1
  24. data/lib/action_mcp/json_rpc.rb +1 -6
  25. data/lib/action_mcp/json_rpc_handler.rb +106 -0
  26. data/lib/action_mcp/logging.rb +19 -0
  27. data/lib/action_mcp/prompt.rb +30 -46
  28. data/lib/action_mcp/prompts_registry.rb +13 -1
  29. data/lib/action_mcp/registry_base.rb +47 -28
  30. data/lib/action_mcp/renderable.rb +26 -0
  31. data/lib/action_mcp/resource.rb +3 -1
  32. data/lib/action_mcp/server.rb +4 -1
  33. data/lib/action_mcp/string_array.rb +5 -0
  34. data/lib/action_mcp/tool.rb +16 -53
  35. data/lib/action_mcp/tools_registry.rb +14 -1
  36. data/lib/action_mcp/transport/capabilities.rb +21 -0
  37. data/lib/action_mcp/transport/messaging.rb +20 -0
  38. data/lib/action_mcp/transport/prompts.rb +19 -0
  39. data/lib/action_mcp/transport/sse_client.rb +309 -0
  40. data/lib/action_mcp/transport/stdio_client.rb +117 -0
  41. data/lib/action_mcp/transport/tools.rb +20 -0
  42. data/lib/action_mcp/transport/transport_base.rb +125 -0
  43. data/lib/action_mcp/transport.rb +1 -235
  44. data/lib/action_mcp/transport_handler.rb +54 -0
  45. data/lib/action_mcp/version.rb +4 -5
  46. data/lib/action_mcp.rb +36 -33
  47. data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
  48. data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
  49. data/lib/tasks/action_mcp_tasks.rake +28 -5
  50. metadata +66 -9
  51. data/exe/action_mcp_stdio +0 -0
  52. data/lib/action_mcp/railtie.rb +0 -27
  53. data/lib/action_mcp/resources_bank.rb +0 -94
@@ -1,55 +1,61 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
+ # Base class for registries.
4
5
  class RegistryBase
6
+ # Error raised when an item is not found in the registry.
5
7
  class NotFound < StandardError; end
6
8
 
7
9
  class << self
10
+ # Returns all registered items.
11
+ #
12
+ # @return [Hash] A hash of registered items.
8
13
  def items
9
- @items ||= {}
14
+ @items = item_klass.descendants.each_with_object({}) do |klass, hash|
15
+ next if klass.abstract?
16
+ hash[klass.capability_name] = klass
17
+ end
10
18
  end
11
19
 
12
- # Register an item by unique name.
13
- def register(name, klass)
14
- raise ArgumentError, "Name can't be blank" if name.blank?
15
- raise ArgumentError, "Name '#{name}' is already registered." if items.key?(name)
16
-
17
- items[name] = { klass: klass, enabled: true }
18
- end
19
-
20
- # Retrieve an item’s metadata by name.
20
+ # Retrieve an item by name.
21
+ #
22
+ # @param name [String] The name of the item to find.
23
+ # @raise [NotFound] if the item is not found.
24
+ # @return [Class] The class of the item.
21
25
  def find(name)
22
26
  item = items[name]
23
27
  raise NotFound, "Item '#{name}' not found." if item.nil?
24
28
 
25
- item[:klass]
29
+ item
26
30
  end
27
31
 
28
32
  # Return the number of registered items, ignoring abstract ones.
33
+ #
34
+ # @return [Integer] The number of registered items.
29
35
  def size
30
- items.values.reject { |item| abstract_item?(item) }.size
31
- end
32
-
33
- def unregister(name)
34
- items.delete(name)
36
+ items.size
35
37
  end
36
38
 
37
- def clear!
38
- items.clear
39
- end
40
-
41
- # Chainable scope: returns only enabled, non-abstract items.
42
- def enabled
39
+ # Chainable scope: returns only non-abstract items.
40
+ #
41
+ # @return [RegistryScope] A RegistryScope instance.
42
+ def non_abstract
43
43
  RegistryScope.new(items)
44
44
  end
45
45
 
46
46
  private
47
47
 
48
48
  # Helper to determine if an item is abstract.
49
- def abstract_item?(item)
50
- klass = item[:klass]
49
+ #
50
+ # @param klass [Class] The class to check.
51
+ # @return [Boolean] True if the class is abstract, false otherwise.
52
+ def abstract_item?(klass)
51
53
  klass.respond_to?(:abstract?) && klass.abstract?
52
54
  end
55
+
56
+ def item_klass
57
+ raise NotImplementedError, "Implement in subclass"
58
+ end
53
59
  end
54
60
 
55
61
  # Query object for chainable registry scopes.
@@ -59,22 +65,35 @@ module ActionMCP
59
65
  # Using a Data type for items.
60
66
  Item = Data.define(:name, :klass)
61
67
 
68
+ # Initializes a new RegistryScope instance.
69
+ #
70
+ # @param items [Hash] The items to scope.
71
+ # @return [void]
62
72
  def initialize(items)
63
- @items = items.reject do |_name, item|
64
- RegistryBase.send(:abstract_item?, item) || !item[:enabled]
65
- end.map { |name, item| Item.new(name, item[:klass]) }
73
+ @items = items.reject do |_name, klass|
74
+ RegistryBase.send(:abstract_item?, klass)
75
+ end.map { |name, klass| Item.new(name, klass) }
66
76
  end
67
77
 
78
+ # Iterates over the items in the scope.
79
+ #
80
+ # @yield [Item] The item to yield.
81
+ # @return [void]
68
82
  def each(&)
69
83
  @items.each(&)
70
84
  end
71
85
 
72
- # Returns the names (keys) of all enabled items.
86
+ # Returns the names (keys) of all non-abstract items.
87
+ #
88
+ # @return [Array<String>] The names of all non-abstract items.
73
89
  def keys
74
90
  @items.map(&:name)
75
91
  end
76
92
 
77
93
  # Chainable finder for available tools by name.
94
+ #
95
+ # @param name [String] The name of the tool to find.
96
+ # @return [Class, nil] The class of the tool, or nil if not found.
78
97
  def find_available_tool(name)
79
98
  item = @items.find { |i| i.name == name }
80
99
  item&.klass
@@ -1,23 +1,49 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
+ # Module for rendering content.
4
5
  module Renderable
6
+ # Renders text content.
7
+ #
8
+ # @param text [String] The text to render.
9
+ # @return [Content::Text] The rendered text content.
5
10
  def render_text(text)
6
11
  Content::Text.new(text)
7
12
  end
8
13
 
14
+ # Renders audio content.
15
+ #
16
+ # @param data [String] The audio data.
17
+ # @param mime_type [String] The MIME type of the audio data.
18
+ # @return [Content::Audio] The rendered audio content.
9
19
  def render_audio(data, mime_type)
10
20
  Content::Audio.new(data, mime_type)
11
21
  end
12
22
 
23
+ # Renders image content.
24
+ #
25
+ # @param data [String] The image data.
26
+ # @param mime_type [String] The MIME type of the image data.
27
+ # @return [Content::Image] The rendered image content.
13
28
  def render_image(data, mime_type)
14
29
  Content::Image.new(data, mime_type)
15
30
  end
16
31
 
32
+ # Renders a resource.
33
+ #
34
+ # @param uri [String] The URI of the resource.
35
+ # @param mime_type [String] The MIME type of the resource.
36
+ # @param text [String, nil] The text associated with the resource.
37
+ # @param blob [String, nil] The blob associated with the resource.
38
+ # @return [Content::Resource] The rendered resource content.
17
39
  def render_resource(uri, mime_type, text: nil, blob: nil)
18
40
  Content::Resource.new(uri, mime_type, text: text, blob: blob)
19
41
  end
20
42
 
43
+ # Renders an error.
44
+ #
45
+ # @param errors [Array<String>] The errors to render.
46
+ # @return [Hash] A hash containing the error information.
21
47
  def render_error(errors)
22
48
  {
23
49
  isError: true,
@@ -1,9 +1,12 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
+ # Represents a resource with its metadata.
4
5
  Resource = Data.define(:uri, :name, :description, :mime_type, :size) do
5
6
  # Convert the resource to a hash with the keys expected by MCP.
6
7
  # Note: The key for mime_type is converted to 'mimeType' as specified.
8
+ #
9
+ # @return [Hash] A hash representation of the resource.
7
10
  def to_h
8
11
  hash = { uri: uri, name: name }
9
12
  hash[:description] = description if description
@@ -12,7 +15,6 @@ module ActionMCP
12
15
  hash
13
16
  end
14
17
 
15
- # Convert the resource to a JSON string.
16
18
  def to_json(*)
17
19
  MultiJson.dump(to_h, *)
18
20
  end
@@ -1,7 +1,10 @@
1
- # frozen_string_literal: true
2
1
 
3
2
  # TODO: move all server related code here before version 1.0.0
4
3
  module ActionMCP
4
+ # Module for server-related functionality.
5
5
  module Server
6
+ module_function def server
7
+ @server ||= ActionCable::Server::Base.new
8
+ end
6
9
  end
7
10
  end
@@ -1,7 +1,12 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
+ # Custom type for handling arrays of strings in ActiveModel.
4
5
  class StringArray < ActiveModel::Type::Value
6
+ # Casts the given value to an array of strings.
7
+ #
8
+ # @param value [Object] The value to cast.
9
+ # @return [Array<String>] The array of strings.
5
10
  def cast(value)
6
11
  Array(value).map(&:to_s) # Ensure all elements are strings
7
12
  end
@@ -5,49 +5,16 @@ module ActionMCP
5
5
  #
6
6
  # Provides a DSL for specifying metadata, properties, and nested collection schemas.
7
7
  # Tools are registered automatically in the ToolsRegistry unless marked as abstract.
8
- class Tool
9
- include ActiveModel::Model
10
- include ActiveModel::Attributes
11
- include Renderable
12
-
8
+ class Tool < Capability
13
9
  # --------------------------------------------------------------------------
14
10
  # Class Attributes for Tool Metadata and Schema
15
11
  # --------------------------------------------------------------------------
16
- class_attribute :_tool_name, instance_accessor: false
17
- class_attribute :_description, instance_accessor: false, default: ""
12
+ # @!attribute _schema_properties
13
+ # @return [Hash] The schema properties of the tool.
14
+ # @!attribute _required_properties
15
+ # @return [Array<String>] The required properties of the tool.
18
16
  class_attribute :_schema_properties, instance_accessor: false, default: {}
19
17
  class_attribute :_required_properties, instance_accessor: false, default: []
20
- class_attribute :abstract_tool, instance_accessor: false, default: false
21
-
22
- # --------------------------------------------------------------------------
23
- # Subclass Registration
24
- # --------------------------------------------------------------------------
25
- # Automatically registers non-abstract subclasses in the ToolsRegistry.
26
- #
27
- # @param subclass [Class] the subclass inheriting from Tool.
28
- def self.inherited(subclass)
29
- super
30
- return if subclass == Tool
31
-
32
- subclass.abstract_tool = false
33
- return if subclass.name == "ApplicationTool"
34
-
35
- ToolsRegistry.register(subclass.tool_name, subclass)
36
- end
37
-
38
- # Marks this tool as abstract so that it won’t be available for use.
39
- # If the tool is registered in ToolsRegistry, it is unregistered.
40
- def self.abstract!
41
- self.abstract_tool = true
42
- ToolsRegistry.unregister(tool_name) if ToolsRegistry.items.key?(tool_name)
43
- end
44
-
45
- # Returns whether this tool is abstract.
46
- #
47
- # @return [Boolean] true if abstract, false otherwise.
48
- def self.abstract?
49
- abstract_tool
50
- end
51
18
 
52
19
  # --------------------------------------------------------------------------
53
20
  # Tool Name and Description DSL
@@ -58,9 +25,9 @@ module ActionMCP
58
25
  # @return [String] The current tool name.
59
26
  def self.tool_name(name = nil)
60
27
  if name
61
- self._tool_name = name
28
+ self._capability_name = name
62
29
  else
63
- _tool_name || default_tool_name
30
+ _capability_name || default_tool_name
64
31
  end
65
32
  end
66
33
 
@@ -68,19 +35,11 @@ module ActionMCP
68
35
  #
69
36
  # @return [String] The default tool name.
70
37
  def self.default_tool_name
71
- name.demodulize.underscore.dasherize.sub(/-tool$/, "")
38
+ name.demodulize.underscore.sub(/_tool$/, "")
72
39
  end
73
40
 
74
- # Sets or retrieves the tool's description.
75
- #
76
- # @param text [String, nil] Optional. The description text to set.
77
- # @return [String] The current description.
78
- def self.description(text = nil)
79
- if text
80
- self._description = text
81
- else
82
- _description
83
- end
41
+ class << self
42
+ alias default_capability_name default_tool_name
84
43
  end
85
44
 
86
45
  # --------------------------------------------------------------------------
@@ -97,6 +56,7 @@ module ActionMCP
97
56
  # @param required [Boolean] Whether the property is required (default: false).
98
57
  # @param default [Object, nil] The default value for the property.
99
58
  # @param opts [Hash] Additional options for the JSON Schema.
59
+ # @return [void]
100
60
  def self.property(prop_name, type: "string", description: nil, required: false, default: nil, **opts)
101
61
  # Build the JSON Schema definition.
102
62
  prop_definition = { type: type }
@@ -127,6 +87,7 @@ module ActionMCP
127
87
  # @param description [String, nil] Optional description for the collection.
128
88
  # @param required [Boolean] Whether the collection is required (default: false).
129
89
  # @param default [Array, nil] The default value for the collection.
90
+ # @return [void]
130
91
  def self.collection(prop_name, type:, description: nil, required: false, default: [])
131
92
  raise ArgumentError, "Type is required for a collection" if type.nil?
132
93
 
@@ -166,6 +127,7 @@ module ActionMCP
166
127
  # Subclasses must implement this method.
167
128
  #
168
129
  # @raise [NotImplementedError] Always raised if not implemented in a subclass.
130
+ # @return [Array<Content>] Array of Content objects is expected as return value
169
131
  def call
170
132
  raise NotImplementedError, "Subclasses must implement the call method"
171
133
  # Default implementation (no-op)
@@ -184,13 +146,14 @@ module ActionMCP
184
146
  # @return [Symbol] The corresponding ActiveModel attribute type.
185
147
  def self.map_json_type_to_active_model_type(type)
186
148
  case type.to_s
187
- when "number" then :float # JSON Schema "number" is a float in Ruby, the spec doesn't have an integer type yet.
149
+ when "number" then :float # JSON Schema "number" is a float in Ruby, the spec doesn't have an integer type yet.
188
150
  when "array_number" then :integer_array
189
151
  when "array_integer" then :string_array
190
152
  when "array_string" then :string_array
191
- else :string
153
+ else :string
192
154
  end
193
155
  end
156
+
194
157
  private_class_method :map_json_type_to_active_model_type
195
158
  end
196
159
  end
@@ -1,11 +1,20 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module ActionMCP
4
+ # Registry for managing tools.
4
5
  class ToolsRegistry < RegistryBase
5
6
  class << self
7
+ # @!method tools
8
+ # Returns all registered tools.
9
+ # @return [Hash] A hash of registered tools.
6
10
  alias tools items
7
- alias available_tools enabled
8
11
 
12
+ # Calls a tool with the given name and arguments.
13
+ #
14
+ # @param tool_name [String] The name of the tool to call.
15
+ # @param arguments [Hash] The arguments to pass to the tool.
16
+ # @param _metadata [Hash] Optional metadata.
17
+ # @return [Hash] A hash containing the tool's response.
9
18
  def tool_call(tool_name, arguments, _metadata = {})
10
19
  tool = find(tool_name)
11
20
  tool = tool.new(arguments)
@@ -19,6 +28,10 @@ module ActionMCP
19
28
  }
20
29
  end
21
30
  end
31
+
32
+ def item_klass
33
+ Tool
34
+ end
22
35
  end
23
36
  end
24
37
  end
@@ -0,0 +1,21 @@
1
+ module ActionMCP
2
+ module Transport
3
+ module Capabilities
4
+ def send_capabilities(request_id, params = {})
5
+ @protocol_version = params["protocolVersion"]
6
+ @client_info = params["clientInfo"]
7
+ @client_capabilities = params["capabilities"]
8
+ capabilities = ActionMCP.configuration.capabilities
9
+
10
+ payload = {
11
+ protocolVersion: PROTOCOL_VERSION,
12
+ serverInfo: {
13
+ name: ActionMCP.configuration.name,
14
+ version: ActionMCP.configuration.version
15
+ }
16
+ }.merge(capabilities)
17
+ send_jsonrpc_response(request_id, result: payload)
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,20 @@
1
+ module ActionMCP
2
+ module Transport
3
+ module Messaging
4
+ def send_jsonrpc_request(method, params: nil, id: SecureRandom.uuid_v7)
5
+ request = JsonRpc::Request.new(id: id, method: method, params: params)
6
+ write_message(request.to_json)
7
+ end
8
+
9
+ def send_jsonrpc_response(request_id, result: nil, error: nil)
10
+ response = JsonRpc::Response.new(id: request_id, result: result, error: error)
11
+ write_message(response.to_json)
12
+ end
13
+
14
+ def send_jsonrpc_notification(method, params = nil)
15
+ notification = JsonRpc::Notification.new(method: method, params: params)
16
+ write_message(notification.to_json)
17
+ end
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,19 @@
1
+ module ActionMCP
2
+ module Transport
3
+ module Prompts
4
+ def send_prompts_list(request_id)
5
+ prompts = format_registry_items(PromptsRegistry.non_abstract)
6
+ send_jsonrpc_response(request_id, result: { prompts: prompts })
7
+ end
8
+
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)
16
+ end
17
+ end
18
+ end
19
+ end