actionmcp 0.14.0 → 0.17.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.
Files changed (65) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +174 -144
  3. data/Rakefile +1 -1
  4. data/app/controllers/action_mcp/{application_controller.rb → mcp_controller.rb} +3 -1
  5. data/app/controllers/action_mcp/messages_controller.rb +7 -5
  6. data/app/controllers/action_mcp/sse_controller.rb +19 -13
  7. data/app/models/action_mcp/session/message.rb +95 -90
  8. data/app/models/action_mcp/session/resource.rb +10 -6
  9. data/app/models/action_mcp/session/subscription.rb +9 -5
  10. data/app/models/action_mcp/session.rb +22 -13
  11. data/app/models/action_mcp.rb +2 -0
  12. data/config/routes.rb +2 -0
  13. data/db/migrate/20250308122801_create_action_mcp_sessions.rb +12 -10
  14. data/db/migrate/20250314230152_add_is_ping_to_session_message.rb +2 -0
  15. data/db/migrate/20250316005021_create_action_mcp_session_subscriptions.rb +3 -1
  16. data/db/migrate/20250316005649_create_action_mcp_session_resources.rb +4 -2
  17. data/exe/actionmcp_cli +57 -55
  18. data/lib/action_mcp/base_json_rpc_handler.rb +97 -0
  19. data/lib/action_mcp/callbacks.rb +122 -0
  20. data/lib/action_mcp/capability.rb +6 -3
  21. data/lib/action_mcp/client.rb +20 -26
  22. data/lib/action_mcp/client_json_rpc_handler.rb +69 -0
  23. data/lib/action_mcp/configuration.rb +8 -8
  24. data/lib/action_mcp/content/resource.rb +1 -1
  25. data/lib/action_mcp/gem_version.rb +2 -0
  26. data/lib/action_mcp/instrumentation/controller_runtime.rb +37 -0
  27. data/lib/action_mcp/instrumentation/instrumentation.rb +26 -0
  28. data/lib/action_mcp/instrumentation/resource_instrumentation.rb +40 -0
  29. data/lib/action_mcp/json_rpc/response.rb +18 -2
  30. data/lib/action_mcp/json_rpc_handler.rb +93 -21
  31. data/lib/action_mcp/log_subscriber.rb +29 -0
  32. data/lib/action_mcp/logging.rb +1 -3
  33. data/lib/action_mcp/prompt.rb +15 -6
  34. data/lib/action_mcp/prompt_response.rb +1 -1
  35. data/lib/action_mcp/prompts_registry.rb +1 -0
  36. data/lib/action_mcp/registry_base.rb +1 -0
  37. data/lib/action_mcp/resource_callbacks.rb +156 -0
  38. data/lib/action_mcp/resource_template.rb +25 -19
  39. data/lib/action_mcp/resource_templates_registry.rb +19 -25
  40. data/lib/action_mcp/sampling_request.rb +113 -0
  41. data/lib/action_mcp/server.rb +4 -1
  42. data/lib/action_mcp/server_json_rpc_handler.rb +90 -0
  43. data/lib/action_mcp/test_helper.rb +6 -2
  44. data/lib/action_mcp/tool.rb +12 -3
  45. data/lib/action_mcp/tool_response.rb +3 -2
  46. data/lib/action_mcp/transport/capabilities.rb +5 -1
  47. data/lib/action_mcp/transport/messaging.rb +2 -0
  48. data/lib/action_mcp/transport/prompts.rb +2 -0
  49. data/lib/action_mcp/transport/resources.rb +23 -6
  50. data/lib/action_mcp/transport/roots.rb +11 -0
  51. data/lib/action_mcp/transport/sampling.rb +14 -0
  52. data/lib/action_mcp/transport/sse_client.rb +11 -15
  53. data/lib/action_mcp/transport/stdio_client.rb +12 -14
  54. data/lib/action_mcp/transport/tools.rb +2 -0
  55. data/lib/action_mcp/transport/transport_base.rb +16 -15
  56. data/lib/action_mcp/transport.rb +2 -0
  57. data/lib/action_mcp/transport_handler.rb +3 -0
  58. data/lib/action_mcp/version.rb +1 -1
  59. data/lib/action_mcp.rb +8 -2
  60. data/lib/generators/action_mcp/install/install_generator.rb +4 -1
  61. data/lib/generators/action_mcp/install/templates/application_mcp_res_template.rb +2 -0
  62. data/lib/generators/action_mcp/resource_template/resource_template_generator.rb +2 -0
  63. data/lib/generators/action_mcp/resource_template/templates/resource_template.rb.erb +1 -1
  64. data/lib/tasks/action_mcp_tasks.rake +11 -6
  65. metadata +26 -14
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # == Schema Information
2
4
  #
