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
data/lib/action_mcp/content.rb
CHANGED
@@ -1,28 +1,33 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
+
# Module for managing content within ActionMCP.
|
4
5
|
module Content
|
5
|
-
extend ActiveSupport::Autoload
|
6
6
|
# Base class for MCP content items.
|
7
7
|
class Base
|
8
|
+
# @return [Symbol] The type of content.
|
8
9
|
attr_reader :type
|
9
10
|
|
11
|
+
# Initializes a new content item.
|
12
|
+
#
|
13
|
+
# @param type [Symbol] The type of content.
|
10
14
|
def initialize(type)
|
11
15
|
@type = type
|
12
16
|
end
|
13
17
|
|
18
|
+
# Returns a hash representation of the content.
|
19
|
+
#
|
20
|
+
# @return [Hash] The hash representation.
|
14
21
|
def to_h
|
15
22
|
{ type: @type }
|
16
23
|
end
|
17
24
|
|
25
|
+
# Returns a JSON representation of the content.
|
26
|
+
#
|
27
|
+
# @return [String] The JSON representation.
|
18
28
|
def to_json(*)
|
19
29
|
MultiJson.dump(to_h, *)
|
20
30
|
end
|
21
31
|
end
|
22
|
-
|
23
|
-
autoload :Image
|
24
|
-
autoload :Text
|
25
|
-
autoload :Audio
|
26
|
-
autoload :Resource
|
27
32
|
end
|
28
33
|
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "rails"
|
4
|
+
require "active_model/railtie"
|
5
|
+
|
6
|
+
module ActionMCP
|
7
|
+
# Engine for integrating ActionMCP with Rails applications.
|
8
|
+
class Engine < ::Rails::Engine
|
9
|
+
isolate_namespace ActionMCP
|
10
|
+
|
11
|
+
ActiveSupport::Inflector.inflections(:en) do |inflect|
|
12
|
+
inflect.acronym "SSE"
|
13
|
+
inflect.acronym "MCP"
|
14
|
+
end
|
15
|
+
# Provide a configuration namespace for ActionMCP
|
16
|
+
config.action_mcp = ActiveSupport::OrderedOptions.new
|
17
|
+
|
18
|
+
initializer "action_mcp.configure" do |app|
|
19
|
+
options = app.config.action_mcp.to_h.symbolize_keys
|
20
|
+
|
21
|
+
# Override the default configuration if specified in the Rails app.
|
22
|
+
ActionMCP.configuration.name = options[:name] if options.key?(:name)
|
23
|
+
ActionMCP.configuration.version = options[:version] if options.key?(:version)
|
24
|
+
ActionMCP.configuration.logging_enabled = options.fetch(:logging_enabled, true)
|
25
|
+
end
|
26
|
+
|
27
|
+
# Initialize the ActionMCP logger.
|
28
|
+
initializer "action_mcp.logger" do
|
29
|
+
ActiveSupport.on_load(:action_mcp) do
|
30
|
+
self.logger = ::Rails.logger
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -3,7 +3,13 @@
|
|
3
3
|
module ActionMCP
|
4
4
|
# This temporary naming extracted from MCPangea
|
5
5
|
# If there is a better name, please suggest it or part of ActiveModel, open a PR
|
6
|
+
#
|
7
|
+
# Custom type for handling arrays of integers in ActiveModel.
|
6
8
|
class IntegerArray < ActiveModel::Type::Value
|
9
|
+
# Casts the given value to an array of integers.
|
10
|
+
#
|
11
|
+
# @param value [Object] The value to cast.
|
12
|
+
# @return [Array<Integer>] The array of integers.
|
7
13
|
def cast(value)
|
8
14
|
Array(value).map(&:to_i) # Ensure all elements are integers
|
9
15
|
end
|
@@ -2,6 +2,7 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
module JsonRpc
|
5
|
+
# Custom exception class for JSON-RPC errors, based on the JSON-RPC 2.0 specification.
|
5
6
|
class JsonRpcError < StandardError
|
6
7
|
# Define the standard JSON-RPC 2.0 error codes
|
7
8
|
ERROR_CODES = {
|
@@ -31,14 +32,25 @@ module ActionMCP
|
|
31
32
|
}
|
32
33
|
}.freeze
|
33
34
|
|
35
|
+
# @return [Integer] The error code.
|
36
|
+
# @return [Object] The error data.
|
34
37
|
attr_reader :code, :data
|
35
38
|
|
36
39
|
# Retrieve error details by symbol.
|
40
|
+
#
|
41
|
+
# @param symbol [Symbol] The error symbol.
|
42
|
+
# @raise [ArgumentError] if the error code is unknown.
|
43
|
+
# @return [Hash] The error details.
|
37
44
|
def self.[](symbol)
|
38
45
|
ERROR_CODES[symbol] or raise ArgumentError, "Unknown error code: #{symbol}"
|
39
46
|
end
|
40
47
|
|
41
48
|
# Build an error hash, allowing custom message or data to override defaults.
|
49
|
+
#
|
50
|
+
# @param symbol [Symbol] The error symbol.
|
51
|
+
# @param message [String, nil] Optional custom message.
|
52
|
+
# @param data [Object, nil] Optional custom data.
|
53
|
+
# @return [Hash] The error hash.
|
42
54
|
def self.build(symbol, message: nil, data: nil)
|
43
55
|
error = self[symbol].dup
|
44
56
|
error[:message] = message if message
|
@@ -47,6 +59,10 @@ module ActionMCP
|
|
47
59
|
end
|
48
60
|
|
49
61
|
# Initialize the error using a symbol key, with optional custom message and data.
|
62
|
+
#
|
63
|
+
# @param symbol [Symbol] The error symbol.
|
64
|
+
# @param message [String, nil] Optional custom message.
|
65
|
+
# @param data [Object, nil] Optional custom data.
|
50
66
|
def initialize(symbol, message: nil, data: nil)
|
51
67
|
error_details = self.class.build(symbol, message: message, data: data)
|
52
68
|
@code = error_details[:code]
|
@@ -55,6 +71,8 @@ module ActionMCP
|
|
55
71
|
end
|
56
72
|
|
57
73
|
# Returns a hash formatted for a JSON-RPC error response.
|
74
|
+
#
|
75
|
+
# @return [Hash] The error hash.
|
58
76
|
def as_json
|
59
77
|
hash = { code: code, message: message }
|
60
78
|
hash[:data] = data if data
|
@@ -62,6 +80,9 @@ module ActionMCP
|
|
62
80
|
end
|
63
81
|
|
64
82
|
# Converts the error hash to a JSON string.
|
83
|
+
#
|
84
|
+
# @param _args [Array] Arguments passed to MultiJson.dump.
|
85
|
+
# @return [String] The JSON string.
|
65
86
|
def to_json(*_args)
|
66
87
|
MultiJson.dump(as_json, *args)
|
67
88
|
end
|
@@ -2,11 +2,19 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
module JsonRpc
|
5
|
+
# Represents a JSON-RPC notification.
|
5
6
|
Notification = Data.define(:method, :params) do
|
7
|
+
# Initializes a new Notification.
|
8
|
+
#
|
9
|
+
# @param method [String] The method name.
|
10
|
+
# @param params [Hash, nil] The parameters (optional).
|
6
11
|
def initialize(method:, params: nil)
|
7
12
|
super
|
8
13
|
end
|
9
14
|
|
15
|
+
# Returns a hash representation of the notification.
|
16
|
+
#
|
17
|
+
# @return [Hash] The hash representation.
|
10
18
|
def to_h
|
11
19
|
{
|
12
20
|
jsonrpc: "2.0",
|
@@ -2,12 +2,22 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
module JsonRpc
|
5
|
+
# Represents a JSON-RPC request.
|
5
6
|
Request = Data.define(:id, :method, :params) do
|
7
|
+
# Initializes a new Request.
|
8
|
+
#
|
9
|
+
# @param id [String, Numeric] The request identifier.
|
10
|
+
# @param method [String] The method name.
|
11
|
+
# @param params [Hash, nil] The parameters (optional).
|
12
|
+
# @raise [JsonRpcError] if the ID is invalid.
|
6
13
|
def initialize(id:, method:, params: nil)
|
7
14
|
validate_id(id)
|
8
15
|
super
|
9
16
|
end
|
10
17
|
|
18
|
+
# Returns a hash representation of the request.
|
19
|
+
#
|
20
|
+
# @return [Hash] The hash representation.
|
11
21
|
def to_h
|
12
22
|
hash = {
|
13
23
|
jsonrpc: "2.0",
|
@@ -20,6 +30,10 @@ module ActionMCP
|
|
20
30
|
|
21
31
|
private
|
22
32
|
|
33
|
+
# Validates the ID.
|
34
|
+
#
|
35
|
+
# @param id [Object] The ID to validate.
|
36
|
+
# @raise [JsonRpcError] if the ID is invalid.
|
23
37
|
def validate_id(id)
|
24
38
|
unless id.is_a?(String) || id.is_a?(Numeric)
|
25
39
|
raise JsonRpcError.new(:invalid_params,
|
@@ -2,14 +2,25 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
module JsonRpc
|
5
|
+
# Represents a JSON-RPC response.
|
5
6
|
Response = Data.define(:id, :result, :error) do
|
7
|
+
# Initializes a new Response.
|
8
|
+
#
|
9
|
+
# @param id [String, Numeric] The request identifier.
|
10
|
+
# @param result [Object, nil] The result data (optional).
|
11
|
+
# @param error [Object, nil] The error data (optional).
|
12
|
+
# @raise [ArgumentError] if neither result nor error is provided, or if both are provided.
|
6
13
|
def initialize(id:, result: nil, error: nil)
|
7
14
|
validate_presence_of_result_or_error!(result, error)
|
8
15
|
validate_absence_of_both_result_and_error!(result, error)
|
16
|
+
result, error = transform_value_to_hash!(result, error)
|
9
17
|
|
10
|
-
super
|
18
|
+
super(id: id, result: result, error: error)
|
11
19
|
end
|
12
20
|
|
21
|
+
# Returns a hash representation of the response.
|
22
|
+
#
|
23
|
+
# @return [Hash] The hash representation.
|
13
24
|
def to_h
|
14
25
|
{
|
15
26
|
jsonrpc: "2.0",
|
@@ -19,15 +30,35 @@ module ActionMCP
|
|
19
30
|
}.compact
|
20
31
|
end
|
21
32
|
|
33
|
+
def is_error?
|
34
|
+
error.present?
|
35
|
+
end
|
36
|
+
|
22
37
|
private
|
23
38
|
|
39
|
+
# Validates that either result or error is present.
|
40
|
+
#
|
41
|
+
# @param result [Object, nil] The result data.
|
42
|
+
# @param error [Object, nil] The error data.
|
43
|
+
# @raise [ArgumentError] if neither result nor error is provided.
|
24
44
|
def validate_presence_of_result_or_error!(result, error)
|
25
45
|
raise ArgumentError, "Either result or error must be provided." if result.nil? && error.nil?
|
26
46
|
end
|
27
47
|
|
48
|
+
# Validates that both result and error are not present simultaneously.
|
49
|
+
#
|
50
|
+
# @param result [Object, nil] The result data.
|
51
|
+
# @param error [Object, nil] The error data.
|
52
|
+
# @raise [ArgumentError] if both result and error are provided.
|
28
53
|
def validate_absence_of_both_result_and_error!(result, error)
|
29
54
|
raise ArgumentError, "Both result and error cannot be provided simultaneously." if result && error
|
30
55
|
end
|
56
|
+
|
57
|
+
def transform_value_to_hash!(result, error)
|
58
|
+
result = result.is_a?(String) ? (MultiJson.load(result) rescue result) : result
|
59
|
+
error = error.is_a?(String) ? (MultiJson.load(error) rescue error) : error
|
60
|
+
[ result, error ]
|
61
|
+
end
|
31
62
|
end
|
32
63
|
end
|
33
64
|
end
|
data/lib/action_mcp/json_rpc.rb
CHANGED
@@ -1,12 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
+
# Module for handling JSON-RPC communication.
|
4
5
|
module JsonRpc
|
5
|
-
extend ActiveSupport::Autoload
|
6
|
-
|
7
|
-
autoload :JsonRpcError
|
8
|
-
autoload :Notification
|
9
|
-
autoload :Request
|
10
|
-
autoload :Response
|
11
6
|
end
|
12
7
|
end
|
@@ -0,0 +1,106 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
class JsonRpcHandler
|
5
|
+
attr_reader :transport
|
6
|
+
|
7
|
+
def initialize(transport)
|
8
|
+
@transport = transport
|
9
|
+
end
|
10
|
+
|
11
|
+
# Process a single line of input.
|
12
|
+
def call(line)
|
13
|
+
request = if line.is_a?(String)
|
14
|
+
line.strip!
|
15
|
+
return if line.empty?
|
16
|
+
begin
|
17
|
+
MultiJson.load(line)
|
18
|
+
rescue MultiJson::ParseError => e
|
19
|
+
Rails.logger.error("Failed to parse JSON: #{e.message}")
|
20
|
+
return
|
21
|
+
end
|
22
|
+
|
23
|
+
else
|
24
|
+
line
|
25
|
+
end
|
26
|
+
process_request(request)
|
27
|
+
end
|
28
|
+
|
29
|
+
private
|
30
|
+
|
31
|
+
def process_request(request)
|
32
|
+
unless request["jsonrpc"] == "2.0"
|
33
|
+
puts "Invalid request: #{request}"
|
34
|
+
return
|
35
|
+
end
|
36
|
+
method = request["method"]
|
37
|
+
id = request["id"]
|
38
|
+
params = request["params"]
|
39
|
+
|
40
|
+
case method
|
41
|
+
when "initialize"
|
42
|
+
puts "\e[31mSending capabilities\e[0m"
|
43
|
+
transport.send_capabilities(id, params)
|
44
|
+
when "ping"
|
45
|
+
transport.send_pong(id)
|
46
|
+
when /^notifications\//
|
47
|
+
puts "\e[31mProcessing notifications\e[0m"
|
48
|
+
process_notifications(method, id, params)
|
49
|
+
when /^prompts\//
|
50
|
+
process_prompts(method, id, params)
|
51
|
+
when /^resources\//
|
52
|
+
process_resources(method, id, params)
|
53
|
+
when /^tools\//
|
54
|
+
process_tools(method, id, params)
|
55
|
+
else
|
56
|
+
puts "\e[31mUnknown method: #{method}\e[0m"
|
57
|
+
Rails.logger.warn("Unknown method: #{method}")
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def process_notifications(method, _id, _params)
|
62
|
+
case method
|
63
|
+
when "notifications/initialized"
|
64
|
+
puts "\e[31mInitialized\e[0m"
|
65
|
+
transport.initialized!
|
66
|
+
else
|
67
|
+
Rails.logger.warn("Unknown notifications method: #{method}")
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
def process_prompts(method, id, params)
|
72
|
+
case method
|
73
|
+
when "prompts/get"
|
74
|
+
transport.send_prompts_get(id, params&.dig("name"), params&.dig("arguments"))
|
75
|
+
when "prompts/list"
|
76
|
+
transport.send_prompts_list(id)
|
77
|
+
else
|
78
|
+
Rails.logger.warn("Unknown prompts method: #{method}")
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def process_resources(method, id, params)
|
83
|
+
case method
|
84
|
+
when "resources/list"
|
85
|
+
transport.send_resources_list(id)
|
86
|
+
when "resources/templates/list"
|
87
|
+
transport.send_resource_templates_list(id)
|
88
|
+
when "resources/read"
|
89
|
+
transport.send_resource_read(id, params)
|
90
|
+
else
|
91
|
+
Rails.logger.warn("Unknown resources method: #{method}")
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
def process_tools(method, id, params)
|
96
|
+
case method
|
97
|
+
when "tools/list"
|
98
|
+
transport.send_tools_list(id)
|
99
|
+
when "tools/call"
|
100
|
+
transport.send_tools_call(id, params&.dig("name"), params&.dig("arguments"))
|
101
|
+
else
|
102
|
+
Rails.logger.warn("Unknown tools method: #{method}")
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
106
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_support/tagged_logging"
|
4
|
+
require "active_support/logger"
|
5
|
+
require "logger"
|
6
|
+
|
7
|
+
module ActionMCP
|
8
|
+
# Module for providing logging functionality to ActionMCP transport.
|
9
|
+
module Logging
|
10
|
+
extend ActiveSupport::Concern
|
11
|
+
|
12
|
+
# Included hook to configure the logger.
|
13
|
+
included do
|
14
|
+
logger_instance = ActiveSupport::Logger.new(STDOUT)
|
15
|
+
logger_instance.level = Logger.const_get(ActionMCP.configuration.logging_level.to_s.upcase)
|
16
|
+
cattr_accessor :logger, default: ActiveSupport::TaggedLogging.new(logger_instance)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
data/lib/action_mcp/prompt.rb
CHANGED
@@ -2,66 +2,44 @@
|
|
2
2
|
|
3
3
|
module ActionMCP
|
4
4
|
# Abstract base class for Prompts
|
5
|
-
|
6
|
-
class Prompt
|
7
|
-
include ActiveModel::Model
|
8
|
-
include ActiveModel::Attributes
|
9
|
-
include Renderable
|
10
|
-
|
11
|
-
class_attribute :_prompt_name, instance_accessor: false
|
12
|
-
class_attribute :_description, instance_accessor: false, default: ""
|
5
|
+
class Prompt < Capability
|
13
6
|
class_attribute :_argument_definitions, instance_accessor: false, default: []
|
14
|
-
class_attribute :abstract_prompt, instance_accessor: false, default: false
|
15
|
-
|
16
|
-
def self.inherited(subclass)
|
17
|
-
super
|
18
|
-
return if subclass == Prompt
|
19
|
-
return if subclass.name == "ApplicationPrompt"
|
20
|
-
|
21
|
-
subclass.abstract_prompt = false
|
22
|
-
|
23
|
-
# Automatically register the subclass with the PromptsRegistry
|
24
|
-
PromptsRegistry.register(subclass.prompt_name, subclass)
|
25
|
-
end
|
26
|
-
|
27
|
-
def self.abstract!
|
28
|
-
self.abstract_prompt = true
|
29
|
-
# If already registered, you might want to unregister it here.
|
30
|
-
end
|
31
|
-
|
32
|
-
def self.abstract?
|
33
|
-
abstract_prompt
|
34
|
-
end
|
35
7
|
|
36
8
|
# ---------------------------------------------------
|
37
9
|
# Prompt Name
|
38
10
|
# ---------------------------------------------------
|
11
|
+
# Gets or sets the prompt name.
|
12
|
+
#
|
13
|
+
# @param name [String, nil] The prompt name to set.
|
14
|
+
# @return [String] The prompt name.
|
39
15
|
def self.prompt_name(name = nil)
|
40
16
|
if name
|
41
|
-
self.
|
17
|
+
self._capability_name = name
|
42
18
|
else
|
43
|
-
|
19
|
+
_capability_name || default_prompt_name
|
44
20
|
end
|
45
21
|
end
|
46
22
|
|
23
|
+
# Returns the default prompt name based on the class name.
|
24
|
+
#
|
25
|
+
# @return [String] The default prompt name.
|
47
26
|
def self.default_prompt_name
|
48
|
-
name.demodulize.underscore.
|
27
|
+
name.demodulize.underscore.sub(/_prompt$/, "")
|
49
28
|
end
|
50
|
-
|
51
|
-
|
52
|
-
# Description
|
53
|
-
# ---------------------------------------------------
|
54
|
-
def self.description(text = nil)
|
55
|
-
if text
|
56
|
-
self._description = text
|
57
|
-
else
|
58
|
-
_description
|
59
|
-
end
|
29
|
+
class << self
|
30
|
+
alias default_capability_name default_prompt_name
|
60
31
|
end
|
61
32
|
|
62
33
|
# ---------------------------------------------------
|
63
34
|
# Argument DSL
|
64
35
|
# ---------------------------------------------------
|
36
|
+
# Defines an argument for the prompt.
|
37
|
+
#
|
38
|
+
# @param arg_name [Symbol] The name of the argument.
|
39
|
+
# @param description [String] The description of the argument.
|
40
|
+
# @param required [Boolean] Whether the argument is required.
|
41
|
+
# @param default [Object] The default value of the argument.
|
42
|
+
# @return [void]
|
65
43
|
def self.argument(arg_name, description: "", required: false, default: nil)
|
66
44
|
arg_def = {
|
67
45
|
name: arg_name.to_s,
|
@@ -78,6 +56,9 @@ module ActionMCP
|
|
78
56
|
validates arg_name, presence: true
|
79
57
|
end
|
80
58
|
|
59
|
+
# Returns the list of argument definitions.
|
60
|
+
#
|
61
|
+
# @return [Array<Hash>] The list of argument definitions.
|
81
62
|
def self.arguments
|
82
63
|
_argument_definitions
|
83
64
|
end
|
@@ -85,6 +66,7 @@ module ActionMCP
|
|
85
66
|
# ---------------------------------------------------
|
86
67
|
# Convert prompt definition to Hash
|
87
68
|
# ---------------------------------------------------
|
69
|
+
# @return [Hash] The prompt definition as a Hash.
|
88
70
|
def self.to_h
|
89
71
|
{
|
90
72
|
name: prompt_name,
|
@@ -106,6 +88,8 @@ module ActionMCP
|
|
106
88
|
# Raises:
|
107
89
|
# ActionMCP::JsonRpc::JsonRpcError(:invalid_params) if validation fails.
|
108
90
|
#
|
91
|
+
# @param params [Hash] The parameters for the prompt.
|
92
|
+
# @return [Object] The result of the prompt's call method.
|
109
93
|
def self.call(params)
|
110
94
|
prompt = new(params) # Initialize an instance with provided params
|
111
95
|
unless prompt.valid?
|
@@ -131,14 +115,14 @@ module ActionMCP
|
|
131
115
|
#
|
132
116
|
# Usage: Called internally after validation in self.call
|
133
117
|
#
|
118
|
+
# @raise [NotImplementedError] Subclasses must implement the call method.
|
119
|
+
# @return [Array<Content>] Array of Content objects is expected as return value
|
134
120
|
def call
|
135
121
|
raise NotImplementedError, "Subclasses must implement the call method"
|
136
122
|
# Default implementation (no-op)
|
137
123
|
# In a real subclass, you might do:
|
138
|
-
#
|
139
|
-
#
|
140
|
-
# # Return something meaningful.
|
141
|
-
# end
|
124
|
+
# # Perform logic, e.g. analyze code, etc.
|
125
|
+
# # Return something meaningful.
|
142
126
|
end
|
143
127
|
end
|
144
128
|
end
|
@@ -1,11 +1,19 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActionMCP
|
4
|
+
# Registry for managing prompts.
|
4
5
|
class PromptsRegistry < RegistryBase
|
5
6
|
class << self
|
7
|
+
# @!method prompts
|
8
|
+
# Returns all registered prompts.
|
9
|
+
# @return [Hash] A hash of registered prompts.
|
6
10
|
alias prompts items
|
7
|
-
alias available_prompts enabled
|
8
11
|
|
12
|
+
# Calls a prompt with the given name and arguments.
|
13
|
+
#
|
14
|
+
# @param prompt_name [String] The name of the prompt to call.
|
15
|
+
# @param arguments [Hash] The arguments to pass to the prompt.
|
16
|
+
# @return [Hash] A hash containing the prompt's response.
|
9
17
|
def prompt_call(prompt_name, arguments)
|
10
18
|
prompt = find(prompt_name)
|
11
19
|
prompt = prompt.new(arguments)
|
@@ -24,6 +32,10 @@ module ActionMCP
|
|
24
32
|
}
|
25
33
|
end
|
26
34
|
end
|
35
|
+
|
36
|
+
def item_klass
|
37
|
+
Prompt
|
38
|
+
end
|
27
39
|
end
|
28
40
|
end
|
29
41
|
end
|