actionmcp 0.13.0 → 0.16.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.
- checksums.yaml +4 -4
- data/README.md +153 -158
- data/Rakefile +1 -1
- data/app/controllers/action_mcp/{application_controller.rb → mcp_controller.rb} +3 -1
- data/app/controllers/action_mcp/messages_controller.rb +7 -5
- data/app/controllers/action_mcp/sse_controller.rb +19 -13
- data/app/models/action_mcp/session/message.rb +95 -90
- data/app/models/action_mcp/session/resource.rb +10 -6
- data/app/models/action_mcp/session/subscription.rb +9 -5
- data/app/models/action_mcp/session.rb +22 -13
- data/app/models/action_mcp.rb +2 -0
- data/config/routes.rb +2 -0
- data/db/migrate/20250308122801_create_action_mcp_sessions.rb +12 -10
- data/db/migrate/20250314230152_add_is_ping_to_session_message.rb +2 -0
- data/db/migrate/20250316005021_create_action_mcp_session_subscriptions.rb +3 -1
- data/db/migrate/20250316005649_create_action_mcp_session_resources.rb +4 -2
- data/exe/actionmcp_cli +57 -55
- data/lib/action_mcp/base_json_rpc_handler.rb +97 -0
- data/lib/action_mcp/callbacks.rb +122 -0
- data/lib/action_mcp/capability.rb +6 -3
- data/lib/action_mcp/client.rb +20 -26
- data/lib/action_mcp/client_json_rpc_handler.rb +69 -0
- data/lib/action_mcp/configuration.rb +8 -8
- data/lib/action_mcp/gem_version.rb +2 -0
- data/lib/action_mcp/instrumentation/controller_runtime.rb +38 -0
- data/lib/action_mcp/instrumentation/instrumentation.rb +26 -0
- data/lib/action_mcp/instrumentation/log_subscriber.rb +39 -0
- data/lib/action_mcp/instrumentation/resource_instrumentation.rb +40 -0
- data/lib/action_mcp/json_rpc/response.rb +18 -2
- data/lib/action_mcp/json_rpc_handler.rb +93 -21
- data/lib/action_mcp/log_subscriber.rb +28 -0
- data/lib/action_mcp/logging.rb +1 -3
- data/lib/action_mcp/prompt.rb +15 -6
- data/lib/action_mcp/prompt_response.rb +1 -1
- data/lib/action_mcp/prompts_registry.rb +1 -0
- data/lib/action_mcp/registry_base.rb +1 -0
- data/lib/action_mcp/resource_callbacks.rb +156 -0
- data/lib/action_mcp/resource_template.rb +18 -19
- data/lib/action_mcp/resource_templates_registry.rb +19 -25
- data/lib/action_mcp/sampling_request.rb +113 -0
- data/lib/action_mcp/server.rb +4 -1
- data/lib/action_mcp/server_json_rpc_handler.rb +90 -0
- data/lib/action_mcp/test_helper.rb +26 -9
- data/lib/action_mcp/tool.rb +12 -3
- data/lib/action_mcp/tool_response.rb +3 -2
- data/lib/action_mcp/tools_registry.rb +1 -1
- data/lib/action_mcp/transport/capabilities.rb +5 -1
- data/lib/action_mcp/transport/messaging.rb +2 -0
- data/lib/action_mcp/transport/prompts.rb +2 -0
- data/lib/action_mcp/transport/resources.rb +23 -6
- data/lib/action_mcp/transport/roots.rb +11 -0
- data/lib/action_mcp/transport/sampling.rb +14 -0
- data/lib/action_mcp/transport/sse_client.rb +11 -15
- data/lib/action_mcp/transport/stdio_client.rb +12 -14
- data/lib/action_mcp/transport/tools.rb +2 -0
- data/lib/action_mcp/transport/transport_base.rb +16 -15
- data/lib/action_mcp/transport.rb +2 -0
- data/lib/action_mcp/transport_handler.rb +3 -0
- data/lib/action_mcp/version.rb +1 -1
- data/lib/action_mcp.rb +8 -2
- data/lib/generators/action_mcp/install/install_generator.rb +4 -1
- data/lib/generators/action_mcp/install/templates/application_mcp_res_template.rb +2 -0
- data/lib/generators/action_mcp/resource_template/resource_template_generator.rb +2 -0
- data/lib/generators/action_mcp/resource_template/templates/resource_template.rb.erb +1 -1
- data/lib/tasks/action_mcp_tasks.rake +11 -6
- metadata +27 -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
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
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
|
-
|
75
|
-
|
76
|
-
|
77
|
+
def data
|
78
|
+
message_json.presence || message_text
|
79
|
+
end
|
77
80
|
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
81
|
+
# Helper methods
|
82
|
+
def request?
|
83
|
+
message_type == "request"
|
84
|
+
end
|
82
85
|
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
+
def notification?
|
87
|
+
message_type == "notification"
|
88
|
+
end
|
86
89
|
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
+
def response?
|
91
|
+
message_type == "response"
|
92
|
+
end
|
90
93
|
|
91
|
-
|
94
|
+
private
|
92
95
|
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
+
def outgoing_message?
|
97
|
+
direction != role
|
98
|
+
end
|
96
99
|
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
+
def broadcast_message
|
101
|
+
adapter.broadcast(session_key, data.to_json)
|
102
|
+
end
|
100
103
|
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
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 = "
|
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
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
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
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
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
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
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
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
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
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
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
|
data/app/models/action_mcp.rb
CHANGED
data/config/routes.rb
CHANGED
@@ -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:
|
5
|
-
t.string :status, null: false, default:
|
6
|
-
t.datetime :ended_at, comment:
|
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:
|
9
|
-
t.jsonb :client_capabilities, comment:
|
10
|
-
t.jsonb :server_info, comment:
|
11
|
-
t.jsonb :client_info, comment:
|
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:
|
22
|
-
t.string :direction, null: false, comment:
|
23
|
-
t.string :message_type, null: false, comment:
|
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,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
|
-
|
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,
|
19
|
-
change_column_comment :action_mcp_session_messages, :is_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:
|
14
|
+
logging_level: 'INFO',
|
15
15
|
auto_initialize: true
|
16
16
|
}
|
17
17
|
|
18
18
|
# Set up logger
|
19
|
-
logger = Logger.new(
|
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 =
|
27
|
-
opts.on(
|
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 =
|
29
|
+
logger.level = begin
|
30
|
+
Logger.const_get(l.upcase)
|
31
|
+
rescue StandardError
|
32
|
+
Logger::INFO
|
33
|
+
end
|
30
34
|
end
|
31
|
-
opts.on(
|
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(
|
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
|
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
|
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(
|
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
|
82
|
+
when 'true'
|
79
83
|
true
|
80
|
-
when
|
84
|
+
when 'false'
|
81
85
|
false
|
82
|
-
when
|
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:
|
97
|
+
method: 'tools/get',
|
94
98
|
params: {
|
95
|
-
|
96
|
-
|
99
|
+
'name' => tool_name,
|
100
|
+
'arguments' => arguments
|
97
101
|
}
|
98
102
|
)
|
99
|
-
when
|
103
|
+
when 'list_tools'
|
100
104
|
ActionMCP::JsonRpc::Request.new(
|
101
105
|
id: generate_request_id,
|
102
|
-
method:
|
106
|
+
method: 'tools/list'
|
103
107
|
)
|
104
|
-
when
|
108
|
+
when 'list_prompts'
|
105
109
|
ActionMCP::JsonRpc::Request.new(
|
106
110
|
id: generate_request_id,
|
107
|
-
method:
|
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
|
117
|
-
puts
|
118
|
-
puts
|
119
|
-
puts
|
120
|
-
puts
|
121
|
-
puts
|
122
|
-
puts
|
123
|
-
puts
|
124
|
-
puts
|
125
|
-
puts
|
126
|
-
puts
|
127
|
-
puts
|
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 ||
|
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
|
140
|
-
puts
|
141
|
-
puts
|
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 =~
|
144
|
-
puts
|
145
|
-
puts
|
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(
|
153
|
+
Signal.trap('INT') do
|
152
154
|
puts "\nReceived Ctrl+C. Disconnecting..."
|
153
155
|
client.disconnect
|
154
|
-
puts
|
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
|
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
|
168
|
+
when 'exit'
|
167
169
|
break
|
168
|
-
when
|
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?(
|
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?(
|
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?(
|
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[
|
190
|
+
if json['method']
|
189
191
|
request = ActionMCP::JsonRpc::Request.new(
|
190
|
-
id: json[
|
191
|
-
method: json[
|
192
|
-
params: json[
|
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
|
221
|
+
puts 'Disconnecting...'
|
220
222
|
client.disconnect
|
221
|
-
puts
|
223
|
+
puts 'MCP Client stopped.'
|