3
5
  # Table name: action_mcp_session_messages
@@ -24,110 +26,113 @@
24
26
  # fk_action_mcp_session_messages_session_id (session_id => action_mcp_sessions.id) ON DELETE => cascade ON UPDATE => cascade
25
27
  #
26
28
  module ActionMCP
27
- ##
28
- # Represents a message exchanged during an MCP session.
29
- # Its role is to store the content and metadata of each message,
30
- # including the direction (client or server), message type (request, response, notification),
31
- # and any associated JSON-RPC ID.
32
- class Session::Message < ApplicationRecord
33
- belongs_to :session,
34
- class_name: "ActionMCP::Session",
35
- inverse_of: :messages,
36
- counter_cache: true
37
-
38
- delegate :adapter,
39
- :role,
40
- :session_key,
41
- to: :session
42
-
43
- # Virtual attribute for data
44
- attr_reader :data
45
-
46
- after_create_commit :broadcast_message, if: :outgoing_message?
47
- # Set is_ping on responses if the original request was a ping
48
- after_create :handle_ping_response, if: -> { %w[response error].include?(message_type) }
49
-
50
- # Scope to exclude both "ping" requests and their responses
51
- scope :without_pings, -> { where(is_ping: false) }
52
-
53
- # @param payload [String, Hash]
54
- def data=(payload)
55
- @data = payload
56
-
57
- # Store original version and attempt to determine type
58
- if payload.is_a?(String)
59
- self.message_text = payload
60
- begin
61
- parsed_json = MultiJson.load(payload)
62
- self.message_json = parsed_json
63
- process_json_content(parsed_json)
64
- rescue MultiJson::ParseError
65
- self.message_type = "text"
29
+ class Session
30
+ #
31
+ # Represents a message exchanged during an MCP session.
32
+ # Its role is to store the content and metadata of each message,
33
+ # including the direction (client or server), message type (request, response, notification),
34
+ # and any associated JSON-RPC ID.
35
+ class Message < ApplicationRecord
36
+ belongs_to :session,
37
+ class_name: "ActionMCP::Session",
38
+ inverse_of: :messages,
39
+ counter_cache: true
40
+
41
+ delegate :adapter,
42
+ :role,
43
+ :session_key,
44
+ to: :session
45
+
46
+ # Virtual attribute for data
47
+ attr_reader :data
48
+
49
+ after_create_commit :broadcast_message, if: :outgoing_message?
50
+ # Set is_ping on responses if the original request was a ping
51
+ after_create :handle_ping_response, if: -> { %w[response error].include?(message_type) }
52
+
53
+ # Scope to exclude both "ping" requests and their responses
54
+ scope :without_pings, -> { where(is_ping: false) }
55
+
56
+ # @param payload [String, Hash]
57
+ def data=(payload)
58
+ @data = payload
59
+
60
+ # Store original version and attempt to determine type
61
+ if payload.is_a?(String)
62
+ self.message_text = payload
63
+ begin
64
+ parsed_json = MultiJson.load(payload)
65
+ self.message_json = parsed_json
66
+ process_json_content(parsed_json)
67
+ rescue MultiJson::ParseError
68
+ self.message_type = "text"
69
+ end
70
+ else
71
+ self.message_json = payload
72
+ self.message_text = MultiJson.dump(payload)
73
+ process_json_content(payload)
66
74
  end
67
- else
68
- self.message_json = payload
69
- self.message_text = MultiJson.dump(payload)
70
- process_json_content(payload)
71
75
  end
72
- end
73
76
 
74
- def data
75
- message_json.presence || message_text
76
- end
77
+ def data
78
+ message_json.presence || message_text
79
+ end
77
80
 
78
- # Helper methods
79
- def request?
80
- message_type == "request"
81
- end
81
+ # Helper methods
82
+ def request?
83
+ message_type == "request"
84
+ end
82
85
 
83
- def notification?
84
- message_type == "notification"
85
- end
86
+ def notification?
87
+ message_type == "notification"
88
+ end
86
89
 
