makit 0.0.111 → 0.0.112

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 (140) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +41 -41
  3. data/exe/makit +5 -5
  4. data/lib/makit/apache.rb +28 -28
  5. data/lib/makit/cli/build_commands.rb +500 -500
  6. data/lib/makit/cli/generators/base_generator.rb +74 -74
  7. data/lib/makit/cli/generators/dotnet_generator.rb +50 -50
  8. data/lib/makit/cli/generators/generator_factory.rb +49 -49
  9. data/lib/makit/cli/generators/node_generator.rb +50 -50
  10. data/lib/makit/cli/generators/ruby_generator.rb +77 -77
  11. data/lib/makit/cli/generators/rust_generator.rb +50 -50
  12. data/lib/makit/cli/generators/templates/dotnet_templates.rb +167 -167
  13. data/lib/makit/cli/generators/templates/node_templates.rb +161 -161
  14. data/lib/makit/cli/generators/templates/ruby/gemfile.rb +26 -26
  15. data/lib/makit/cli/generators/templates/ruby/gemspec.rb +40 -40
  16. data/lib/makit/cli/generators/templates/ruby/main_lib.rb +33 -33
  17. data/lib/makit/cli/generators/templates/ruby/rakefile.rb +35 -35
  18. data/lib/makit/cli/generators/templates/ruby/readme.rb +63 -63
  19. data/lib/makit/cli/generators/templates/ruby/test.rb +39 -39
  20. data/lib/makit/cli/generators/templates/ruby/test_helper.rb +29 -29
  21. data/lib/makit/cli/generators/templates/ruby/version.rb +29 -29
  22. data/lib/makit/cli/generators/templates/rust_templates.rb +128 -128
  23. data/lib/makit/cli/main.rb +62 -62
  24. data/lib/makit/cli/project_commands.rb +868 -868
  25. data/lib/makit/cli/repository_commands.rb +661 -661
  26. data/lib/makit/cli/utility_commands.rb +521 -521
  27. data/lib/makit/commands/factory.rb +359 -359
  28. data/lib/makit/commands/middleware/base.rb +73 -73
  29. data/lib/makit/commands/middleware/cache.rb +248 -248
  30. data/lib/makit/commands/middleware/command_logger.rb +320 -323
  31. data/lib/makit/commands/middleware/unified_logger.rb +243 -243
  32. data/lib/makit/commands/middleware/validator.rb +269 -269
  33. data/lib/makit/commands/request.rb +254 -254
  34. data/lib/makit/commands/result.rb +323 -323
  35. data/lib/makit/commands/runner.rb +337 -317
  36. data/lib/makit/commands/strategies/base.rb +160 -160
  37. data/lib/makit/commands/strategies/synchronous.rb +134 -134
  38. data/lib/makit/commands.rb +51 -42
  39. data/lib/makit/configuration/gitlab_helper.rb +60 -60
  40. data/lib/makit/configuration/project.rb +127 -127
  41. data/lib/makit/configuration/rakefile_helper.rb +43 -43
  42. data/lib/makit/configuration/step.rb +34 -34
  43. data/lib/makit/configuration.rb +14 -14
  44. data/lib/makit/content/default_gitignore.rb +7 -7
  45. data/lib/makit/content/default_rakefile.rb +13 -13
  46. data/lib/makit/content/gem_rakefile.rb +16 -16
  47. data/lib/makit/context.rb +1 -1
  48. data/lib/makit/data.rb +49 -49
  49. data/lib/makit/directories.rb +141 -141
  50. data/lib/makit/directory.rb +262 -262
  51. data/lib/makit/docs/files.rb +89 -89
  52. data/lib/makit/docs/rake.rb +102 -102
  53. data/lib/makit/dotnet/project.rb +153 -153
  54. data/lib/makit/dotnet/solution.rb +38 -38
  55. data/lib/makit/dotnet/solution_classlib.rb +239 -239
  56. data/lib/makit/dotnet/solution_console.rb +264 -264
  57. data/lib/makit/dotnet/solution_maui.rb +354 -354
  58. data/lib/makit/dotnet/solution_wasm.rb +275 -275
  59. data/lib/makit/dotnet/solution_wpf.rb +304 -304
  60. data/lib/makit/dotnet.rb +102 -102
  61. data/lib/makit/email.rb +90 -90
  62. data/lib/makit/environment.rb +142 -142
  63. data/lib/makit/examples/runner.rb +370 -370
  64. data/lib/makit/exceptions.rb +45 -45
  65. data/lib/makit/fileinfo.rb +24 -24
  66. data/lib/makit/files.rb +43 -43
  67. data/lib/makit/gems.rb +40 -40
  68. data/lib/makit/git/cli.rb +54 -54
  69. data/lib/makit/git/repository.rb +90 -90
  70. data/lib/makit/git.rb +98 -98
  71. data/lib/makit/gitlab_runner.rb +59 -59
  72. data/lib/makit/humanize.rb +137 -137
  73. data/lib/makit/indexer.rb +47 -47
  74. data/lib/makit/logging/configuration.rb +305 -305
  75. data/lib/makit/logging/formatters/base.rb +39 -39
  76. data/lib/makit/logging/formatters/console_formatter.rb +140 -127
  77. data/lib/makit/logging/formatters/json_formatter.rb +65 -65
  78. data/lib/makit/logging/formatters/plain_text_formatter.rb +71 -71
  79. data/lib/makit/logging/formatters/text_formatter.rb +64 -64
  80. data/lib/makit/logging/log_request.rb +115 -115
  81. data/lib/makit/logging/logger.rb +163 -159
  82. data/lib/makit/logging/sinks/console.rb +72 -72
  83. data/lib/makit/logging/sinks/file_sink.rb +92 -92
  84. data/lib/makit/logging/sinks/unified_file_sink.rb +303 -303
  85. data/lib/makit/logging.rb +530 -521
  86. data/lib/makit/markdown.rb +75 -75
  87. data/lib/makit/mp/basic_object_mp.rb +17 -17
  88. data/lib/makit/mp/command_mp.rb +13 -13
  89. data/lib/makit/mp/command_request.mp.rb +17 -17
  90. data/lib/makit/mp/project_mp.rb +199 -199
  91. data/lib/makit/mp/string_mp.rb +193 -348
  92. data/lib/makit/nuget.rb +74 -74
  93. data/lib/makit/port.rb +32 -32
  94. data/lib/makit/process.rb +163 -163
  95. data/lib/makit/protoc.rb +107 -107
  96. data/lib/makit/rake/cli.rb +196 -196
  97. data/lib/makit/rake.rb +25 -25
  98. data/lib/makit/ruby/cli.rb +185 -185
  99. data/lib/makit/ruby.rb +25 -25
  100. data/lib/makit/secrets.rb +51 -51
  101. data/lib/makit/serializer.rb +130 -117
  102. data/lib/makit/services/builder.rb +186 -186
  103. data/lib/makit/services/error_handler.rb +226 -226
  104. data/lib/makit/services/repository_manager.rb +229 -229
  105. data/lib/makit/services/validator.rb +112 -112
  106. data/lib/makit/setup/classlib.rb +53 -53
  107. data/lib/makit/setup/gem.rb +30 -17
  108. data/lib/makit/setup/runner.rb +45 -40
  109. data/lib/makit/setup.rb +5 -0
  110. data/lib/makit/show.rb +110 -110
  111. data/lib/makit/storage.rb +126 -126
  112. data/lib/makit/symbols.rb +170 -161
  113. data/lib/makit/task_info.rb +128 -128
  114. data/lib/makit/tasks/at_exit.rb +13 -13
  115. data/lib/makit/tasks/build.rb +19 -18
  116. data/lib/makit/tasks/clean.rb +11 -11
  117. data/lib/makit/tasks/hook_manager.rb +393 -239
  118. data/lib/makit/tasks/init.rb +47 -47
  119. data/lib/makit/tasks/integrate.rb +17 -15
  120. data/lib/makit/tasks/pull_incoming.rb +11 -12
  121. data/lib/makit/tasks/setup.rb +6 -6
  122. data/lib/makit/tasks/sync.rb +12 -11
  123. data/lib/makit/tasks/tag.rb +15 -0
  124. data/lib/makit/tasks/task_monkey_patch.rb +79 -79
  125. data/lib/makit/tasks.rb +10 -0
  126. data/lib/makit/test_cache.rb +239 -239
  127. data/lib/makit/tree.rb +37 -37
  128. data/lib/makit/v1/makit.v1_pb.rb +34 -34
  129. data/lib/makit/v1/makit.v1_services_pb.rb +27 -27
  130. data/lib/makit/version.rb +5 -5
  131. data/lib/makit/version_util.rb +21 -0
  132. data/lib/makit/wix.rb +95 -95
  133. data/lib/makit/yaml.rb +29 -29
  134. data/lib/makit/zip.rb +17 -17
  135. data/lib/makit copy.rb +44 -0
  136. data/lib/makit.rb +40 -8
  137. metadata +50 -7
  138. data/lib/makit/command_runner.rb +0 -463
  139. data/lib/makit/commands/compatibility.rb +0 -365
  140. data/lib/makit/task_hooks.rb +0 -125
