makit 0.0.99 → 0.0.111

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 (148) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +41 -0
  3. data/exe/makit +5 -0
  4. data/lib/makit/apache.rb +7 -11
  5. data/lib/makit/cli/build_commands.rb +500 -0
  6. data/lib/makit/cli/generators/base_generator.rb +74 -0
  7. data/lib/makit/cli/generators/dotnet_generator.rb +50 -0
  8. data/lib/makit/cli/generators/generator_factory.rb +49 -0
  9. data/lib/makit/cli/generators/node_generator.rb +50 -0
  10. data/lib/makit/cli/generators/ruby_generator.rb +77 -0
  11. data/lib/makit/cli/generators/rust_generator.rb +50 -0
  12. data/lib/makit/cli/generators/templates/dotnet_templates.rb +167 -0
  13. data/lib/makit/cli/generators/templates/node_templates.rb +161 -0
  14. data/lib/makit/cli/generators/templates/ruby/gemfile.rb +26 -0
  15. data/lib/makit/cli/generators/templates/ruby/gemspec.rb +40 -0
  16. data/lib/makit/cli/generators/templates/ruby/main_lib.rb +33 -0
  17. data/lib/makit/cli/generators/templates/ruby/rakefile.rb +35 -0
  18. data/lib/makit/cli/generators/templates/ruby/readme.rb +63 -0
  19. data/lib/makit/cli/generators/templates/ruby/test.rb +39 -0
  20. data/lib/makit/cli/generators/templates/ruby/test_helper.rb +29 -0
  21. data/lib/makit/cli/generators/templates/ruby/version.rb +29 -0
  22. data/lib/makit/cli/generators/templates/rust_templates.rb +128 -0
  23. data/lib/makit/cli/main.rb +48 -19
  24. data/lib/makit/cli/project_commands.rb +868 -0
  25. data/lib/makit/cli/repository_commands.rb +661 -0
  26. data/lib/makit/cli/utility_commands.rb +521 -0
  27. data/lib/makit/command_runner.rb +187 -128
  28. data/lib/makit/commands/compatibility.rb +365 -0
  29. data/lib/makit/commands/factory.rb +359 -0
  30. data/lib/makit/commands/middleware/base.rb +73 -0
  31. data/lib/makit/commands/middleware/cache.rb +248 -0
  32. data/lib/makit/commands/middleware/command_logger.rb +323 -0
  33. data/lib/makit/commands/middleware/unified_logger.rb +243 -0
  34. data/lib/makit/commands/middleware/validator.rb +269 -0
  35. data/lib/makit/commands/request.rb +254 -0
  36. data/lib/makit/commands/result.rb +323 -0
  37. data/lib/makit/commands/runner.rb +317 -0
  38. data/lib/makit/commands/strategies/base.rb +160 -0
  39. data/lib/makit/commands/strategies/synchronous.rb +134 -0
  40. data/lib/makit/commands.rb +24 -3
  41. data/lib/makit/configuration/gitlab_helper.rb +60 -0
  42. data/lib/makit/configuration/project.rb +127 -0
  43. data/lib/makit/configuration/rakefile_helper.rb +43 -0
  44. data/lib/makit/configuration/step.rb +34 -0
  45. data/lib/makit/configuration.rb +14 -0
  46. data/lib/makit/content/default_gitignore.rb +4 -2
  47. data/lib/makit/content/default_rakefile.rb +4 -2
  48. data/lib/makit/content/gem_rakefile.rb +4 -2
  49. data/lib/makit/context.rb +1 -0
  50. data/lib/makit/data.rb +9 -10
  51. data/lib/makit/directories.rb +48 -52
  52. data/lib/makit/directory.rb +38 -52
  53. data/lib/makit/docs/files.rb +5 -10
  54. data/lib/makit/docs/rake.rb +16 -20
  55. data/lib/makit/dotnet/cli.rb +65 -0
  56. data/lib/makit/dotnet/project.rb +153 -0
  57. data/lib/makit/dotnet/solution.rb +38 -0
  58. data/lib/makit/dotnet/solution_classlib.rb +239 -0
  59. data/lib/makit/dotnet/solution_console.rb +264 -0
  60. data/lib/makit/dotnet/solution_maui.rb +354 -0
  61. data/lib/makit/dotnet/solution_wasm.rb +275 -0
  62. data/lib/makit/dotnet/solution_wpf.rb +304 -0
  63. data/lib/makit/dotnet.rb +54 -171
  64. data/lib/makit/email.rb +46 -17
  65. data/lib/makit/environment.rb +22 -19
  66. data/lib/makit/examples/runner.rb +370 -0
  67. data/lib/makit/exceptions.rb +45 -0
  68. data/lib/makit/fileinfo.rb +3 -5
  69. data/lib/makit/files.rb +12 -16
  70. data/lib/makit/gems.rb +40 -39
  71. data/lib/makit/git/cli.rb +54 -0
  72. data/lib/makit/git/repository.rb +90 -0
  73. data/lib/makit/git.rb +44 -91
  74. data/lib/makit/gitlab_runner.rb +0 -1
  75. data/lib/makit/humanize.rb +31 -23
  76. data/lib/makit/indexer.rb +15 -24
  77. data/lib/makit/logging/configuration.rb +305 -0
  78. data/lib/makit/logging/format_registry.rb +84 -0
  79. data/lib/makit/logging/formatters/base.rb +39 -0
  80. data/lib/makit/logging/formatters/console_formatter.rb +127 -0
  81. data/lib/makit/logging/formatters/json_formatter.rb +65 -0
  82. data/lib/makit/logging/formatters/plain_text_formatter.rb +71 -0
  83. data/lib/makit/logging/formatters/text_formatter.rb +64 -0
  84. data/lib/makit/logging/log_request.rb +115 -0
  85. data/lib/makit/logging/logger.rb +159 -0
  86. data/lib/makit/logging/sinks/base.rb +91 -0
  87. data/lib/makit/logging/sinks/console.rb +72 -0
  88. data/lib/makit/logging/sinks/file_sink.rb +92 -0
  89. data/lib/makit/logging/sinks/structured.rb +129 -0
  90. data/lib/makit/logging/sinks/unified_file_sink.rb +303 -0
  91. data/lib/makit/logging.rb +452 -37
  92. data/lib/makit/markdown.rb +18 -18
  93. data/lib/makit/mp/basic_object_mp.rb +5 -4
  94. data/lib/makit/mp/command_mp.rb +5 -5
  95. data/lib/makit/mp/command_request.mp.rb +3 -2
  96. data/lib/makit/mp/project_mp.rb +85 -96
  97. data/lib/makit/mp/string_mp.rb +245 -73
  98. data/lib/makit/nuget.rb +27 -25
  99. data/lib/makit/port.rb +25 -27
  100. data/lib/makit/process.rb +127 -29
  101. data/lib/makit/protoc.rb +27 -24
  102. data/lib/makit/rake/cli.rb +196 -0
  103. data/lib/makit/rake.rb +6 -6
  104. data/lib/makit/ruby/cli.rb +185 -0
  105. data/lib/makit/ruby.rb +25 -0
  106. data/lib/makit/secrets.rb +18 -18
  107. data/lib/makit/serializer.rb +29 -27
  108. data/lib/makit/services/builder.rb +186 -0
  109. data/lib/makit/services/error_handler.rb +226 -0
  110. data/lib/makit/services/repository_manager.rb +229 -0
  111. data/lib/makit/services/validator.rb +112 -0
  112. data/lib/makit/setup/classlib.rb +53 -0
  113. data/lib/makit/setup/gem.rb +250 -0
  114. data/lib/makit/setup/runner.rb +40 -0
  115. data/lib/makit/show.rb +16 -16
  116. data/lib/makit/storage.rb +32 -37
  117. data/lib/makit/symbols.rb +12 -0
  118. data/lib/makit/task_hooks.rb +125 -0
  119. data/lib/makit/task_info.rb +63 -21
  120. data/lib/makit/tasks/at_exit.rb +13 -0
  121. data/lib/makit/tasks/build.rb +18 -0
  122. data/lib/makit/tasks/clean.rb +11 -0
  123. data/lib/makit/tasks/hook_manager.rb +239 -0
  124. data/lib/makit/tasks/init.rb +47 -0
  125. data/lib/makit/tasks/integrate.rb +15 -0
  126. data/lib/makit/tasks/pull_incoming.rb +12 -0
  127. data/lib/makit/tasks/setup.rb +6 -0
  128. data/lib/makit/tasks/sync.rb +11 -0
  129. data/lib/makit/tasks/task_monkey_patch.rb +79 -0
  130. data/lib/makit/tasks.rb +5 -150
  131. data/lib/makit/test_cache.rb +239 -0
  132. data/lib/makit/v1/makit.v1_pb.rb +34 -35
  133. data/lib/makit/v1/makit.v1_services_pb.rb +2 -0
  134. data/lib/makit/version.rb +1 -57
  135. data/lib/makit/wix.rb +23 -23
  136. data/lib/makit/yaml.rb +18 -6
  137. data/lib/makit.rb +2 -261
  138. metadata +109 -145
  139. data/lib/makit/cli/clean.rb +0 -14
  140. data/lib/makit/cli/clone.rb +0 -59
  141. data/lib/makit/cli/init.rb +0 -38
  142. data/lib/makit/cli/make.rb +0 -54
  143. data/lib/makit/cli/new.rb +0 -37
  144. data/lib/makit/cli/nuget_cache.rb +0 -38
  145. data/lib/makit/cli/pull.rb +0 -31
  146. data/lib/makit/cli/setup.rb +0 -71
  147. data/lib/makit/cli/work.rb +0 -21
  148. data/lib/makit/content/default_gitignore.txt +0 -222
