makit 0.0.98 → 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.
- checksums.yaml +4 -4
- data/README.md +41 -0
- data/exe/makit +5 -0
- data/lib/makit/apache.rb +7 -11
- data/lib/makit/cli/build_commands.rb +500 -0
- data/lib/makit/cli/generators/base_generator.rb +74 -0
- data/lib/makit/cli/generators/dotnet_generator.rb +50 -0
- data/lib/makit/cli/generators/generator_factory.rb +49 -0
- data/lib/makit/cli/generators/node_generator.rb +50 -0
- data/lib/makit/cli/generators/ruby_generator.rb +77 -0
- data/lib/makit/cli/generators/rust_generator.rb +50 -0
- data/lib/makit/cli/generators/templates/dotnet_templates.rb +167 -0
- data/lib/makit/cli/generators/templates/node_templates.rb +161 -0
- data/lib/makit/cli/generators/templates/ruby/gemfile.rb +26 -0
- data/lib/makit/cli/generators/templates/ruby/gemspec.rb +40 -0
- data/lib/makit/cli/generators/templates/ruby/main_lib.rb +33 -0
- data/lib/makit/cli/generators/templates/ruby/rakefile.rb +35 -0
- data/lib/makit/cli/generators/templates/ruby/readme.rb +63 -0
- data/lib/makit/cli/generators/templates/ruby/test.rb +39 -0
- data/lib/makit/cli/generators/templates/ruby/test_helper.rb +29 -0
- data/lib/makit/cli/generators/templates/ruby/version.rb +29 -0
- data/lib/makit/cli/generators/templates/rust_templates.rb +128 -0
- data/lib/makit/cli/main.rb +48 -19
- data/lib/makit/cli/project_commands.rb +868 -0
- data/lib/makit/cli/repository_commands.rb +661 -0
- data/lib/makit/cli/utility_commands.rb +521 -0
- data/lib/makit/command_runner.rb +187 -128
- data/lib/makit/commands/compatibility.rb +365 -0
- data/lib/makit/commands/factory.rb +359 -0
- data/lib/makit/commands/middleware/base.rb +73 -0
- data/lib/makit/commands/middleware/cache.rb +248 -0
- data/lib/makit/commands/middleware/command_logger.rb +323 -0
- data/lib/makit/commands/middleware/unified_logger.rb +243 -0
- data/lib/makit/commands/middleware/validator.rb +269 -0
- data/lib/makit/commands/request.rb +254 -0
- data/lib/makit/commands/result.rb +323 -0
- data/lib/makit/commands/runner.rb +317 -0
- data/lib/makit/commands/strategies/base.rb +160 -0
- data/lib/makit/commands/strategies/synchronous.rb +134 -0
- data/lib/makit/commands.rb +24 -3
- data/lib/makit/configuration/gitlab_helper.rb +60 -0
- data/lib/makit/configuration/project.rb +127 -0
- data/lib/makit/configuration/rakefile_helper.rb +43 -0
- data/lib/makit/configuration/step.rb +34 -0
- data/lib/makit/configuration.rb +14 -0
- data/lib/makit/content/default_gitignore.rb +4 -2
- data/lib/makit/content/default_rakefile.rb +4 -2
- data/lib/makit/content/gem_rakefile.rb +4 -2
- data/lib/makit/context.rb +1 -0
- data/lib/makit/data.rb +9 -10
- data/lib/makit/directories.rb +48 -52
- data/lib/makit/directory.rb +38 -52
- data/lib/makit/docs/files.rb +5 -10
- data/lib/makit/docs/rake.rb +16 -20
- data/lib/makit/dotnet/cli.rb +65 -0
- data/lib/makit/dotnet/project.rb +153 -0
- data/lib/makit/dotnet/solution.rb +38 -0
- data/lib/makit/dotnet/solution_classlib.rb +239 -0
- data/lib/makit/dotnet/solution_console.rb +264 -0
- data/lib/makit/dotnet/solution_maui.rb +354 -0
- data/lib/makit/dotnet/solution_wasm.rb +275 -0
- data/lib/makit/dotnet/solution_wpf.rb +304 -0
- data/lib/makit/dotnet.rb +54 -171
- data/lib/makit/email.rb +46 -17
- data/lib/makit/environment.rb +22 -19
- data/lib/makit/examples/runner.rb +370 -0
- data/lib/makit/exceptions.rb +45 -0
- data/lib/makit/fileinfo.rb +3 -5
- data/lib/makit/files.rb +12 -16
- data/lib/makit/gems.rb +40 -39
- data/lib/makit/git/cli.rb +54 -0
- data/lib/makit/git/repository.rb +90 -0
- data/lib/makit/git.rb +44 -91
- data/lib/makit/gitlab_runner.rb +0 -1
- data/lib/makit/humanize.rb +31 -23
- data/lib/makit/indexer.rb +15 -24
- data/lib/makit/logging/configuration.rb +305 -0
- data/lib/makit/logging/format_registry.rb +84 -0
- data/lib/makit/logging/formatters/base.rb +39 -0
- data/lib/makit/logging/formatters/console_formatter.rb +127 -0
- data/lib/makit/logging/formatters/json_formatter.rb +65 -0
- data/lib/makit/logging/formatters/plain_text_formatter.rb +71 -0
- data/lib/makit/logging/formatters/text_formatter.rb +64 -0
- data/lib/makit/logging/log_request.rb +115 -0
- data/lib/makit/logging/logger.rb +159 -0
- data/lib/makit/logging/sinks/base.rb +91 -0
- data/lib/makit/logging/sinks/console.rb +72 -0
- data/lib/makit/logging/sinks/file_sink.rb +92 -0
- data/lib/makit/logging/sinks/structured.rb +129 -0
- data/lib/makit/logging/sinks/unified_file_sink.rb +303 -0
- data/lib/makit/logging.rb +452 -37
- data/lib/makit/markdown.rb +18 -18
- data/lib/makit/mp/basic_object_mp.rb +5 -4
- data/lib/makit/mp/command_mp.rb +5 -5
- data/lib/makit/mp/command_request.mp.rb +3 -2
- data/lib/makit/mp/project_mp.rb +85 -96
- data/lib/makit/mp/string_mp.rb +245 -73
- data/lib/makit/nuget.rb +27 -25
- data/lib/makit/port.rb +25 -27
- data/lib/makit/process.rb +127 -29
- data/lib/makit/protoc.rb +27 -24
- data/lib/makit/rake/cli.rb +196 -0
- data/lib/makit/rake.rb +6 -6
- data/lib/makit/ruby/cli.rb +185 -0
- data/lib/makit/ruby.rb +25 -0
- data/lib/makit/secrets.rb +18 -18
- data/lib/makit/serializer.rb +29 -27
- data/lib/makit/services/builder.rb +186 -0
- data/lib/makit/services/error_handler.rb +226 -0
- data/lib/makit/services/repository_manager.rb +229 -0
- data/lib/makit/services/validator.rb +112 -0
- data/lib/makit/setup/classlib.rb +53 -0
- data/lib/makit/setup/gem.rb +250 -0
- data/lib/makit/setup/runner.rb +40 -0
- data/lib/makit/show.rb +16 -16
- data/lib/makit/storage.rb +32 -37
- data/lib/makit/symbols.rb +12 -0
- data/lib/makit/task_hooks.rb +125 -0
- data/lib/makit/task_info.rb +63 -21
- data/lib/makit/tasks/at_exit.rb +13 -0
- data/lib/makit/tasks/build.rb +18 -0
- data/lib/makit/tasks/clean.rb +11 -0
- data/lib/makit/tasks/hook_manager.rb +239 -0
- data/lib/makit/tasks/init.rb +47 -0
- data/lib/makit/tasks/integrate.rb +15 -0
- data/lib/makit/tasks/pull_incoming.rb +12 -0
- data/lib/makit/tasks/setup.rb +6 -0
- data/lib/makit/tasks/sync.rb +11 -0
- data/lib/makit/tasks/task_monkey_patch.rb +79 -0
- data/lib/makit/tasks.rb +5 -150
- data/lib/makit/test_cache.rb +239 -0
- data/lib/makit/v1/makit.v1_pb.rb +34 -35
- data/lib/makit/v1/makit.v1_services_pb.rb +2 -0
- data/lib/makit/version.rb +1 -60
- data/lib/makit/wix.rb +23 -23
- data/lib/makit/yaml.rb +18 -6
- data/lib/makit.rb +2 -261
- metadata +109 -145
- data/lib/makit/cli/clean.rb +0 -14
- data/lib/makit/cli/clone.rb +0 -59
- data/lib/makit/cli/init.rb +0 -38
- data/lib/makit/cli/make.rb +0 -54
- data/lib/makit/cli/new.rb +0 -37
- data/lib/makit/cli/nuget_cache.rb +0 -38
- data/lib/makit/cli/pull.rb +0 -31
- data/lib/makit/cli/setup.rb +0 -71
- data/lib/makit/cli/work.rb +0 -21
- 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
|