@@ -1,317 +1,337 @@
1
- # frozen_string_literal: true
2
-
3
- require_relative "request"
4
- require_relative "result"
5
- require_relative "strategies/synchronous"
6
- require_relative "middleware/base"
7
- require_relative "middleware/unified_logger"
8
- require_relative "middleware/command_logger"
9
-
10
- module Makit
11
- module Commands
12
- # Modern command execution engine with proper separation of concerns.
13
- #
14
- # The Runner coordinates command execution through a middleware chain and
15
- # execution strategies. It provides a clean, extensible architecture for
16
- # command processing with support for caching, logging, validation, and
17
- # custom execution patterns.
18
- #
19
- # @example Basic usage
20
- # runner = Runner.new
21
- # request = Request.from_string("git --version")
22
- # result = runner.execute(request)
23
- # puts result.stdout
24
- #
25
- # @example With custom middleware
26
- # runner = Runner.new(middleware: [Logger.new, Cache.new])
27
- # result = runner.execute(request)
28
- #
29
- # @example With custom strategy
30
- # runner = Runner.new(strategy: Strategies::Parallel.new)
31
- # results = runner.execute_batch(requests)
32
- class Runner
33
- # @!attribute [r] middleware
34
- # @return [Array<Middleware::Base>] middleware chain
35
- # @!attribute [r] strategy
36
- # @return [Strategies::Base] execution strategy
37
- attr_reader :middleware, :strategy
38
-
39
- # Get the default configured runner instance.
40
- #
41
- # @return [Runner] default runner with standard middleware
42
- def self.default
43
- # Recreate the runner if the log level has changed
44
- current_log_level = Makit::Logging.current_log_level
45
- if @default.nil? || @last_log_level != current_log_level
46
- @last_log_level = current_log_level
47
- @default = new(
48
- middleware: [
49
- Middleware::CommandLogger.new(
50
- log_stdout: true,
51
- log_stderr: true,
52
- log_performance: true,
53
- max_output_lines: 50,
54
- ),
55
- ],
56
- )
57
- end
58
- @default
59
- end
60
-
61
- # Initialize a new command runner.
62
- #
63
- # @param middleware [Array<Middleware::Base>] middleware chain
64
- # @param strategy [Strategies::Base] execution strategy
65
- # @param options [Hash] additional configuration
66
- def initialize(middleware: nil, strategy: nil, **options)
67
- @middleware = Array(middleware || default_middleware)
68
- @strategy = strategy || Strategies::Synchronous.new
69
- @options = options
70
-
71
- validate_middleware
72
- validate_strategy
73
-
74
- # Log runner initialization
75
- log_runner_initialization
76
- end
77
-
78
- # Execute a single command request.
79
- #
80
- # @param request [Request, String, Hash] command request to execute
81
- # @return [Result] execution result
82
- # @raise [ArgumentError] if request is invalid
83
- def execute(request)
84
- # Normalize request to Request object
85
- normalized_request = normalize_request(request)
86
-
87
- # Apply middleware chain
88
- execute_with_middleware(normalized_request) do |processed_request|
89
- @strategy.execute(processed_request)
90
- end
91
- end
92
-
93
- # Execute multiple command requests.
94
- #
95
- # @param requests [Array<Request, String, Hash>] requests to execute
96
- # @return [Array<Result>] execution results
97
- def execute_batch(requests)
98
- # Normalize all requests
99
- normalized_requests = requests.map { |req| normalize_request(req) }
100
-
101
- # Use strategy's batch execution if available, otherwise execute individually
102
- if @strategy.respond_to?(:execute_batch)
103
- # Apply middleware to the entire batch
104
- execute_with_middleware_batch(normalized_requests) do |processed_requests|
105
- @strategy.execute_batch(processed_requests)
106
- end
107
- else
108
- # Execute each request individually with middleware
109
- normalized_requests.map { |request| execute(request) }
110
- end
111
- end
112
-
113
- # Add middleware to the execution chain.
114
- #
115
- # @param middleware_instance [Middleware::Base] middleware to add
116
- # @return [self] for method chaining
117
- def add_middleware(middleware_instance)
118
- validate_middleware_instance(middleware_instance)
119
- @middleware << middleware_instance
120
- self
121
- end
122
-
123
- # Remove middleware from the execution chain.
124
- #
125
- # @param middleware_class [Class] middleware class to remove
126
- # @return [self] for method chaining
127
- def remove_middleware(middleware_class)
128
- @middleware.reject! { |m| m.is_a?(middleware_class) }
129
- self
130
- end
131
-
132
- # Check if specific middleware is present.
133
- #
134
- # @param middleware_class [Class] middleware class to check
135
- # @return [Boolean] true if middleware is present
136
- def has_middleware?(middleware_class)
137
- @middleware.any? { |m| m.is_a?(middleware_class) }
138
- end
139
-
140
- # Get runner configuration for debugging.
141
- #
142
- # @return [Hash] runner configuration
143
- def config
144
- {
145
- middleware: @middleware.map(&:config),
146
- strategy: @strategy.config,
147
- options: @options,
148
- }
149
- end
150
-
151
- # Get execution statistics.
152
- #
153
- # @return [Hash] execution statistics
154
- def stats
155
- @stats ||= {
156
- total_executions: 0,
157
- successful_executions: 0,
158
- failed_executions: 0,
159
- total_duration: 0.0,
160
- }
161
- end
162
-
163
- private
164
-
165
- # Execute request with middleware chain.
166
- #
167
- # @param request [Request] normalized request
168
- # @yield [Request] yields processed request to execution
169
- # @return [Result] execution result
170
- def execute_with_middleware(request, &block)
171
- # Build middleware chain
172
- chain = build_middleware_chain(request)
173
-
174
- # Execute chain
175
- result = chain.call(request, &block)
176
-
177
- # Update statistics
178
- update_stats(result)
179
-
180
- result
181
- end
182
-
183
- # Execute batch with middleware chain.
184
- #
185
- # @param requests [Array<Request>] normalized requests
186
- # @yield [Array<Request>] yields processed requests to execution
187
- # @return [Array<Result>] execution results
188
- def execute_with_middleware_batch(requests, &block)
189
- # For batch execution, apply middleware to each request individually
190
- # This maintains the middleware contract while supporting batch execution
191
- results = requests.map do |request|
192
- execute_with_middleware(request) { |req| [req] }
193
- end.flatten
194
-
195
- # Then execute the batch
196
- processed_requests = results.map(&:command).map { |cmd| Request.from_string(cmd) }
197
- batch_results = block.call(processed_requests)
198
-
199
- # Update statistics for batch
200
- batch_results.each { |result| update_stats(result) }
201
-
202
- batch_results
203
- end
204
-
205
- # Build middleware execution chain.
206
- #
207
- # @param request [Request] the request to process
208
- # @return [Proc] middleware chain
209
- def build_middleware_chain(request)
210
- # Filter middleware that applies to this request
211
- applicable_middleware = @middleware.select { |m| m.applicable?(request) }
212
-
213
- # Build chain in reverse order so first middleware wraps everything
214
- applicable_middleware.reverse.reduce(method(:execute_final)) do |chain, middleware|
215
- ->(req) { middleware.call(req, &chain) }
216
- end
217
- end
218
-
219
- # Final execution step (after all middleware).
220
- #
221
- # @param request [Request] processed request
222
- # @return [Result] execution result
223
- def execute_final(request)
224
- @strategy.execute(request)
225
- end
226
-
227
- # Normalize various input types to Request objects.
228
- #
229
- # @param request [Request, String, Hash] request in various formats
230
- # @return [Request] normalized request
231
- # @raise [ArgumentError] if request cannot be normalized
232
- def normalize_request(request)
233
- case request
234
- when Request
235
- request
236
- when String
237
- Request.from_string(request)
238
- when Hash
239
- Request.from_hash(request)
240
- else
241
- raise ArgumentError, "Invalid request type: #{request.class}. Expected Request, String, or Hash."
242
- end
243
- end
244
-
245
- # Update execution statistics.
246
- #
247
- # @param result [Result] execution result
248
- def update_stats(result)
249
- stats[:total_executions] += 1
250
-
251
- if result.success?
252
- stats[:successful_executions] += 1
253
- else
254
- stats[:failed_executions] += 1
255
- end
256
-
257
- stats[:total_duration] += result.duration if result.duration
258
- end
259
-
260
- # Get default middleware chain.
261
- #
262
- # @return [Array<Middleware::Base>] default middleware
263
- def default_middleware
264
- # No default middleware for now - keep it simple
265
- # Subclasses or configuration can add middleware as needed
266
- []
267
- end
268
-
269
- # Validate middleware array.
270
- def validate_middleware
271
- @middleware.each { |m| validate_middleware_instance(m) }
272
- end
273
-
274
- # Validate individual middleware instance.
275
- #
276
- # @param middleware [Object] middleware to validate
277
- # @raise [ArgumentError] if middleware is invalid
278
- def validate_middleware_instance(middleware)
279
- unless middleware.respond_to?(:call)
280
- raise ArgumentError, "Middleware must respond to #call: #{middleware.class}"
281
- end
282
-
283
- return if middleware.respond_to?(:applicable?)
284
-
285
- raise ArgumentError, "Middleware must respond to #applicable?: #{middleware.class}"
286
- end
287
-
288
- # Validate execution strategy.
289
- #
290
- # @raise [ArgumentError] if strategy is invalid
291
- def validate_strategy
292
- unless @strategy.respond_to?(:execute)
293
- raise ArgumentError, "Strategy must respond to #execute: #{@strategy.class}"
294
- end
295
-
296
- return if @strategy.respond_to?(:supports?)
297
-
298
- raise ArgumentError, "Strategy must respond to #supports?: #{@strategy.class}"
299
- end
300
-
301
- # Log runner initialization for debugging.
302
- #
303
- # @return [void]
304
- def log_runner_initialization
305
- # Only log if we have a unified logger middleware
306
- logger_middleware = @middleware.find { |m| m.is_a?(Middleware::UnifiedLogger) }
307
- return unless logger_middleware
308
-
309
- logger_middleware.logger.debug("Command runner initialized",
310
- middleware_count: @middleware.length,
311
- strategy: @strategy.class.name,
312
- working_directory: Dir.pwd,
313
- options: @options)
314
- end
315
- end
316
- end
317
- end
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "request"
4
+ require_relative "result"
5
+ require_relative "strategies/synchronous"
6
+ require_relative "middleware/base"
7
+ require_relative "middleware/unified_logger"
8
+ require_relative "middleware/command_logger"
9
+
10
+ module Makit
11
+ module Commands
12
+ # Modern command execution engine with proper separation of concerns.
13
+ #
14
+ # The Runner coordinates command execution through a middleware chain and
15
+ # execution strategies. It provides a clean, extensible architecture for
16
+ # command processing with support for caching, logging, validation, and
17
+ # custom execution patterns.
18
+ #
19
+ # @example Basic usage
20
+ # runner = Runner.new
21
+ # request = Request.from_string("git --version")
22
+ # result = runner.execute(request)
23
+ # puts result.stdout
24
+ #
25
+ # @example With custom middleware
26
+ # runner = Runner.new(middleware: [Logger.new, Cache.new])
27
+ # result = runner.execute(request)
28
+ #
29
+ # @example With custom strategy
30
+ # runner = Runner.new(strategy: Strategies::Parallel.new)
31
+ # results = runner.execute_batch(requests)
32
+ class Runner
33
+ # @!attribute [r] middleware
34
+ # @return [Array<Middleware::Base>] middleware chain
35
+ # @!attribute [r] strategy
36
+ # @return [Strategies::Base] execution strategy
37
+ attr_reader :middleware, :strategy
38
+
39
+ # Get the default configured runner instance.
40
+ #
41
+ # @return [Runner] default runner with standard middleware
42
+ def self.default
43
+ # Recreate the runner if the log level has changed
44
+ current_log_level = Makit::Logging.current_log_level
45
+ if @default.nil? || @last_log_level != current_log_level
46
+ @last_log_level = current_log_level
47
+ @default = new(
48
+ middleware: [
49
+ Middleware::CommandLogger.new(
50
+ log_stdout: true,
51
+ log_stderr: true,
52
+ log_performance: true,
53
+ max_output_lines: 50,
54
+ ),
55
+ ],
56
+ )
57
+ end
58
+ @default
59
+ end
60
+
61
+ # Initialize a new command runner.
62
+ #
63
+ # @param middleware [Array<Middleware::Base>] middleware chain
64
+ # @param strategy [Strategies::Base] execution strategy
65
+ # @param options [Hash] additional configuration
66
+ def initialize(middleware: nil, strategy: nil, **options)
67
+ @middleware = Array(middleware || default_middleware)
68
+ @strategy = strategy || Strategies::Synchronous.new
69
+ @options = options
70
+
71
+ validate_middleware
72
+ validate_strategy
73
+
74
+ # Log runner initialization
75
+ log_runner_initialization
76
+ end
77
+
78
+ # Execute a single command request.
79
+ #
80
+ # @param request [Request, String, Hash] command request to execute
81
+ # @return [Result] execution result
82
+ # @raise [ArgumentError] if request is invalid
83
+ def execute(request)
84
+ # Normalize request to Request object
85
+ normalized_request = normalize_request(request)
86
+
87
+ # Apply middleware chain
88
+ execute_with_middleware(normalized_request) do |processed_request|
89
+ @strategy.execute(processed_request)
90
+ end
91
+ end
92
+
93
+ # Execute multiple command requests.
94
+ #
95
+ # @param requests [Array<Request, String, Hash>] requests to execute
96
+ # @return [Array<Result>] execution results
97
+ def execute_batch(requests)
98
+ # Normalize all requests
99
+ normalized_requests = requests.map { |req| normalize_request(req) }
100
+
101
+ # Use strategy's batch execution if available, otherwise execute individually
102
+ if @strategy.respond_to?(:execute_batch)
103
+ # Apply middleware to the entire batch
104
+ execute_with_middleware_batch(normalized_requests) do |processed_requests|
105
+ @strategy.execute_batch(processed_requests)
106
+ end
107
+ else
108
+ # Execute each request individually with middleware
109
+ normalized_requests.map { |request| execute(request) }
110
+ end
111
+ end
112
+
113
+ # Add middleware to the execution chain.
114
+ #
115
+ # @param middleware_instance [Middleware::Base] middleware to add
116
+ # @return [self] for method chaining
117
+ def add_middleware(middleware_instance)
118
+ validate_middleware_instance(middleware_instance)
119
+ @middleware << middleware_instance
120
+ self
121
+ end
122
+
123
+ # Remove middleware from the execution chain.
124
+ #
125
+ # @param middleware_class [Class] middleware class to remove
126
+ # @return [self] for method chaining
127
+ def remove_middleware(middleware_class)
128
+ @middleware.reject! { |m| m.is_a?(middleware_class) }
129
+ self
130
+ end
131
+
132
+ # Check if specific middleware is present.
133
+ #
134
+ # @param middleware_class [Class] middleware class to check
135
+ # @return [Boolean] true if middleware is present
136
+ def has_middleware?(middleware_class)
137
+ @middleware.any? { |m| m.is_a?(middleware_class) }
138
+ end
139
+
140
+ # Get runner configuration for debugging.
141
+ #
142
+ # @return [Hash] runner configuration
143
+ def config
144
+ {
145
+ middleware: @middleware.map(&:config),
146
+ strategy: @strategy.config,
147
+ options: @options,
148
+ }
149
+ end
150
+
151
+ # Get execution statistics.
152
+ #
153
+ # @return [Hash] execution statistics
154
+ def stats
155
+ @stats ||= {
156
+ total_executions: 0,
157
+ successful_executions: 0,
158
+ failed_executions: 0,
159
+ total_duration: 0.0,
160
+ }
161
+ end
162
+
163
+ private
164
+
165
+ # Execute request with middleware chain.
166
+ #
167
+ # @param request [Request] normalized request
168
+ # @yield [Request] yields processed request to execution
169
+ # @return [Result] execution result
170
+ def execute_with_middleware(request, &block)
171
+ # Build middleware chain
172
+ chain = build_middleware_chain(request)
173
+
174
+ # Execute chain
175
+ result = chain.call(request, &block)
176
+
177
+ # Log high-level success/error using the default logger
178
+ log_command_result(request, result)
179
+
180
+ # Update statistics
181
+ update_stats(result)
182
+
183
+ result
184
+ end
185
+
186
+ # Execute batch with middleware chain.
187
+ #
188
+ # @param requests [Array<Request>] normalized requests
189
+ # @yield [Array<Request>] yields processed requests to execution
190
+ # @return [Array<Result>] execution results
191
+ def execute_with_middleware_batch(requests, &block)
192
+ # For batch execution, apply middleware to each request individually
193
+ # This maintains the middleware contract while supporting batch execution
194
+ results = requests.map do |request|
195
+ execute_with_middleware(request) { |req| [req] }
196
+ end.flatten
197
+
198
+ # Then execute the batch
199
+ processed_requests = results.map(&:command).map { |cmd| Request.from_string(cmd) }
200
+ batch_results = block.call(processed_requests)
201
+
202
+ # Update statistics for batch
203
+ batch_results.each { |result| update_stats(result) }
204
+
205
+ batch_results
206
+ end
207
+
208
+ # Build middleware execution chain.
209
+ #
210
+ # @param request [Request] the request to process
211
+ # @return [Proc] middleware chain
212
+ def build_middleware_chain(request)
213
+ # Filter middleware that applies to this request
214
+ applicable_middleware = @middleware.select { |m| m.applicable?(request) }
215
+
216
+ # Build chain in reverse order so first middleware wraps everything
217
+ applicable_middleware.reverse.reduce(method(:execute_final)) do |chain, middleware|
218
+ ->(req) { middleware.call(req, &chain) }
219
+ end
220
+ end
221
+
222
+ # Final execution step (after all middleware).
223
+ #
224
+ # @param request [Request] processed request
225
+ # @return [Result] execution result
226
+ def execute_final(request)
227
+ @strategy.execute(request)
228
+ end
229
+
230
+ # Normalize various input types to Request objects.
231
+ #
232
+ # @param request [Request, String, Hash] request in various formats
233
+ # @return [Request] normalized request
234
+ # @raise [ArgumentError] if request cannot be normalized
235
+ def normalize_request(request)
236
+ case request
237
+ when Request
238
+ request
239
+ when String
240
+ Request.from_string(request)
241
+ when Hash
242
+ Request.from_hash(request)
243
+ else
244
+ raise ArgumentError, "Invalid request type: #{request.class}. Expected Request, String, or Hash."
245
+ end
246
+ end
247
+
248
+ # Update execution statistics.
249
+ #
250
+ # @param result [Result] execution result
251
+ def update_stats(result)
252
+ stats[:total_executions] += 1
253
+
254
+ if result.success?
255
+ stats[:successful_executions] += 1
256
+ else
257
+ stats[:failed_executions] += 1
258
+ end
259
+
260
+ stats[:total_duration] += result.duration if result.duration
261
+ end
262
+
263
+ # Get default middleware chain.
264
+ #
265
+ # @return [Array<Middleware::Base>] default middleware
266
+ def default_middleware
267
+ # No default middleware for now - keep it simple
268
+ # Subclasses or configuration can add middleware as needed
269
+ []
270
+ end
271
+
272
+ # Validate middleware array.
273
+ def validate_middleware
274
+ @middleware.each { |m| validate_middleware_instance(m) }
275
+ end
276
+
277
+ # Validate individual middleware instance.
278
+ #
279
+ # @param middleware [Object] middleware to validate
280
+ # @raise [ArgumentError] if middleware is invalid
281
+ def validate_middleware_instance(middleware)
282
+ unless middleware.respond_to?(:call)
283
+ raise ArgumentError, "Middleware must respond to #call: #{middleware.class}"
284
+ end
285
+
286
+ return if middleware.respond_to?(:applicable?)
287
+
288
+ raise ArgumentError, "Middleware must respond to #applicable?: #{middleware.class}"
289
+ end
290
+
291
+ # Validate execution strategy.
292
+ #
293
+ # @raise [ArgumentError] if strategy is invalid
294
+ def validate_strategy
295
+ unless @strategy.respond_to?(:execute)
296
+ raise ArgumentError, "Strategy must respond to #execute: #{@strategy.class}"
297
+ end
298
+
299
+ return if @strategy.respond_to?(:supports?)
300
+
301
+ raise ArgumentError, "Strategy must respond to #supports?: #{@strategy.class}"
302
+ end
303
+
304
+ # Log command execution result using the default logger.
305
+ #
306
+ # @param request [Request] the executed request
307
+ # @param result [Result] the execution result
308
+ # @return [void]
309
+ def log_command_result(request, result)
310
+ # Build the full command string
311
+ command_string = "#{request.command} #{request.arguments&.join(' ')}".strip
312
+
313
+ # Log success or error using the default logger
314
+ if result.success?
315
+ Makit::Logging.success(command_string)
316
+ else
317
+ Makit::Logging.error(command_string)
318
+ end
319
+ end
320
+
321
+ # Log runner initialization for debugging.
322
+ #
323
+ # @return [void]
324
+ def log_runner_initialization
325
+ # Only log if we have a unified logger middleware
326
+ logger_middleware = @middleware.find { |m| m.is_a?(Middleware::UnifiedLogger) }
327
+ return unless logger_middleware
328
+
329
+ logger_middleware.logger.debug("Command runner initialized",
330
+ middleware_count: @middleware.length,
331
+ strategy: @strategy.class.name,
332
+ working_directory: Dir.pwd,
333
+ options: @options)
334
+ end
335
+ end
336
+ end
337
+ end