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,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
@@ -1,7 +1,7 @@
1
- # frozen_string_literal: true
2
-
3
1
  module ActionMCP
4
2
  # Returns the currently loaded version of Active MCP as a +Gem::Version+.
3
+ #
4
+ # @return [Gem::Version] the currently loaded version of Active MCP
5
5
  def self.gem_version
6
6
  Gem::Version.new VERSION
7
7
  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
@@ -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
@@ -2,66 +2,44 @@
2
2
 
3
3
  module ActionMCP
4
4
  # Abstract base class for Prompts
5
- # Defines: name, description, arguments, plus auto-registration.
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._prompt_name = name
17
+ self._capability_name = name
42
18
  else
43
- _prompt_name || default_prompt_name
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.dasherize.sub(/-prompt$/, "")
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
- # def call
139
- # # Perform logic, e.g. analyze code, etc.
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