87
- def response?
88
- message_type == "response"
89
- end
90
+ def response?
91
+ message_type == "response"
92
+ end
90
93
 
91
- private
94
+ private
92
95
 
93
- def outgoing_message?
94
- direction != role
95
- end
96
+ def outgoing_message?
97
+ direction != role
98
+ end
96
99
 
97
- def broadcast_message
98
- adapter.broadcast(session_key, data.to_json)
99
- end
100
+ def broadcast_message
101
+ adapter.broadcast(session_key, data.to_json)
102
+ end
100
103
 
101
- def process_json_content(content)
102
- if content.is_a?(Hash) && content["jsonrpc"] == "2.0"
103
- if content.key?("id") && content.key?("method")
104
- self.message_type = "request"
105
- self.jsonrpc_id = content["id"].to_s
106
- # Set is_ping to true if the method is "ping"
107
- self.is_ping = true if content["method"] == "ping"
108
- elsif content.key?("method") && !content.key?("id")
109
- self.message_type = "notification"
110
- elsif content.key?("id") && content.key?("result")
111
- self.message_type = "response"
112
- self.jsonrpc_id = content["id"].to_s
113
- elsif content.key?("id") && content.key?("error")
114
- self.message_type = "error"
115
- self.jsonrpc_id = content["id"].to_s
104
+ def process_json_content(content)
105
+ if content.is_a?(Hash) && content["jsonrpc"] == "2.0"
106
+ if content.key?("id") && content.key?("method")
107
+ self.message_type = "request"
108
+ self.jsonrpc_id = content["id"].to_s
109
+ # Set is_ping to true if the method is "ping"
110
+ self.is_ping = true if content["method"] == "ping"
111
+ elsif content.key?("method") && !content.key?("id")
112
+ self.message_type = "notification"
113
+ elsif content.key?("id") && content.key?("result")
114
+ self.message_type = "response"
115
+ self.jsonrpc_id = content["id"].to_s
116
+ elsif content.key?("id") && content.key?("error")
117
+ self.message_type = "error"
118
+ self.jsonrpc_id = content["id"].to_s
119
+ else
120
+ self.message_type = "invalid_jsonrpc"
121
+ end
116
122
  else
117
- self.message_type = "invalid_jsonrpc"
123
+ self.message_type = "non_jsonrpc_json"
118
124
  end
119
- else
120
- self.message_type = "non_jsonrpc_json"
121
125
  end
122
- end
123
126
 
124
- def handle_ping_response
125
- return unless jsonrpc_id.present?
126
- request_message = session.messages.find_by(
127
- jsonrpc_id: jsonrpc_id,
128
- message_type: "request"
129
- )
130
- if request_message&.is_ping
127
+ def handle_ping_response
128
+ return unless jsonrpc_id.present?
129
+
130
+ request_message = session.messages.find_by(
131
+ jsonrpc_id: jsonrpc_id,
132
+ message_type: "request"
133
+ )
134
+ return unless request_message&.is_ping
135
+
131
136
  self.is_ping = true
132
137
  request_message.update(request_acknowledged: true)
133
138
  save! if changed?
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # == Schema Information
2
4
  #
3
5
  # Table name: action_mcp_session_resources
@@ -23,11 +25,13 @@
23
25
  # fk_rails_... (session_id => action_mcp_sessions.id) ON DELETE => cascade
24
26
  #
25
27
  module ActionMCP
26
- ##
27
- # Represents a resource associated with an MCP session.
28
- # Its role is to store information about a resource, such as its URI, MIME type, description,
29
- # and any associated metadata. It also tracks whether the resource was created by a tool and the last time it was accessed.
30
- class Session::Resource < ApplicationRecord
31
- belongs_to :session
28
+ class Session
29
+ #
30
+ # Represents a resource associated with an MCP session.
31
+ # Its role is to store information about a resource, such as its URI, MIME type, description,
32
+ # and any associated metadata. It also tracks whether the resource was created by a tool and the last time it was accessed.
33
+ class Resource < ApplicationRecord
34
+ belongs_to :session
35
+ end
32
36
  end
33
37
  end
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # == Schema Information
2
4
  #
3
5
  # Table name: action_mcp_session_subscriptions
@@ -18,13 +20,15 @@
18
20
  # fk_rails_... (session_id => action_mcp_sessions.id) ON DELETE => cascade
