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,303 +1,303 @@
1
- # frozen_string_literal: true
2
-
3
- require_relative "base"
4
- require_relative "../format_registry"
5
- require "fileutils"
6
-
7
- module Makit
8
- module Logging
9
- module Sinks
10
- # Unified file sink with configurable multiple outputs
11
- #
12
- # This sink replaces the need for multiple separate sinks by providing
13
- # a single, highly configurable sink that can write to multiple files
14
- # with different formats, filtering, and options. It supports console
15
- # output, file logging, and structured logging through configuration.
16
- #
17
- # @example Basic usage with console and file output
18
- # sink = UnifiedFileSink.new(
19
- # configurations: [
20
- # { file: $stdout, format: :console },
21
- # { file: "logs/app.log", format: :json }
22
- # ]
23
- # )
24
- #
25
- # @example Advanced configuration with filtering and formatting
26
- # sink = UnifiedFileSink.new(
27
- # configurations: [
28
- # {
29
- # file: $stdout,
30
- # format: :console,
31
- # show_timestamp: true,
32
- # show_level: true
33
- # },
34
- # {
35
- # file: "logs/debug.log",
36
- # format: :text,
37
- # min_level: :debug,
38
- # include_context: true
39
- # },
40
- # {
41
- # file: "logs/audit.log",
42
- # format: :json,
43
- # min_level: :info,
44
- # include_metadata: true
45
- # }
46
- # ]
47
- # )
48
- class UnifiedFileSink < Base
49
- # @return [Array<Hash>] list of output configurations
50
- attr_reader :configurations
51
-
52
- # Initialize unified file sink
53
- #
54
- # @param configurations [Array<Hash>] list of output configurations
55
- def initialize(configurations: [])
56
- @configurations = validate_configurations(configurations)
57
- @formatters = {}
58
- @files = {}
59
-
60
- # Pre-initialize formatters and files
61
- initialize_formatters
62
- initialize_files
63
- end
64
-
65
- # Execute sink logic to write log entry to configured outputs
66
- #
67
- # @param log_request [LogRequest] the log request to process
68
- # @yield [LogRequest] yields the processed log request to the next sink
69
- # @yieldreturn [LogRequest] the processed log request
70
- # @return [LogRequest] the final processed log request
71
- def call(log_request, &block)
72
- write_to_outputs(log_request)
73
- block&.call(log_request) if block_given?
74
- log_request
75
- end
76
-
77
- # Get sink configuration
78
- #
79
- # @return [Hash] sink configuration
80
- def config
81
- {
82
- name: self.class.name.split("::").last,
83
- configurations_count: @configurations.length,
84
- configurations: @configurations.map { |config| sanitize_config(config) },
85
- }
86
- end
87
-
88
- private
89
-
90
- # Validate configuration array
91
- #
92
- # @param configurations [Array<Hash>] configurations to validate
93
- # @return [Array<Hash>] validated configurations
94
- # @raise [ArgumentError] if configuration is invalid
95
- def validate_configurations(configurations)
96
- raise ArgumentError, "configurations must be an array" unless configurations.is_a?(Array)
97
- raise ArgumentError, "at least one configuration is required" if configurations.empty?
98
-
99
- configurations.each_with_index do |config, index|
100
- validate_single_configuration(config, index)
101
- end
102
-
103
- configurations
104
- end
105
-
106
- # Validate a single configuration
107
- #
108
- # @param config [Hash] configuration to validate
109
- # @param index [Integer] configuration index for error messages
110
- # @raise [ArgumentError] if configuration is invalid
111
- def validate_single_configuration(config, index)
112
- raise ArgumentError, "configuration #{index} must be a hash" unless config.is_a?(Hash)
113
-
114
- required_keys = [:file, :format]
115
- missing_keys = required_keys - config.keys
116
- raise ArgumentError, "configuration #{index} missing required keys: #{missing_keys.join(", ")}" unless missing_keys.empty?
117
-
118
- # Validate file
119
- file = config[:file]
120
- unless file.is_a?(String) || file.is_a?(IO) || file.respond_to?(:write)
121
- raise ArgumentError, "configuration #{index} file must be a String path, IO object, or writable object"
122
- end
123
-
124
- # Validate format
125
- format = config[:format]
126
- unless format.is_a?(Symbol) || format.is_a?(String)
127
- raise ArgumentError, "configuration #{index} format must be a Symbol or String"
128
- end
129
-
130
- # Validate log levels if specified
131
- if config[:min_level] && !valid_log_level?(config[:min_level])
132
- raise ArgumentError, "configuration #{index} min_level must be a valid log level"
133
- end
134
-
135
- if config[:max_level] && !valid_log_level?(config[:max_level])
136
- raise ArgumentError, "configuration #{index} max_level must be a valid log level"
137
- end
138
- end
139
-
140
- # Check if log level is valid
141
- #
142
- # @param level [Symbol] log level to check
143
- # @return [Boolean] true if valid
144
- def valid_log_level?(level)
145
- %i[debug info warn error fatal success].include?(level)
146
- end
147
-
148
- # Initialize formatters for each configuration
149
- #
150
- # @return [void]
151
- def initialize_formatters
152
- @configurations.each_with_index do |config, index|
153
- format = config[:format].to_sym
154
- formatter_options = config[:formatter_options] || {}
155
-
156
- # Merge configuration options into formatter options
157
- formatter_options[:include_context] = config[:include_context] if config.key?(:include_context)
158
- formatter_options[:include_metadata] = config[:include_metadata] if config.key?(:include_metadata)
159
- formatter_options[:show_timestamp] = config[:show_timestamp] if config.key?(:show_timestamp)
160
- formatter_options[:show_level] = config[:show_level] if config.key?(:show_level)
161
-
162
- begin
163
- formatter_class = FormatRegistry.get(format)
164
-
165
- # Handle different formatter initialization patterns
166
- if formatter_class == Formatters::ConsoleFormatter
167
- # ConsoleFormatter expects keyword arguments
168
- @formatters[index] = formatter_class.new(
169
- show_timestamp: formatter_options[:show_timestamp] || false,
170
- show_level: formatter_options[:show_level] || false,
171
- )
172
- elsif formatter_class == Formatters::TextFormatter
173
- # TextFormatter expects keyword arguments
174
- @formatters[index] = formatter_class.new(
175
- timestamp_format: formatter_options[:timestamp_format] || "%Y-%m-%d %H:%M:%S",
176
- include_context: formatter_options[:include_context] || false,
177
- )
178
- elsif formatter_class == Formatters::JsonFormatter
179
- # JsonFormatter expects keyword arguments
180
- @formatters[index] = formatter_class.new(
181
- options: formatter_options,
182
- )
183
- elsif formatter_class == Formatters::PlainTextFormatter
184
- # PlainTextFormatter expects keyword arguments
185
- @formatters[index] = formatter_class.new(
186
- include_context: formatter_options[:include_context] || false,
187
- )
188
- else
189
- # Other formatters expect hash or no arguments
190
- if formatter_options.empty?
191
- @formatters[index] = formatter_class.new
192
- else
193
- @formatters[index] = formatter_class.new(formatter_options)
194
- end
195
- end
196
- rescue ArgumentError => e
197
- raise ArgumentError, "configuration #{index} has invalid format '#{format}': #{e.message}"
198
- end
199
- end
200
- end
201
-
202
- # Initialize file handles for each configuration
203
- #
204
- # @return [void]
205
- def initialize_files
206
- @configurations.each_with_index do |config, index|
207
- file = config[:file]
208
-
209
- if file.is_a?(String)
210
- # Ensure directory exists
211
- FileUtils.mkdir_p(File.dirname(file)) unless File.dirname(file) == "."
212
-
213
- # Open file with appropriate mode
214
- mode = config[:append] != false ? "a" : "w"
215
- @files[index] = File.open(file, mode)
216
- else
217
- # Use IO object directly
218
- @files[index] = file
219
- end
220
- end
221
- end
222
-
223
- # Write log request to all applicable outputs
224
- #
225
- # @param log_request [LogRequest] the log request to write
226
- def write_to_outputs(log_request)
227
- @configurations.each_with_index do |config, index|
228
- next unless should_output?(log_request, config)
229
-
230
- formatter = @formatters[index]
231
- file = @files[index]
232
-
233
- formatted_message = formatter.format(log_request)
234
- file.write(formatted_message)
235
- file.write("\n") unless config[:no_newline]
236
- file.flush
237
- end
238
- end
239
-
240
- # Check if log request should be output to this configuration
241
- #
242
- # @param log_request [LogRequest] the log request
243
- # @param config [Hash] the output configuration
244
- # @return [Boolean] true if should output
245
- def should_output?(log_request, config)
246
- # Check level filtering
247
- if config[:min_level] && !level_greater_or_equal?(log_request.level, config[:min_level])
248
- return false
249
- end
250
-
251
- if config[:max_level] && !level_less_or_equal?(log_request.level, config[:max_level])
252
- return false
253
- end
254
-
255
- # Check custom condition
256
- if config[:condition] && !config[:condition].call(log_request)
257
- return false
258
- end
259
-
260
- true
261
- end
262
-
263
- # Check if log level is greater than or equal to minimum level
264
- #
265
- # @param level [Symbol] log level to check
266
- # @param min_level [Symbol] minimum level
267
- # @return [Boolean] true if level >= min_level
268
- def level_greater_or_equal?(level, min_level)
269
- levels = %i[debug info warn error fatal success]
270
- levels.index(level) >= levels.index(min_level)
271
- end
272
-
273
- # Check if log level is less than or equal to maximum level
274
- #
275
- # @param level [Symbol] log level to check
276
- # @param max_level [Symbol] maximum level
277
- # @return [Boolean] true if level <= max_level
278
- def level_less_or_equal?(level, max_level)
279
- levels = %i[debug info warn error fatal success]
280
- levels.index(level) <= levels.index(max_level)
281
- end
282
-
283
- # Sanitize configuration for display (remove sensitive data)
284
- #
285
- # @param config [Hash] configuration to sanitize
286
- # @return [Hash] sanitized configuration
287
- def sanitize_config(config)
288
- sanitized = config.dup
289
-
290
- # Remove condition procs and other non-serializable items
291
- sanitized.delete(:condition)
292
-
293
- # Convert IO objects to string representation
294
- if sanitized[:file].is_a?(IO)
295
- sanitized[:file] = "<#{sanitized[:file].class.name}>"
296
- end
297
-
298
- sanitized
299
- end
300
- end
301
- end
302
- end
303
- end
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "base"
4
+ require_relative "../format_registry"
5
+ require "fileutils"
6
+
7
+ module Makit
8
+ module Logging
9
+ module Sinks
10
+ # Unified file sink with configurable multiple outputs
11
+ #
12
+ # This sink replaces the need for multiple separate sinks by providing
13
+ # a single, highly configurable sink that can write to multiple files
14
+ # with different formats, filtering, and options. It supports console
15
+ # output, file logging, and structured logging through configuration.
16
+ #
17
+ # @example Basic usage with console and file output
18
+ # sink = UnifiedFileSink.new(
19
+ # configurations: [
20
+ # { file: $stdout, format: :console },
21
+ # { file: "logs/app.log", format: :json }
22
+ # ]
23
+ # )
24
+ #
25
+ # @example Advanced configuration with filtering and formatting
26
+ # sink = UnifiedFileSink.new(
27
+ # configurations: [
28
+ # {
29
+ # file: $stdout,
30
+ # format: :console,
31
+ # show_timestamp: true,
32
+ # show_level: true
33
+ # },
34
+ # {
35
+ # file: "logs/debug.log",
36
+ # format: :text,
37
+ # min_level: :debug,
38
+ # include_context: true
39
+ # },
40
+ # {
41
+ # file: "logs/audit.log",
42
+ # format: :json,
43
+ # min_level: :info,
44
+ # include_metadata: true
45
+ # }
46
+ # ]
47
+ # )
48
+ class UnifiedFileSink < Base
49
+ # @return [Array<Hash>] list of output configurations
50
+ attr_reader :configurations
51
+
52
+ # Initialize unified file sink
53
+ #
54
+ # @param configurations [Array<Hash>] list of output configurations
55
+ def initialize(configurations: [])
56
+ @configurations = validate_configurations(configurations)
57
+ @formatters = {}
58
+ @files = {}
59
+
60
+ # Pre-initialize formatters and files
61
+ initialize_formatters
62
+ initialize_files
63
+ end
64
+
65
+ # Execute sink logic to write log entry to configured outputs
66
+ #
67
+ # @param log_request [LogRequest] the log request to process
68
+ # @yield [LogRequest] yields the processed log request to the next sink
69
+ # @yieldreturn [LogRequest] the processed log request
70
+ # @return [LogRequest] the final processed log request
71
+ def call(log_request, &block)
72
+ write_to_outputs(log_request)
73
+ block&.call(log_request) if block_given?
74
+ log_request
75
+ end
76
+
77
+ # Get sink configuration
78
+ #
79
+ # @return [Hash] sink configuration
80
+ def config
81
+ {
82
+ name: self.class.name.split("::").last,
83
+ configurations_count: @configurations.length,
84
+ configurations: @configurations.map { |config| sanitize_config(config) },
85
+ }
86
+ end
87
+
88
+ private
89
+
90
+ # Validate configuration array
91
+ #
92
+ # @param configurations [Array<Hash>] configurations to validate
93
+ # @return [Array<Hash>] validated configurations
94
+ # @raise [ArgumentError] if configuration is invalid
95
+ def validate_configurations(configurations)
96
+ raise ArgumentError, "configurations must be an array" unless configurations.is_a?(Array)
97
+ raise ArgumentError, "at least one configuration is required" if configurations.empty?
98
+
99
+ configurations.each_with_index do |config, index|
100
+ validate_single_configuration(config, index)
101
+ end
102
+
103
+ configurations
104
+ end
105
+
106
+ # Validate a single configuration
107
+ #
108
+ # @param config [Hash] configuration to validate
109
+ # @param index [Integer] configuration index for error messages
110
+ # @raise [ArgumentError] if configuration is invalid
111
+ def validate_single_configuration(config, index)
112
+ raise ArgumentError, "configuration #{index} must be a hash" unless config.is_a?(Hash)
113
+
114
+ required_keys = [:file, :format]
115
+ missing_keys = required_keys - config.keys
116
+ raise ArgumentError, "configuration #{index} missing required keys: #{missing_keys.join(", ")}" unless missing_keys.empty?
117
+
118
+ # Validate file
119
+ file = config[:file]
120
+ unless file.is_a?(String) || file.is_a?(IO) || file.respond_to?(:write)
121
+ raise ArgumentError, "configuration #{index} file must be a String path, IO object, or writable object"
122
+ end
123
+
124
+ # Validate format
125
+ format = config[:format]
126
+ unless format.is_a?(Symbol) || format.is_a?(String)
127
+ raise ArgumentError, "configuration #{index} format must be a Symbol or String"
128
+ end
129
+
130
+ # Validate log levels if specified
131
+ if config[:min_level] && !valid_log_level?(config[:min_level])
132
+ raise ArgumentError, "configuration #{index} min_level must be a valid log level"
133
+ end
134
+
135
+ if config[:max_level] && !valid_log_level?(config[:max_level])
136
+ raise ArgumentError, "configuration #{index} max_level must be a valid log level"
137
+ end
138
+ end
139
+
140
+ # Check if log level is valid
141
+ #
142
+ # @param level [Symbol] log level to check
143
+ # @return [Boolean] true if valid
144
+ def valid_log_level?(level)
145
+ %i[debug info warn error fatal success].include?(level)
146
+ end
147
+
148
+ # Initialize formatters for each configuration
149
+ #
150
+ # @return [void]
151
+ def initialize_formatters
152
+ @configurations.each_with_index do |config, index|
153
+ format = config[:format].to_sym
154
+ formatter_options = config[:formatter_options] || {}
155
+
156
+ # Merge configuration options into formatter options
157
+ formatter_options[:include_context] = config[:include_context] if config.key?(:include_context)
158
+ formatter_options[:include_metadata] = config[:include_metadata] if config.key?(:include_metadata)
159
+ formatter_options[:show_timestamp] = config[:show_timestamp] if config.key?(:show_timestamp)
160
+ formatter_options[:show_level] = config[:show_level] if config.key?(:show_level)
161
+
162
+ begin
163
+ formatter_class = FormatRegistry.get(format)
164
+
165
+ # Handle different formatter initialization patterns
166
+ if formatter_class == Formatters::ConsoleFormatter
167
+ # ConsoleFormatter expects keyword arguments
168
+ @formatters[index] = formatter_class.new(
169
+ show_timestamp: formatter_options[:show_timestamp] || false,
170
+ show_level: formatter_options[:show_level] || false,
171
+ )
172
+ elsif formatter_class == Formatters::TextFormatter
173
+ # TextFormatter expects keyword arguments
174
+ @formatters[index] = formatter_class.new(
175
+ timestamp_format: formatter_options[:timestamp_format] || "%Y-%m-%d %H:%M:%S",
176
+ include_context: formatter_options[:include_context] || false,
177
+ )
178
+ elsif formatter_class == Formatters::JsonFormatter
179
+ # JsonFormatter expects keyword arguments
180
+ @formatters[index] = formatter_class.new(
181
+ options: formatter_options,
182
+ )
183
+ elsif formatter_class == Formatters::PlainTextFormatter
184
+ # PlainTextFormatter expects keyword arguments
185
+ @formatters[index] = formatter_class.new(
186
+ include_context: formatter_options[:include_context] || false,
187
+ )
188
+ else
189
+ # Other formatters expect hash or no arguments
190
+ if formatter_options.empty?
191
+ @formatters[index] = formatter_class.new
192
+ else
193
+ @formatters[index] = formatter_class.new(formatter_options)
194
+ end
195
+ end
196
+ rescue ArgumentError => e
197
+ raise ArgumentError, "configuration #{index} has invalid format '#{format}': #{e.message}"
198
+ end
199
+ end
200
+ end
201
+
202
+ # Initialize file handles for each configuration
203
+ #
204
+ # @return [void]
205
+ def initialize_files
206
+ @configurations.each_with_index do |config, index|
207
+ file = config[:file]
208
+
209
+ if file.is_a?(String)
210
+ # Ensure directory exists
211
+ FileUtils.mkdir_p(File.dirname(file)) unless File.dirname(file) == "."
212
+
213
+ # Open file with appropriate mode
214
+ mode = config[:append] != false ? "a" : "w"
215
+ @files[index] = File.open(file, mode)
216
+ else
217
+ # Use IO object directly
218
+ @files[index] = file
219
+ end
220
+ end
221
+ end
222
+
223
+ # Write log request to all applicable outputs
224
+ #
225
+ # @param log_request [LogRequest] the log request to write
226
+ def write_to_outputs(log_request)
227
+ @configurations.each_with_index do |config, index|
228
+ next unless should_output?(log_request, config)
229
+
230
+ formatter = @formatters[index]
231
+ file = @files[index]
232
+
233
+ formatted_message = formatter.format(log_request)
234
+ file.write(formatted_message)
235
+ file.write("\n") unless config[:no_newline]
236
+ file.flush
237
+ end
238
+ end
239
+
240
+ # Check if log request should be output to this configuration
241
+ #
242
+ # @param log_request [LogRequest] the log request
243
+ # @param config [Hash] the output configuration
244
+ # @return [Boolean] true if should output
245
+ def should_output?(log_request, config)
246
+ # Check level filtering
247
+ if config[:min_level] && !level_greater_or_equal?(log_request.level, config[:min_level])
248
+ return false
249
+ end
250
+
251
+ if config[:max_level] && !level_less_or_equal?(log_request.level, config[:max_level])
252
+ return false
253
+ end
254
+
255
+ # Check custom condition
256
+ if config[:condition] && !config[:condition].call(log_request)
257
+ return false
258
+ end
259
+
260
+ true
261
+ end
262
+
263
+ # Check if log level is greater than or equal to minimum level
264
+ #
265
+ # @param level [Symbol] log level to check
266
+ # @param min_level [Symbol] minimum level
267
+ # @return [Boolean] true if level >= min_level
268
+ def level_greater_or_equal?(level, min_level)
269
+ levels = %i[debug info warn error fatal success]
270
+ levels.index(level) >= levels.index(min_level)
271
+ end
272
+
273
+ # Check if log level is less than or equal to maximum level
274
+ #
275
+ # @param level [Symbol] log level to check
276
+ # @param max_level [Symbol] maximum level
277
+ # @return [Boolean] true if level <= max_level
278
+ def level_less_or_equal?(level, max_level)
279
+ levels = %i[debug info warn error fatal success]
280
+ levels.index(level) <= levels.index(max_level)
281
+ end
282
+
283
+ # Sanitize configuration for display (remove sensitive data)
284
+ #
285
+ # @param config [Hash] configuration to sanitize
286
+ # @return [Hash] sanitized configuration
287
+ def sanitize_config(config)
288
+ sanitized = config.dup
289
+
290
+ # Remove condition procs and other non-serializable items
291
+ sanitized.delete(:condition)
292
+
293
+ # Convert IO objects to string representation
294
+ if sanitized[:file].is_a?(IO)
295
+ sanitized[:file] = "<#{sanitized[:file].class.name}>"
296
+ end
297
+
298
+ sanitized
299
+ end
300
+ end
301
+ end
302
+ end
303
+ end