makit 0.0.169 → 0.0.170

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 (179) 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/auto.rb +48 -48
  6. data/lib/makit/azure/blob_storage.rb +257 -257
  7. data/lib/makit/azure/cli.rb +284 -284
  8. data/lib/makit/azure-pipelines.rb +187 -187
  9. data/lib/makit/cli/base.rb +17 -17
  10. data/lib/makit/cli/build_commands.rb +500 -500
  11. data/lib/makit/cli/generators/base_generator.rb +74 -74
  12. data/lib/makit/cli/generators/dotnet_generator.rb +50 -50
  13. data/lib/makit/cli/generators/generator_factory.rb +49 -49
  14. data/lib/makit/cli/generators/node_generator.rb +50 -50
  15. data/lib/makit/cli/generators/ruby_generator.rb +77 -77
  16. data/lib/makit/cli/generators/rust_generator.rb +50 -50
  17. data/lib/makit/cli/generators/templates/dotnet_templates.rb +167 -167
  18. data/lib/makit/cli/generators/templates/node_templates.rb +161 -161
  19. data/lib/makit/cli/generators/templates/ruby/gemfile.rb +26 -26
  20. data/lib/makit/cli/generators/templates/ruby/gemspec.rb +41 -41
  21. data/lib/makit/cli/generators/templates/ruby/main_lib.rb +33 -33
  22. data/lib/makit/cli/generators/templates/ruby/rakefile.rb +35 -35
  23. data/lib/makit/cli/generators/templates/ruby/readme.rb +63 -63
  24. data/lib/makit/cli/generators/templates/ruby/test.rb +39 -39
  25. data/lib/makit/cli/generators/templates/ruby/test_helper.rb +29 -29
  26. data/lib/makit/cli/generators/templates/ruby/version.rb +29 -29
  27. data/lib/makit/cli/generators/templates/rust_templates.rb +128 -128
  28. data/lib/makit/cli/main.rb +78 -78
  29. data/lib/makit/cli/pipeline_commands.rb +311 -311
  30. data/lib/makit/cli/project_commands.rb +868 -868
  31. data/lib/makit/cli/repository_commands.rb +661 -661
  32. data/lib/makit/cli/strategy_commands.rb +207 -207
  33. data/lib/makit/cli/utility_commands.rb +521 -521
  34. data/lib/makit/commands/factory.rb +359 -359
  35. data/lib/makit/commands/middleware/base.rb +73 -73
  36. data/lib/makit/commands/middleware/cache.rb +248 -248
  37. data/lib/makit/commands/middleware/command_logger.rb +312 -312
  38. data/lib/makit/commands/middleware/validator.rb +269 -269
  39. data/lib/makit/commands/request.rb +316 -316
  40. data/lib/makit/commands/result.rb +323 -323
  41. data/lib/makit/commands/runner.rb +386 -386
  42. data/lib/makit/commands/strategies/base.rb +171 -171
  43. data/lib/makit/commands/strategies/child_process.rb +162 -162
  44. data/lib/makit/commands/strategies/factory.rb +136 -136
  45. data/lib/makit/commands/strategies/synchronous.rb +139 -139
  46. data/lib/makit/commands.rb +50 -50
  47. data/lib/makit/configuration/dotnet_project.rb +48 -48
  48. data/lib/makit/configuration/gitlab_helper.rb +61 -61
  49. data/lib/makit/configuration/project.rb +292 -292
  50. data/lib/makit/configuration/rakefile_helper.rb +43 -43
  51. data/lib/makit/configuration/step.rb +34 -34
  52. data/lib/makit/configuration/timeout.rb +74 -74
  53. data/lib/makit/configuration.rb +21 -21
  54. data/lib/makit/content/default_gitignore.rb +7 -7
  55. data/lib/makit/content/default_gitignore.txt +225 -225
  56. data/lib/makit/content/default_rakefile.rb +13 -13
  57. data/lib/makit/content/gem_rakefile.rb +16 -16
  58. data/lib/makit/context.rb +1 -1
  59. data/lib/makit/data.rb +49 -49
  60. data/lib/makit/directories.rb +170 -170
  61. data/lib/makit/directory.rb +262 -262
  62. data/lib/makit/docs/files.rb +89 -89
  63. data/lib/makit/docs/rake.rb +102 -102
  64. data/lib/makit/dotnet/cli.rb +224 -224
  65. data/lib/makit/dotnet/project.rb +217 -217
  66. data/lib/makit/dotnet/solution.rb +38 -38
  67. data/lib/makit/dotnet/solution_classlib.rb +239 -239
  68. data/lib/makit/dotnet/solution_console.rb +264 -264
  69. data/lib/makit/dotnet/solution_maui.rb +354 -354
  70. data/lib/makit/dotnet/solution_wasm.rb +275 -275
  71. data/lib/makit/dotnet/solution_wpf.rb +304 -304
  72. data/lib/makit/dotnet.rb +110 -110
  73. data/lib/makit/email.rb +90 -90
  74. data/lib/makit/environment.rb +142 -142
  75. data/lib/makit/examples/runner.rb +370 -370
  76. data/lib/makit/exceptions.rb +45 -45
  77. data/lib/makit/fileinfo.rb +32 -32
  78. data/lib/makit/files.rb +43 -43
  79. data/lib/makit/gems.rb +49 -49
  80. data/lib/makit/git/cli.rb +103 -103
  81. data/lib/makit/git/repository.rb +100 -100
  82. data/lib/makit/git.rb +104 -104
  83. data/lib/makit/github_actions.rb +202 -202
  84. data/lib/makit/gitlab/pipeline.rb +857 -857
  85. data/lib/makit/gitlab/pipeline_service_impl.rb +1535 -1535
  86. data/lib/makit/gitlab_runner.rb +59 -59
  87. data/lib/makit/humanize.rb +218 -218
  88. data/lib/makit/indexer.rb +47 -47
  89. data/lib/makit/io/filesystem.rb +111 -111
  90. data/lib/makit/io/filesystem_service_impl.rb +337 -337
  91. data/lib/makit/lint.rb +212 -212
  92. data/lib/makit/logging/configuration.rb +309 -309
  93. data/lib/makit/logging/format_registry.rb +84 -84
  94. data/lib/makit/logging/formatters/base.rb +39 -39
  95. data/lib/makit/logging/formatters/console_formatter.rb +140 -140
  96. data/lib/makit/logging/formatters/json_formatter.rb +65 -65
  97. data/lib/makit/logging/formatters/plain_text_formatter.rb +71 -71
  98. data/lib/makit/logging/formatters/text_formatter.rb +64 -64
  99. data/lib/makit/logging/log_request.rb +119 -119
  100. data/lib/makit/logging/logger.rb +199 -199
  101. data/lib/makit/logging/sinks/base.rb +91 -91
  102. data/lib/makit/logging/sinks/console.rb +72 -72
  103. data/lib/makit/logging/sinks/file_sink.rb +92 -92
  104. data/lib/makit/logging/sinks/structured.rb +123 -123
  105. data/lib/makit/logging/sinks/unified_file_sink.rb +296 -296
  106. data/lib/makit/logging.rb +578 -578
  107. data/lib/makit/markdown.rb +75 -75
  108. data/lib/makit/mp/basic_object_mp.rb +17 -17
  109. data/lib/makit/mp/command_mp.rb +13 -13
  110. data/lib/makit/mp/command_request.mp.rb +17 -17
  111. data/lib/makit/mp/project_mp.rb +199 -199
  112. data/lib/makit/mp/string_mp.rb +205 -205
  113. data/lib/makit/nuget.rb +460 -460
  114. data/lib/makit/podman/podman.rb +458 -458
  115. data/lib/makit/podman/podman_service_impl.rb +1081 -1081
  116. data/lib/makit/port.rb +32 -32
  117. data/lib/makit/process.rb +377 -377
  118. data/lib/makit/protoc.rb +112 -112
  119. data/lib/makit/rake/cli.rb +196 -196
  120. data/lib/makit/rake/trace_controller.rb +174 -174
  121. data/lib/makit/rake.rb +81 -81
  122. data/lib/makit/ruby/cli.rb +185 -185
  123. data/lib/makit/ruby.rb +25 -25
  124. data/lib/makit/rubygems.rb +137 -137
  125. data/lib/makit/secrets/azure_key_vault.rb +322 -322
  126. data/lib/makit/secrets/azure_secrets.rb +221 -221
  127. data/lib/makit/secrets/local_secrets.rb +72 -72
  128. data/lib/makit/secrets/secrets_manager.rb +105 -105
  129. data/lib/makit/secrets.rb +96 -96
  130. data/lib/makit/serializer.rb +130 -130
  131. data/lib/makit/services/builder.rb +186 -186
  132. data/lib/makit/services/error_handler.rb +226 -226
  133. data/lib/makit/services/repository_manager.rb +367 -367
  134. data/lib/makit/services/validator.rb +112 -112
  135. data/lib/makit/setup/classlib.rb +101 -101
  136. data/lib/makit/setup/gem.rb +268 -268
  137. data/lib/makit/setup/pages.rb +11 -11
  138. data/lib/makit/setup/razorclasslib.rb +101 -101
  139. data/lib/makit/setup/runner.rb +54 -54
  140. data/lib/makit/setup.rb +5 -5
  141. data/lib/makit/show.rb +110 -110
  142. data/lib/makit/storage.rb +126 -126
  143. data/lib/makit/symbols.rb +175 -175
  144. data/lib/makit/task_info.rb +130 -130
  145. data/lib/makit/tasks/at_exit.rb +15 -15
  146. data/lib/makit/tasks/build.rb +22 -22
  147. data/lib/makit/tasks/bump.rb +7 -7
  148. data/lib/makit/tasks/clean.rb +13 -13
  149. data/lib/makit/tasks/configure.rb +10 -10
  150. data/lib/makit/tasks/format.rb +10 -10
  151. data/lib/makit/tasks/hook_manager.rb +443 -443
  152. data/lib/makit/tasks/info.rb +368 -368
  153. data/lib/makit/tasks/init.rb +49 -49
  154. data/lib/makit/tasks/integrate.rb +60 -60
  155. data/lib/makit/tasks/pull_incoming.rb +13 -13
  156. data/lib/makit/tasks/secrets.rb +7 -7
  157. data/lib/makit/tasks/setup.rb +16 -16
  158. data/lib/makit/tasks/sync.rb +14 -14
  159. data/lib/makit/tasks/tag.rb +27 -27
  160. data/lib/makit/tasks/task_monkey_patch.rb +81 -81
  161. data/lib/makit/tasks/test.rb +22 -22
  162. data/lib/makit/tasks/update.rb +21 -21
  163. data/lib/makit/tasks/version.rb +6 -6
  164. data/lib/makit/tasks.rb +24 -24
  165. data/lib/makit/test_cache.rb +239 -239
  166. data/lib/makit/tree.rb +37 -37
  167. data/lib/makit/v1/configuration/project_service_impl.rb +370 -370
  168. data/lib/makit/v1/git/git_repository_service_impl.rb +295 -295
  169. data/lib/makit/v1/makit.v1_pb.rb +35 -35
  170. data/lib/makit/v1/makit.v1_services_pb.rb +27 -27
  171. data/lib/makit/v1/services/repository_manager_service_impl.rb +572 -572
  172. data/lib/makit/version.rb +661 -661
  173. data/lib/makit/version_util.rb +21 -21
  174. data/lib/makit/wix.rb +95 -95
  175. data/lib/makit/yaml.rb +29 -29
  176. data/lib/makit/zip.rb +17 -17
  177. data/lib/makit copy.rb +44 -44
  178. data/lib/makit.rb +121 -121
  179. metadata +2 -2