19
21
  #
20
22
  module ActionMCP
21
- #
22
- # Represents a client's subscription to a resource for real-time updates.
23
- # Its role is to store the URI of the resource being subscribed to and track the last time a notification was sent for the subscription.
24
- # All Subscriptions are deleted when the session is closed.
25
- class Session::Subscription < ApplicationRecord
23
+ class Session
24
+ #
25
+ # Represents a client's subscription to a resource for real-time updates.
26
+ # Its role is to store the URI of the resource being subscribed to and track the last time a notification was sent for the subscription.
27
+ # All Subscriptions are deleted when the session is closed.
28
+ class Subscription < ApplicationRecord
26
29
  belongs_to :session,
27
30
  class_name: "ActionMCP::Session",
28
31
  inverse_of: :subscriptions
32
+ end
29
33
  end
30
34
  end
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # == Schema Information
2
4
  #
3
5
  # Table name: action_mcp_sessions
@@ -30,15 +32,15 @@ module ActionMCP
30
32
  dependent: :delete_all,
31
33
  inverse_of: :session
32
34
  has_many :subscriptions,
33
- class_name: "ActionMCP::Session::Subscription",
34
- foreign_key: "session_id",
35
- dependent: :delete_all,
36
- inverse_of: :session
35
+ class_name: "ActionMCP::Session::Subscription",
36
+ foreign_key: "session_id",
37
+ dependent: :delete_all,
38
+ inverse_of: :session
37
39
  has_many :resources,
38
- class_name: "ActionMCP::Session::Resource",
39
- foreign_key: "session_id",
40
- dependent: :delete_all,
41
- inverse_of: :session
40
+ class_name: "ActionMCP::Session::Resource",
41
+ foreign_key: "session_id",
42
+ dependent: :delete_all,
43
+ inverse_of: :session
42
44
 
43
45
  scope :pre_initialize, -> { where(status: "pre_initialize") }
44
46
  scope :closed, -> { where(status: "closed") }
@@ -60,9 +62,7 @@ module ActionMCP
60
62
  if data.is_a?(JsonRpc::Request) || data.is_a?(JsonRpc::Response) || data.is_a?(JsonRpc::Notification)
61
63
  data = data.to_json
62
64
  end
63
- if data.is_a?(Hash)
64
- data = MultiJson.dump(data)
65
- end
65
+ data = MultiJson.dump(data) if data.is_a?(Hash)
66
66
 
67
67
  messages.create!(data: data, direction: writer_role)
68
68
  end
@@ -101,9 +101,10 @@ module ActionMCP
101
101
 
102
102
  def initialize!
103
103
  # update the session initialized to true if client_capabilities are present
104
+ return unless client_capabilities.present?
105
+
104
106
  update!(initialized: true,
105
- status: "initialized"
106
- ) if client_capabilities.present?
107
+ status: "initialized")
107
108
  end
108
109
 
109
110
  def message_flow
@@ -122,6 +123,14 @@ module ActionMCP
122
123
  end
123
124
  end
124
125
 
126
+ def resource_subscribe(uri)
127
+ subscriptions.find_or_create_by(uri: uri)
128
+ end
129
+
130
+ def resource_unsubscribe(uri)
131
+ subscriptions.find_by(uri: uri)&.destroy
132
+ end
133
+
125
134
  private
126
135
 
127
136
  # if this session is from a server, the writer is the client
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActionMCP
2
4
  def self.table_name_prefix
3
5
  "action_mcp_"
data/config/routes.rb CHANGED
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  ActionMCP::Engine.routes.draw do
2
4
  get "/", to: "sse#events", as: :sse_out
3
5
  post "/", to: "messages#create", as: :sse_in
@@ -1,14 +1,16 @@
1
+ # frozen_string_literal: true
2
+
1
3
  class CreateActionMCPSessions < ActiveRecord::Migration[8.0]
2
4
  def change
3
5
  create_table :action_mcp_sessions, id: :string do |t|
4
- t.string :role, null: false, default: "server", comment: "The role of the session"
5
- t.string :status, null: false, default: "pre_initialize"
6
- t.datetime :ended_at, comment: "The time the session ended"
6
+ t.string :role, null: false, default: 'server', comment: 'The role of the session'
7
+ t.string :status, null: false, default: 'pre_initialize'
8
+ t.datetime :ended_at, comment: 'The time the session ended'
7
9
  t.string :protocol_version
