Module: ReactOnRails::PackerUtils
- Defined in:
- lib/react_on_rails/packer_utils.rb
Class Method Summary collapse
-
.asset_uri_from_packer(asset_name) ⇒ Object
The function doesn’t ensure that the asset exists.
-
.bundle_js_uri_from_packer(bundle_name) ⇒ Object
This returns either a URL for the webpack-dev-server, non-server bundle or the hashed server bundle if using the same bundle for the client.
- .check_manifest_not_cached ⇒ Object
- .dev_server_running? ⇒ Boolean
- .dev_server_url ⇒ Object
- .extract_precompile_hook ⇒ Object
- .hook_contains_generate_packs?(hook_value) ⇒ Boolean
- .manifest_exists? ⇒ Boolean
- .nested_entries? ⇒ Boolean
- .packer_public_output_path ⇒ Object
- .packer_source_entry_path ⇒ Object
- .packer_source_path ⇒ Object
- .packer_source_path_explicit? ⇒ Boolean
- .precompile? ⇒ Boolean
- .public_output_uri_path ⇒ Object
- .raise_nested_entries_disabled ⇒ Object
- .raise_shakapacker_version_incompatible_for_autobundling ⇒ Object
- .raise_shakapacker_version_incompatible_for_basic_pack_generation ⇒ Object
- .resolve_hook_script_path(hook_value) ⇒ Object
-
.shakapacker_precompile_hook_configured? ⇒ Boolean
Check if shakapacker.yml has a precompile hook configured This prevents react_on_rails from running generate_packs twice.
-
.shakapacker_precompile_hook_value ⇒ Object
Returns the configured precompile hook value for logging/debugging Returns nil if no hook is configured.
- .shakapacker_version ⇒ Object
- .shakapacker_version_as_array ⇒ Object
- .shakapacker_version_requirement_met?(required_version) ⇒ Boolean
- .supports_async_loading? ⇒ Boolean
- .supports_autobundling? ⇒ Boolean
- .supports_basic_pack_generation? ⇒ Boolean
- .webpack_assets_status_checker ⇒ Object
Class Method Details
.asset_uri_from_packer(asset_name) ⇒ Object
The function doesn’t ensure that the asset exists.
-
It just returns url to the asset if dev server is running
-
Otherwise it returns file path to the asset
80 81 82 83 84 85 86 |
# File 'lib/react_on_rails/packer_utils.rb', line 80 def self.asset_uri_from_packer(asset_name) if dev_server_running? "#{dev_server_url}/#{public_output_uri_path}#{asset_name}" else File.join(packer_public_output_path, asset_name).to_s end end |
.bundle_js_uri_from_packer(bundle_name) ⇒ Object
This returns either a URL for the webpack-dev-server, non-server bundle or the hashed server bundle if using the same bundle for the client. Otherwise returns a file path.
51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 |
# File 'lib/react_on_rails/packer_utils.rb', line 51 def self.bundle_js_uri_from_packer(bundle_name) hashed_bundle_name = ::Shakapacker.manifest.lookup!(bundle_name) # Support for hashing the server-bundle and having that built # the webpack-dev-server is provided by the config value # "same_bundle_for_client_and_server" where a value of true # would mean that the bundle is created by the webpack-dev-server is_bundle_running_on_server = bundle_name == ReactOnRails.configuration.server_bundle_js_file # Check Pro RSC bundle if Pro is available if ReactOnRails::Utils.react_on_rails_pro? is_bundle_running_on_server ||= (bundle_name == ReactOnRailsPro.configuration.rsc_bundle_js_file) end if ::Shakapacker.dev_server.running? && (!is_bundle_running_on_server || ReactOnRails.configuration.same_bundle_for_client_and_server) "#{dev_server_url}#{hashed_bundle_name}" else File.(File.join("public", hashed_bundle_name)).to_s end end |
.check_manifest_not_cached ⇒ Object
116 117 118 119 120 121 122 123 124 125 126 |
# File 'lib/react_on_rails/packer_utils.rb', line 116 def self.check_manifest_not_cached return unless ::Shakapacker.config.cache_manifest? msg = <<-MSG.strip_heredoc ERROR: you have enabled cache_manifest in the #{Rails.env} env when using the ReactOnRails::TestHelper.configure_rspec_to_compile_assets helper To fix this: edit your config/shakapacker.yml file and set cache_manifest to false for test. MSG puts (msg) exit! end |
.dev_server_running? ⇒ Boolean
7 8 9 |
# File 'lib/react_on_rails/packer_utils.rb', line 7 def self.dev_server_running? Shakapacker.dev_server.running? end |
.dev_server_url ⇒ Object
11 12 13 |
# File 'lib/react_on_rails/packer_utils.rb', line 11 def self.dev_server_url "#{Shakapacker.dev_server.protocol}://#{Shakapacker.dev_server.host_with_port}" end |
.extract_precompile_hook ⇒ Object
194 195 196 197 198 199 200 201 202 |
# File 'lib/react_on_rails/packer_utils.rb', line 194 def self.extract_precompile_hook # Access config data using private :data method since there's no public API # to access the raw configuration hash needed for hook detection config_data = ::Shakapacker.config.send(:data) # Try symbol keys first (Shakapacker's internal format), then fall back to string keys # The key is 'precompile_hook' at the top level of the config config_data&.[](:precompile_hook) || config_data&.[]("precompile_hook") end |
.hook_contains_generate_packs?(hook_value) ⇒ Boolean
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 |
# File 'lib/react_on_rails/packer_utils.rb', line 204 def self.hook_contains_generate_packs?(hook_value) # The hook value can be either: # 1. A direct command containing the rake task # 2. A path to a script file that needs to be read return false if hook_value.blank? # Check if it's a direct command first return true if hook_value.to_s.match?(/\breact_on_rails:generate_packs\b/) # Check if it's a script file path script_path = resolve_hook_script_path(hook_value) return false unless script_path && File.exist?(script_path) # Read and check script contents script_contents = File.read(script_path) script_contents.match?(/\breact_on_rails:generate_packs\b/) rescue StandardError # If we can't read the script, assume it doesn't contain generate_packs false end |
.manifest_exists? ⇒ Boolean
108 109 110 |
# File 'lib/react_on_rails/packer_utils.rb', line 108 def self.manifest_exists? ::Shakapacker.config.public_manifest_path.exist? end |
.nested_entries? ⇒ Boolean
100 101 102 |
# File 'lib/react_on_rails/packer_utils.rb', line 100 def self.nested_entries? ::Shakapacker.config.nested_entries? end |
.packer_public_output_path ⇒ Object
104 105 106 |
# File 'lib/react_on_rails/packer_utils.rb', line 104 def self.packer_public_output_path ::Shakapacker.config.public_output_path.to_s end |
.packer_source_entry_path ⇒ Object
96 97 98 |
# File 'lib/react_on_rails/packer_utils.rb', line 96 def self.packer_source_entry_path ::Shakapacker.config.source_entry_path end |
.packer_source_path ⇒ Object
92 93 94 |
# File 'lib/react_on_rails/packer_utils.rb', line 92 def self.packer_source_path ::Shakapacker.config.source_path end |
.packer_source_path_explicit? ⇒ Boolean
112 113 114 |
# File 'lib/react_on_rails/packer_utils.rb', line 112 def self.packer_source_path_explicit? ::Shakapacker.config.send(:data)[:source_path].present? end |
.precompile? ⇒ Boolean
88 89 90 |
# File 'lib/react_on_rails/packer_utils.rb', line 88 def self.precompile? ::Shakapacker.config.shakapacker_precompile? end |
.public_output_uri_path ⇒ Object
73 74 75 |
# File 'lib/react_on_rails/packer_utils.rb', line 73 def self.public_output_uri_path "#{::Shakapacker.config.public_output_path.relative_path_from(::Shakapacker.config.public_path)}/" end |
.raise_nested_entries_disabled ⇒ Object
140 141 142 143 144 145 146 147 148 |
# File 'lib/react_on_rails/packer_utils.rb', line 140 def self.raise_nested_entries_disabled msg = <<~MSG **ERROR** ReactOnRails: `nested_entries` is configured to be disabled in shakapacker. Please update \ config/shakapacker.yml to enable nested entries. for more information read https://www.shakacode.com/react-on-rails/docs/guides/file-system-based-automated-bundle-generation.md#enable-nested_entries-for-shakapacker MSG raise ReactOnRails::Error, msg end |
.raise_shakapacker_version_incompatible_for_autobundling ⇒ Object
150 151 152 153 154 155 156 157 158 159 |
# File 'lib/react_on_rails/packer_utils.rb', line 150 def self.raise_shakapacker_version_incompatible_for_autobundling msg = <<~MSG **ERROR** ReactOnRails: Please upgrade ::Shakapacker to version #{ReactOnRails::PacksGenerator::MINIMUM_SHAKAPACKER_VERSION_FOR_AUTO_BUNDLING} or \ above to use the automated bundle generation feature (which requires nested_entries support). \ The currently installed version is #{ReactOnRails::PackerUtils.shakapacker_version}. \ Basic pack generation requires ::Shakapacker #{ReactOnRails::PacksGenerator::MINIMUM_SHAKAPACKER_VERSION} or above. MSG raise ReactOnRails::Error, msg end |
.raise_shakapacker_version_incompatible_for_basic_pack_generation ⇒ Object
161 162 163 164 165 166 167 168 |
# File 'lib/react_on_rails/packer_utils.rb', line 161 def self.raise_shakapacker_version_incompatible_for_basic_pack_generation msg = <<~MSG **ERROR** ReactOnRails: Please upgrade ::Shakapacker to version #{ReactOnRails::PacksGenerator::MINIMUM_SHAKAPACKER_VERSION} or \ above to use basic pack generation features. The currently installed version is #{ReactOnRails::PackerUtils.shakapacker_version}. MSG raise ReactOnRails::Error, msg end |
.resolve_hook_script_path(hook_value) ⇒ Object
225 226 227 228 229 230 231 |
# File 'lib/react_on_rails/packer_utils.rb', line 225 def self.resolve_hook_script_path(hook_value) # Hook value might be a script path relative to Rails root return nil unless defined?(Rails) && Rails.respond_to?(:root) potential_path = Rails.root.join(hook_value.to_s.strip) potential_path if potential_path.file? end |
.shakapacker_precompile_hook_configured? ⇒ Boolean
Check if shakapacker.yml has a precompile hook configured This prevents react_on_rails from running generate_packs twice
Returns false if detection fails for any reason (missing shakapacker, malformed config, etc.) to ensure generate_packs runs rather than being incorrectly skipped
Note: Currently checks a single hook value. Future enhancement will support hook lists to allow prepending/appending multiple commands. See related Shakapacker issue for details.
178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 |
# File 'lib/react_on_rails/packer_utils.rb', line 178 def self.shakapacker_precompile_hook_configured? return false unless defined?(::Shakapacker) hook_value = extract_precompile_hook return false if hook_value.nil? hook_contains_generate_packs?(hook_value) rescue StandardError => e # Swallow errors during hook detection to fail safe - if we can't detect the hook, # we should run generate_packs rather than skip it incorrectly. # Possible errors: NoMethodError (config method missing), TypeError (unexpected data structure), # or errors from shakapacker's internal implementation changes warn "Warning: Unable to detect shakapacker precompile hook: #{e.}" if ENV["DEBUG"] false end |
.shakapacker_precompile_hook_value ⇒ Object
Returns the configured precompile hook value for logging/debugging Returns nil if no hook is configured
235 236 237 238 239 240 241 |
# File 'lib/react_on_rails/packer_utils.rb', line 235 def self.shakapacker_precompile_hook_value return nil unless defined?(::Shakapacker) extract_precompile_hook rescue StandardError nil end |
.shakapacker_version ⇒ Object
15 16 17 18 19 |
# File 'lib/react_on_rails/packer_utils.rb', line 15 def self.shakapacker_version return @shakapacker_version if defined?(@shakapacker_version) @shakapacker_version = Gem.loaded_specs["shakapacker"].version.to_s end |
.shakapacker_version_as_array ⇒ Object
21 22 23 24 25 26 27 28 |
# File 'lib/react_on_rails/packer_utils.rb', line 21 def self.shakapacker_version_as_array return @shakapacker_version_as_array if defined?(@shakapacker_version_as_array) match = shakapacker_version.match(ReactOnRails::VersionChecker::VERSION_PARTS_REGEX) # match[4] is the pre-release version, not normally a number but something like "beta.1" or `nil` @shakapacker_version_as_array = [match[1].to_i, match[2].to_i, match[3].to_i, match[4]].compact end |
.shakapacker_version_requirement_met?(required_version) ⇒ Boolean
30 31 32 33 |
# File 'lib/react_on_rails/packer_utils.rb', line 30 def self.shakapacker_version_requirement_met?(required_version) @version_checks ||= {} @version_checks[required_version] ||= Gem::Version.new(shakapacker_version) >= Gem::Version.new(required_version) end |
.supports_async_loading? ⇒ Boolean
35 36 37 |
# File 'lib/react_on_rails/packer_utils.rb', line 35 def self.supports_async_loading? shakapacker_version_requirement_met?("8.2.0") end |
.supports_autobundling? ⇒ Boolean
43 44 45 46 |
# File 'lib/react_on_rails/packer_utils.rb', line 43 def self.supports_autobundling? min_version = ReactOnRails::PacksGenerator::MINIMUM_SHAKAPACKER_VERSION_FOR_AUTO_BUNDLING ::Shakapacker.config.respond_to?(:nested_entries?) && shakapacker_version_requirement_met?(min_version) end |
.supports_basic_pack_generation? ⇒ Boolean
39 40 41 |
# File 'lib/react_on_rails/packer_utils.rb', line 39 def self.supports_basic_pack_generation? shakapacker_version_requirement_met?(ReactOnRails::PacksGenerator::MINIMUM_SHAKAPACKER_VERSION) end |
.webpack_assets_status_checker ⇒ Object
128 129 130 131 132 133 134 135 136 137 138 |
# File 'lib/react_on_rails/packer_utils.rb', line 128 def self.webpack_assets_status_checker source_path = ReactOnRails::Utils.source_path generated_assets_full_path = ReactOnRails::Utils.generated_assets_full_path webpack_generated_files = ReactOnRails.configuration.webpack_generated_files @webpack_assets_status_checker ||= ReactOnRails::TestHelper::WebpackAssetsStatusChecker.new( source_path: source_path, generated_assets_full_path: generated_assets_full_path, webpack_generated_files: webpack_generated_files ) end |