actionmcp 0.22.0 → 0.25.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 +113 -2
- data/app/controllers/action_mcp/messages_controller.rb +2 -14
- data/app/controllers/action_mcp/sse_controller.rb +113 -45
- data/app/models/action_mcp/session/message.rb +21 -16
- data/app/models/action_mcp/session.rb +3 -2
- data/config/routes.rb +1 -1
- data/db/migrate/20250324203409_remove_session_message_text.rb +7 -0
- data/lib/action_mcp/client/base.rb +12 -14
- data/lib/action_mcp/client/blueprint.rb +5 -71
- data/lib/action_mcp/client/catalog.rb +10 -74
- data/lib/action_mcp/client/collection.rb +93 -0
- data/lib/action_mcp/client/json_rpc_handler.rb +12 -7
- data/lib/action_mcp/client/logging.rb +1 -2
- data/lib/action_mcp/client/prompt_book.rb +5 -71
- data/lib/action_mcp/client/prompts.rb +9 -4
- data/lib/action_mcp/client/request_timeouts.rb +74 -0
- data/lib/action_mcp/client/resources.rb +23 -11
- data/lib/action_mcp/client/server.rb +3 -3
- data/lib/action_mcp/client/toolbox.rb +12 -54
- data/lib/action_mcp/client/tools.rb +9 -4
- data/lib/action_mcp/configuration.rb +134 -24
- data/lib/action_mcp/engine.rb +6 -0
- data/lib/action_mcp/json_rpc_handler_base.rb +1 -0
- data/lib/action_mcp/registry_base.rb +3 -1
- data/lib/action_mcp/server/capabilities.rb +1 -1
- data/lib/action_mcp/server/json_rpc_handler.rb +1 -1
- data/lib/action_mcp/server/messaging.rb +32 -9
- data/lib/action_mcp/version.rb +1 -1
- data/lib/generators/action_mcp/install/install_generator.rb +4 -0
- data/lib/generators/action_mcp/install/templates/mcp.yml +11 -0
- data/lib/tasks/action_mcp_tasks.rake +77 -6
- metadata +6 -2
@@ -24,35 +24,16 @@ module ActionMCP
|
|
24
24
|
# # Generate a concrete URI from a blueprint with parameters
|
25
25
|
# uri = Blueprint.construct("file://{path}", { path: "/logs/app.log" })
|
26
26
|
#
|
27
|
-
class Blueprint
|
27
|
+
class Blueprint < Collection
|
28
28
|
# Initialize a new Blueprints collection with URI template definitions
|
29
29
|
#
|
30
30
|
# @param templates [Array<Hash>] Array of URI template definition hashes, each containing
|
31
31
|
# uriTemplate, name, description, and optionally mimeType keys
|
32
32
|
# @param client [Object, nil] Optional client for lazy loading of templates
|
33
|
-
attr_reader :client
|
34
|
-
|
35
33
|
def initialize(templates, client)
|
36
|
-
|
37
|
-
|
38
|
-
@
|
39
|
-
end
|
40
|
-
|
41
|
-
# Return all URI templates in the collection. If initialized with a client and templates
|
42
|
-
# haven't been loaded yet, this will trigger lazy loading from the client.
|
43
|
-
#
|
44
|
-
# @return [Array<Blueprint>] All blueprint objects in the collection
|
45
|
-
def all
|
46
|
-
load_templates unless @loaded
|
47
|
-
@templates
|
48
|
-
end
|
49
|
-
|
50
|
-
# Force reload all templates from the client and return them
|
51
|
-
#
|
52
|
-
# @return [Array<Blueprint>] All blueprint objects in the collection
|
53
|
-
def all!
|
54
|
-
load_templates(force: true)
|
55
|
-
@templates
|
34
|
+
super(templates, client)
|
35
|
+
self.templates = @collection_data
|
36
|
+
@load_method = :list_resource_templates
|
56
37
|
end
|
57
38
|
|
58
39
|
# Find a blueprint by its URI pattern
|
@@ -84,23 +65,6 @@ module ActionMCP
|
|
84
65
|
blueprint.construct(params)
|
85
66
|
end
|
86
67
|
|
87
|
-
# Filter blueprints based on a given block
|
88
|
-
#
|
89
|
-
# @yield [blueprint] Block that determines whether to include a blueprint
|
90
|
-
# @yieldparam blueprint [Blueprint] A blueprint from the collection
|
91
|
-
# @yieldreturn [Boolean] true to include the blueprint, false to exclude it
|
92
|
-
# @return [Array<Blueprint>] Blueprints that match the filter criteria
|
93
|
-
def filter(&block)
|
94
|
-
all.select(&block)
|
95
|
-
end
|
96
|
-
|
97
|
-
# Number of blueprints in the collection
|
98
|
-
#
|
99
|
-
# @return [Integer] The number of blueprints
|
100
|
-
def size
|
101
|
-
all.size
|
102
|
-
end
|
103
|
-
|
104
68
|
# Check if the collection contains a blueprint with the given pattern
|
105
69
|
#
|
106
70
|
# @param pattern [String] The blueprint pattern to check for
|
@@ -116,41 +80,11 @@ module ActionMCP
|
|
116
80
|
all.group_by(&:protocol)
|
117
81
|
end
|
118
82
|
|
119
|
-
# Implements enumerable functionality for the collection
|
120
|
-
include Enumerable
|
121
|
-
|
122
|
-
# Yield each blueprint in the collection to the given block
|
123
|
-
#
|
124
|
-
# @yield [blueprint] Block to execute for each blueprint
|
125
|
-
# @yieldparam blueprint [Blueprint] A blueprint from the collection
|
126
|
-
# @return [Enumerator] If no block is given
|
127
|
-
def each(&block)
|
128
|
-
all.each(&block)
|
129
|
-
end
|
130
|
-
|
131
83
|
# Convert raw template data into ResourceTemplate objects
|
132
84
|
#
|
133
85
|
# @param templates [Array<Hash>] Array of template definition hashes
|
134
86
|
def templates=(templates)
|
135
|
-
@
|
136
|
-
end
|
137
|
-
|
138
|
-
private
|
139
|
-
|
140
|
-
# Load or reload templates using the client
|
141
|
-
#
|
142
|
-
# @param force [Boolean] Whether to force reload even if templates are already loaded
|
143
|
-
# @return [void]
|
144
|
-
def load_templates(force: false)
|
145
|
-
return if @loaded && !force
|
146
|
-
|
147
|
-
begin
|
148
|
-
@client.list_resource_templates
|
149
|
-
@loaded = true
|
150
|
-
rescue StandardError => e
|
151
|
-
Rails.logger.error("Failed to load templates: #{e.message}")
|
152
|
-
@loaded = true unless @templates.empty?
|
153
|
-
end
|
87
|
+
@collection_data = templates.map { |template_data| ResourceTemplate.new(template_data) }
|
154
88
|
end
|
155
89
|
|
156
90
|
# Internal Blueprint class to represent individual URI templates
|
@@ -24,35 +24,16 @@ module ActionMCP
|
|
24
24
|
# # Get all resources matching a criteria
|
25
25
|
# rust_files = catalog.filter { |r| r.mime_type == "text/x-rust" }
|
26
26
|
#
|
27
|
-
class Catalog
|
27
|
+
class Catalog < Collection
|
28
28
|
# Initialize a new Catalog with resource definitions
|
29
29
|
#
|
30
30
|
# @param resources [Array<Hash>] Array of resource definition hashes, each containing
|
31
31
|
# uri, name, description, and mimeType keys
|
32
32
|
# @param client [Object, nil] Optional client for lazy loading of resources
|
33
|
-
attr_reader :client
|
34
|
-
|
35
33
|
def initialize(resources, client)
|
36
|
-
|
37
|
-
|
38
|
-
@
|
39
|
-
end
|
40
|
-
|
41
|
-
# Return all resources in the collection. If initialized with a client and resources
|
42
|
-
# haven't been loaded yet, this will trigger lazy loading from the client.
|
43
|
-
#
|
44
|
-
# @return [Array<Resource>] All resource objects in the collection
|
45
|
-
def all
|
46
|
-
load_resources unless @loaded
|
47
|
-
@resources
|
48
|
-
end
|
49
|
-
|
50
|
-
# Force reload all resources from the client and return them
|
51
|
-
#
|
52
|
-
# @return [Array<Resource>] All resource objects in the collection
|
53
|
-
def all!
|
54
|
-
load_resources(force: true)
|
55
|
-
@resources
|
34
|
+
super(resources, client)
|
35
|
+
self.resources = @collection_data
|
36
|
+
@load_method = :list_resources
|
56
37
|
end
|
57
38
|
|
58
39
|
# Find a resource by URI
|
@@ -79,16 +60,6 @@ module ActionMCP
|
|
79
60
|
all.select { |resource| resource.mime_type == mime_type }
|
80
61
|
end
|
81
62
|
|
82
|
-
# Filter resources based on a given block
|
83
|
-
#
|
84
|
-
# @yield [resource] Block that determines whether to include a resource
|
85
|
-
# @yieldparam resource [Resource] A resource from the collection
|
86
|
-
# @yieldreturn [Boolean] true to include the resource, false to exclude it
|
87
|
-
# @return [Array<Resource>] Resources that match the filter criteria
|
88
|
-
def filter(&block)
|
89
|
-
all.select(&block)
|
90
|
-
end
|
91
|
-
|
92
63
|
# Get a list of all resource URIs
|
93
64
|
#
|
94
65
|
# @return [Array<String>] URIs of all resources in the collection
|
@@ -96,13 +67,6 @@ module ActionMCP
|
|
96
67
|
all.map(&:uri)
|
97
68
|
end
|
98
69
|
|
99
|
-
# Number of resources in the collection
|
100
|
-
#
|
101
|
-
# @return [Integer] The number of resources
|
102
|
-
def size
|
103
|
-
all.size
|
104
|
-
end
|
105
|
-
|
106
70
|
# Check if the collection contains a resource with the given URI
|
107
71
|
#
|
108
72
|
# @param uri [String] The resource URI to check for
|
@@ -126,45 +90,15 @@ module ActionMCP
|
|
126
90
|
keyword = keyword.downcase
|
127
91
|
all.select do |resource|
|
128
92
|
resource.name.downcase.include?(keyword) ||
|
129
|
-
|
93
|
+
resource.description&.downcase&.include?(keyword)
|
130
94
|
end
|
131
95
|
end
|
132
96
|
|
133
|
-
# Implements enumerable functionality for the collection
|
134
|
-
include Enumerable
|
135
|
-
|
136
|
-
# Yield each resource in the collection to the given block
|
137
|
-
#
|
138
|
-
# @yield [resource] Block to execute for each resource
|
139
|
-
# @yieldparam resource [Resource] A resource from the collection
|
140
|
-
# @return [Enumerator] If no block is given
|
141
|
-
def each(&block)
|
142
|
-
all.each(&block)
|
143
|
-
end
|
144
|
-
|
145
97
|
# Convert raw resource data into Resource objects
|
146
98
|
#
|
147
99
|
# @param raw_resources [Array<Hash>] Array of resource definition hashes
|
148
100
|
def resources=(raw_resources)
|
149
|
-
@
|
150
|
-
end
|
151
|
-
|
152
|
-
private
|
153
|
-
|
154
|
-
# Load or reload resources using the client
|
155
|
-
#
|
156
|
-
# @param force [Boolean] Whether to force reload even if resources are already loaded
|
157
|
-
# @return [void]
|
158
|
-
def load_resources(force: false)
|
159
|
-
return if @loaded && !force
|
160
|
-
|
161
|
-
begin
|
162
|
-
@client.list_resources
|
163
|
-
@loaded = true
|
164
|
-
rescue StandardError => e
|
165
|
-
Rails.logger.error("Failed to load resources: #{e.message}")
|
166
|
-
@loaded = true unless @resources.empty?
|
167
|
-
end
|
101
|
+
@collection_data = raw_resources.map { |resource_data| Resource.new(resource_data) }
|
168
102
|
end
|
169
103
|
|
170
104
|
# Internal Resource class to represent individual resources
|
@@ -185,7 +119,7 @@ module ActionMCP
|
|
185
119
|
#
|
186
120
|
# @return [String, nil] The file extension or nil if no extension
|
187
121
|
def extension
|
188
|
-
File.extname(@name)[1
|
122
|
+
File.extname(@name)[1..] if @name.include?(".")
|
189
123
|
end
|
190
124
|
|
191
125
|
# Check if this resource is a text file based on MIME type
|
@@ -206,7 +140,9 @@ module ActionMCP
|
|
206
140
|
#
|
207
141
|
# @return [String, nil] The path component of the URI
|
208
142
|
def path
|
209
|
-
URI(@uri).path
|
143
|
+
URI(@uri).path
|
144
|
+
rescue StandardError
|
145
|
+
nil
|
210
146
|
end
|
211
147
|
|
212
148
|
# Generate a hash representation of the resource
|
@@ -0,0 +1,93 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
module Client
|
5
|
+
# Base collection class for MCP client collections
|
6
|
+
class Collection
|
7
|
+
include RequestTimeouts
|
8
|
+
|
9
|
+
attr_reader :client, :loaded
|
10
|
+
|
11
|
+
def initialize(items, client, silence_sql: true)
|
12
|
+
@collection_data = items || []
|
13
|
+
@client = client
|
14
|
+
@loaded = !@collection_data.empty?
|
15
|
+
@silence_sql = silence_sql
|
16
|
+
end
|
17
|
+
|
18
|
+
def all
|
19
|
+
silence_logs { load_items unless @loaded }
|
20
|
+
@collection_data
|
21
|
+
end
|
22
|
+
|
23
|
+
def all!(timeout: DEFAULT_TIMEOUT)
|
24
|
+
silence_logs { load_items(force: true, timeout: timeout) }
|
25
|
+
@collection_data
|
26
|
+
end
|
27
|
+
|
28
|
+
# Filter items based on a given block
|
29
|
+
#
|
30
|
+
# @yield [item] Block that determines whether to include an item
|
31
|
+
# @yieldparam item [Object] An item from the collection
|
32
|
+
# @yieldreturn [Boolean] true to include the item, false to exclude it
|
33
|
+
# @return [Array<Object>] Items that match the filter criteria
|
34
|
+
def filter(&block)
|
35
|
+
all.select(&block)
|
36
|
+
end
|
37
|
+
|
38
|
+
# Number of items in the collection
|
39
|
+
#
|
40
|
+
# @return [Integer] The number of items
|
41
|
+
def size
|
42
|
+
all.size
|
43
|
+
end
|
44
|
+
|
45
|
+
# Implements enumerable functionality
|
46
|
+
include Enumerable
|
47
|
+
|
48
|
+
def each(&block)
|
49
|
+
all.each(&block)
|
50
|
+
end
|
51
|
+
|
52
|
+
alias loaded? loaded
|
53
|
+
|
54
|
+
protected
|
55
|
+
|
56
|
+
def load_items(force: false, timeout: DEFAULT_TIMEOUT)
|
57
|
+
return if @loaded && !force
|
58
|
+
|
59
|
+
# Make sure @load_method is defined in the subclass
|
60
|
+
raise NotImplementedError, "Subclass must define @load_method" unless defined?(@load_method)
|
61
|
+
|
62
|
+
# Use the RequestTimeouts module to handle the request
|
63
|
+
load_with_timeout(@load_method, force: force, timeout: timeout)
|
64
|
+
end
|
65
|
+
|
66
|
+
private
|
67
|
+
|
68
|
+
def silence_logs
|
69
|
+
return yield unless @silence_sql
|
70
|
+
|
71
|
+
original_log_level = Session.logger&.level
|
72
|
+
begin
|
73
|
+
# Temporarily increase log level to suppress SQL queries
|
74
|
+
Session.logger.level = Logger::WARN if Session.logger
|
75
|
+
yield
|
76
|
+
ensure
|
77
|
+
# Restore original log level
|
78
|
+
Session.logger.level = original_log_level if Session.logger
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def log_error(message)
|
83
|
+
# Safely handle logging - don't assume Rails.logger exists
|
84
|
+
if defined?(Rails) && Rails.respond_to?(:logger) && Rails.logger
|
85
|
+
Rails.logger.error("[#{self.class.name}] #{message}")
|
86
|
+
else
|
87
|
+
# Fall back to puts if Rails.logger is not available
|
88
|
+
puts "[ERROR] [#{self.class.name}] #{message}"
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
@@ -4,6 +4,7 @@ module ActionMCP
|
|
4
4
|
module Client
|
5
5
|
class JsonRpcHandler < JsonRpcHandlerBase
|
6
6
|
attr_reader :client
|
7
|
+
|
7
8
|
def initialize(transport, client)
|
8
9
|
super(transport)
|
9
10
|
@client = client
|
@@ -70,25 +71,30 @@ module ActionMCP
|
|
70
71
|
|
71
72
|
def process_response(id, result)
|
72
73
|
if transport.id == id
|
73
|
-
## This
|
74
|
+
## This initializes the transport
|
74
75
|
client.server = Client::Server.new(result)
|
75
76
|
return send_initialized_notification
|
76
77
|
end
|
78
|
+
|
77
79
|
request = transport.messages.requests.find_by(jsonrpc_id: id)
|
78
80
|
return unless request
|
81
|
+
|
82
|
+
# Mark the request as acknowledged
|
83
|
+
request.update(request_acknowledged: true)
|
84
|
+
|
79
85
|
case request.rpc_method
|
80
86
|
when "tools/list"
|
81
|
-
|
82
|
-
|
87
|
+
client.toolbox.tools = result["tools"]
|
88
|
+
return true
|
83
89
|
when "prompts/list"
|
84
90
|
client.prompt_book.prompts = result["prompts"]
|
85
|
-
return
|
91
|
+
return true
|
86
92
|
when "resources/list"
|
87
93
|
client.catalog.resources = result["resources"]
|
88
|
-
return
|
94
|
+
return true
|
89
95
|
when "resources/templates/list"
|
90
96
|
client.blueprint.templates = result["resourceTemplates"]
|
91
|
-
return
|
97
|
+
return true
|
92
98
|
end
|
93
99
|
|
94
100
|
puts "\e[31mUnknown response: #{id} #{result}\e[0m"
|
@@ -99,7 +105,6 @@ module ActionMCP
|
|
99
105
|
puts "\e[31mUnknown error: #{id} #{error}\e[0m"
|
100
106
|
end
|
101
107
|
|
102
|
-
|
103
108
|
def send_initialized_notification
|
104
109
|
transport.initialize!
|
105
110
|
client.send_jsonrpc_notification("notifications/initialized")
|
@@ -24,27 +24,16 @@ module ActionMCP
|
|
24
24
|
# # Get all prompts matching a criteria
|
25
25
|
# text_prompts = book.filter { |p| p.name.include?("text") }
|
26
26
|
#
|
27
|
-
class PromptBook
|
27
|
+
class PromptBook < Collection
|
28
28
|
# Initialize a new PromptBook with prompt definitions
|
29
29
|
#
|
30
30
|
# @param prompts [Array<Hash>] Array of prompt definition hashes, each containing
|
31
31
|
# name, description, and arguments keys
|
32
32
|
# @param client [Object, nil] Optional client for lazy loading of prompts
|
33
|
-
attr_reader :client
|
34
|
-
|
35
33
|
def initialize(prompts, client)
|
36
|
-
|
37
|
-
|
38
|
-
@
|
39
|
-
end
|
40
|
-
|
41
|
-
# Return all prompts in the collection. If initialized with a client and prompts
|
42
|
-
# haven't been loaded yet, this will trigger lazy loading from the client.
|
43
|
-
#
|
44
|
-
# @return [Array<Prompt>] All prompt objects in the collection
|
45
|
-
def all
|
46
|
-
load_prompts unless @loaded
|
47
|
-
@prompts
|
34
|
+
super(prompts, client)
|
35
|
+
self.prompts = @collection_data
|
36
|
+
@load_method = :list_prompts
|
48
37
|
end
|
49
38
|
|
50
39
|
# Find a prompt by name
|
@@ -55,16 +44,6 @@ module ActionMCP
|
|
55
44
|
all.find { |prompt| prompt.name == name }
|
56
45
|
end
|
57
46
|
|
58
|
-
# Filter prompts based on a given block
|
59
|
-
#
|
60
|
-
# @yield [prompt] Block that determines whether to include a prompt
|
61
|
-
# @yieldparam prompt [Prompt] A prompt from the collection
|
62
|
-
# @yieldreturn [Boolean] true to include the prompt, false to exclude it
|
63
|
-
# @return [Array<Prompt>] Prompts that match the filter criteria
|
64
|
-
def filter(&block)
|
65
|
-
all.select(&block)
|
66
|
-
end
|
67
|
-
|
68
47
|
# Get a list of all prompt names
|
69
48
|
#
|
70
49
|
# @return [Array<String>] Names of all prompts in the collection
|
@@ -72,13 +51,6 @@ module ActionMCP
|
|
72
51
|
all.map(&:name)
|
73
52
|
end
|
74
53
|
|
75
|
-
# Number of prompts in the collection
|
76
|
-
#
|
77
|
-
# @return [Integer] The number of prompts
|
78
|
-
def size
|
79
|
-
all.size
|
80
|
-
end
|
81
|
-
|
82
54
|
# Check if the collection contains a prompt with the given name
|
83
55
|
#
|
84
56
|
# @param name [String] The prompt name to check for
|
@@ -87,49 +59,11 @@ module ActionMCP
|
|
87
59
|
all.any? { |prompt| prompt.name == name }
|
88
60
|
end
|
89
61
|
|
90
|
-
# Implements enumerable functionality for the collection
|
91
|
-
include Enumerable
|
92
|
-
|
93
|
-
# Yield each prompt in the collection to the given block
|
94
|
-
#
|
95
|
-
# @yield [prompt] Block to execute for each prompt
|
96
|
-
# @yieldparam prompt [Prompt] A prompt from the collection
|
97
|
-
# @return [Enumerator] If no block is given
|
98
|
-
def each(&block)
|
99
|
-
all.each(&block)
|
100
|
-
end
|
101
|
-
|
102
|
-
# Force reload all prompts from the client and return them
|
103
|
-
#
|
104
|
-
# @return [Array<Prompt>] All prompt objects in the collection
|
105
|
-
def all!
|
106
|
-
load_prompts(force: true)
|
107
|
-
all
|
108
|
-
end
|
109
|
-
|
110
62
|
# Convert raw prompt data into Prompt objects
|
111
63
|
#
|
112
64
|
# @param prompts [Array<Hash>] Array of prompt definition hashes
|
113
65
|
def prompts=(prompts)
|
114
|
-
@
|
115
|
-
end
|
116
|
-
|
117
|
-
private
|
118
|
-
|
119
|
-
# Load or reload prompts using the client
|
120
|
-
#
|
121
|
-
# @param force [Boolean] Whether to force reload even if prompts are already loaded
|
122
|
-
# @return [void]
|
123
|
-
def load_prompts(force: false)
|
124
|
-
return if @loaded && !force
|
125
|
-
|
126
|
-
begin
|
127
|
-
@client.list_prompts
|
128
|
-
@loaded = true
|
129
|
-
rescue StandardError => e
|
130
|
-
Rails.logger.error("Failed to load prompts: #{e.message}")
|
131
|
-
@loaded = true unless @prompts.empty?
|
132
|
-
end
|
66
|
+
@collection_data = prompts.map { |data| Prompt.new(data) }
|
133
67
|
end
|
134
68
|
|
135
69
|
# Internal Prompt class to represent individual prompts
|
@@ -4,18 +4,21 @@ module ActionMCP
|
|
4
4
|
module Client
|
5
5
|
module Prompts
|
6
6
|
# List all available prompts from the server
|
7
|
-
# @return [
|
7
|
+
# @return [String] Request ID for tracking the request
|
8
8
|
def list_prompts
|
9
9
|
request_id = SecureRandom.uuid_v7
|
10
10
|
|
11
11
|
# Send request
|
12
12
|
send_jsonrpc_request("prompts/list", id: request_id)
|
13
|
+
|
14
|
+
# Return request ID for tracking the request
|
15
|
+
request_id
|
13
16
|
end
|
14
17
|
|
15
18
|
# Get a specific prompt with arguments
|
16
19
|
# @param name [String] Name of the prompt to get
|
17
20
|
# @param arguments [Hash] Arguments to pass to the prompt
|
18
|
-
# @return [
|
21
|
+
# @return [String] Request ID for tracking the request
|
19
22
|
def get_prompt(name, arguments = {})
|
20
23
|
request_id = SecureRandom.uuid_v7
|
21
24
|
|
@@ -25,8 +28,10 @@ module ActionMCP
|
|
25
28
|
name: name,
|
26
29
|
arguments: arguments
|
27
30
|
},
|
28
|
-
id: request_id
|
29
|
-
|
31
|
+
id: request_id)
|
32
|
+
|
33
|
+
# Return request ID for tracking the request
|
34
|
+
request_id
|
30
35
|
end
|
31
36
|
end
|
32
37
|
end
|
@@ -0,0 +1,74 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActionMCP
|
4
|
+
module Client
|
5
|
+
module RequestTimeouts
|
6
|
+
# Default timeout in seconds
|
7
|
+
DEFAULT_TIMEOUT = 1.0
|
8
|
+
|
9
|
+
# Load resources with timeout support - blocking until response or timeout
|
10
|
+
# @param method_name [Symbol] The method to call for loading (e.g., :list_resources)
|
11
|
+
# @param force [Boolean] Whether to force reload even if already loaded
|
12
|
+
# @param timeout [Float] Timeout in seconds
|
13
|
+
# @return [Boolean] Success status
|
14
|
+
def load_with_timeout(method_name, force: false, timeout: DEFAULT_TIMEOUT)
|
15
|
+
return true if @loaded && !force
|
16
|
+
|
17
|
+
# Make the request and store its ID
|
18
|
+
request_id = client.send(method_name)
|
19
|
+
|
20
|
+
start_time = Time.now
|
21
|
+
|
22
|
+
# Wait until either:
|
23
|
+
# 1. The collection is loaded (@loaded becomes true from JsonRpcHandler)
|
24
|
+
# 2. The timeout is reached
|
25
|
+
sleep(0.1) while !@loaded && (Time.now - start_time) < timeout
|
26
|
+
|
27
|
+
# If we timed out
|
28
|
+
unless @loaded
|
29
|
+
request = client.session.messages.requests.find_by(jsonrpc_id: request_id)
|
30
|
+
|
31
|
+
if request && !request.request_acknowledged?
|
32
|
+
# Send cancel notification
|
33
|
+
client.send_jsonrpc_notification("notifications/cancelled", {
|
34
|
+
requestId: request_id,
|
35
|
+
reason: "Request timed out after #{timeout} seconds"
|
36
|
+
})
|
37
|
+
|
38
|
+
# Mark as cancelled in the database
|
39
|
+
request.update(request_cancelled: true)
|
40
|
+
|
41
|
+
log_error("Request #{method_name} timed out after #{timeout} seconds")
|
42
|
+
end
|
43
|
+
|
44
|
+
# Mark as loaded even though we timed out
|
45
|
+
@loaded = true
|
46
|
+
return false
|
47
|
+
end
|
48
|
+
|
49
|
+
# Collection was successfully loaded
|
50
|
+
true
|
51
|
+
end
|
52
|
+
|
53
|
+
private
|
54
|
+
|
55
|
+
def handle_timeout(request_id, method_name, timeout)
|
56
|
+
# Find the request
|
57
|
+
request = client.session.messages.requests.find_by(jsonrpc_id: request_id)
|
58
|
+
|
59
|
+
return unless request && !request.request_acknowledged?
|
60
|
+
|
61
|
+
# Send cancel notification
|
62
|
+
client.send_jsonrpc_notification("notifications/cancelled", {
|
63
|
+
requestId: request_id,
|
64
|
+
reason: "Request timed out after #{timeout} seconds"
|
65
|
+
})
|
66
|
+
|
67
|
+
# Mark as cancelled in the database
|
68
|
+
request.update(request_cancelled: true)
|
69
|
+
|
70
|
+
log_error("Request #{method_name} timed out after #{timeout} seconds")
|
71
|
+
end
|
72
|
+
end
|
73
|
+
end
|
74
|
+
end
|