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
data/lib/makit/nuget.rb CHANGED
@@ -1,74 +1,74 @@
1
- # frozen_string_literal: true
2
-
3
- require "tmpdir"
4
- # This module provides classes for the Makit gem.
5
- module Makit
6
- # This class provide methods for working with the Nuget package cache
7
- #
8
- # Example:
9
- #
10
- # Makit::Directory.cache("Google.Protobuf", "3.27.2")
11
- #
12
- # dotnet nuget locals all --list
13
- # dotnet nuget locals all --clear
14
- #
15
- class NuGet
16
- def self.get_cache_dir(package, version)
17
- File.join(Makit::Directories::NUGET_PACKAGE_CACHE, package, version)
18
- end
19
-
20
- def self.cache(package, version)
21
- # if the package is already cached, there is nothing to do
22
- return if Dir.exist?(get_cache_dir(package, version))
23
-
24
- Dir.mktmpdir do |dir|
25
- # Use the temp directory here
26
- Dir.chdir(dir) do
27
- system("dotnet new classlib -n ClassLib")
28
- Dir.chdir("ClassLib") do
29
- # display a list of directories in the current directory
30
- puts Dir.entries(Dir.pwd)
31
- # display a list of files in the current directory
32
- puts Dir.entries(Dir.pwd)
33
- puts "dotnet add ClassLib.csproj package #{package} --version #{version}"
34
- system("dotnet add ClassLib.csproj package #{package} --version #{version}")
35
- end
36
- end
37
- # The directory and its contents will be removed automatically after the block
38
- end
39
- end
40
-
41
- def self.clear_cache(package, version)
42
- return unless Dir.exist?(get_cache_dir(package, version))
43
-
44
- FileUtils.rm_rf(get_cache_dir(package, version))
45
- end
46
-
47
- # get the latest version of the package
48
- def self.get_latest_version(package)
49
- dir = File.join(Makit::Directories::NUGET_PACKAGE_CACHE, package)
50
- if Dir.exist?(dir)
51
- versions = Dir.entries(dir).select do |entry|
52
- File.directory?(File.join(dir, entry)) && ![".", ".."].include?(entry)
53
- end
54
- highest_version = Makit::Version.get_highest_version(versions)
55
- return highest_version
56
- end
57
- nil
58
- end
59
-
60
- # publish a package to a nuget directory feed
61
- def self.add_package(_package, path)
62
- system("dotnet nuget push #{path} --source #{path}")
63
- end
64
-
65
- def self.publish_to_directory(nuget_package_path, directory, package_name, version)
66
- target_package_path = "#{directory}/#{package_name}/#{version}/#{package_name}.#{version}.nupkg".downcase
67
- if File.exist?(target_package_path)
68
- puts " #{target_package_path} already exists".colorize(:grey)
69
- else
70
- "dotnet nuget push #{nuget_package_path} --source #{directory}".run
71
- end
72
- end
73
- end
74
- end
1
+ # frozen_string_literal: true
2
+
3
+ require "tmpdir"
4
+ # This module provides classes for the Makit gem.
5
+ module Makit
6
+ # This class provide methods for working with the Nuget package cache
7
+ #
8
+ # Example:
9
+ #
10
+ # Makit::Directory.cache("Google.Protobuf", "3.27.2")
11
+ #
12
+ # dotnet nuget locals all --list
13
+ # dotnet nuget locals all --clear
14
+ #
15
+ class NuGet
16
+ def self.get_cache_dir(package, version)
17
+ File.join(Makit::Directories::NUGET_PACKAGE_CACHE, package, version)
18
+ end
19
+
20
+ def self.cache(package, version)
21
+ # if the package is already cached, there is nothing to do
22
+ return if Dir.exist?(get_cache_dir(package, version))
23
+
24
+ Dir.mktmpdir do |dir|
25
+ # Use the temp directory here
26
+ Dir.chdir(dir) do
27
+ system("dotnet new classlib -n ClassLib")
28
+ Dir.chdir("ClassLib") do
29
+ # display a list of directories in the current directory
30
+ puts Dir.entries(Dir.pwd)
31
+ # display a list of files in the current directory
32
+ puts Dir.entries(Dir.pwd)
33
+ puts "dotnet add ClassLib.csproj package #{package} --version #{version}"
34
+ system("dotnet add ClassLib.csproj package #{package} --version #{version}")
35
+ end
36
+ end
37
+ # The directory and its contents will be removed automatically after the block
38
+ end
39
+ end
40
+
41
+ def self.clear_cache(package, version)
42
+ return unless Dir.exist?(get_cache_dir(package, version))
43
+
44
+ FileUtils.rm_rf(get_cache_dir(package, version))
45
+ end
46
+
47
+ # get the latest version of the package
48
+ def self.get_latest_version(package)
49
+ dir = File.join(Makit::Directories::NUGET_PACKAGE_CACHE, package)
50
+ if Dir.exist?(dir)
51
+ versions = Dir.entries(dir).select do |entry|
52
+ File.directory?(File.join(dir, entry)) && ![".", ".."].include?(entry)
53
+ end
54
+ highest_version = Makit::Version.get_highest_version(versions)
55
+ return highest_version
56
+ end
57
+ nil
58
+ end
59
+
60
+ # publish a package to a nuget directory feed
61
+ def self.add_package(_package, path)
62
+ system("dotnet nuget push #{path} --source #{path}")
63
+ end
64
+
65
+ def self.publish_to_directory(nuget_package_path, directory, package_name, version)
66
+ target_package_path = "#{directory}/#{package_name}/#{version}/#{package_name}.#{version}.nupkg".downcase
67
+ if File.exist?(target_package_path)
68
+ puts " #{target_package_path} already exists".colorize(:grey)
69
+ else
70
+ "dotnet nuget push #{nuget_package_path} --source #{directory}".run
71
+ end
72
+ end
73
+ end
74
+ end
data/lib/makit/port.rb CHANGED
@@ -1,32 +1,32 @@
1
- # frozen_string_literal: true
2
-
3
- # This module provides classes for the Makit gem.
4
- module Makit
5
- # This class provide methods for working with ports
6
- #
7
- class Port
8
- def self.is_port_available?(port)
9
- socket = Socket.new(:INET, :STREAM)
10
- socket.bind(Socket.sockaddr_in(port, ""))
11
- socket.close
12
- true # Port is available for use
13
- rescue Errno::EADDRINUSE, Errno::EACCES
14
- false # Port is not available for use
15
- end
16
-
17
- def self.get_available_port(start_port = 1024, end_port = 65_535)
18
- (start_port..end_port).each do |port|
19
- return port if is_port_available?(port)
20
- end
21
- nil # No available port found in the range
22
- end
23
-
24
- def self.get_random_port
25
- rand(1024..65_535)
26
- end
27
-
28
- def self.get_random_available_port
29
- get_available_port(get_random_port)
30
- end
31
- end
32
- end
1
+ # frozen_string_literal: true
2
+
3
+ # This module provides classes for the Makit gem.
4
+ module Makit
5
+ # This class provide methods for working with ports
6
+ #
7
+ class Port
8
+ def self.is_port_available?(port)
9
+ socket = Socket.new(:INET, :STREAM)
10
+ socket.bind(Socket.sockaddr_in(port, ""))
11
+ socket.close
12
+ true # Port is available for use
13
+ rescue Errno::EADDRINUSE, Errno::EACCES
14
+ false # Port is not available for use
15
+ end
16
+
17
+ def self.get_available_port(start_port = 1024, end_port = 65_535)
18
+ (start_port..end_port).each do |port|
19
+ return port if is_port_available?(port)
20
+ end
21
+ nil # No available port found in the range
22
+ end
23
+
24
+ def self.get_random_port
25
+ rand(1024..65_535)
26
+ end
27
+
28
+ def self.get_random_available_port
29
+ get_available_port(get_random_port)
30
+ end
31
+ end
32
+ end
data/lib/makit/process.rb CHANGED
@@ -1,163 +1,163 @@
1
- # frozen_string_literal: true
2
-
3
- # This module provides classes for the Makit gem.
4
- module Makit
5
- # Cross-platform process management utilities
6
- #
7
- # This class provides methods for checking, listing, and terminating processes
8
- # across different operating systems (Windows, Linux, macOS). It abstracts
9
- # platform-specific commands into a unified interface.
10
- class Process
11
- # Check if a process with the given name is currently running
12
- #
13
- # Uses platform-specific commands:
14
- # - Windows: tasklist command
15
- # - Unix: ps command with grep filtering
16
- #
17
- # @param name [String] Name of the process to check (without .exe extension)
18
- # @return [Boolean] True if the process is running, false otherwise
19
- def self.is_running?(name)
20
- return false if name.nil? || name.strip.empty?
21
-
22
- # provide a cross-platform way to check if a process is running
23
- results = if Makit::Environment.is_windows?
24
- # on windows, use the tasklist command
25
- `tasklist /FI "imagename eq #{name}.exe" 2>nul`
26
- else
27
- # on linux/mac, use the ps command
28
- `ps aux | grep "#{name}" | grep -v grep 2>/dev/null`
29
- end
30
- results.include?(name)
31
- end
32
-
33
- # Terminate all processes that match a given name
34
- #
35
- # Uses platform-specific termination commands:
36
- # - Windows: taskkill command
37
- # - Unix: kill command with SIGKILL (-9)
38
- #
39
- # @param name [String] Name of the process(es) to terminate
40
- # @return [Array<String>] Array of killed process IDs (empty if none found/killed)
41
- def self.kill(name)
42
- return [] if name.nil? || name.strip.empty?
43
-
44
- killed_pids = []
45
-
46
- if Makit::Environment.is_windows?
47
- # Windows: Use tasklist to find PIDs first, then kill them
48
- begin
49
- # Find processes by name
50
- tasklist_output = `tasklist /FI "imagename eq #{name}.exe" /FO CSV 2>nul`
51
-
52
- # Parse CSV output to extract PIDs
53
- lines = tasklist_output.split("\n")
54
- if lines.length > 1 # Skip header line
55
- lines[1..].each do |line|
56
- parts = line.split(",")
57
- next unless parts.length >= 2
58
-
59
- pid = parts[1]&.gsub('"', "")&.strip
60
- next unless pid && !pid.empty? && pid.match?(/^\d+$/)
61
-
62
- begin
63
- `taskkill /PID #{pid} /F 2>nul`
64
- killed_pids << pid
65
- rescue StandardError => e
66
- # Log error but continue with other processes
67
- puts "Failed to kill process #{pid}: #{e.message}"
68
- end
69
- end
70
- end
71
- rescue StandardError => e
72
- # If tasklist fails, try direct taskkill by name
73
- begin
74
- `taskkill /IM "#{name}.exe" /F 2>nul`
75
- # Since we can't get PIDs from direct kill, return empty array
76
- # but don't raise error
77
- rescue StandardError => e2
78
- # Both methods failed, return empty array
79
- puts "Failed to kill processes by name #{name}: #{e2.message}"
80
- end
81
- end
82
- else
83
- # Unix (Linux/Mac): Use pgrep to find PIDs, then kill them
84
- begin
85
- # Use pgrep to find PIDs, handle both Linux and Mac
86
- pids = if Makit::Environment.is_mac?
87
- # Mac: pgrep -f might not work the same way
88
- `pgrep -x "#{name}" 2>/dev/null`.split("\n").reject(&:empty?)
89
- else
90
- # Linux: pgrep -f works well
91
- `pgrep -f "#{name}" 2>/dev/null`.split("\n").reject(&:empty?)
92
- end
93
-
94
- pids.each do |pid|
95
- next unless pid.match?(/^\d+$/)
96
-
97
- begin
98
- `kill -9 #{pid} 2>/dev/null`
99
- killed_pids << pid
100
- rescue StandardError => e
101
- puts "Failed to kill process #{pid}: #{e.message}"
102
- end
103
- end
104
- rescue StandardError => e
105
- # If pgrep fails, return empty array but don't raise error
106
- puts "Failed to find processes by name #{name}: #{e.message}"
107
- end
108
- end
109
-
110
- killed_pids
111
- end
112
-
113
- # List all running processes on the system
114
- #
115
- # Uses platform-specific commands to get a complete process listing:
116
- # - Windows: tasklist command
117
- # - Unix: ps aux command
118
- #
119
- # @return [String] Raw output from the system process listing command
120
- def self.list
121
- if Makit::Environment.is_windows?
122
- `tasklist 2>nul`
123
- else
124
- `ps aux 2>/dev/null`
125
- end
126
- end
127
-
128
- # List all running processes for the current user
129
- #
130
- # Filters the process list to show only processes owned by the current user.
131
- # Uses platform-specific filtering:
132
- # - Windows: tasklist with USERNAME filter
133
- # - Unix: ps aux with grep filtering
134
- #
135
- # @return [String] Raw output from the filtered process listing command
136
- def self.list_current_user
137
- if Makit::Environment.is_windows?
138
- `tasklist /FI "STATUS eq running and USERNAME eq #{Makit::Environment.current_user}" 2>nul`
139
- else
140
- `ps aux | grep "#{Makit::Environment.current_user}" | grep -v grep 2>/dev/null`
141
- end
142
- end
143
-
144
- # List processes for the current user filtered by name
145
- #
146
- # Returns an array of process lines for consistency across platforms.
147
- # Filters by both current user and process name.
148
- #
149
- # @param name [String] Process name to filter by
150
- # @return [Array<String>] Array of process lines matching the criteria
151
- def self.list_current_user_processes(name)
152
- if Makit::Environment.is_windows?
153
- # filter the results to only include the current user
154
- results = `tasklist /FI "STATUS eq running and USERNAME eq #{Makit::Environment.current_user}" 2>nul`
155
- results.split("\n").select { |line| line.include?(Makit::Environment.current_user) }
156
- else
157
- # Unix (Linux/Mac): return as array for consistency
158
- results = `ps aux | grep "#{name}" | grep -v grep 2>/dev/null`
159
- results.split("\n").reject(&:empty?)
160
- end
161
- end
162
- end
163
- end
1
+ # frozen_string_literal: true
2
+
3
+ # This module provides classes for the Makit gem.
4
+ module Makit
5
+ # Cross-platform process management utilities
6
+ #
7
+ # This class provides methods for checking, listing, and terminating processes
8
+ # across different operating systems (Windows, Linux, macOS). It abstracts
9
+ # platform-specific commands into a unified interface.
10
+ class Process
11
+ # Check if a process with the given name is currently running
12
+ #
13
+ # Uses platform-specific commands:
14
+ # - Windows: tasklist command
15
+ # - Unix: ps command with grep filtering
16
+ #
17
+ # @param name [String] Name of the process to check (without .exe extension)
18
+ # @return [Boolean] True if the process is running, false otherwise
19
+ def self.is_running?(name)
20
+ return false if name.nil? || name.strip.empty?
21
+
22
+ # provide a cross-platform way to check if a process is running
23
+ results = if Makit::Environment.is_windows?
24
+ # on windows, use the tasklist command
25
+ `tasklist /FI "imagename eq #{name}.exe" 2>nul`
26
+ else
27
+ # on linux/mac, use the ps command
28
+ `ps aux | grep "#{name}" | grep -v grep 2>/dev/null`
29
+ end
30
+ results.include?(name)
31
+ end
32
+
33
+ # Terminate all processes that match a given name
34
+ #
35
+ # Uses platform-specific termination commands:
36
+ # - Windows: taskkill command
37
+ # - Unix: kill command with SIGKILL (-9)
38
+ #
39
+ # @param name [String] Name of the process(es) to terminate
40
+ # @return [Array<String>] Array of killed process IDs (empty if none found/killed)
41
+ def self.kill(name)
42
+ return [] if name.nil? || name.strip.empty?
43
+
44
+ killed_pids = []
45
+
46
+ if Makit::Environment.is_windows?
47
+ # Windows: Use tasklist to find PIDs first, then kill them
48
+ begin
49
+ # Find processes by name
50
+ tasklist_output = `tasklist /FI "imagename eq #{name}.exe" /FO CSV 2>nul`
51
+
52
+ # Parse CSV output to extract PIDs
53
+ lines = tasklist_output.split("\n")
54
+ if lines.length > 1 # Skip header line
55
+ lines[1..].each do |line|
56
+ parts = line.split(",")
57
+ next unless parts.length >= 2
58
+
59
+ pid = parts[1]&.gsub('"', "")&.strip
60
+ next unless pid && !pid.empty? && pid.match?(/^\d+$/)
61
+
62
+ begin
63
+ `taskkill /PID #{pid} /F 2>nul`
64
+ killed_pids << pid
65
+ rescue StandardError => e
66
+ # Log error but continue with other processes
67
+ puts "Failed to kill process #{pid}: #{e.message}"
68
+ end
69
+ end
70
+ end
71
+ rescue StandardError => e
72
+ # If tasklist fails, try direct taskkill by name
73
+ begin
74
+ `taskkill /IM "#{name}.exe" /F 2>nul`
75
+ # Since we can't get PIDs from direct kill, return empty array
76
+ # but don't raise error
77
+ rescue StandardError => e2
78
+ # Both methods failed, return empty array
79
+ puts "Failed to kill processes by name #{name}: #{e2.message}"
80
+ end
81
+ end
82
+ else
83
+ # Unix (Linux/Mac): Use pgrep to find PIDs, then kill them
84
+ begin
85
+ # Use pgrep to find PIDs, handle both Linux and Mac
86
+ pids = if Makit::Environment.is_mac?
87
+ # Mac: pgrep -f might not work the same way
88
+ `pgrep -x "#{name}" 2>/dev/null`.split("\n").reject(&:empty?)
89
+ else
90
+ # Linux: pgrep -f works well
91
+ `pgrep -f "#{name}" 2>/dev/null`.split("\n").reject(&:empty?)
92
+ end
93
+
94
+ pids.each do |pid|
95
+ next unless pid.match?(/^\d+$/)
96
+
97
+ begin
98
+ `kill -9 #{pid} 2>/dev/null`
99
+ killed_pids << pid
100
+ rescue StandardError => e
101
+ puts "Failed to kill process #{pid}: #{e.message}"
102
+ end
103
+ end
104
+ rescue StandardError => e
105
+ # If pgrep fails, return empty array but don't raise error
106
+ puts "Failed to find processes by name #{name}: #{e.message}"
107
+ end
108
+ end
109
+
110
+ killed_pids
111
+ end
112
+
113
+ # List all running processes on the system
114
+ #
115
+ # Uses platform-specific commands to get a complete process listing:
116
+ # - Windows: tasklist command
117
+ # - Unix: ps aux command
118
+ #
119
+ # @return [String] Raw output from the system process listing command
120
+ def self.list
121
+ if Makit::Environment.is_windows?
122
+ `tasklist 2>nul`
123
+ else
124
+ `ps aux 2>/dev/null`
125
+ end
126
+ end
127
+
128
+ # List all running processes for the current user
129
+ #
130
+ # Filters the process list to show only processes owned by the current user.
131
+ # Uses platform-specific filtering:
132
+ # - Windows: tasklist with USERNAME filter
133
+ # - Unix: ps aux with grep filtering
134
+ #
135
+ # @return [String] Raw output from the filtered process listing command
136
+ def self.list_current_user
137
+ if Makit::Environment.is_windows?
138
+ `tasklist /FI "STATUS eq running and USERNAME eq #{Makit::Environment.current_user}" 2>nul`
139
+ else
140
+ `ps aux | grep "#{Makit::Environment.current_user}" | grep -v grep 2>/dev/null`
141
+ end
142
+ end
143
+
144
+ # List processes for the current user filtered by name
145
+ #
146
+ # Returns an array of process lines for consistency across platforms.
147
+ # Filters by both current user and process name.
148
+ #
149
+ # @param name [String] Process name to filter by
150
+ # @return [Array<String>] Array of process lines matching the criteria
151
+ def self.list_current_user_processes(name)
152
+ if Makit::Environment.is_windows?
153
+ # filter the results to only include the current user
154
+ results = `tasklist /FI "STATUS eq running and USERNAME eq #{Makit::Environment.current_user}" 2>nul`
155
+ results.split("\n").select { |line| line.include?(Makit::Environment.current_user) }
156
+ else
157
+ # Unix (Linux/Mac): return as array for consistency
158
+ results = `ps aux | grep "#{name}" | grep -v grep 2>/dev/null`
159
+ results.split("\n").reject(&:empty?)
160
+ end
161
+ end
162
+ end
163
+ end