8
- t.jsonb :server_capabilities, comment: "The capabilities of the server"
9
- t.jsonb :client_capabilities, comment: "The capabilities of the client"
10
- t.jsonb :server_info, comment: "The information about the server"
11
- t.jsonb :client_info, comment: "The information about the client"
10
+ t.jsonb :server_capabilities, comment: 'The capabilities of the server'
11
+ t.jsonb :client_capabilities, comment: 'The capabilities of the client'
12
+ t.jsonb :server_info, comment: 'The information about the server'
13
+ t.jsonb :client_info, comment: 'The information about the client'
12
14
  t.boolean :initialized, null: false, default: false
13
15
  t.integer :messages_count, null: false, default: 0
14
16
  t.timestamps
@@ -18,9 +20,9 @@ class CreateActionMCPSessions < ActiveRecord::Migration[8.0]
18
20
  t.references :session, null: false, foreign_key: { to_table: :action_mcp_sessions,
19
21
  on_delete: :cascade,
20
22
  on_update: :cascade,
21
- name: "fk_action_mcp_session_messages_session_id" }, type: :string
22
- t.string :direction, null: false, comment: "The session direction", default: "client"
23
- t.string :message_type, null: false, comment: "The type of the message"
23
+ name: 'fk_action_mcp_session_messages_session_id' }, type: :string
24
+ t.string :direction, null: false, comment: 'The session direction', default: 'client'
25
+ t.string :message_type, null: false, comment: 'The type of the message'
24
26
  t.string :jsonrpc_id
25
27
  t.string :message_text
26
28
  t.jsonb :message_json
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  class AddIsPingToSessionMessage < ActiveRecord::Migration[8.0]
2
4
  def change
3
5
  add_column :action_mcp_session_messages, :is_ping, :boolean, default: false, null: false
@@ -1,10 +1,12 @@
1
+ # frozen_string_literal: true
2
+
1
3
  class CreateActionMCPSessionSubscriptions < ActiveRecord::Migration[8.0]
2
4
  def change
3
5
  create_table :action_mcp_session_subscriptions do |t|
4
6
  t.references :session,
5
7
  null: false,
6
8
  foreign_key: { to_table: :action_mcp_sessions, on_delete: :cascade },
7
- type: :string
9
+ type: :string
8
10
  t.string :uri, null: false
9
11
  t.datetime :last_notification_at
10
12
 
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  class CreateActionMCPSessionResources < ActiveRecord::Migration[8.0]
2
4
  def change
3
5
  create_table :action_mcp_session_resources do |t|
@@ -15,8 +17,8 @@ class CreateActionMCPSessionResources < ActiveRecord::Migration[8.0]
15
17
 
16
18
  t.timestamps
17
19
  end
18
- change_column_comment :action_mcp_session_messages, :direction, "The message recipient"
19
- change_column_comment :action_mcp_session_messages, :is_ping, "Whether the message is a ping"
20
+ change_column_comment :action_mcp_session_messages, :direction, 'The message recipient'
21
+ change_column_comment :action_mcp_session_messages, :is_ping, 'Whether the message is a ping'
20
22
  rename_column :action_mcp_session_messages, :ping_acknowledged, :request_acknowledged
21
23
  add_column :action_mcp_session_messages, :request_cancelled, :boolean, null: false, default: false
22
24
  end
data/exe/actionmcp_cli CHANGED
@@ -11,27 +11,31 @@ require 'logger'
11
11
 
12
12
  # Default options
13
13
  options = {
14
- logging_level: "INFO",
14
+ logging_level: 'INFO',
15
15
  auto_initialize: true
16
16
  }
17
17
 
18
18
  # Set up logger
19
- logger = Logger.new(STDOUT)
19
+ logger = Logger.new($stdout)
20
20
  logger.formatter = proc do |severity, _, _, msg|
21
21
  "#{severity}: #{msg}\n"
22
22
  end
23
23
 
24
24
  # Parse command-line arguments
25
25
  parser = OptionParser.new do |opts|
26
- opts.banner = "Usage: mcp_client ENDPOINT [options]"
27
- opts.on("-l", "--log-level LEVEL", "Set log level (DEBUG, INFO, WARN, ERROR)") do |l|
26
+ opts.banner = 'Usage: mcp_client ENDPOINT [options]'
27
+ opts.on('-l', '--log-level LEVEL', 'Set log level (DEBUG, INFO, WARN, ERROR)') do |l|
28
28
  options[:logging_level] = l.upcase