@@ -1,316 +1,316 @@
1
- # frozen_string_literal: true
2
-
3
- require "shellwords"
4
- require "json"
5
-
6
- module Makit
7
- module Commands
8
- # Enhanced request handling with validation and type safety.
9
- # Uses plain Ruby objects - no protobuf dependency for better performance.
10
- #
11
- # @example Create a simple request
12
- # request = Request.new(command: "git", arguments: ["--version"])
13
- #
14
- # @example Create request from string
15
- # request = Request.from_string("git clone https://github.com/user/repo.git")
16
- #
17
- # @example Create request with full options
18
- # request = Request.new(
19
- # command: "bundle",
20
- # arguments: ["install"],
21
- # directory: "/path/to/project",
22
- # environment: { "BUNDLE_JOBS" => "4" },
23
- # timeout: 300,
24
- # metadata: { operation: "dependency_install" }
25
- # )
26
- class Request
27
- # Command execution request with validation and metadata.
28
- #
29
- # @!attribute [r] command
30
- # @return [String] the command to execute
31
- # @!attribute [r] arguments
32
- # @return [Array<String>] command line arguments
33
- # @!attribute [r] environment
34
- # @return [Hash<String,String>] environment variables
35
- # @!attribute [r] directory
36
- # @return [String] working directory for execution
37
- # @!attribute [r] timeout
38
- # @return [Integer] timeout in seconds
39
- # @!attribute [r] metadata
40
- # @return [Hash] additional metadata for logging/tracking
41
- attr_reader :command, :arguments, :environment, :directory, :timeout, :metadata
42
- attr_writer :exit_on_failure, :show_stderr, :show_stdout
43
-
44
- # Initialize a new command request.
45
- #
46
- # @param command [String] the command to execute
47
- # @param arguments [Array<String>] command arguments
48
- # @param environment [Hash<String,String>] environment variables
49
- # @param directory [String] working directory
50
- # @param timeout [Integer] timeout in seconds
51
- # @param metadata [Hash] additional metadata
52
- # @raise [ArgumentError] if command is invalid
53
- def initialize(command:, arguments: [], **options)
54
- @command = validate_command(command)
55
- @arguments = validate_arguments(arguments)
56
- @environment = options[:environment] || {}
57
- @directory = options[:directory] || Dir.pwd
58
- @timeout = options[:timeout] || Makit::Configuration::Timeout.global_default
59
- @metadata = options[:metadata] || {}
60
- @exit_on_failure = options[:exit_on_failure] || true
61
- @show_stderr = options[:show_stderr] || true
62
- @show_stdout = options[:show_stdout] || false
63
- validate_directory(@directory)
64
- validate_timeout(@timeout)
65
- end
66
-
67
- # Convert request to shell-executable command string.
68
- #
69
- # @return [String] shell command string
70
- def to_shell_command
71
- cmd_parts = [command] + arguments
72
- Shellwords.join(cmd_parts)
73
- end
74
-
75
- # Convert request to hash representation.
76
- #
77
- # @return [Hash] hash representation of the request
78
- def to_h
79
- {
80
- command: command,
81
- arguments: arguments,
82
- environment: environment,
83
- directory: directory,
84
- timeout: timeout,
85
- metadata: metadata,
86
- exit_on_failure: exit_on_failure?,
87
- show_stderr: show_stderr?,
88
- show_stdout: show_stdout?,
89
- }
90
- end
91
-
92
- # Convert request to JSON representation.
93
- #
94
- # @param args [Array] arguments passed to JSON.generate
95
- # @return [String] JSON representation
96
- def to_json(*args)
97
- JSON.generate(to_h, *args)
98
- end
99
-
100
- def to_json_pretty(*args)
101
- JSON.pretty_generate(to_h, *args)
102
- end
103
-
104
- # Create a request from a shell command string.
105
- #
106
- # @param command_string [String] shell command string to parse
107
- # @param options [Hash] additional options
108
- # @return [Request] new request object
109
- # @raise [ArgumentError] if command string is invalid
110
- # @example
111
- # Request.from_string("git clone https://github.com/user/repo.git")
112
- # Request.from_string("bundle install --jobs 4")
113
- def self.from_string(command_string, **options)
114
- raise ArgumentError, "Command string cannot be empty" if command_string.nil? || command_string.strip.empty?
115
-
116
- # Parse shell command string into command and arguments
117
- parts = Shellwords.split(command_string.strip)
118
- command = parts.shift
119
- arguments = parts
120
-
121
- new(command: command, arguments: arguments, **options)
122
- end
123
-
124
- # Create a request from hash representation.
125
- #
126
- # @param command_hash [Hash] hash containing command information
127
- # @return [Request] new request object
128
- # @raise [ArgumentError] if hash is invalid
129
- # @example
130
- # Request.from_hash({
131
- # "command" => "git",
132
- # "arguments" => ["clone", "https://github.com/user/repo.git"],
133
- # "timeout" => 300
134
- # })
135
- def self.from_hash(command_hash)
136
- raise ArgumentError, "Command hash cannot be nil" if command_hash.nil?
137
-
138
- # Convert string keys to symbols for consistency
139
- hash = command_hash.is_a?(Hash) ? normalize_hash_keys(command_hash) : command_hash
140
-
141
- new(
142
- command: hash[:command] || hash["command"],
143
- arguments: hash[:arguments] || hash["arguments"] || [],
144
- environment: hash[:environment] || hash["environment"] || {},
145
- directory: hash[:directory] || hash["directory"],
146
- timeout: hash[:timeout] || hash["timeout"],
147
- metadata: hash[:metadata] || hash["metadata"] || {},
148
- exit_on_failure: hash[:exit_on_failure] || hash["exit_on_failure"],
149
- show_stderr: hash[:show_stderr] || hash["show_stderr"],
150
- show_stdout: hash[:show_stdout] || hash["show_stdout"],
151
- )
152
- end
153
-
154
- # Create a request from JSON string.
155
- #
156
- # @param json_string [String] JSON string representation
157
- # @return [Request] new request object
158
- # @raise [JSON::ParserError, ArgumentError] if JSON is invalid
159
- def self.from_json(json_string)
160
- hash = JSON.parse(json_string)
161
- from_hash(hash)
162
- end
163
-
164
- # Check if this request represents the same command.
165
- #
166
- # @param other [Request] another request to compare
167
- # @return [Boolean] true if commands are equivalent
168
- def equivalent_to?(other)
169
- return false unless other.is_a?(Request)
170
-
171
- command == other.command &&
172
- arguments == other.arguments &&
173
- environment == other.environment &&
174
- directory == other.directory
175
- end
176
-
177
- # Generate a cache key for this request.
178
- #
179
- # @return [String] cache key based on command, arguments, and context
180
- def cache_key
181
- require "digest"
182
- content = "#{command}:#{arguments.join(":")}:#{directory}"
183
- Digest::SHA256.hexdigest(content)[0..16]
184
- end
185
-
186
- def exit_on_failure?
187
- @exit_on_failure
188
- end
189
-
190
- def exit_on_failure(exit_on_failure)
191
- @exit_on_failure = exit_on_failure
192
- self
193
- end
194
-
195
- def show_stderr?
196
- @show_stderr
197
- end
198
-
199
- def show_stderr(show_stderr)
200
- @show_stderr = show_stderr
201
- self
202
- end
203
-
204
- def show_stdout?
205
- @show_stdout
206
- end
207
-
208
- def show_stdout(show_stdout)
209
- @show_stdout = show_stdout
210
- self
211
- end
212
-
213
- # Execute the command.
214
- #
215
- # @return [Result] execution result
216
- # @raise [ArgumentError] if runner is not set
217
- def run
218
- # self.exit_on_failure = exit_on_failure
219
- # self.show_stderr = true
220
- result = Makit::Commands::Runner.default.execute(show_stderr(true))
221
-
222
- # Display stderr if the command failed and show_stderr is enabled
223
- puts result.stderr if result.failure? && show_stderr? && !result.stderr.empty?
224
-
225
- # Exit with the command's exit code if it failed and exit_on_failure is enabled
226
- exit result.exit_code if result.failure? && exit_on_failure?
227
-
228
- result
229
- end
230
-
231
- def try
232
- Makit::Commands::Runner.default.execute(exit_on_failure(false))
233
- end
234
-
235
- private
236
-
237
- # Validate command parameter.
238
- #
239
- # @param command [String] command to validate
240
- # @return [String] validated command
241
- # @raise [ArgumentError] if command is invalid
242
- def validate_command(command)
243
- raise ArgumentError, "Command cannot be nil" if command.nil?
244
- raise ArgumentError, "Command cannot be empty" if command.strip.empty?
245
-
246
- cmd = command.strip
247
-
248
- # Check for dangerous characters that could indicate command injection
249
- raise ArgumentError, "Command contains potentially dangerous characters" if cmd.match?(/[;&|`$(){}]/)
250
-
251
- cmd
252
- end
253
-
254
- # Validate arguments array.
255
- #
256
- # @param arguments [Array] arguments to validate
257
- # @return [Array<String>] validated arguments
258
- # @raise [ArgumentError] if arguments are invalid
259
- def validate_arguments(arguments)
260
- return [] if arguments.nil?
261
- raise ArgumentError, "Arguments must be an array" unless arguments.is_a?(Array)
262
-
263
- # Convert all arguments to strings and validate
264
- arguments.map do |arg|
265
- raise ArgumentError, "Argument cannot be nil" if arg.nil?
266
-
267
- arg.to_s
268
- end
269
- end
270
-
271
- # Validate directory parameter.
272
- #
273
- # @param directory [String] directory to validate
274
- # @raise [ArgumentError] if directory is invalid
275
- def validate_directory(directory)
276
- return unless directory
277
-
278
- raise ArgumentError, "Directory must be a string" unless directory.is_a?(String)
279
-
280
- # NOTE: We don't validate directory existence here as it might be created later
281
- # We just validate it's a reasonable path format
282
- return unless directory.include?("\0")
283
-
284
- raise ArgumentError, "Directory path contains null bytes"
285
- end
286
-
287
- # Validate timeout parameter.
288
- #
289
- # @param timeout [Integer] timeout to validate
290
- # @raise [ArgumentError] if timeout is invalid
291
- def validate_timeout(timeout)
292
- raise ArgumentError, "Timeout must be a positive integer" unless timeout.is_a?(Integer) && timeout.positive?
293
-
294
- return unless timeout > 3600 # 1 hour max
295
-
296
- raise ArgumentError, "Timeout cannot exceed 3600 seconds"
297
- end
298
-
299
- # Normalize hash keys to support both string and symbol keys.
300
- #
301
- # @param hash [Hash] hash to normalize
302
- # @return [Hash] hash with normalized keys
303
- def self.normalize_hash_keys(hash)
304
- normalized = {}
305
- hash.each do |key, value|
306
- normalized_key = key.is_a?(String) ? key.to_sym : key
307
- normalized[key] = value # Keep original key
308
- normalized[normalized_key] = value # Add symbol version
309
- end
310
- normalized
311
- end
312
-
313
- private_class_method :normalize_hash_keys
314
- end
315
- end
316
- end
1
+ # frozen_string_literal: true
2
+
3
+ require "shellwords"
4
+ require "json"
5
+
6
+ module Makit
7
+ module Commands
8
+ # Enhanced request handling with validation and type safety.
9
+ # Uses plain Ruby objects - no protobuf dependency for better performance.
10
+ #
11
+ # @example Create a simple request
12
+ # request = Request.new(command: "git", arguments: ["--version"])
13
+ #
14
+ # @example Create request from string
15
+ # request = Request.from_string("git clone https://github.com/user/repo.git")
16
+ #
17
+ # @example Create request with full options
18
+ # request = Request.new(
19
+ # command: "bundle",
20
+ # arguments: ["install"],
21
+ # directory: "/path/to/project",
22
+ # environment: { "BUNDLE_JOBS" => "4" },
23
+ # timeout: 300,
24
+ # metadata: { operation: "dependency_install" }
25
+ # )
26
+ class Request
27
+ # Command execution request with validation and metadata.
28
+ #
29
+ # @!attribute [r] command
30
+ # @return [String] the command to execute
31
+ # @!attribute [r] arguments
32
+ # @return [Array<String>] command line arguments
33
+ # @!attribute [r] environment
34
+ # @return [Hash<String,String>] environment variables
35
+ # @!attribute [r] directory
36
+ # @return [String] working directory for execution
37
+ # @!attribute [r] timeout
38
+ # @return [Integer] timeout in seconds
39
+ # @!attribute [r] metadata
40
+ # @return [Hash] additional metadata for logging/tracking
41
+ attr_reader :command, :arguments, :environment, :directory, :timeout, :metadata
42
+ attr_writer :exit_on_failure, :show_stderr, :show_stdout
43
+
44
+ # Initialize a new command request.
45
+ #
46
+ # @param command [String] the command to execute
47
+ # @param arguments [Array<String>] command arguments
48
+ # @param environment [Hash<String,String>] environment variables
49
+ # @param directory [String] working directory
50
+ # @param timeout [Integer] timeout in seconds
51
+ # @param metadata [Hash] additional metadata
52
+ # @raise [ArgumentError] if command is invalid
53
+ def initialize(command:, arguments: [], **options)
54
+ @command = validate_command(command)
55
+ @arguments = validate_arguments(arguments)
56
+ @environment = options[:environment] || {}
57
+ @directory = options[:directory] || Dir.pwd
58
+ @timeout = options[:timeout] || Makit::Configuration::Timeout.global_default
59
+ @metadata = options[:metadata] || {}
60
+ @exit_on_failure = options[:exit_on_failure] || true
61
+ @show_stderr = options[:show_stderr] || true
62
+ @show_stdout = options[:show_stdout] || false
63
+ validate_directory(@directory)
64
+ validate_timeout(@timeout)
65
+ end
66
+
67
+ # Convert request to shell-executable command string.
68
+ #
69
+ # @return [String] shell command string
70
+ def to_shell_command
71
+ cmd_parts = [command] + arguments
72
+ Shellwords.join(cmd_parts)
73
+ end
74
+
75
+ # Convert request to hash representation.
76
+ #
77
+ # @return [Hash] hash representation of the request
78
+ def to_h
79
+ {
80
+ command: command,
81
+ arguments: arguments,
82
+ environment: environment,
83
+ directory: directory,
84
+ timeout: timeout,
85
+ metadata: metadata,
86
+ exit_on_failure: exit_on_failure?,
87
+ show_stderr: show_stderr?,
88
+ show_stdout: show_stdout?,
89
+ }
90
+ end
91
+
92
+ # Convert request to JSON representation.
93
+ #
94
+ # @param args [Array] arguments passed to JSON.generate
95
+ # @return [String] JSON representation
96
+ def to_json(*args)
97
+ JSON.generate(to_h, *args)
98
+ end
99
+
100
+ def to_json_pretty(*args)
101
+ JSON.pretty_generate(to_h, *args)
102
+ end
103
+
104
+ # Create a request from a shell command string.
105
+ #
106
+ # @param command_string [String] shell command string to parse
107
+ # @param options [Hash] additional options
108
+ # @return [Request] new request object
109
+ # @raise [ArgumentError] if command string is invalid
110
+ # @example
111
+ # Request.from_string("git clone https://github.com/user/repo.git")
112
+ # Request.from_string("bundle install --jobs 4")
113
+ def self.from_string(command_string, **options)
114
+ raise ArgumentError, "Command string cannot be empty" if command_string.nil? || command_string.strip.empty?
115
+
116
+ # Parse shell command string into command and arguments
117
+ parts = Shellwords.split(command_string.strip)
118
+ command = parts.shift
119
+ arguments = parts
120
+
121
+ new(command: command, arguments: arguments, **options)
122
+ end
123
+
124
+ # Create a request from hash representation.
125
+ #
126
+ # @param command_hash [Hash] hash containing command information
127
+ # @return [Request] new request object
128
+ # @raise [ArgumentError] if hash is invalid
129
+ # @example
130
+ # Request.from_hash({
131
+ # "command" => "git",
132
+ # "arguments" => ["clone", "https://github.com/user/repo.git"],
133
+ # "timeout" => 300
134
+ # })
135
+ def self.from_hash(command_hash)
136
+ raise ArgumentError, "Command hash cannot be nil" if command_hash.nil?
137
+
138
+ # Convert string keys to symbols for consistency
139
+ hash = command_hash.is_a?(Hash) ? normalize_hash_keys(command_hash) : command_hash
140
+
141
+ new(
142
+ command: hash[:command] || hash["command"],
143
+ arguments: hash[:arguments] || hash["arguments"] || [],
144
+ environment: hash[:environment] || hash["environment"] || {},
145
+ directory: hash[:directory] || hash["directory"],
146
+ timeout: hash[:timeout] || hash["timeout"],
147
+ metadata: hash[:metadata] || hash["metadata"] || {},
148
+ exit_on_failure: hash[:exit_on_failure] || hash["exit_on_failure"],
149
+ show_stderr: hash[:show_stderr] || hash["show_stderr"],
150
+ show_stdout: hash[:show_stdout] || hash["show_stdout"],
151
+ )
152
+ end
153
+
154
+ # Create a request from JSON string.
155
+ #
156
+ # @param json_string [String] JSON string representation
157
+ # @return [Request] new request object
158
+ # @raise [JSON::ParserError, ArgumentError] if JSON is invalid
159
+ def self.from_json(json_string)
160
+ hash = JSON.parse(json_string)
161
+ from_hash(hash)
162
+ end
163
+
164
+ # Check if this request represents the same command.
165
+ #
166
+ # @param other [Request] another request to compare
167
+ # @return [Boolean] true if commands are equivalent
168
+ def equivalent_to?(other)
169
+ return false unless other.is_a?(Request)
170
+
171
+ command == other.command &&
172
+ arguments == other.arguments &&
173
+ environment == other.environment &&
174
+ directory == other.directory
175
+ end
176
+
177
+ # Generate a cache key for this request.
178
+ #
179
+ # @return [String] cache key based on command, arguments, and context
180
+ def cache_key
181
+ require "digest"
182
+ content = "#{command}:#{arguments.join(":")}:#{directory}"
183
+ Digest::SHA256.hexdigest(content)[0..16]
184
+ end
185
+
186
+ def exit_on_failure?
187
+ @exit_on_failure
188
+ end
189
+
190
+ def exit_on_failure(exit_on_failure)
191
+ @exit_on_failure = exit_on_failure
192
+ self
193
+ end
194
+
195
+ def show_stderr?
196
+ @show_stderr
197
+ end
198
+
199
+ def show_stderr(show_stderr)
200
+ @show_stderr = show_stderr
201
+ self
202
+ end
203
+
204
+ def show_stdout?
205
+ @show_stdout
206
+ end
207
+
208
+ def show_stdout(show_stdout)
209
+ @show_stdout = show_stdout
210
+ self
211
+ end
212
+
213
+ # Execute the command.
214
+ #
215
+ # @return [Result] execution result
216
+ # @raise [ArgumentError] if runner is not set
217
+ def run
218
+ # self.exit_on_failure = exit_on_failure
219
+ # self.show_stderr = true
220
+ result = Makit::Commands::Runner.default.execute(show_stderr(true))
221
+
222
+ # Display stderr if the command failed and show_stderr is enabled
223
+ puts result.stderr if result.failure? && show_stderr? && !result.stderr.empty?
224
+
225
+ # Exit with the command's exit code if it failed and exit_on_failure is enabled
226
+ exit result.exit_code if result.failure? && exit_on_failure?
227
+
228
+ result
229
+ end
230
+
231
+ def try
232
+ Makit::Commands::Runner.default.execute(exit_on_failure(false))
233
+ end
234
+
235
+ private
236
+
237
+ # Validate command parameter.
238
+ #
239
+ # @param command [String] command to validate
240
+ # @return [String] validated command
241
+ # @raise [ArgumentError] if command is invalid
242
+ def validate_command(command)
243
+ raise ArgumentError, "Command cannot be nil" if command.nil?
244
+ raise ArgumentError, "Command cannot be empty" if command.strip.empty?
245
+
246
+ cmd = command.strip
247
+
248
+ # Check for dangerous characters that could indicate command injection
249
+ raise ArgumentError, "Command contains potentially dangerous characters" if cmd.match?(/[;&|`$(){}]/)
250
+
251
+ cmd
252
+ end
253
+
254
+ # Validate arguments array.
255
+ #
256
+ # @param arguments [Array] arguments to validate
257
+ # @return [Array<String>] validated arguments
258
+ # @raise [ArgumentError] if arguments are invalid
259
+ def validate_arguments(arguments)
260
+ return [] if arguments.nil?
261
+ raise ArgumentError, "Arguments must be an array" unless arguments.is_a?(Array)
262
+
263
+ # Convert all arguments to strings and validate
264
+ arguments.map do |arg|
265
+ raise ArgumentError, "Argument cannot be nil" if arg.nil?
266
+
267
+ arg.to_s
268
+ end
269
+ end
270
+
271
+ # Validate directory parameter.
272
+ #
273
+ # @param directory [String] directory to validate
274
+ # @raise [ArgumentError] if directory is invalid
275
+ def validate_directory(directory)
276
+ return unless directory
277
+
278
+ raise ArgumentError, "Directory must be a string" unless directory.is_a?(String)
279
+
280
+ # NOTE: We don't validate directory existence here as it might be created later
281
+ # We just validate it's a reasonable path format
282
+ return unless directory.include?("\0")
283
+
284
+ raise ArgumentError, "Directory path contains null bytes"
285
+ end
286
+
287
+ # Validate timeout parameter.
288
+ #
289
+ # @param timeout [Integer] timeout to validate
290
+ # @raise [ArgumentError] if timeout is invalid
291
+ def validate_timeout(timeout)
292
+ raise ArgumentError, "Timeout must be a positive integer" unless timeout.is_a?(Integer) && timeout.positive?
293
+
294
+ return unless timeout > 3600 # 1 hour max
295
+
296
+ raise ArgumentError, "Timeout cannot exceed 3600 seconds"
297
+ end
298
+
299
+ # Normalize hash keys to support both string and symbol keys.
300
+ #
301
+ # @param hash [Hash] hash to normalize
302
+ # @return [Hash] hash with normalized keys
303
+ def self.normalize_hash_keys(hash)
304
+ normalized = {}
305
+ hash.each do |key, value|
306
+ normalized_key = key.is_a?(String) ? key.to_sym : key
307
+ normalized[key] = value # Keep original key
308
+ normalized[normalized_key] = value # Add symbol version
309
+ end
310
+ normalized
311
+ end
312
+
313
+ private_class_method :normalize_hash_keys
314
+ end
315
+ end
316
+ end