Module: ReactOnRails::Utils
- Defined in:
- lib/react_on_rails/utils.rb
Defined Under Namespace
Modules: Required
Constant Summary collapse
- TRUNCATION_FILLER =
"\n... TRUNCATED #{ Rainbow('To see the full output, set FULL_TEXT_ERRORS=true.').red } ...\n".freeze
Class Method Summary collapse
- .bundle_js_file_path(bundle_name) ⇒ Object
- .default_troubleshooting_section ⇒ Object
-
.detect_package_manager ⇒ Symbol
Detects which package manager is being used.
- .find_most_recent_mtime(files) ⇒ Object
- .full_text_errors_enabled? ⇒ Boolean
- .gem_available?(name) ⇒ Boolean
-
.generated_assets_full_path ⇒ Object
DEPRECATED: Use public_bundles_full_path for clarity about public vs private bundle paths.
-
.invoke_and_exit_if_failed(cmd, failure_message) ⇒ Object
Invokes command, exiting with a detailed message if there’s a failure.
- .object_to_boolean(value) ⇒ Object
-
.package_manager_install_exact_command(package_name, version) ⇒ String
Returns the appropriate install command for the detected package manager.
-
.package_manager_remove_command(package_name) ⇒ String
Returns the appropriate remove command for the detected package manager.
- .prepend_cd_node_modules_directory(cmd) ⇒ Object
- .prepend_to_file_if_text_not_present(file:, text_to_prepend:, regex:) ⇒ Object
- .public_bundles_full_path ⇒ Object
- .rails_version_less_than(version) ⇒ Object
-
.react_on_rails_pro? ⇒ Boolean
Checks if React on Rails Pro is installed and licensed.
-
.react_on_rails_pro_version ⇒ Object
Return an empty string if React on Rails Pro is not installed.
-
.rsc_support_enabled? ⇒ Boolean
RSC support detection has been moved to React on Rails Pro See react_on_rails_pro/lib/react_on_rails_pro/utils.rb.
- .running_on_windows? ⇒ Boolean
- .server_bundle_js_file_path ⇒ Object
- .server_bundle_path_is_http? ⇒ Boolean
- .server_rendering_is_enabled? ⇒ Boolean
- .smart_trim(str, max_length = 1000) ⇒ Object
- .source_path ⇒ Object
-
.truthy_presence(obj) ⇒ Object
forum.shakacode.com/t/yak-of-the-week-ruby-2-4-pathname-empty-changed-to-look-at-file-size/901 return object if truthy, else return nil.
- .using_packer_source_path_is_not_defined_and_custom_node_modules? ⇒ Boolean
-
.wrap_message(msg, color = :red) ⇒ Object
Wraps message and makes it colored.
Class Method Details
.bundle_js_file_path(bundle_name) ⇒ Object
75 76 77 78 79 80 81 82 83 84 85 86 87 |
# File 'lib/react_on_rails/utils.rb', line 75 def self.bundle_js_file_path(bundle_name) # Priority order depends on bundle type: # SERVER BUNDLES (normal case): Try private non-public locations first, then manifest, then legacy # CLIENT BUNDLES (normal case): Try manifest first, then fallback locations if bundle_name == "manifest.json" # Default to the non-hashed name in the specified output directory, which, for legacy # React on Rails, this is the output directory picked up by the asset pipeline. # For Shakapacker, this is the public output path defined in the (shaka/web)packer.yml file. File.join(public_bundles_full_path, bundle_name) else bundle_js_file_path_with_packer(bundle_name) end end |
.default_troubleshooting_section ⇒ Object
406 407 408 409 410 411 412 413 414 |
# File 'lib/react_on_rails/utils.rb', line 406 def self.default_troubleshooting_section <<~DEFAULT 📞 Get Help & Support: • 🚀 Professional Support: react_on_rails@shakacode.com (fastest resolution) • 💬 React + Rails Slack: https://invite.reactrails.com • 🆓 GitHub Issues: https://github.com/shakacode/react_on_rails/issues • 📖 Discussions: https://github.com/shakacode/react_on_rails/discussions DEFAULT end |
.detect_package_manager ⇒ Symbol
Detects which package manager is being used. First checks the packageManager field in package.json (Node.js Corepack standard), then falls back to checking for lock files.
292 293 294 295 |
# File 'lib/react_on_rails/utils.rb', line 292 def self.detect_package_manager manager = detect_package_manager_from_package_json || detect_package_manager_from_lock_files manager || :yarn # Default to yarn if no detection succeeds end |
.find_most_recent_mtime(files) ⇒ Object
265 266 267 268 269 270 |
# File 'lib/react_on_rails/utils.rb', line 265 def self.find_most_recent_mtime(files) files.reduce(1.year.ago) do |newest_time, file| mt = File.mtime(file) [mt, newest_time].max end end |
.full_text_errors_enabled? ⇒ Boolean
245 246 247 |
# File 'lib/react_on_rails/utils.rb', line 245 def self.full_text_errors_enabled? ENV["FULL_TEXT_ERRORS"] == "true" end |
.gem_available?(name) ⇒ Boolean
199 200 201 202 203 204 205 206 207 208 209 |
# File 'lib/react_on_rails/utils.rb', line 199 def self.gem_available?(name) Gem.loaded_specs[name].present? rescue Gem::LoadError false rescue StandardError begin Gem.available?(name).present? rescue NoMethodError false end end |
.generated_assets_full_path ⇒ Object
DEPRECATED: Use public_bundles_full_path for clarity about public vs private bundle paths
195 196 197 |
# File 'lib/react_on_rails/utils.rb', line 195 def self.generated_assets_full_path public_bundles_full_path end |
.invoke_and_exit_if_failed(cmd, failure_message) ⇒ Object
Invokes command, exiting with a detailed message if there’s a failure.
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
# File 'lib/react_on_rails/utils.rb', line 49 def self.invoke_and_exit_if_failed(cmd, ) stdout, stderr, status = Open3.capture3(cmd) unless status.success? stdout_msg = stdout.present? ? "\nstdout:\n#{stdout.strip}\n" : "" stderr_msg = stderr.present? ? "\nstderr:\n#{stderr.strip}\n" : "" msg = <<~MSG React on Rails FATAL ERROR! #{} cmd: #{cmd} exitstatus: #{status.exitstatus}#{stdout_msg}#{stderr_msg} MSG puts (msg) puts "" puts default_troubleshooting_section # Rspec catches exit without! in the exit callbacks exit!(1) end [stdout, stderr, status] end |
.object_to_boolean(value) ⇒ Object
40 41 42 |
# File 'lib/react_on_rails/utils.rb', line 40 def self.object_to_boolean(value) [true, "true", "yes", 1, "1", "t"].include?(value.instance_of?(String) ? value.downcase : value) end |
.package_manager_install_exact_command(package_name, version) ⇒ String
Returns the appropriate install command for the detected package manager. Generates the correct command with exact version syntax.
364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 |
# File 'lib/react_on_rails/utils.rb', line 364 def self.package_manager_install_exact_command(package_name, version) validate_package_command_inputs!(package_name, version) manager = detect_package_manager # Escape shell arguments to prevent command injection safe_package = Shellwords.escape("#{package_name}@#{version}") case manager when :pnpm "pnpm add #{safe_package} --save-exact" when :bun "bun add #{safe_package} --exact" when :npm "npm install #{safe_package} --save-exact" else # :yarn or unknown, default to yarn "yarn add #{safe_package} --exact" end end |
.package_manager_remove_command(package_name) ⇒ String
Returns the appropriate remove command for the detected package manager.
387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 |
# File 'lib/react_on_rails/utils.rb', line 387 def self.package_manager_remove_command(package_name) validate_package_name!(package_name) manager = detect_package_manager # Escape shell arguments to prevent command injection safe_package = Shellwords.escape(package_name) case manager when :pnpm "pnpm remove #{safe_package}" when :bun "bun remove #{safe_package}" when :npm "npm uninstall #{safe_package}" else # :yarn or unknown, default to yarn "yarn remove #{safe_package}" end end |
.prepend_cd_node_modules_directory(cmd) ⇒ Object
177 178 179 |
# File 'lib/react_on_rails/utils.rb', line 177 def self.prepend_cd_node_modules_directory(cmd) "cd \"#{ReactOnRails.configuration.node_modules_location}\" && #{cmd}" end |
.prepend_to_file_if_text_not_present(file:, text_to_prepend:, regex:) ⇒ Object
272 273 274 275 276 277 278 279 280 281 282 283 284 285 |
# File 'lib/react_on_rails/utils.rb', line 272 def self.prepend_to_file_if_text_not_present(file:, text_to_prepend:, regex:) if File.exist?(file) file_content = File.read(file) return if file_content.match(regex) content_with_prepended_text = text_to_prepend + file_content File.write(file, content_with_prepended_text, mode: "w") else File.write(file, text_to_prepend, mode: "w+") end puts "Prepended\n#{text_to_prepend}to #{file}." end |
.public_bundles_full_path ⇒ Object
190 191 192 |
# File 'lib/react_on_rails/utils.rb', line 190 def self.public_bundles_full_path ReactOnRails::PackerUtils.packer_public_output_path end |
.rails_version_less_than(version) ⇒ Object
161 162 163 164 165 166 167 168 169 |
# File 'lib/react_on_rails/utils.rb', line 161 def self.rails_version_less_than(version) @rails_version_less_than ||= {} return @rails_version_less_than[version] if @rails_version_less_than.key?(version) @rails_version_less_than[version] = begin Gem::Version.new(Rails.version) < Gem::Version.new(version) end end |
.react_on_rails_pro? ⇒ Boolean
Checks if React on Rails Pro is installed and licensed. This method validates the license and will raise an exception if invalid.
216 217 218 219 220 221 222 223 224 |
# File 'lib/react_on_rails/utils.rb', line 216 def self.react_on_rails_pro? return @react_on_rails_pro if defined?(@react_on_rails_pro) @react_on_rails_pro = begin return false unless gem_available?("react_on_rails_pro") ReactOnRailsPro::Utils.validated_license_data!.present? end end |
.react_on_rails_pro_version ⇒ Object
Return an empty string if React on Rails Pro is not installed
227 228 229 230 231 232 233 234 235 |
# File 'lib/react_on_rails/utils.rb', line 227 def self.react_on_rails_pro_version return @react_on_rails_pro_version if defined?(@react_on_rails_pro_version) @react_on_rails_pro_version = if react_on_rails_pro? Gem.loaded_specs["react_on_rails_pro"].version.to_s else "" end end |
.rsc_support_enabled? ⇒ Boolean
RSC support detection has been moved to React on Rails Pro See react_on_rails_pro/lib/react_on_rails_pro/utils.rb
239 240 241 242 243 |
# File 'lib/react_on_rails/utils.rb', line 239 def self.rsc_support_enabled? return false unless react_on_rails_pro? ReactOnRailsPro::Utils.rsc_support_enabled? end |
.running_on_windows? ⇒ Boolean
157 158 159 |
# File 'lib/react_on_rails/utils.rb', line 157 def self.running_on_windows? (/cygwin|mswin|mingw|bccwin|wince|emx/ =~ RUBY_PLATFORM) != nil end |
.server_bundle_js_file_path ⇒ Object
150 151 152 153 154 155 |
# File 'lib/react_on_rails/utils.rb', line 150 def self.server_bundle_js_file_path return @server_bundle_path if @server_bundle_path && !Rails.env.development? bundle_name = ReactOnRails.configuration.server_bundle_js_file @server_bundle_path = bundle_js_file_path(bundle_name) end |
.server_bundle_path_is_http? ⇒ Boolean
71 72 73 |
# File 'lib/react_on_rails/utils.rb', line 71 def self.server_bundle_path_is_http? server_bundle_js_file_path =~ %r{https?://} end |
.server_rendering_is_enabled? ⇒ Boolean
44 45 46 |
# File 'lib/react_on_rails/utils.rb', line 44 def self.server_rendering_is_enabled? ReactOnRails.configuration.server_bundle_js_file.present? end |
.smart_trim(str, max_length = 1000) ⇒ Object
249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 |
# File 'lib/react_on_rails/utils.rb', line 249 def self.smart_trim(str, max_length = 1000) # From https://stackoverflow.com/a/831583/1009332 str = str.to_s return str if full_text_errors_enabled? return str unless str.present? && max_length >= 1 return str if str.length <= max_length return str[0, 1] + TRUNCATION_FILLER if max_length == 1 midpoint = (str.length / 2.0).ceil to_remove = str.length - max_length lstrip = (to_remove / 2.0).ceil rstrip = to_remove - lstrip str[0..(midpoint - lstrip - 1)] + TRUNCATION_FILLER + str[(midpoint + rstrip)..] end |
.source_path ⇒ Object
181 182 183 |
# File 'lib/react_on_rails/utils.rb', line 181 def self.source_path ReactOnRails::PackerUtils.packer_source_path end |
.truthy_presence(obj) ⇒ Object
forum.shakacode.com/t/yak-of-the-week-ruby-2-4-pathname-empty-changed-to-look-at-file-size/901 return object if truthy, else return nil
19 20 21 22 23 24 25 |
# File 'lib/react_on_rails/utils.rb', line 19 def self.truthy_presence(obj) if obj.nil? || obj == false nil else obj end end |
.using_packer_source_path_is_not_defined_and_custom_node_modules? ⇒ Boolean
185 186 187 188 |
# File 'lib/react_on_rails/utils.rb', line 185 def self.using_packer_source_path_is_not_defined_and_custom_node_modules? !ReactOnRails::PackerUtils.packer_source_path_explicit? && ReactOnRails.configuration.node_modules_location.present? end |
.wrap_message(msg, color = :red) ⇒ Object
Wraps message and makes it colored. Pass in the msg and color as a symbol.
29 30 31 32 33 34 35 36 37 38 |
# File 'lib/react_on_rails/utils.rb', line 29 def self.(msg, color = :red) wrapper_line = ("=" * 80).to_s fenced_msg = <<~MSG #{wrapper_line} #{msg.strip} #{wrapper_line} MSG Rainbow(fenced_msg).color(color) end |