@@ -0,0 +1,226 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Makit
4
+ module Services
5
+ # Service class responsible for handling and formatting errors
6
+ # Provides user-friendly error messages and recovery suggestions
7
+ class ErrorHandler
8
+ # Error message templates with suggestions
9
+ ERROR_MESSAGES = {
10
+ invalid_url: {
11
+ message: "The repository URL '%<url>s' is not valid.",
12
+ suggestions: [
13
+ "Use HTTPS format: https://github.com/user/repo.git",
14
+ "Use SSH format: git@github.com:user/repo.git",
15
+ "Use simple format: user/repo",
16
+ "Ensure the URL points to a git repository",
17
+ ],
18
+ },
19
+ invalid_directory: {
20
+ message: "The directory path '%<directory>s' is not valid.",
21
+ suggestions: [
22
+ "Use absolute paths like /path/to/directory",
23
+ "Use relative paths like ./relative/path",
24
+ "Ensure the path doesn't contain null characters",
25
+ "Keep directory paths under 255 characters",
26
+ ],
27
+ },
28
+ invalid_commit: {
29
+ message: "The commit '%<commit>s' is not valid.",
30
+ suggestions: [
31
+ "Use 'latest' for the most recent commit",
32
+ "Use a full 40-character commit hash",
33
+ "Use a short 7+ character commit hash",
34
+ "Verify the commit exists in the repository",
35
+ ],
36
+ },
37
+ directory_not_found: {
38
+ message: "The directory '%<directory>s' does not exist or is not accessible.",
39
+ suggestions: [
40
+ "Check that the directory path is correct",
41
+ "Verify you have read permissions for the directory",
42
+ "Create the directory first if it doesn't exist",
43
+ "Use 'makit init' to create a new project directory",
44
+ ],
45
+ },
46
+ git_command_failed: {
47
+ message: "Git command failed: %<command>s",
48
+ suggestions: [
49
+ "Check that git is installed and available in PATH",
50
+ "Verify you have network connectivity for remote operations",
51
+ "Ensure you have proper permissions for the repository",
52
+ "Check that the remote repository URL is accessible",
53
+ ],
54
+ },
55
+ clone_failed: {
56
+ message: "Failed to clone repository '%<url>s'.",
57
+ suggestions: [
58
+ "Check that the repository URL is correct and accessible",
59
+ "Verify your network connectivity",
60
+ "Ensure you have proper authentication (SSH keys, tokens)",
61
+ "Check that you have write permissions to the destination directory",
62
+ ],
63
+ },
64
+ pull_failed: {
65
+ message: "Failed to pull changes from repository.",
66
+ suggestions: [
67
+ "Check that you have network connectivity",
68
+ "Verify the remote repository is accessible",
69
+ "Ensure your local repository is in a clean state",
70
+ "Try 'git status' to check for conflicts",
71
+ ],
72
+ },
73
+ build_failed: {
74
+ message: "Build process failed for repository '%<url>s' at commit '%<commit>s'.",
75
+ suggestions: [
76
+ "Check the build logs for specific error messages",
77
+ "Verify all dependencies are installed",
78
+ "Ensure the commit hash is valid and exists",
79
+ "Try building manually to isolate the issue",
80
+ ],
81
+ },
82
+ }.freeze
83
+
84
+ class << self
85
+ # Format an error with user-friendly message and suggestions
86
+ #
87
+ # @param error_type [Symbol] the type of error from ERROR_MESSAGES
88
+ # @param details [Hash] details to interpolate into the message
89
+ # @param original_error [Exception] the original exception (optional)
90
+ # @return [String] formatted error message with suggestions
91
+ def format_error(error_type, details = {}, original_error = nil)
92
+ error_config = ERROR_MESSAGES[error_type]
93
+ return "Unknown error type: #{error_type}" unless error_config
94
+
95
+ message = error_config[:message] % details
96
+ suggestions = error_config[:suggestions]
97
+
98
+ formatted_message = ["❌ Error: #{message}"]
99
+
100
+ if suggestions && !suggestions.empty?
101
+ formatted_message << ""
102
+ formatted_message << "💡 Suggestions:"
103
+ suggestions.each_with_index do |suggestion, index|
104
+ formatted_message << " #{index + 1}. #{suggestion}"
105
+ end
106
+ end
107
+
108
+ if original_error && !original_error.message.empty?
109
+ formatted_message << ""
110
+ formatted_message << "🔍 Technical details:"
111
+ formatted_message << " #{original_error.class}: #{original_error.message}"
112
+ end
113
+
114
+ formatted_message.join("\n")
115
+ end
116
+
117
+ # Handle ArgumentError exceptions with user-friendly formatting
118
+ #
119
+ # @param error [ArgumentError] the argument error to format
120
+ # @param context [Hash] additional context about where the error occurred
121
+ # @return [String] formatted error message
122
+ def handle_argument_error(error, context = {})
123
+ case error.message
124
+ when /URL parameter cannot be nil|URL parameter cannot be empty|Invalid URL format/
125
+ format_error(:invalid_url, { url: context[:url] || "unknown" }, error)
126
+ when /directory parameter cannot be nil|directory parameter cannot be empty|directory path/
127
+ format_error(:invalid_directory, { directory: context[:directory] || "unknown" }, error)
128
+ when /commit parameter cannot be nil|commit parameter cannot be empty|Invalid commit format/
129
+ format_error(:invalid_commit, { commit: context[:commit] || "unknown" }, error)
130
+ else
131
+ "❌ Error: #{error.message}\n💡 Please check your input parameters and try again."
132
+ end
133
+ end
134
+
135
+ # Handle Makit-specific exceptions with user-friendly formatting
136
+ #
137
+ # @param error [Makit::Error] the Makit error to format
138
+ # @param context [Hash] additional context about where the error occurred
139
+ # @return [String] formatted error message
140
+ def handle_makit_error(error, context = {})
141
+ case error
142
+ when Makit::DirectoryError
143
+ format_error(:directory_not_found, { directory: context[:directory] || "unknown" }, error)
144
+ when Makit::CloneError
145
+ format_error(:clone_failed, { url: context[:url] || "unknown" }, error)
146
+ when Makit::PullError
147
+ format_error(:pull_failed, {}, error)
148
+ when Makit::BuildError
149
+ format_error(:build_failed, { url: context[:url] || "unknown", commit: context[:commit] || "unknown" },
150
+ error)
151
+ when Makit::GitError
152
+ format_error(:git_command_failed, { command: context[:command] || "unknown" }, error)
153
+ else
154
+ "❌ Error: #{error.message}\n💡 Please refer to the documentation or run with --verbose for more details."
155
+ end
156
+ end
157
+
158
+ # Log error with context for debugging
159
+ #
160
+ # @param error [Exception] the error to log
161
+ # @param context [Hash] additional context information
162
+ def log_error(error, context = {})
163
+ return unless defined?(Makit::LOGGER) && Makit::LOGGER
164
+
165
+ Makit::LOGGER.error "Error occurred: #{error.class} - #{error.message}"
166
+ Makit::LOGGER.error "Context: #{context.inspect}" unless context.empty?
167
+ Makit::LOGGER.debug "Backtrace: #{error.backtrace&.join("\n")}" if error.backtrace
168
+ end
169
+
170
+ # Wrap a block with comprehensive error handling
171
+ #
172
+ # @param context [Hash] context information for error messages
173
+ # @yield block to execute with error handling
174
+ # @return [Object] the result of the block
175
+ def with_error_handling(context = {})
176
+ yield
177
+ rescue ArgumentError => e
178
+ log_error(e, context)
179
+ formatted_message = handle_argument_error(e, context)
180
+ raise Makit::ValidationError, formatted_message
181
+ rescue Makit::Error => e
182
+ log_error(e, context)
183
+ formatted_message = handle_makit_error(e, context)
184
+ raise e.class, formatted_message
185
+ rescue StandardError => e
186
+ log_error(e, context)
187
+ formatted_message = "❌ An unexpected error occurred: #{e.message}\n💡 Please check the logs or run with --verbose for more details."
188
+ raise Makit::Error, formatted_message
189
+ end
190
+
191
+ # Validate input and provide helpful error messages
192
+ #
193
+ # @param validations [Array<Hash>] array of validation rules
194
+ # @example
195
+ # validate_inputs([
196
+ # { value: url, validator: :url, name: 'repository URL' },
197
+ # { value: commit, validator: :commit, name: 'commit hash' }
198
+ # ])
199
+ def validate_inputs(validations)
200
+ validations.each do |validation|
201
+ value = validation[:value]
202
+ validator = validation[:validator]
203
+ name = validation[:name] || "parameter"
204
+
205
+ case validator
206
+ when :url
207
+ Validator.validate_url_parameter(value)
208
+ when :directory
209
+ Validator.validate_directory_parameter(value)
210
+ when :commit
211
+ Validator.validate_commit_parameter(value)
212
+ when :required_string
213
+ Validator.validate_required_string(value, name)
214
+ when :boolean
215
+ Validator.validate_boolean_parameter(value, name)
216
+ end
217
+ rescue ArgumentError => e
218
+ context = { validator => value }
219
+ formatted_message = handle_argument_error(e, context)
220
+ raise Makit::ValidationError, formatted_message
221
+ end
222
+ end
223
+ end
224
+ end
225
+ end
226
+ end
@@ -0,0 +1,229 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "validator"
4
+
5
+ module Makit
6
+ module Services
7
+ # Service class responsible for git repository management operations
8
+ # Handles cloning, pulling, logging, and working directory setup
9
+ class RepositoryManager
10
+ class << self
11
+ # Initialize a new git repository in the specified directory
12
+ #
13
+ # @param directory [String] the directory path to initialize
14
+ # @return [Makit::V1::Command] the git init command result
15
+ # @raise [Makit::DirectoryError] if the directory cannot be created
16
+ # @raise [Makit::GitError] if git initialization fails
17
+ def initialize_repository(directory)
18
+ Validator.validate_directory_parameter(directory)
19
+ FileUtils.mkdir_p(directory)
20
+
21
+ raise Makit::DirectoryError, "directory does not exist: #{directory}" unless Dir.exist?(directory)
22
+
23
+ Dir.chdir(directory) do
24
+ File.write(".gitignore", Makit::Content::GITIGNORE) unless File.exist?(".gitignore")
25
+ init_command = Makit::RUNNER.execute("git init")
26
+
27
+ if init_command.exit_code != 0
28
+ raise Makit::GitError, "failed to initialize local repository: #{directory}\\n#{Makit::Humanize.get_command_summary(init_command)}"
29
+ end
30
+
31
+ init_command
32
+ end
33
+ end
34
+
35
+ # Clone a git repository to a local directory
36
+ #
37
+ # @param git_repository [String] the git repository URL
38
+ # @return [Array<Makit::V1::Command>] array of command results
39
+ def clone_repository(git_repository)
40
+ Validator.validate_url_parameter(git_repository)
41
+ commands = []
42
+
43
+ clone_dir = Directories.get_clone_directory(git_repository)
44
+ commands << Makit::RUNNER.execute("git clone #{git_repository} #{clone_dir}") unless Dir.exist?(clone_dir)
45
+
46
+ commands
47
+ end
48
+
49
+ # Pull the latest changes from the remote repository
50
+ #
51
+ # @param git_repository [String] the git repository URL
52
+ # @return [Makit::V1::Command] the git pull command result
53
+ def pull_repository(git_repository)
54
+ Validator.validate_url_parameter(git_repository)
55
+ clone_dir = Directories.get_clone_directory(git_repository)
56
+
57
+ Validator.validate_directory_exists(clone_dir, "clone directory")
58
+
59
+ Dir.chdir(clone_dir) do
60
+ execute_git_pull(clone_dir)
61
+ end
62
+ end
63
+
64
+ # Clone repository if it doesn't exist, otherwise pull latest changes
65
+ #
66
+ # @param git_repository [String] the git repository URL
67
+ # @return [Array<Makit::V1::Command>] array of command results
68
+ def clone_or_pull_repository(git_repository)
69
+ commands = []
70
+ clone_dir = Directories.get_clone_directory(git_repository)
71
+
72
+ commands << if Dir.exist?(clone_dir)
73
+ pull_repository(git_repository)
74
+ else
75
+ clone_repository(git_repository)
76
+ end
77
+ commands
78
+ end
79
+
80
+ # Get git log information for a repository
81
+ #
82
+ # @param git_repository [String] the git repository URL
83
+ # @param limit [Integer] maximum number of commits to retrieve
84
+ # @param skip [Integer] number of commits to skip
85
+ # @return [Array<Makit::V1::GitLogEntry>] array of log entries
86
+ def get_repository_log(git_repository, limit, skip)
87
+ Validator.validate_url_parameter(git_repository)
88
+ Validator.validate_pagination_parameters(limit, skip)
89
+
90
+ clone_dir = Directories.get_clone_directory(git_repository)
91
+ Validator.validate_directory_exists(clone_dir, "clone directory")
92
+
93
+ Dir.chdir(clone_dir) do
94
+ log_command = Makit::RUNNER.execute("git log -n #{limit} --skip #{skip} --date=iso")
95
+ parse_git_log_output(log_command)
96
+ end
97
+ end
98
+
99
+ # Set up a work directory for a repository
100
+ #
101
+ # @param repository [String] the git repository URL
102
+ # @return [nil] always returns nil
103
+ def setup_work_directory(repository)
104
+ Validator.validate_url_parameter(repository)
105
+ clone_or_pull_repository(repository)
106
+ create_work_directory(repository)
107
+ nil
108
+ end
109
+
110
+ # Get the latest commit hash from a repository
111
+ #
112
+ # @param clone_dir [String] the local clone directory
113
+ # @param commands [Array] array to append command results to
114
+ # @return [String] the latest commit hash
115
+ def get_latest_commit(clone_dir, commands)
116
+ Dir.chdir(clone_dir) do
117
+ git_log = Makit::RUNNER.execute("git log -n 1 --date=iso")
118
+ commands << git_log
119
+
120
+ commit = git_log.output.match(/^commit ([0-9a-f]{#{Validator::COMMIT_HASH_LENGTH}})$/i)&.captures&.first
121
+
122
+ if commit.nil? || commit.empty? || !commit.match?(/\\A[0-9a-f]{#{Validator::COMMIT_HASH_LENGTH}}\\z/i)
123
+ raise Makit::InvalidCommitError, "invalid commit: #{commit}"
124
+ end
125
+
126
+ commit
127
+ end
128
+ end
129
+
130
+ # Validate that a clone directory exists for the given repository
131
+ #
132
+ # @param url [String] the git repository URL
133
+ # @return [String] the clone directory path
134
+ def validate_clone_directory(url)
135
+ clone_dir = Directories.get_clone_directory(url)
136
+ Validator.validate_directory_exists(clone_dir, "clone directory")
137
+ clone_dir
138
+ end
139
+
140
+ private
141
+
142
+ # Execute git pull command with proper error handling
143
+ def execute_git_pull(clone_dir)
144
+ request = Makit::V1::CommandRequest.new(
145
+ name: "git",
146
+ arguments: ["pull"],
147
+ directory: clone_dir,
148
+ )
149
+ pull_command = Makit::RUNNER.execute(request)
150
+
151
+ raise Makit::PullError, Makit::Humanize.get_command_details(pull_command) if pull_command.exit_code != 0
152
+
153
+ pull_command
154
+ end
155
+
156
+ # Create work directory for development
157
+ def create_work_directory(repository)
158
+ work_dir = Makit::Directories.get_work_directory(repository)
159
+ clone_dir = Makit::Directories.get_clone_directory(repository)
160
+
161
+ unless Dir.exist?(work_dir)
162
+ FileUtils.mkdir_p(File.dirname(work_dir))
163
+ Makit::RUNNER.execute("git clone #{clone_dir} #{work_dir}")
164
+ end
165
+
166
+ Dir.chdir(work_dir) do
167
+ File.write(".gitignore", Makit::Content::GITIGNORE) unless File.exist?(".gitignore")
168
+ end
169
+ end
170
+
171
+ # Parse git log output into structured log entries
172
+ def parse_git_log_output(log_command)
173
+ entries = []
174
+ return entries if log_command.exit_code != 0
175
+
176
+ lines = log_command.output.split("\\n")
177
+ current_entry = nil
178
+
179
+ lines.each do |line|
180
+ current_entry = process_git_log_line(line, current_entry, entries)
181
+ end
182
+
183
+ entries
184
+ end
185
+
186
+ # Process a single line from git log output
187
+ def process_git_log_line(line, current_entry, entries)
188
+ case line
189
+ when /^commit/
190
+ create_new_log_entry(line, entries)
191
+ when /^Author:/
192
+ current_entry ? update_entry_author(line, current_entry) : current_entry
193
+ when /^Date:/
194
+ current_entry ? update_entry_date(line, current_entry) : current_entry
195
+ when /^ /
196
+ current_entry ? update_entry_message(line, current_entry) : current_entry
197
+ else
198
+ current_entry
199
+ end
200
+ end
201
+
202
+ # Create new git log entry from commit line
203
+ def create_new_log_entry(line, entries)
204
+ current_entry = GitLogEntry.new(line.split[1])
205
+ entries << current_entry
206
+ current_entry
207
+ end
208
+
209
+ # Update log entry with author information
210
+ def update_entry_author(line, current_entry)
211
+ current_entry.author = line.split[1..].join(" ")
212
+ current_entry
213
+ end
214
+
215
+ # Update log entry with date information
216
+ def update_entry_date(line, current_entry)
217
+ current_entry.date = line.split[1..].join(" ")
218
+ current_entry
219
+ end
220
+
221
+ # Update log entry with commit message
222
+ def update_entry_message(line, current_entry)
223
+ current_entry.message += line[4..]
224
+ current_entry
225
+ end
226
+ end
227
+ end
228
+ end
229
+ end
@@ -0,0 +1,112 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Makit
4
+ module Services
5
+ # Service class responsible for validating all input parameters
6
+ # Used by various Makit operations to ensure data integrity
7
+ class Validator
8
+ # Git-related validation constants
9
+ COMMIT_HASH_LENGTH = 40
10
+ COMMIT_LATEST = "latest"
11
+ MAX_PAGINATION_LIMIT = 1000
12
+ MAX_DIRECTORY_PATH_LENGTH = 255
13
+
14
+ class << self
15
+ # Validate directory parameter for initialization operations
16
+ #
17
+ # @param directory [String] the directory path to validate
18
+ # @raise [ArgumentError] if directory parameter is invalid
19
+ def validate_directory_parameter(directory)
20
+ raise ArgumentError, "directory parameter cannot be nil" if directory.nil?
21
+ raise ArgumentError, "directory parameter cannot be empty" if directory.to_s.strip.empty?
22
+ raise ArgumentError, "directory path contains invalid characters" if directory.to_s.include?("\0")
23
+ raise ArgumentError, "directory path is too long" if directory.to_s.length > MAX_DIRECTORY_PATH_LENGTH
24
+ end
25
+
26
+ # Validate URL parameter for git repository operations
27
+ #
28
+ # @param url [String] the repository URL to validate
29
+ # @raise [ArgumentError] if URL parameter is invalid
30
+ def validate_url_parameter(url)
31
+ raise ArgumentError, "URL parameter cannot be nil" if url.nil?
32
+ raise ArgumentError, "URL parameter cannot be empty" if url.to_s.strip.empty?
33
+
34
+ url_str = url.to_s.strip
35
+ # Basic URL format validation - accept git@, https://, http://, and simple names like "user/repo"
36
+ valid_patterns = [
37
+ %r{\Agit@[\w.-]+:[\w._/-]+\.git\z}, # SSH format: git@github.com:user/repo.git
38
+ %r{\Ahttps?://[\w.-]+/[\w._/-]+(\.git)?\z}, # HTTPS format: https://github.com/user/repo
39
+ %r{\A[\w.-]+/[\w._-]+\z}, # Simple format: user/repo
40
+ ]
41
+
42
+ return if valid_patterns.any? { |pattern| url_str.match?(pattern) }
43
+
44
+ raise ArgumentError, "Invalid URL format: #{url_str}"
45
+ end
46
+
47
+ # Validate commit parameter for git operations
48
+ #
49
+ # @param commit [String] the commit hash or "latest" to validate
50
+ # @raise [ArgumentError] if commit parameter is invalid
51
+ def validate_commit_parameter(commit)
52
+ raise ArgumentError, "commit parameter cannot be nil" if commit.nil?
53
+
54
+ commit_str = commit.to_s.strip
55
+ raise ArgumentError, "commit parameter cannot be empty" if commit_str.empty?
56
+
57
+ # Allow "latest" or valid commit hash formats
58
+ return if commit_str == COMMIT_LATEST
59
+ return if commit_str.match?(/\A[0-9a-f]{7,#{COMMIT_HASH_LENGTH}}\z/i)
60
+
61
+ raise ArgumentError, "Invalid commit format: #{commit_str}. Must be 'latest' or a valid git commit hash."
62
+ end
63
+
64
+ # Validate pagination parameters for log operations
65
+ #
66
+ # @param limit [Integer] maximum number of items to retrieve
67
+ # @param skip [Integer] number of items to skip
68
+ # @raise [ArgumentError] if pagination parameters are invalid
69
+ def validate_pagination_parameters(limit, skip)
70
+ unless limit.is_a?(Integer) && limit.positive?
71
+ raise ArgumentError,
72
+ "limit parameter must be a positive integer"
73
+ end
74
+ raise ArgumentError, "skip parameter must be a non-negative integer" unless skip.is_a?(Integer) && skip >= 0
75
+ raise ArgumentError, "limit parameter is too large" if limit > MAX_PAGINATION_LIMIT
76
+ end
77
+
78
+ # Validate that a directory exists and is accessible
79
+ #
80
+ # @param directory [String] the directory path to check
81
+ # @param description [String] human-readable description for error messages
82
+ # @raise [Makit::DirectoryError] if directory doesn't exist or isn't accessible
83
+ def validate_directory_exists(directory, description = "directory")
84
+ return if Dir.exist?(directory)
85
+
86
+ raise Makit::DirectoryError, "#{description} does not exist: #{directory}"
87
+ end
88
+
89
+ # Validate boolean parameters
90
+ #
91
+ # @param value [Object] the value to check
92
+ # @param parameter_name [String] the parameter name for error messages
93
+ # @raise [ArgumentError] if value is not boolean
94
+ def validate_boolean_parameter(value, parameter_name)
95
+ return if [true, false].include?(value)
96
+
97
+ raise ArgumentError, "#{parameter_name} must be true or false, got #{value.class}"
98
+ end
99
+
100
+ # Validate string parameters that cannot be empty
101
+ #
102
+ # @param value [String] the string value to validate
103
+ # @param parameter_name [String] the parameter name for error messages
104
+ # @raise [ArgumentError] if string is nil or empty
105
+ def validate_required_string(value, parameter_name)
106
+ raise ArgumentError, "#{parameter_name} cannot be nil" if value.nil?
107
+ raise ArgumentError, "#{parameter_name} cannot be empty" if value.to_s.strip.empty?
108
+ end
109
+ end
110
+ end
111
+ end
112
+ end
@@ -0,0 +1,53 @@
1
+ # frozen_string_literal: true
2
+ require_relative "../configuration/project"
3
+
4
+ module Makit
5
+ module Setup
6
+ class ClassLib
7
+ def self.run
8
+ #puts "Setting up Nuget project..."
9
+ project = Makit::Configuration::Project.default
10
+ Makit::DotNet::Project.new_project("classlib", project.name, "source/#{project.name}", "--framework net8.0")
11
+ Makit::DotNet::Project.new_project("TUnit", "#{project.name}.Tests", "tests/#{project.name}")
12
+ Makit::DotNet::Project.add_reference("tests/#{project.name}/#{project.name}.Tests.csproj", "source/#{project.name}/#{project.name}.csproj")
13
+ update_build_step(project)
14
+ update_test_step(project)
15
+
16
+ project.save
17
+ Makit::Logging.default_logger.info("Project setup completed")
18
+ end
19
+
20
+ def self.update_build_step(project)
21
+ build_commands = Array.new
22
+ build_commands << "dotnet build source/#{project.name}/#{project.name}.csproj --configuration Release"
23
+ steps = project.steps
24
+ if steps.any? { |step| step.name == "build" }
25
+ build_step = steps.find { |step| step.name == "build" }
26
+ build_step.commands = build_commands
27
+ else
28
+ build_step = Makit::Configuration::Step.new(name: "build", description: "Build the project artifacts", commands: build_commands)
29
+ build_step.commands = build_commands
30
+ project.add_step(build_step)
31
+ end
32
+ build_step.commands = build_commands
33
+
34
+ project.save
35
+ end
36
+
37
+ def self.update_test_step(project)
38
+ test_commands = Array.new
39
+ test_commands << "dotnet test tests/#{project.name}/#{project.name}.Tests.csproj"
40
+ steps = project.steps
41
+ if steps.any? { |step| step.name == "test" }
42
+ test_step = steps.find { |step| step.name == "test" }
43
+ test_step.commands = test_commands
44
+ else
45
+ test_step = Makit::Configuration::Step.new(name: "test", description: "Run the project tests", commands: test_commands)
46
+ test_step.commands = test_commands
47
+ project.add_step(test_step)
48
+ end
49
+ project.save
50
+ end
51
+ end
52
+ end
53
+ end