makit 0.0.144 → 0.0.145

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 (165) 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/cli/base.rb +17 -0
  7. data/lib/makit/cli/build_commands.rb +500 -500
  8. data/lib/makit/cli/generators/base_generator.rb +74 -74
  9. data/lib/makit/cli/generators/dotnet_generator.rb +50 -50
  10. data/lib/makit/cli/generators/generator_factory.rb +49 -49
  11. data/lib/makit/cli/generators/node_generator.rb +50 -50
  12. data/lib/makit/cli/generators/ruby_generator.rb +77 -77
  13. data/lib/makit/cli/generators/rust_generator.rb +50 -50
  14. data/lib/makit/cli/generators/templates/dotnet_templates.rb +167 -167
  15. data/lib/makit/cli/generators/templates/node_templates.rb +161 -161
  16. data/lib/makit/cli/generators/templates/ruby/gemfile.rb +26 -26
  17. data/lib/makit/cli/generators/templates/ruby/gemspec.rb +41 -40
  18. data/lib/makit/cli/generators/templates/ruby/main_lib.rb +33 -33
  19. data/lib/makit/cli/generators/templates/ruby/rakefile.rb +35 -35
  20. data/lib/makit/cli/generators/templates/ruby/readme.rb +63 -63
  21. data/lib/makit/cli/generators/templates/ruby/test.rb +39 -39
  22. data/lib/makit/cli/generators/templates/ruby/test_helper.rb +29 -29
  23. data/lib/makit/cli/generators/templates/ruby/version.rb +29 -29
  24. data/lib/makit/cli/generators/templates/rust_templates.rb +128 -128
  25. data/lib/makit/cli/main.rb +78 -69
  26. data/lib/makit/cli/pipeline_commands.rb +311 -0
  27. data/lib/makit/cli/project_commands.rb +868 -868
  28. data/lib/makit/cli/repository_commands.rb +661 -661
  29. data/lib/makit/cli/strategy_commands.rb +207 -212
  30. data/lib/makit/cli/utility_commands.rb +521 -521
  31. data/lib/makit/commands/factory.rb +359 -359
  32. data/lib/makit/commands/middleware/base.rb +73 -73
  33. data/lib/makit/commands/middleware/cache.rb +248 -248
  34. data/lib/makit/commands/middleware/command_logger.rb +312 -312
  35. data/lib/makit/commands/middleware/validator.rb +269 -269
  36. data/lib/makit/commands/request.rb +316 -316
  37. data/lib/makit/commands/result.rb +323 -323
  38. data/lib/makit/commands/runner.rb +386 -386
  39. data/lib/makit/commands/strategies/base.rb +171 -171
  40. data/lib/makit/commands/strategies/child_process.rb +162 -162
  41. data/lib/makit/commands/strategies/factory.rb +136 -136
  42. data/lib/makit/commands/strategies/synchronous.rb +139 -139
  43. data/lib/makit/commands.rb +50 -50
  44. data/lib/makit/configuration/dotnet_project.rb +48 -48
  45. data/lib/makit/configuration/gitlab_helper.rb +61 -58
  46. data/lib/makit/configuration/project.rb +446 -168
  47. data/lib/makit/configuration/rakefile_helper.rb +43 -43
  48. data/lib/makit/configuration/step.rb +34 -34
  49. data/lib/makit/configuration/timeout.rb +74 -74
  50. data/lib/makit/configuration.rb +21 -16
  51. data/lib/makit/content/default_gitignore.rb +7 -7
  52. data/lib/makit/content/default_gitignore.txt +225 -225
  53. data/lib/makit/content/default_rakefile.rb +13 -13
  54. data/lib/makit/content/gem_rakefile.rb +16 -16
  55. data/lib/makit/context.rb +1 -1
  56. data/lib/makit/data.rb +49 -49
  57. data/lib/makit/directories.rb +140 -140
  58. data/lib/makit/directory.rb +262 -262
  59. data/lib/makit/docs/files.rb +89 -89
  60. data/lib/makit/docs/rake.rb +102 -102
  61. data/lib/makit/dotnet/cli.rb +69 -69
  62. data/lib/makit/dotnet/project.rb +217 -217
  63. data/lib/makit/dotnet/solution.rb +38 -38
  64. data/lib/makit/dotnet/solution_classlib.rb +239 -239
  65. data/lib/makit/dotnet/solution_console.rb +264 -264
  66. data/lib/makit/dotnet/solution_maui.rb +354 -354
  67. data/lib/makit/dotnet/solution_wasm.rb +275 -275
  68. data/lib/makit/dotnet/solution_wpf.rb +304 -304
  69. data/lib/makit/dotnet.rb +102 -102
  70. data/lib/makit/email.rb +90 -90
  71. data/lib/makit/environment.rb +142 -142
  72. data/lib/makit/examples/runner.rb +370 -370
  73. data/lib/makit/exceptions.rb +45 -45
  74. data/lib/makit/fileinfo.rb +32 -24
  75. data/lib/makit/files.rb +43 -43
  76. data/lib/makit/gems.rb +40 -40
  77. data/lib/makit/git/cli.rb +54 -54
  78. data/lib/makit/git/repository.rb +266 -90
  79. data/lib/makit/git.rb +104 -98
  80. data/lib/makit/gitlab/pipeline.rb +857 -0
  81. data/lib/makit/gitlab/pipeline_service_impl.rb +1536 -0
  82. data/lib/makit/gitlab_runner.rb +59 -59
  83. data/lib/makit/humanize.rb +218 -137
  84. data/lib/makit/indexer.rb +47 -47
  85. data/lib/makit/io/filesystem.rb +111 -0
  86. data/lib/makit/io/filesystem_service_impl.rb +337 -0
  87. data/lib/makit/logging/configuration.rb +308 -308
  88. data/lib/makit/logging/format_registry.rb +84 -84
  89. data/lib/makit/logging/formatters/base.rb +39 -39
  90. data/lib/makit/logging/formatters/console_formatter.rb +140 -140
  91. data/lib/makit/logging/formatters/json_formatter.rb +65 -65
  92. data/lib/makit/logging/formatters/plain_text_formatter.rb +71 -71
  93. data/lib/makit/logging/formatters/text_formatter.rb +64 -64
  94. data/lib/makit/logging/log_request.rb +119 -119
  95. data/lib/makit/logging/logger.rb +199 -199
  96. data/lib/makit/logging/sinks/base.rb +91 -91
  97. data/lib/makit/logging/sinks/console.rb +72 -72
  98. data/lib/makit/logging/sinks/file_sink.rb +92 -92
  99. data/lib/makit/logging/sinks/structured.rb +123 -123
  100. data/lib/makit/logging/sinks/unified_file_sink.rb +296 -296
  101. data/lib/makit/logging.rb +565 -565
  102. data/lib/makit/markdown.rb +75 -75
  103. data/lib/makit/mp/basic_object_mp.rb +17 -17
  104. data/lib/makit/mp/command_mp.rb +13 -13
  105. data/lib/makit/mp/command_request.mp.rb +17 -17
  106. data/lib/makit/mp/project_mp.rb +199 -199
  107. data/lib/makit/mp/string_mp.rb +205 -199
  108. data/lib/makit/nuget.rb +74 -74
  109. data/lib/makit/podman/podman.rb +458 -0
  110. data/lib/makit/podman/podman_service_impl.rb +1081 -0
  111. data/lib/makit/port.rb +32 -32
  112. data/lib/makit/process.rb +377 -377
  113. data/lib/makit/protoc.rb +112 -107
  114. data/lib/makit/rake/cli.rb +196 -196
  115. data/lib/makit/rake/trace_controller.rb +174 -174
  116. data/lib/makit/rake.rb +81 -81
  117. data/lib/makit/ruby/cli.rb +185 -185
  118. data/lib/makit/ruby.rb +25 -25
  119. data/lib/makit/secrets.rb +51 -51
  120. data/lib/makit/serializer.rb +130 -130
  121. data/lib/makit/services/builder.rb +186 -186
  122. data/lib/makit/services/error_handler.rb +226 -226
  123. data/lib/makit/services/repository_manager.rb +367 -231
  124. data/lib/makit/services/validator.rb +112 -112
  125. data/lib/makit/setup/classlib.rb +101 -101
  126. data/lib/makit/setup/gem.rb +268 -268
  127. data/lib/makit/setup/pages.rb +11 -11
  128. data/lib/makit/setup/razorclasslib.rb +101 -101
  129. data/lib/makit/setup/runner.rb +54 -54
  130. data/lib/makit/setup.rb +5 -5
  131. data/lib/makit/show.rb +110 -110
  132. data/lib/makit/storage.rb +126 -126
  133. data/lib/makit/symbols.rb +175 -170
  134. data/lib/makit/task_info.rb +130 -130
  135. data/lib/makit/tasks/at_exit.rb +15 -15
  136. data/lib/makit/tasks/build.rb +22 -22
  137. data/lib/makit/tasks/clean.rb +13 -13
  138. data/lib/makit/tasks/configure.rb +10 -10
  139. data/lib/makit/tasks/format.rb +10 -10
  140. data/lib/makit/tasks/hook_manager.rb +443 -443
  141. data/lib/makit/tasks/init.rb +49 -49
  142. data/lib/makit/tasks/integrate.rb +29 -29
  143. data/lib/makit/tasks/pull_incoming.rb +13 -13
  144. data/lib/makit/tasks/setup.rb +16 -16
  145. data/lib/makit/tasks/sync.rb +17 -17
  146. data/lib/makit/tasks/tag.rb +16 -16
  147. data/lib/makit/tasks/task_monkey_patch.rb +81 -81
  148. data/lib/makit/tasks/test.rb +22 -22
  149. data/lib/makit/tasks/update.rb +18 -18
  150. data/lib/makit/tasks.rb +20 -20
  151. data/lib/makit/test_cache.rb +239 -239
  152. data/lib/makit/tree.rb +37 -37
  153. data/lib/makit/v1/configuration/project_service_impl.rb +371 -0
  154. data/lib/makit/v1/git/git_repository_service_impl.rb +295 -0
  155. data/lib/makit/v1/makit.v1_pb.rb +35 -35
  156. data/lib/makit/v1/makit.v1_services_pb.rb +27 -27
  157. data/lib/makit/v1/services/repository_manager_service_impl.rb +572 -0
  158. data/lib/makit/version.rb +100 -100
  159. data/lib/makit/version_util.rb +21 -21
  160. data/lib/makit/wix.rb +95 -95
  161. data/lib/makit/yaml.rb +29 -29
  162. data/lib/makit/zip.rb +17 -17
  163. data/lib/makit copy.rb +44 -44
  164. data/lib/makit.rb +111 -43
  165. metadata +61 -36
