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.
- checksums.yaml +4 -4
- data/README.md +133 -30
- data/Rakefile +0 -2
- data/app/controllers/action_mcp/application_controller.rb +13 -0
- data/app/controllers/action_mcp/messages_controller.rb +51 -0
- data/app/controllers/action_mcp/sse_controller.rb +151 -0
- data/config/routes.rb +4 -0
- data/exe/actionmcp_cli +221 -0
- data/lib/action_mcp/capability.rb +52 -0
- data/lib/action_mcp/client.rb +243 -1
- data/lib/action_mcp/configuration.rb +50 -1
- data/lib/action_mcp/content/audio.rb +9 -0
- data/lib/action_mcp/content/image.rb +9 -0
- data/lib/action_mcp/content/resource.rb +13 -0
- data/lib/action_mcp/content/text.rb +7 -0
- data/lib/action_mcp/content.rb +11 -6
- data/lib/action_mcp/engine.rb +34 -0
- data/lib/action_mcp/gem_version.rb +2 -2
- data/lib/action_mcp/integer_array.rb +6 -0
- data/lib/action_mcp/json_rpc/json_rpc_error.rb +21 -0
- data/lib/action_mcp/json_rpc/notification.rb +8 -0
- data/lib/action_mcp/json_rpc/request.rb +14 -0
- data/lib/action_mcp/json_rpc/response.rb +32 -1
- data/lib/action_mcp/json_rpc.rb +1 -6
- data/lib/action_mcp/json_rpc_handler.rb +106 -0
- data/lib/action_mcp/logging.rb +19 -0
- data/lib/action_mcp/prompt.rb +30 -46
- data/lib/action_mcp/prompts_registry.rb +13 -1
- data/lib/action_mcp/registry_base.rb +47 -28
- data/lib/action_mcp/renderable.rb +26 -0
- data/lib/action_mcp/resource.rb +3 -1
- data/lib/action_mcp/server.rb +4 -1
- data/lib/action_mcp/string_array.rb +5 -0
- data/lib/action_mcp/tool.rb +16 -53
- data/lib/action_mcp/tools_registry.rb +14 -1
- data/lib/action_mcp/transport/capabilities.rb +21 -0
- data/lib/action_mcp/transport/messaging.rb +20 -0
- data/lib/action_mcp/transport/prompts.rb +19 -0
- data/lib/action_mcp/transport/sse_client.rb +309 -0
- data/lib/action_mcp/transport/stdio_client.rb +117 -0
- data/lib/action_mcp/transport/tools.rb +20 -0
- data/lib/action_mcp/transport/transport_base.rb +125 -0
- data/lib/action_mcp/transport.rb +1 -235
- data/lib/action_mcp/transport_handler.rb +54 -0
- data/lib/action_mcp/version.rb +4 -5
- data/lib/action_mcp.rb +36 -33
- data/lib/generators/action_mcp/prompt/templates/prompt.rb.erb +3 -1
- data/lib/generators/action_mcp/tool/templates/tool.rb.erb +5 -1
- data/lib/tasks/action_mcp_tasks.rake +28 -5
- metadata +66 -9
- data/exe/action_mcp_stdio +0 -0
- data/lib/action_mcp/railtie.rb +0 -27
- 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
|
-
#
|
13
|
-
|
14
|
-
|
15
|
-
|
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
|
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.
|
31
|
-
end
|
32
|
-
|
33
|
-
def unregister(name)
|
34
|
-
items.delete(name)
|
36
|
+
items.size
|
35
37
|
end
|
36
38
|
|
37
|
-
|
38
|
-
|
39
|
-
|
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
|
-
|
50
|
-
|
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,
|
64
|
-
RegistryBase.send(:abstract_item?,
|
65
|
-
end.map { |name,
|
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
|
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,
|
data/lib/action_mcp/resource.rb
CHANGED
@@ -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
|
data/lib/action_mcp/server.rb
CHANGED
@@ -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
|
data/lib/action_mcp/tool.rb
CHANGED
@@ -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
|
-
|
17
|
-
|
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.
|
28
|
+
self._capability_name = name
|
62
29
|
else
|
63
|
-
|
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.
|
38
|
+
name.demodulize.underscore.sub(/_tool$/, "")
|
72
39
|
end
|
73
40
|
|
74
|
-
|
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
|
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
|
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
|