29
- logger.level = Logger.const_get(l.upcase) rescue Logger::INFO
29
+ logger.level = begin
30
+ Logger.const_get(l.upcase)
31
+ rescue StandardError
32
+ Logger::INFO
33
+ end
30
34
  end
31
- opts.on("--no-auto-init", "Don't automatically initialize the connection") do
35
+ opts.on('--no-auto-init', "Don't automatically initialize the connection") do
32
36
  options[:auto_initialize] = false
33
37
  end
34
- opts.on("-h", "--help", "Show this help message") do
38
+ opts.on('-h', '--help', 'Show this help message') do
35
39
  puts opts
36
40
  exit
37
41
  end
@@ -44,7 +48,7 @@ endpoint = ARGV.shift
44
48
  parser.parse!(ARGV)
45
49
 
46
50
  if endpoint.nil?
47
- puts "Error: You must provide an MCP endpoint."
51
+ puts 'Error: You must provide an MCP endpoint.'
48
52
  puts parser
49
53
  exit 1
50
54
  end
@@ -60,13 +64,13 @@ def parse_command(input)
60
64
  command = parts.shift
61
65
 
62
66
  case command
63
- when "call_tool"
67
+ when 'call_tool'
64
68
  tool_name = parts.shift
65
69
  return nil unless tool_name
66
70
 
67
71
  arguments = {}
68
72
  parts.each do |arg|
69
- key, value = arg.split(":", 2)
73
+ key, value = arg.split(':', 2)
70
74
  next unless value
71
75
 
72
76
  # Try to convert the value to appropriate type
@@ -75,11 +79,11 @@ def parse_command(input)
75
79
  value.to_i
76
80
  when /^\d+\.\d+$/
77
81
  value.to_f
78
- when "true"
82
+ when 'true'
79
83
  true
80
- when "false"
84
+ when 'false'
81
85
  false
82
- when "null"
86
+ when 'null'
83
87
  nil
84
88
  else
85
89
  value
@@ -90,41 +94,39 @@ def parse_command(input)
90
94
 
91
95
  ActionMCP::JsonRpc::Request.new(
92
96
  id: generate_request_id,
93
- method: "tools/get",
97
+ method: 'tools/get',
94
98
  params: {
95
- "name" => tool_name,
96
- "arguments" => arguments
99
+ 'name' => tool_name,
100
+ 'arguments' => arguments
97
101
  }
98
102
  )
99
- when "list_tools"
103
+ when 'list_tools'
100
104
  ActionMCP::JsonRpc::Request.new(
101
105
  id: generate_request_id,
102
- method: "tools/list"
106
+ method: 'tools/list'
103
107
  )
104
- when "list_prompts"
108
+ when 'list_prompts'
105
109
  ActionMCP::JsonRpc::Request.new(
106
110
  id: generate_request_id,
107
- method: "prompts/list"
111
+ method: 'prompts/list'
108
112
  )
109
- else
110
- nil
111
113
  end
112
114
  end
113
115
 
114
116
  # Help message for shortcuts
115
117
  def print_help
116
- puts "Available shortcuts:"
117
- puts " list_tools"
118
- puts " - Get a list of available tools"
119
- puts " call_tool TOOL_NAME PARAM1:VALUE1 PARAM2:VALUE2 ..."
120
- puts " - Sends a tools/get request with the specified tool and parameters"
121
- puts " list_prompts"
122
- puts " - Get a list of available prompts"
123
- puts " get_prompt PROMPT_NAME PARAM1:VALUE1 PARAM2:VALUE2 ..."
124
- puts " - Sends a prompts/get request with the specified prompt and arguments"
125
- puts " help - Show this help message"
126
- puts " exit - Quit the client"
127
- puts "Otherwise, enter a raw JSON-RPC request to send directly"
118
+ puts 'Available shortcuts:'
119
+ puts ' list_tools'
120
+ puts ' - Get a list of available tools'
121
+ puts ' call_tool TOOL_NAME PARAM1:VALUE1 PARAM2:VALUE2 ...'
122
+ puts ' - Sends a tools/get request with the specified tool and parameters'
123
+ puts ' list_prompts'
124
+ puts ' - Get a list of available prompts'
125
+ puts ' get_prompt PROMPT_NAME PARAM1:VALUE1 PARAM2:VALUE2 ...'
126
+ puts ' - Sends a prompts/get request with the specified prompt and arguments'
127
+ puts ' help - Show this help message'
128
+ puts ' exit - Quit the client'
129
+ puts 'Otherwise, enter a raw JSON-RPC request to send directly'
128
130
  end
