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,262 +1,262 @@
1
- # frozen_string_literal: true
2
-
3
- require "find"
4
- require "pathname"
5
-
6
- # This module provides classes for the Makit gem.
7
- module Makit
8
- # This class provide methods for working with Directories/
9
- #
10
- # Example:
11
- #
12
- # Makit::Directory.find_directory_with_pattern("/home/user", "*.rb")
13
- #
14
- class Directory
15
- def self.show_dir_size(dir)
16
- total_bytes = Dir.glob(File.join(dir, "**", "*"))
17
- .select { |f| File.file?(f) }
18
- .sum { |f| File.size(f) }
19
-
20
- human_size(total_bytes)
21
- puts " directory #{dir.to_s.colorize(:green)} size is #{human_size(total_bytes).to_s.colorize(:green)}"
22
- end
23
-
24
- def self.show_largest_files(dir, limit)
25
- puts " directory #{dir.to_s.colorize(:green)} #{limit} largest files"
26
- files = Dir.glob(File.join(dir, "**", "*"), File::FNM_DOTMATCH)
27
- .select { |f| File.file?(f) }
28
-
29
- sorted = files.map { |f| [f, File.size(f)] }
30
- .sort_by { |_, size| -size }
31
- .first(limit)
32
-
33
- sorted.each do |path, size|
34
- # right justify human_size to be at least 10 characters wide
35
- padded_size = human_size(size).rjust(10)
36
- puts " #{padded_size}" + " #{path}".colorize(:green)
37
- end
38
- end
39
-
40
- def human_size(bytes)
41
- units = %w[B KB MB GB TB]
42
- return "0 B" if bytes.zero?
43
-
44
- exp = (Math.log(bytes) / Math.log(1024)).to_i
45
- exp = units.size - 1 if exp >= units.size
46
-
47
- format("%.2f %s", bytes.to_f / (1024 ** exp), units[exp])
48
- end
49
-
50
- def self.human_size(bytes)
51
- units = %w[B KB MB GB TB]
52
- return "0 B" if bytes.zero?
53
-
54
- exp = (Math.log(bytes) / Math.log(1024)).to_i
55
- exp = units.size - 1 if exp >= units.size
56
-
57
- format("%.2f %s", bytes.to_f / (1024 ** exp), units[exp])
58
- end
59
-
60
- def self.get_line_count(file)
61
- line_count = 0
62
- File.foreach(file) { line_count += 1 }
63
- line_count
64
- end
65
-
66
- def self.find_directory_with_patterns(starting_directory, patterns)
67
- patterns.each do |pattern|
68
- result = find_directory_with_pattern(starting_directory, pattern)
69
- return result if Dir.exist?(result)
70
- end
71
-
72
- nil
73
- end
74
-
75
- def self.find_directory_with_pattern(starting_directory, pattern)
76
- current_directory = File.expand_path(starting_directory)
77
-
78
- loop do
79
- return current_directory if contains_pattern?(current_directory, pattern)
80
-
81
- parent_directory = File.dirname(current_directory)
82
- break if parent_directory == current_directory # Reached the root directory
83
-
84
- current_directory = parent_directory
85
- end
86
-
87
- nil
88
- end
89
-
90
- def self.contains_pattern?(directory, pattern)
91
- Dir.foreach(directory) do |entry|
92
- next if [".", ".."].include?(entry)
93
- return true if File.fnmatch(pattern, entry)
94
- end
95
- false
96
- end
97
-
98
- def self.get_size(directory)
99
- total_size = 0
100
-
101
- Find.find(directory) do |file|
102
- total_size += File.size(file) if File.file?(file)
103
- end
104
-
105
- total_size
106
- end
107
-
108
- def self.get_humanized_size(size_in_bytes, precision = 2); end
109
-
110
- def self.zip_source_files(directory_path, zip_file_name)
111
- raise ArgumentError, "Directory path cannot be nil" if directory_path.nil?
112
- raise ArgumentError, "Zip file name cannot be nil or empty" if zip_file_name.nil? || zip_file_name.strip.empty?
113
-
114
- raise ArgumentError, "Directory '#{directory_path}' does not exist." unless Dir.exist?(directory_path)
115
-
116
- tracked_files = get_git_tracked_files(directory_path)
117
-
118
- raise "No tracked files found in the directory." if tracked_files.empty?
119
-
120
- Zip::File.open(zip_file_name, Zip::File::CREATE) do |zipfile|
121
- tracked_files.each do |file|
122
- full_path = File.join(directory_path, file)
123
- zipfile.add(file, full_path) if File.exist?(full_path)
124
- end
125
- end
126
- end
127
-
128
- def self.get_git_tracked_files(directory_path)
129
- # raise "do not use this method"
130
- # output, status = Open3.capture2("git ls-files directory", chdir: directory_path)
131
- # raise "Failed to list git-tracked files" unless status.success?
132
- # command = "git ls-files #{directory_path}".run
133
-
134
- # command.output.split("\n")
135
- `git ls-files #{directory_path}`.split("\n")
136
- end
137
-
138
- def self.get_newest_git_file(directory_path)
139
- get_git_tracked_files(directory_path).select { |f| File.file?(f) }.max_by { |f| File.mtime(f) }
140
- end
141
-
142
- def self.get_newest_git_file_timestamp(directory_path)
143
- get_newest_git_file(directory_path).nil? ? Time.now : File.mtime(get_newest_git_file(directory_path))
144
- end
145
-
146
- def self.get_newest_file_timestamp(directory_path)
147
- newest_file = get_git_tracked_files(directory_path).select { |f| File.file?(f) }.max_by { |f| File.mtime(f) }
148
- newest_file.nil? ? Time.now : File.mtime(newest_file)
149
- end
150
-
151
- # Normalize the path by removing any leading or trailing slashes
152
- # and replacing any backslashes with forward slashes
153
- def self.normalize(path)
154
- path = path.gsub("\\", "/")
155
- # path = path[1..-1] if path.start_with?("/")
156
- path = path[0..-2] if path.end_with?("/")
157
- path
158
- end
159
-
160
- # for a given path, collect all the subdirectories that match a version pattern.
161
- # for example 2.57.0, 2.62, or 2.65.0
162
- def self.get_version_directories(path)
163
- version_directories = []
164
- Dir.foreach(path) do |entry|
165
- next if [".", ".."].include?(entry)
166
-
167
- version_directories << entry if entry.match?(/^\d+\.\d+\.\d+$/)
168
- end
169
- version_directories
170
- end
171
-
172
- def self.modified(path)
173
- get_newest_file(path)
174
- end
175
-
176
- # for a given path, get the most recent file change date for any file in the directory
177
- def self.get_latest_file_change_date(path)
178
- # loop over all files in the directory
179
- latest_date = nil
180
- Find.find(path) do |file|
181
- if File.file?(file)
182
- date = File.mtime(file)
183
- latest_date = date if latest_date.nil? || date > latest_date
184
- end
185
- end
186
- latest_date
187
- end
188
-
189
- # for a given path, return the filename of the most recently modified file
190
- def self.get_newest_file(path)
191
- newest_file = nil
192
- latest_date = nil
193
- if Dir.exist?(path)
194
- Find.find(path) do |file|
195
- next unless File.file?(file)
196
-
197
- date = File.mtime(file)
198
- latest_date = date if latest_date.nil? || date > latest_date
199
- newest_file = file if date == latest_date
200
- end
201
- end
202
- newest_file
203
- end
204
-
205
- def self.copy(src_dir, dst_dir, verbose = false)
206
- FileUtils.cp_r(src_dir, dst_dir, verbose: verbose)
207
- end
208
-
209
- def self.generate_manifest(dir)
210
- FileUtils.rm_f("#{dir}/manifest.txt")
211
- File.open("#{dir}/manifest.txt", "w") do |f|
212
- Dir.glob("#{dir}/**/*").each do |file|
213
- next if File.directory?(file)
214
-
215
- f.puts file.sub("#{dir}/", "")
216
- end
217
- end
218
- end
219
-
220
- def self.remove_empty_directories(dir)
221
- Find.find(dir) do |path|
222
- FileUtils.rm_rf(path) if File.directory?(path) && Dir.empty?(path)
223
- end
224
- end
225
-
226
- def self.remove_empty_directories_recursively(dir)
227
- Find.find(dir) do |path|
228
- FileUtils.rm_rf(path) if File.directory?(path) && Dir.empty?(path)
229
- end
230
- end
231
-
232
- # cleans all git repositories found in the given directory
233
- def self.deep_clean(dir)
234
- git_dirs = []
235
- # Dir.chdir(dir) do
236
- # scan all subdirectories for git repositories
237
- puts " scanning #{dir} for git repositories".colorize(:green)
238
- Find.find(dir) do |path|
239
- git_dirs << path if File.directory?(path) && File.exist?("#{path}/.git")
240
- end
241
- # end
242
- git_dirs.each do |path|
243
- puts " cleaning #{path} of untracked files".colorize(:green)
244
- Dir.chdir(path) do
245
- puts `git clean -fdx`
246
- end
247
- end
248
- end
249
-
250
- def self.get_relative_paths(directory_path, pattern = "**/*")
251
- relative_paths = []
252
-
253
- Dir.chdir(directory_path) do
254
- Dir.glob(pattern).each do |file|
255
- relative_paths << file
256
- end
257
- end
258
-
259
- relative_paths
260
- end
261
- end
262
- end
1
+ # frozen_string_literal: true
2
+
3
+ require "find"
4
+ require "pathname"
5
+
6
+ # This module provides classes for the Makit gem.
7
+ module Makit
8
+ # This class provide methods for working with Directories/
9
+ #
10
+ # Example:
11
+ #
12
+ # Makit::Directory.find_directory_with_pattern("/home/user", "*.rb")
13
+ #
14
+ class Directory
15
+ def self.show_dir_size(dir)
16
+ total_bytes = Dir.glob(File.join(dir, "**", "*"))
17
+ .select { |f| File.file?(f) }
18
+ .sum { |f| File.size(f) }
19
+
20
+ human_size(total_bytes)
21
+ puts " directory #{dir.to_s.colorize(:green)} size is #{human_size(total_bytes).to_s.colorize(:green)}"
22
+ end
23
+
24
+ def self.show_largest_files(dir, limit)
25
+ puts " directory #{dir.to_s.colorize(:green)} #{limit} largest files"
26
+ files = Dir.glob(File.join(dir, "**", "*"), File::FNM_DOTMATCH)
27
+ .select { |f| File.file?(f) }
28
+
29
+ sorted = files.map { |f| [f, File.size(f)] }
30
+ .sort_by { |_, size| -size }
31
+ .first(limit)
32
+
33
+ sorted.each do |path, size|
34
+ # right justify human_size to be at least 10 characters wide
35
+ padded_size = human_size(size).rjust(10)
36
+ puts " #{padded_size}" + " #{path}".colorize(:green)
37
+ end
38
+ end
39
+
40
+ def human_size(bytes)
41
+ units = %w[B KB MB GB TB]
42
+ return "0 B" if bytes.zero?
43
+
44
+ exp = (Math.log(bytes) / Math.log(1024)).to_i
45
+ exp = units.size - 1 if exp >= units.size
46
+
47
+ format("%.2f %s", bytes.to_f / (1024 ** exp), units[exp])
48
+ end
49
+
50
+ def self.human_size(bytes)
51
+ units = %w[B KB MB GB TB]
52
+ return "0 B" if bytes.zero?
53
+
54
+ exp = (Math.log(bytes) / Math.log(1024)).to_i
55
+ exp = units.size - 1 if exp >= units.size
56
+
57
+ format("%.2f %s", bytes.to_f / (1024 ** exp), units[exp])
58
+ end
59
+
60
+ def self.get_line_count(file)
61
+ line_count = 0
62
+ File.foreach(file) { line_count += 1 }
63
+ line_count
64
+ end
65
+
66
+ def self.find_directory_with_patterns(starting_directory, patterns)
67
+ patterns.each do |pattern|
68
+ result = find_directory_with_pattern(starting_directory, pattern)
69
+ return result if Dir.exist?(result)
70
+ end
71
+
72
+ nil
73
+ end
74
+
75
+ def self.find_directory_with_pattern(starting_directory, pattern)
76
+ current_directory = File.expand_path(starting_directory)
77
+
78
+ loop do
79
+ return current_directory if contains_pattern?(current_directory, pattern)
80
+
81
+ parent_directory = File.dirname(current_directory)
82
+ break if parent_directory == current_directory # Reached the root directory
83
+
84
+ current_directory = parent_directory
85
+ end
86
+
87
+ nil
88
+ end
89
+
90
+ def self.contains_pattern?(directory, pattern)
91
+ Dir.foreach(directory) do |entry|
92
+ next if [".", ".."].include?(entry)
93
+ return true if File.fnmatch(pattern, entry)
94
+ end
95
+ false
96
+ end
97
+
98
+ def self.get_size(directory)
99
+ total_size = 0
100
+
101
+ Find.find(directory) do |file|
102
+ total_size += File.size(file) if File.file?(file)
103
+ end
104
+
105
+ total_size
106
+ end
107
+
108
+ def self.get_humanized_size(size_in_bytes, precision = 2); end
109
+
110
+ def self.zip_source_files(directory_path, zip_file_name)
111
+ raise ArgumentError, "Directory path cannot be nil" if directory_path.nil?
112
+ raise ArgumentError, "Zip file name cannot be nil or empty" if zip_file_name.nil? || zip_file_name.strip.empty?
113
+
114
+ raise ArgumentError, "Directory '#{directory_path}' does not exist." unless Dir.exist?(directory_path)
115
+
116
+ tracked_files = get_git_tracked_files(directory_path)
117
+
118
+ raise "No tracked files found in the directory." if tracked_files.empty?
119
+
120
+ Zip::File.open(zip_file_name, Zip::File::CREATE) do |zipfile|
121
+ tracked_files.each do |file|
122
+ full_path = File.join(directory_path, file)
123
+ zipfile.add(file, full_path) if File.exist?(full_path)
124
+ end
125
+ end
126
+ end
127
+
128
+ def self.get_git_tracked_files(directory_path)
129
+ # raise "do not use this method"
130
+ # output, status = Open3.capture2("git ls-files directory", chdir: directory_path)
131
+ # raise "Failed to list git-tracked files" unless status.success?
132
+ # command = "git ls-files #{directory_path}".run
133
+
134
+ # command.output.split("\n")
135
+ `git ls-files #{directory_path}`.split("\n")
136
+ end
137
+
138
+ def self.get_newest_git_file(directory_path)
139
+ get_git_tracked_files(directory_path).select { |f| File.file?(f) }.max_by { |f| File.mtime(f) }
140
+ end
141
+
142
+ def self.get_newest_git_file_timestamp(directory_path)
143
+ get_newest_git_file(directory_path).nil? ? Time.now : File.mtime(get_newest_git_file(directory_path))
144
+ end
145
+
146
+ def self.get_newest_file_timestamp(directory_path)
147
+ newest_file = get_git_tracked_files(directory_path).select { |f| File.file?(f) }.max_by { |f| File.mtime(f) }
148
+ newest_file.nil? ? Time.now : File.mtime(newest_file)
149
+ end
150
+
151
+ # Normalize the path by removing any leading or trailing slashes
152
+ # and replacing any backslashes with forward slashes
153
+ def self.normalize(path)
154
+ path = path.gsub("\\", "/")
155
+ # path = path[1..-1] if path.start_with?("/")
156
+ path = path[0..-2] if path.end_with?("/")
157
+ path
158
+ end
159
+
160
+ # for a given path, collect all the subdirectories that match a version pattern.
161
+ # for example 2.57.0, 2.62, or 2.65.0
162
+ def self.get_version_directories(path)
163
+ version_directories = []
164
+ Dir.foreach(path) do |entry|
165
+ next if [".", ".."].include?(entry)
166
+
167
+ version_directories << entry if entry.match?(/^\d+\.\d+\.\d+$/)
168
+ end
169
+ version_directories
170
+ end
171
+
172
+ def self.modified(path)
173
+ get_newest_file(path)
174
+ end
175
+
176
+ # for a given path, get the most recent file change date for any file in the directory
177
+ def self.get_latest_file_change_date(path)
178
+ # loop over all files in the directory
179
+ latest_date = nil
180
+ Find.find(path) do |file|
181
+ if File.file?(file)
182
+ date = File.mtime(file)
183
+ latest_date = date if latest_date.nil? || date > latest_date
184
+ end
185
+ end
186
+ latest_date
187
+ end
188
+
189
+ # for a given path, return the filename of the most recently modified file
190
+ def self.get_newest_file(path)
191
+ newest_file = nil
192
+ latest_date = nil
193
+ if Dir.exist?(path)
194
+ Find.find(path) do |file|
195
+ next unless File.file?(file)
196
+
197
+ date = File.mtime(file)
198
+ latest_date = date if latest_date.nil? || date > latest_date
199
+ newest_file = file if date == latest_date
200
+ end
201
+ end
202
+ newest_file
203
+ end
204
+
205
+ def self.copy(src_dir, dst_dir, verbose = false)
206
+ FileUtils.cp_r(src_dir, dst_dir, verbose: verbose)
207
+ end
208
+
209
+ def self.generate_manifest(dir)
210
+ FileUtils.rm_f("#{dir}/manifest.txt")
211
+ File.open("#{dir}/manifest.txt", "w") do |f|
212
+ Dir.glob("#{dir}/**/*").each do |file|
213
+ next if File.directory?(file)
214
+
215
+ f.puts file.sub("#{dir}/", "")
216
+ end
217
+ end
218
+ end
219
+
220
+ def self.remove_empty_directories(dir)
221
+ Find.find(dir) do |path|
222
+ FileUtils.rm_rf(path) if File.directory?(path) && Dir.empty?(path)
223
+ end
224
+ end
225
+
226
+ def self.remove_empty_directories_recursively(dir)
227
+ Find.find(dir) do |path|
228
+ FileUtils.rm_rf(path) if File.directory?(path) && Dir.empty?(path)
229
+ end
230
+ end
231
+
232
+ # cleans all git repositories found in the given directory
233
+ def self.deep_clean(dir)
234
+ git_dirs = []
235
+ # Dir.chdir(dir) do
236
+ # scan all subdirectories for git repositories
237
+ puts " scanning #{dir} for git repositories".colorize(:green)
238
+ Find.find(dir) do |path|
239
+ git_dirs << path if File.directory?(path) && File.exist?("#{path}/.git")
240
+ end
241
+ # end
242
+ git_dirs.each do |path|
243
+ puts " cleaning #{path} of untracked files".colorize(:green)
244
+ Dir.chdir(path) do
245
+ puts `git clean -fdx`
246
+ end
247
+ end
248
+ end
249
+
250
+ def self.get_relative_paths(directory_path, pattern = "**/*")
251
+ relative_paths = []
252
+
253
+ Dir.chdir(directory_path) do
254
+ Dir.glob(pattern).each do |file|
255
+ relative_paths << file
256
+ end
257
+ end
258
+
259
+ relative_paths
260
+ end
261
+ end
262
+ end