@@ -1,239 +1,239 @@
1
- # frozen_string_literal: true
2
-
3
- module Makit
4
- class TestCache
5
- # Cache configuration
6
- CACHE_BASE_DIR = File.join(Makit::Directories::PROJECT_ARTIFACTS, "test_cache").freeze
7
- CACHE_METADATA_FILE = ".cache_metadata.json"
8
- DEFAULT_TTL = 3600 # 1 hour cache by default
9
-
10
- class << self
11
- def setup
12
- FileUtils.mkdir_p(CACHE_BASE_DIR)
13
- end
14
-
15
- # Main caching method - wraps expensive test operations
16
- def with_cache(test_instance, cache_key, dependencies: [], ttl: DEFAULT_TTL, &block)
17
- return yield unless should_cache?
18
-
19
- cache_dir = cache_directory_for(test_instance, cache_key)
20
- metadata = read_cache_metadata(cache_dir)
21
-
22
- if cache_valid?(cache_dir, metadata, dependencies, ttl)
23
- skip_with_cache_message(test_instance, cache_key, metadata)
24
- return :cached
25
- end
26
-
27
- # Cache miss or invalid - run the expensive operation
28
- cleanup_cache_directory(cache_dir)
29
- FileUtils.mkdir_p(cache_dir)
30
-
31
- # Execute the block in the cache directory context
32
- result = Dir.chdir(cache_dir, &block)
33
-
34
- # Write cache metadata
35
- write_cache_metadata(cache_dir, dependencies)
36
-
37
- result
38
- end
39
-
40
- # Directory-based caching - checks if directory exists and is valid
41
- def cache_directory(test_instance, cache_key, dependencies: [], ttl: DEFAULT_TTL)
42
- return nil unless should_cache?
43
-
44
- cache_dir = cache_directory_for(test_instance, cache_key)
45
- metadata = read_cache_metadata(cache_dir)
46
-
47
- if cache_valid?(cache_dir, metadata, dependencies, ttl)
48
- skip_with_cache_message(test_instance, cache_key, metadata)
49
- return cache_dir
50
- end
51
-
52
- # Prepare fresh cache directory
53
- cleanup_cache_directory(cache_dir)
54
- FileUtils.mkdir_p(cache_dir)
55
-
56
- cache_dir
57
- end
58
-
59
- # File-based caching - checks if specific files exist and are valid
60
- def cache_files(test_instance, cache_key, expected_files, dependencies: [], ttl: DEFAULT_TTL)
61
- return false unless should_cache?
62
-
63
- cache_dir = cache_directory_for(test_instance, cache_key)
64
- metadata = read_cache_metadata(cache_dir)
65
-
66
- # Check if all expected files exist
67
- files_exist = expected_files.all? do |file|
68
- File.exist?(File.join(cache_dir, file))
69
- end
70
-
71
- if files_exist && cache_valid?(cache_dir, metadata, dependencies, ttl)
72
- skip_with_cache_message(test_instance, cache_key, metadata)
73
- return true
74
- end
75
-
76
- false
77
- end
78
-
79
- # Clear all test caches
80
- def clear_all
81
- FileUtils.rm_rf(CACHE_BASE_DIR)
82
- puts "🗑️ Cleared all test caches".colorize(:grey)
83
- end
84
-
85
- # Clear caches for specific test class
86
- def clear_for_class(test_class)
87
- class_cache_dir = File.join(CACHE_BASE_DIR, class_cache_key(test_class))
88
- FileUtils.rm_rf(class_cache_dir)
89
- puts "🗑️ Cleared test caches for #{test_class}".colorize(:grey)
90
- end
91
-
92
- # Get cache statistics
93
- def cache_stats
94
- return { total_size: 0, cache_count: 0, cache_dirs: [] } unless Dir.exist?(CACHE_BASE_DIR)
95
-
96
- cache_dirs = Dir.glob(File.join(CACHE_BASE_DIR, "**", "*")).select { |d| Dir.exist?(d) }
97
- total_size = cache_dirs.sum { |dir| directory_size(dir) }
98
-
99
- {
100
- total_size: total_size,
101
- cache_count: cache_dirs.length,
102
- cache_dirs: cache_dirs.map { |d| d.sub("#{CACHE_BASE_DIR}/", "") },
103
- }
104
- end
105
-
106
- private
107
-
108
- def should_cache?
109
- # Don't cache in CI environments by default (ensures clean runs)
110
- return false if ENV["CI"] == "true"
111
-
112
- # Don't cache if explicitly disabled
113
- return false if ENV["MAKIT_DISABLE_TEST_CACHE"] == "true"
114
-
115
- # Don't cache during test development/debugging
116
- return false if ENV["MAKIT_TEST_DEBUG"] == "true"
117
-
118
- # Don't cache in test environment unless explicitly testing the cache
119
- return false if ENV["MAKIT_TESTING"] == "true" && ENV["MAKIT_TESTING_CACHE"] != "true"
120
-
121
- true
122
- end
123
-
124
- def cache_directory_for(test_instance, cache_key)
125
- test_class = test_instance.class.name.gsub("::", "/")
126
- test_method = test_instance.name
127
-
128
- File.join(CACHE_BASE_DIR, test_class, test_method, cache_key)
129
- end
130
-
131
- def class_cache_key(test_class)
132
- test_class.name.gsub("::", "/")
133
- end
134
-
135
- def cache_valid?(cache_dir, metadata, dependencies, ttl)
136
- return false unless Dir.exist?(cache_dir)
137
- return false unless metadata
138
-
139
- # Check TTL
140
- cache_age = Time.now - Time.parse(metadata["created_at"])
141
- return false if cache_age > ttl
142
-
143
- # Check dependencies
144
- return false unless dependencies_current?(metadata["dependencies"], dependencies)
145
-
146
- # Check if cache directory has content
147
- entries = Dir.entries(cache_dir) - [".", "..", CACHE_METADATA_FILE]
148
- entries.any?
149
- end
150
-
151
- def dependencies_current?(cached_deps, current_deps)
152
- return true if current_deps.empty?
153
-
154
- current_deps.each do |dep_file|
155
- next unless File.exist?(dep_file)
156
-
157
- current_mtime = File.mtime(dep_file).to_i
158
- cached_mtime = cached_deps&.dig(dep_file)
159
-
160
- return false if cached_mtime.nil? || current_mtime > cached_mtime
161
- end
162
-
163
- true
164
- end
165
-
166
- def read_cache_metadata(cache_dir)
167
- metadata_file = File.join(cache_dir, CACHE_METADATA_FILE)
168
- return nil unless File.exist?(metadata_file)
169
-
170
- JSON.parse(File.read(metadata_file))
171
- rescue JSON::ParserError
172
- nil
173
- end
174
-
175
- def write_cache_metadata(cache_dir, dependencies)
176
- metadata = {
177
- created_at: Time.now.iso8601,
178
- dependencies: dependencies.each_with_object({}) do |dep_file, hash|
179
- hash[dep_file] = File.exist?(dep_file) ? File.mtime(dep_file).to_i : nil
180
- end,
181
- }
182
-
183
- metadata_file = File.join(cache_dir, CACHE_METADATA_FILE)
184
- File.write(metadata_file, JSON.pretty_generate(metadata))
185
- end
186
-
187
- def skip_with_cache_message(test_instance, cache_key, metadata)
188
- cache_age = Time.now - Time.parse(metadata["created_at"])
189
- age_text = format_duration(cache_age)
190
-
191
- test_instance.skip("💨 Using cached result for '#{cache_key}' (cached #{age_text} ago)")
192
- end
193
-
194
- def cleanup_cache_directory(cache_dir)
195
- FileUtils.rm_rf(cache_dir)
196
- end
197
-
198
- def directory_size(dir_path)
199
- Dir.glob(File.join(dir_path, "**", "*"))
200
- .select { |f| File.file?(f) }
201
- .sum { |f| File.size(f) }
202
- end
203
-
204
- def format_duration(seconds)
205
- if seconds < 60
206
- "#{seconds.round}s"
207
- elsif seconds < 3600
208
- minutes = (seconds / 60).round
209
- "#{minutes}m"
210
- else
211
- hours = (seconds / 3600).round
212
- "#{hours}h"
213
- end
214
- end
215
- end
216
- end
217
-
218
- # Mixin module for test classes to provide caching helpers
219
- module TestCacheHelpers
220
- def cache_expensive_operation(cache_key, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL, &block)
221
- Makit::TestCache.with_cache(self, cache_key,
222
- dependencies: dependencies,
223
- ttl: ttl,
224
- &block)
225
- end
226
-
227
- def cached_directory(cache_key, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL)
228
- Makit::TestCache.cache_directory(self, cache_key,
229
- dependencies: dependencies,
230
- ttl: ttl)
231
- end
232
-
233
- def cached_files_exist?(cache_key, expected_files, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL)
234
- Makit::TestCache.cache_files(self, cache_key, expected_files,
235
- dependencies: dependencies,
236
- ttl: ttl)
237
- end
238
- end
239
- end
1
+ # frozen_string_literal: true
2
+
3
+ module Makit
4
+ class TestCache
5
+ # Cache configuration
6
+ CACHE_BASE_DIR = File.join(Makit::Directories::PROJECT_ARTIFACTS, "test_cache").freeze
7
+ CACHE_METADATA_FILE = ".cache_metadata.json"
8
+ DEFAULT_TTL = 3600 # 1 hour cache by default
9
+
10
+ class << self
11
+ def setup
12
+ FileUtils.mkdir_p(CACHE_BASE_DIR)
13
+ end
14
+
15
+ # Main caching method - wraps expensive test operations
16
+ def with_cache(test_instance, cache_key, dependencies: [], ttl: DEFAULT_TTL, &block)
17
+ return yield unless should_cache?
18
+
19
+ cache_dir = cache_directory_for(test_instance, cache_key)
20
+ metadata = read_cache_metadata(cache_dir)
21
+
22
+ if cache_valid?(cache_dir, metadata, dependencies, ttl)
23
+ skip_with_cache_message(test_instance, cache_key, metadata)
24
+ return :cached
25
+ end
26
+
27
+ # Cache miss or invalid - run the expensive operation
28
+ cleanup_cache_directory(cache_dir)
29
+ FileUtils.mkdir_p(cache_dir)
30
+
31
+ # Execute the block in the cache directory context
32
+ result = Dir.chdir(cache_dir, &block)
33
+
34
+ # Write cache metadata
35
+ write_cache_metadata(cache_dir, dependencies)
36
+
37
+ result
38
+ end
39
+
40
+ # Directory-based caching - checks if directory exists and is valid
41
+ def cache_directory(test_instance, cache_key, dependencies: [], ttl: DEFAULT_TTL)
42
+ return nil unless should_cache?
43
+
44
+ cache_dir = cache_directory_for(test_instance, cache_key)
45
+ metadata = read_cache_metadata(cache_dir)
46
+
47
+ if cache_valid?(cache_dir, metadata, dependencies, ttl)
48
+ skip_with_cache_message(test_instance, cache_key, metadata)
49
+ return cache_dir
50
+ end
51
+
52
+ # Prepare fresh cache directory
53
+ cleanup_cache_directory(cache_dir)
54
+ FileUtils.mkdir_p(cache_dir)
55
+
56
+ cache_dir
57
+ end
58
+
59
+ # File-based caching - checks if specific files exist and are valid
60
+ def cache_files(test_instance, cache_key, expected_files, dependencies: [], ttl: DEFAULT_TTL)
61
+ return false unless should_cache?
62
+
63
+ cache_dir = cache_directory_for(test_instance, cache_key)
64
+ metadata = read_cache_metadata(cache_dir)
65
+
66
+ # Check if all expected files exist
67
+ files_exist = expected_files.all? do |file|
68
+ File.exist?(File.join(cache_dir, file))
69
+ end
70
+
71
+ if files_exist && cache_valid?(cache_dir, metadata, dependencies, ttl)
72
+ skip_with_cache_message(test_instance, cache_key, metadata)
73
+ return true
74
+ end
75
+
76
+ false
77
+ end
78
+
79
+ # Clear all test caches
80
+ def clear_all
81
+ FileUtils.rm_rf(CACHE_BASE_DIR)
82
+ puts "🗑️ Cleared all test caches".colorize(:grey)
83
+ end
84
+
85
+ # Clear caches for specific test class
86
+ def clear_for_class(test_class)
87
+ class_cache_dir = File.join(CACHE_BASE_DIR, class_cache_key(test_class))
88
+ FileUtils.rm_rf(class_cache_dir)
89
+ puts "🗑️ Cleared test caches for #{test_class}".colorize(:grey)
90
+ end
91
+
92
+ # Get cache statistics
93
+ def cache_stats
94
+ return { total_size: 0, cache_count: 0, cache_dirs: [] } unless Dir.exist?(CACHE_BASE_DIR)
95
+
96
+ cache_dirs = Dir.glob(File.join(CACHE_BASE_DIR, "**", "*")).select { |d| Dir.exist?(d) }
97
+ total_size = cache_dirs.sum { |dir| directory_size(dir) }
98
+
99
+ {
100
+ total_size: total_size,
101
+ cache_count: cache_dirs.length,
102
+ cache_dirs: cache_dirs.map { |d| d.sub("#{CACHE_BASE_DIR}/", "") },
103
+ }
104
+ end
105
+
106
+ private
107
+
108
+ def should_cache?
109
+ # Don't cache in CI environments by default (ensures clean runs)
110
+ return false if ENV["CI"] == "true"
111
+
112
+ # Don't cache if explicitly disabled
113
+ return false if ENV["MAKIT_DISABLE_TEST_CACHE"] == "true"
114
+
115
+ # Don't cache during test development/debugging
116
+ return false if ENV["MAKIT_TEST_DEBUG"] == "true"
117
+
118
+ # Don't cache in test environment unless explicitly testing the cache
119
+ return false if ENV["MAKIT_TESTING"] == "true" && ENV["MAKIT_TESTING_CACHE"] != "true"
120
+
121
+ true
122
+ end
123
+
124
+ def cache_directory_for(test_instance, cache_key)
125
+ test_class = test_instance.class.name.gsub("::", "/")
126
+ test_method = test_instance.name
127
+
128
+ File.join(CACHE_BASE_DIR, test_class, test_method, cache_key)
129
+ end
130
+
131
+ def class_cache_key(test_class)
132
+ test_class.name.gsub("::", "/")
133
+ end
134
+
135
+ def cache_valid?(cache_dir, metadata, dependencies, ttl)
136
+ return false unless Dir.exist?(cache_dir)
137
+ return false unless metadata
138
+
139
+ # Check TTL
140
+ cache_age = Time.now - Time.parse(metadata["created_at"])
141
+ return false if cache_age > ttl
142
+
143
+ # Check dependencies
144
+ return false unless dependencies_current?(metadata["dependencies"], dependencies)
145
+
146
+ # Check if cache directory has content
147
+ entries = Dir.entries(cache_dir) - [".", "..", CACHE_METADATA_FILE]
148
+ entries.any?
149
+ end
150
+
151
+ def dependencies_current?(cached_deps, current_deps)
152
+ return true if current_deps.empty?
153
+
154
+ current_deps.each do |dep_file|
155
+ next unless File.exist?(dep_file)
156
+
157
+ current_mtime = File.mtime(dep_file).to_i
158
+ cached_mtime = cached_deps&.dig(dep_file)
159
+
160
+ return false if cached_mtime.nil? || current_mtime > cached_mtime
161
+ end
162
+
163
+ true
164
+ end
165
+
166
+ def read_cache_metadata(cache_dir)
167
+ metadata_file = File.join(cache_dir, CACHE_METADATA_FILE)
168
+ return nil unless File.exist?(metadata_file)
169
+
170
+ JSON.parse(File.read(metadata_file))
171
+ rescue JSON::ParserError
172
+ nil
173
+ end
174
+
175
+ def write_cache_metadata(cache_dir, dependencies)
176
+ metadata = {
177
+ created_at: Time.now.iso8601,
178
+ dependencies: dependencies.each_with_object({}) do |dep_file, hash|
179
+ hash[dep_file] = File.exist?(dep_file) ? File.mtime(dep_file).to_i : nil
180
+ end,
181
+ }
182
+
183
+ metadata_file = File.join(cache_dir, CACHE_METADATA_FILE)
184
+ File.write(metadata_file, JSON.pretty_generate(metadata))
185
+ end
186
+
187
+ def skip_with_cache_message(test_instance, cache_key, metadata)
188
+ cache_age = Time.now - Time.parse(metadata["created_at"])
189
+ age_text = format_duration(cache_age)
190
+
191
+ test_instance.skip("💨 Using cached result for '#{cache_key}' (cached #{age_text} ago)")
192
+ end
193
+
194
+ def cleanup_cache_directory(cache_dir)
195
+ FileUtils.rm_rf(cache_dir)
196
+ end
197
+
198
+ def directory_size(dir_path)
199
+ Dir.glob(File.join(dir_path, "**", "*"))
200
+ .select { |f| File.file?(f) }
201
+ .sum { |f| File.size(f) }
202
+ end
203
+
204
+ def format_duration(seconds)
205
+ if seconds < 60
206
+ "#{seconds.round}s"
207
+ elsif seconds < 3600
208
+ minutes = (seconds / 60).round
209
+ "#{minutes}m"
210
+ else
211
+ hours = (seconds / 3600).round
212
+ "#{hours}h"
213
+ end
214
+ end
215
+ end
216
+ end
217
+
218
+ # Mixin module for test classes to provide caching helpers
219
+ module TestCacheHelpers
220
+ def cache_expensive_operation(cache_key, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL, &block)
221
+ Makit::TestCache.with_cache(self, cache_key,
222
+ dependencies: dependencies,
223
+ ttl: ttl,
224
+ &block)
225
+ end
226
+
227
+ def cached_directory(cache_key, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL)
228
+ Makit::TestCache.cache_directory(self, cache_key,
229
+ dependencies: dependencies,
230
+ ttl: ttl)
231
+ end
232
+
233
+ def cached_files_exist?(cache_key, expected_files, dependencies: [], ttl: Makit::TestCache::DEFAULT_TTL)
234
+ Makit::TestCache.cache_files(self, cache_key, expected_files,
235
+ dependencies: dependencies,
236
+ ttl: ttl)
237
+ end
238
+ end
239
+ end
data/lib/makit/tree.rb CHANGED
@@ -1,37 +1,37 @@
1
- # frozen_string_literal: true
2
-
3
- # This module provides classes for the Makit gem.
4
- module Makit
5
- class Tree
6
- def get_source_tree
7
- files = `git ls-files`.split("\n")
8
- tree = {}
9
-
10
- # Build a nested hash representing the file structure
11
- files.each do |file|
12
- parts = file.split("/")
13
- current = tree
14
-
15
- parts.each_with_index do |part, index|
16
- current[part] ||= (index == parts.length - 1 ? nil : {})
17
- current = current[part] unless current[part].nil?
18
- end
19
- end
20
-
21
- generate_tree_string(tree)
22
- end
23
-
24
- def generate_tree_string(tree, indent = "")
25
- result = ""
26
- tree.each do |key, value|
27
- if value.is_a?(Hash)
28
- result << "#{indent}#{key}/\n"
29
- result << generate_tree_string(value, "#{indent} ")
30
- else
31
- result << "#{indent}#{key}\n"
32
- end
33
- end
34
- result
35
- end
36
- end
37
- end
1
+ # frozen_string_literal: true
2
+
3
+ # This module provides classes for the Makit gem.
4
+ module Makit
5
+ class Tree
6
+ def get_source_tree
7
+ files = `git ls-files`.split("\n")
8
+ tree = {}
9
+
10
+ # Build a nested hash representing the file structure
11
+ files.each do |file|
12
+ parts = file.split("/")
13
+ current = tree
14
+
15
+ parts.each_with_index do |part, index|
16
+ current[part] ||= (index == parts.length - 1 ? nil : {})
17
+ current = current[part] unless current[part].nil?
18
+ end
19
+ end
20
+
21
+ generate_tree_string(tree)
22
+ end
23
+
24
+ def generate_tree_string(tree, indent = "")
25
+ result = ""
26
+ tree.each do |key, value|
27
+ if value.is_a?(Hash)
28
+ result << "#{indent}#{key}/\n"
29
+ result << generate_tree_string(value, "#{indent} ")
30
+ else
31
+ result << "#{indent}#{key}\n"
32
+ end
33
+ end
34
+ result
35
+ end
36
+ end
37
+ end