129
131
 
130
132
  # Initialize and start the client
@@ -132,52 +134,52 @@ client = ActionMCP.create_client(endpoint, logger: logger)
132
134
 
133
135
  # Start the transport
134
136
  unless client.connect
135
- error_msg = client.connection_error || "Unknown connection error"
137
+ error_msg = client.connection_error || 'Unknown connection error'
136
138
  puts "\nERROR: Failed to connect to MCP server at #{endpoint}"
137
139
  puts "Reason: #{error_msg}"
138
140
  puts "\nPlease check that:"
139
- puts " 1. The server is running"
140
- puts " 2. The endpoint URL/address is correct"
141
- puts " 3. Any required firewall ports are open"
141
+ puts ' 1. The server is running'
142
+ puts ' 2. The endpoint URL/address is correct'
143
+ puts ' 3. Any required firewall ports are open'
142
144
 
143
- if endpoint =~ /\Ahttps?:\/\//
144
- puts " 4. The URL includes the correct protocol, host, and port"
145
- puts " For example: http://localhost:3000/action_mcp"
145
+ if endpoint =~ %r{\Ahttps?://}
146
+ puts ' 4. The URL includes the correct protocol, host, and port'
147
+ puts ' For example: http://localhost:3000/action_mcp'
146
148
  end
147
149
 
148
150
  exit 1
149
151
  end
150
152
 
151
- Signal.trap("INT") do
153
+ Signal.trap('INT') do
152
154
  puts "\nReceived Ctrl+C. Disconnecting..."
153
155
  client.disconnect
154
- puts "MCP Client stopped."
156
+ puts 'MCP Client stopped.'
155
157
  exit 0
156
158
  end
157
159
 
158
160
  # Main REPL loop
159
161
  loop do
160
- print "mcp> "
162
+ print 'mcp> '
161
163
  input = gets&.chomp
162
164
  break unless input # Handle EOF
163
165
  next if input.empty?
164
166
 
165
167
  case input.downcase
166
- when "exit"
168
+ when 'exit'
167
169
  break
168
- when "help"
170
+ when 'help'
169
171
  print_help
170
172
  next
171
173
  else
172
174
  begin
173
175
  # Check if input is a command shortcut
174
- if input.start_with?("call_tool")
176
+ if input.start_with?('call_tool')
175
177
  request = parse_command(input)
176
178
  logger.debug("Parsed shortcut to: #{request.to_h}") if request
177
- elsif input.start_with?("connect") || input.start_with?("initialize")
179
+ elsif input.start_with?('connect') || input.start_with?('initialize')
178
180
  request = parse_command(input)
179
181
  logger.debug("Initializing connection with: #{request.to_h}") if request
180
- elsif input.start_with?("list_tools") || input.start_with?("list_prompts")
182
+ elsif input.start_with?('list_tools') || input.start_with?('list_prompts')
181
183
  request = parse_command(input)
182
184
  logger.debug("Requesting tool list: #{request.to_h}") if request
183
185
  else
@@ -185,11 +187,11 @@ loop do
185
187
  begin
186
188
  json = MultiJson.load(input)
187
189
  # Validate that the parsed JSON has the required fields
188
- if json["method"]
190
+ if json['method']
189
191
  request = ActionMCP::JsonRpc::Request.new(
190
- id: json["id"] || generate_request_id,
191
- method: json["method"],
192
- params: json["params"]
192
+ id: json['id'] || generate_request_id,
193
+ method: json['method'],
194
+ params: json['params']
193
195
  )
194
196
  else
195
197
  puts "Invalid JSON-RPC request: missing 'method' field"
@@ -216,6 +218,6 @@ loop do
216
218
  end
217
219
  end
218
220
 
219
- puts "Disconnecting..."
221
+ puts 'Disconnecting...'
220
222
  client.disconnect
221
- puts "MCP Client stopped."
223
+ puts 'MCP Client stopped.'