├── test ├── fixtures │ ├── assets │ │ ├── mapped │ │ │ ├── source.css.map │ │ │ ├── source.js.map │ │ │ ├── nested │ │ │ │ ├── another-source.js.map │ │ │ │ ├── sourceMappingURL-already-prefixed-nested.js.map │ │ │ │ ├── another-source.js │ │ │ │ └── sourceMappingURL-already-prefixed-nested.js │ │ │ ├── sourceMappingURL-already-prefixed.js.map │ │ │ ├── sourceMappingURL-not-at-start.css.map │ │ │ ├── sourceMappingURL-not-at-start.js.map │ │ │ ├── sourceMappingURL-already-prefixed-invalid.js.map │ │ │ ├── source.js │ │ │ ├── sourceless.js │ │ │ ├── source.css │ │ │ ├── sourceless.css │ │ │ ├── sourceMappingURL-not-at-start.js │ │ │ ├── sourceMappingURL-already-prefixed.js │ │ │ ├── sourceMappingURL-not-at-end.css │ │ │ ├── sourceMappingURL-not-at-start.css │ │ │ ├── sourceMappingURL-outside-comment.css │ │ │ └── sourceMappingURL-already-prefixed-invalid.js │ │ ├── vendor │ │ │ ├── foobar │ │ │ │ ├── source │ │ │ │ │ ├── file.svg │ │ │ │ │ ├── test.css │ │ │ │ │ ├── file.jpg │ │ │ │ │ ├── database.jpg │ │ │ │ │ ├── http-diagram.jpg │ │ │ │ │ └── images │ │ │ │ │ │ └── file.jpg │ │ │ │ ├── file.jpg │ │ │ │ └── sibling │ │ │ │ │ └── file.jpg │ │ │ └── file.jpg │ │ ├── first_path │ │ │ ├── .stuff │ │ │ ├── file-is-a-sourcemap.js.map │ │ │ ├── one.txt │ │ │ ├── again.js │ │ │ ├── dependent │ │ │ │ ├── a.css │ │ │ │ ├── b.css │ │ │ │ └── c.css │ │ │ ├── nested │ │ │ │ └── three.txt │ │ │ ├── another.css │ │ │ ├── dhh.jpg │ │ │ ├── file-not.digested.css │ │ │ ├── file-already-abcdefVWXYZ0123456789_-.digested.css │ │ │ ├── file-already-abcdefVWXYZ0123456789_-.digested.debug.css │ │ │ └── archive.svg │ │ └── second_path │ │ │ ├── one.txt │ │ │ └── two.txt │ ├── output │ │ ├── one-f2e1ec14.txt │ │ ├── one-f2e1ec15.txt.map │ │ └── .manifest.json │ └── new_manifest_format │ │ ├── one-f2e1ec14.txt │ │ ├── one-f2e1ec15.txt.map │ │ └── .manifest.json ├── dummy │ ├── app │ │ ├── assets │ │ │ ├── javascripts │ │ │ │ └── hello_world.js │ │ │ └── stylesheets │ │ │ │ ├── goodbye.css │ │ │ │ └── hello_world.css │ │ ├── controllers │ │ │ ├── application_controller.rb │ │ │ └── sample_controller.rb │ │ └── views │ │ │ ├── sample │ │ │ ├── load_nonexistent_assets.html.erb │ │ │ └── load_real_assets.html.erb │ │ │ └── layouts │ │ │ └── application.html.erb │ ├── lib │ │ └── assets │ │ │ ├── javascripts │ │ │ └── actioncable.js │ │ │ └── stylesheets │ │ │ └── library.css │ ├── config │ │ ├── routes.rb │ │ ├── environment.rb │ │ ├── boot.rb │ │ ├── application.rb │ │ └── environments │ │ │ ├── development.rb │ │ │ ├── test.rb │ │ │ └── production.rb │ ├── bin │ │ └── rails │ └── config.ru ├── test_helper.rb ├── propshaft │ ├── quiet_assets_test.rb │ ├── compilers_test.rb │ ├── assembly_test.rb │ ├── compiler │ │ ├── js_asset_urls_test.rb │ │ ├── source_mapping_urls_test.rb │ │ └── css_asset_urls_test.rb │ ├── resolver │ │ ├── dynamic_test.rb │ │ └── static_test.rb │ ├── processor_test.rb │ ├── output_path_test.rb │ ├── server_test.rb │ ├── load_path_test.rb │ ├── asset_test.rb │ ├── manifest_test.rb │ └── helper_test.rb └── propshaft_integration_test.rb ├── lib ├── propshaft │ ├── version.rb │ ├── quiet_assets.rb │ ├── errors.rb │ ├── compiler.rb │ ├── resolver │ │ ├── dynamic.rb │ │ └── static.rb │ ├── compilers.rb │ ├── compiler │ │ ├── source_mapping_urls.rb │ │ ├── css_asset_urls.rb │ │ └── js_asset_urls.rb │ ├── railties │ │ └── assets.rake │ ├── output_path.rb │ ├── server.rb │ ├── assembly.rb │ ├── processor.rb │ ├── asset.rb │ ├── railtie.rb │ ├── load_path.rb │ ├── manifest.rb │ └── helper.rb └── propshaft.rb ├── Gemfile ├── bin ├── test └── release ├── .devcontainer ├── Dockerfile └── devcontainer.json ├── gemfiles ├── Gemfile.rails-7.1 ├── Gemfile.rails-7.2 ├── Gemfile.rails-8.0 └── Gemfile.rails-7.0 ├── .gitignore ├── Rakefile ├── propshaft.gemspec ├── benchmarks ├── dynamic_resolver ├── static_resolver ├── css_asset_urls └── trackrod.rb ├── .github └── workflows │ └── ci.yml ├── MIT-LICENSE ├── Gemfile.lock ├── README.md └── UPGRADING.md /test/fixtures/assets/mapped/source.css.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/source.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/file.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/.stuff: -------------------------------------------------------------------------------- 1 | Won't be included -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/file-is-a-sourcemap.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/one.txt: -------------------------------------------------------------------------------- 1 | One from first path -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/nested/another-source.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/output/one-f2e1ec14.txt: -------------------------------------------------------------------------------- 1 | One from first path -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/again.js: -------------------------------------------------------------------------------- 1 | // This is JS! 2 | -------------------------------------------------------------------------------- /test/fixtures/assets/second_path/one.txt: -------------------------------------------------------------------------------- 1 | One from second path -------------------------------------------------------------------------------- /test/fixtures/assets/second_path/two.txt: -------------------------------------------------------------------------------- 1 | Two from second path -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-already-prefixed.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-not-at-start.css.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-not-at-start.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/output/one-f2e1ec15.txt.map: -------------------------------------------------------------------------------- 1 | One from first path map -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/dependent/a.css: -------------------------------------------------------------------------------- 1 | @import url('b.css') 2 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/nested/three.txt: -------------------------------------------------------------------------------- 1 | Three from first path -------------------------------------------------------------------------------- /test/fixtures/new_manifest_format/one-f2e1ec14.txt: -------------------------------------------------------------------------------- 1 | One from first path -------------------------------------------------------------------------------- /test/fixtures/output/.manifest.json: -------------------------------------------------------------------------------- 1 | { "one.txt": "one-f2e1ec14.txt" } 2 | -------------------------------------------------------------------------------- /lib/propshaft/version.rb: -------------------------------------------------------------------------------- 1 | module Propshaft 2 | VERSION = "1.3.1" 3 | end 4 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-already-prefixed-invalid.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/fixtures/new_manifest_format/one-f2e1ec15.txt.map: -------------------------------------------------------------------------------- 1 | One from first path map -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/hello_world.js: -------------------------------------------------------------------------------- 1 | console.log("Hello world!") 2 | -------------------------------------------------------------------------------- /test/dummy/lib/assets/javascripts/actioncable.js: -------------------------------------------------------------------------------- 1 | console.log("actioncable.js") 2 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/nested/sourceMappingURL-already-prefixed-nested.js.map: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/goodbye.css: -------------------------------------------------------------------------------- 1 | h2:after { 2 | content: "Goodbye!"; 3 | } 4 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/source.js: -------------------------------------------------------------------------------- 1 | var fun; 2 | //# sourceMappingURL=source.js.map 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/test.css: -------------------------------------------------------------------------------- 1 | .hero { background: url(file.jpg) } 2 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/hello_world.css: -------------------------------------------------------------------------------- 1 | h1:after { 2 | content: " Hello world!"; 3 | } 4 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/dependent/b.css: -------------------------------------------------------------------------------- 1 | @import url('c.css') 2 | @import url('missing.css') 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceless.js: -------------------------------------------------------------------------------- 1 | var failure; 2 | //# sourceMappingURL=sourceless.js.map 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/dependent/c.css: -------------------------------------------------------------------------------- 1 | @import url('a.css') 2 | 3 | p { 4 | color: red; 5 | } -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/source.css: -------------------------------------------------------------------------------- 1 | .class { color: green; } 2 | /*# sourceMappingURL=source.css.map */ 3 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/application_controller.rb: -------------------------------------------------------------------------------- 1 | class ApplicationController < ActionController::Base 2 | end 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/nested/another-source.js: -------------------------------------------------------------------------------- 1 | var extraFun; 2 | //# sourceMappingURL=another-source.js.map 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceless.css: -------------------------------------------------------------------------------- 1 | .failure { color: red; } 2 | /*# sourceMappingURL=sourceless.css.map */ 3 | -------------------------------------------------------------------------------- /Gemfile: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gemspec 4 | 5 | gem "rails", ">= 7.0.1" 6 | gem "rake" 7 | gem "debug" 8 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/file.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/file.jpg -------------------------------------------------------------------------------- /test/dummy/lib/assets/stylesheets/library.css: -------------------------------------------------------------------------------- 1 | # Library css that should not be included by :app but should be included by :all 2 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/another.css: -------------------------------------------------------------------------------- 1 | /* this is css */ 2 | 3 | .btn { 4 | background-image: url("archive.svg"); 5 | } 6 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/dhh.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/first_path/dhh.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-not-at-start.js: -------------------------------------------------------------------------------- 1 | var fun; //# sourceMappingURL=sourceMappingURL-not-at-start.js.map 2 | 3 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/file.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/file.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-already-prefixed.js: -------------------------------------------------------------------------------- 1 | var fun; //# sourceMappingURL=/assets/sourceMappingURL-already-prefixed.js.map -------------------------------------------------------------------------------- /bin/test: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | $: << File.expand_path("../test", __dir__) 3 | 4 | require "bundler/setup" 5 | require "rails/plugin/test" 6 | -------------------------------------------------------------------------------- /.devcontainer/Dockerfile: -------------------------------------------------------------------------------- 1 | # [Choice] Ruby version: 3.4, 3.3, 3.2 2 | ARG VARIANT="3.4.4" 3 | FROM ghcr.io/rails/devcontainer/images/ruby:${VARIANT} 4 | -------------------------------------------------------------------------------- /gemfiles/Gemfile.rails-7.1: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gemspec path: ".." 4 | 5 | gem "rails", "~> 7.1.0" 6 | gem "rake" 7 | gem "debug" 8 | -------------------------------------------------------------------------------- /gemfiles/Gemfile.rails-7.2: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gemspec path: ".." 4 | 5 | gem "rails", "~> 7.2.0" 6 | gem "rake" 7 | gem "debug" 8 | -------------------------------------------------------------------------------- /gemfiles/Gemfile.rails-8.0: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gemspec path: ".." 4 | 5 | gem "rails", "~> 8.0" 6 | gem "rake" 7 | gem "debug" 8 | -------------------------------------------------------------------------------- /test/dummy/config/routes.rb: -------------------------------------------------------------------------------- 1 | Rails.application.routes.draw do 2 | get 'sample/load_real_assets' 3 | get 'sample/load_nonexistent_assets' 4 | end 5 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/file-not.digested.css: -------------------------------------------------------------------------------- 1 | /* this is css */ 2 | 3 | .btn { 4 | background-image: asset-path("archive.svg"); 5 | } 6 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/sibling/file.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/sibling/file.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/file.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/source/file.jpg -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Ignore all logfiles and tempfiles. 2 | /test/dummy/log/* 3 | /test/dummy/tmp/* 4 | 5 | # Ignore compiled assets 6 | /test/dummy/public/assets 7 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-not-at-end.css: -------------------------------------------------------------------------------- 1 | .class { 2 | /*# sourceMappingURL=sourceMappingURL-not-at-end.css.map */ 3 | color: green; 4 | } 5 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-not-at-start.css: -------------------------------------------------------------------------------- 1 | .class{color:green;}/*# sourceMappingURL=sourceMappingURL-not-at-start.css.map */ 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/database.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/source/database.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/nested/sourceMappingURL-already-prefixed-nested.js: -------------------------------------------------------------------------------- 1 | var fun; //# sourceMappingURL=/assets/sourceMappingURL-already-prefixed-nested.js.map 2 | -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/http-diagram.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/source/http-diagram.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/vendor/foobar/source/images/file.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rails/propshaft/main/test/fixtures/assets/vendor/foobar/source/images/file.jpg -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-outside-comment.css: -------------------------------------------------------------------------------- 1 | .class::before { 2 | content: "# sourceMappingURL=sourceMappingURL-outside-comment.css.map"; 3 | } 4 | -------------------------------------------------------------------------------- /test/dummy/bin/rails: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | APP_PATH = File.expand_path("../config/application", __dir__) 3 | require_relative "../config/boot" 4 | require "rails/commands" 5 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/file-already-abcdefVWXYZ0123456789_-.digested.css: -------------------------------------------------------------------------------- 1 | /* this is css */ 2 | 3 | .btn { 4 | background-image: asset-path("archive.svg"); 5 | } 6 | -------------------------------------------------------------------------------- /test/fixtures/assets/mapped/sourceMappingURL-already-prefixed-invalid.js: -------------------------------------------------------------------------------- 1 | var fun; //# sourceMappingURL=thisisinvalidassets/sourceMappingURL-already-prefixed-invalid.js.map 2 | -------------------------------------------------------------------------------- /test/dummy/config/environment.rb: -------------------------------------------------------------------------------- 1 | # Load the Rails application. 2 | require_relative "application" 3 | 4 | # Initialize the Rails application. 5 | Rails.application.initialize! 6 | -------------------------------------------------------------------------------- /gemfiles/Gemfile.rails-7.0: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gemspec path: ".." 4 | 5 | gem "rails", github: "rails/rails", branch: "7-0-stable" 6 | gem "rake" 7 | gem "debug" 8 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/file-already-abcdefVWXYZ0123456789_-.digested.debug.css: -------------------------------------------------------------------------------- 1 | /* this is css */ 2 | 3 | .btn { 4 | background-image: asset-path("archive.svg"); 5 | } 6 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/sample_controller.rb: -------------------------------------------------------------------------------- 1 | class SampleController < ApplicationController 2 | def load_real_assets 3 | end 4 | 5 | def load_nonexistent_assets 6 | end 7 | end 8 | -------------------------------------------------------------------------------- /test/fixtures/new_manifest_format/.manifest.json: -------------------------------------------------------------------------------- 1 | {"one.txt": {"digested_path": "one-f2e1ec14.txt","integrity": "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe"}} 2 | -------------------------------------------------------------------------------- /test/dummy/config.ru: -------------------------------------------------------------------------------- 1 | # This file is used by Rack-based servers to start the application. 2 | 3 | require_relative "config/environment" 4 | 5 | run Rails.application 6 | Rails.application.load_server 7 | -------------------------------------------------------------------------------- /test/dummy/config/boot.rb: -------------------------------------------------------------------------------- 1 | # Set up gems listed in the Gemfile. 2 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../../Gemfile", __dir__) 3 | 4 | require "bundler/setup" if File.exist?(ENV["BUNDLE_GEMFILE"]) 5 | $LOAD_PATH.unshift File.expand_path("../../../lib", __dir__) 6 | -------------------------------------------------------------------------------- /test/dummy/app/views/sample/load_nonexistent_assets.html.erb: -------------------------------------------------------------------------------- 1 | <%= stylesheet_link_tag "nonexistent" %> 2 | 3 |

Sample#load_nonexistent_assets

4 |

Find me in app/views/sample/load_nonexistent_assets.html.erb

5 | 6 | <%= javascript_include_tag "nonexistent" %> 7 | -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | require "bundler/setup" 2 | require "bundler/gem_tasks" 3 | require "rake/testtask" 4 | 5 | Rake::TestTask.new do |test| 6 | test.libs << "test" 7 | test.test_files = FileList["test/**/*_test.rb"] 8 | test.warning = true 9 | end 10 | 11 | task default: :test 12 | -------------------------------------------------------------------------------- /test/dummy/app/views/sample/load_real_assets.html.erb: -------------------------------------------------------------------------------- 1 | <%= stylesheet_link_tag "hello_world", integrity: true %> 2 | 3 |

Sample#load_real_assets

4 |

Find me in app/views/sample/load_real_assets.html.erb

5 | 6 | <%= javascript_include_tag "hello_world", integrity: true %> 7 | <%= javascript_include_tag "actioncable" %> 8 | -------------------------------------------------------------------------------- /test/fixtures/assets/first_path/archive.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /lib/propshaft/quiet_assets.rb: -------------------------------------------------------------------------------- 1 | class Propshaft::QuietAssets 2 | def initialize(app) 3 | @app = app 4 | @assets_regex = %r(\A/{0,2}#{::Rails.application.config.assets.prefix}) 5 | end 6 | 7 | def call(env) 8 | if env['PATH_INFO'] =~ @assets_regex 9 | ::Rails.logger.silence { @app.call(env) } 10 | else 11 | @app.call(env) 12 | end 13 | end 14 | end 15 | -------------------------------------------------------------------------------- /test/dummy/app/views/layouts/application.html.erb: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Dummy 5 | <%= csrf_meta_tags %> 6 | <%= csp_meta_tag %> 7 | 8 | <%= stylesheet_link_tag :all, data: { custom_attribute: true } %> 9 | <%= stylesheet_link_tag :app, data: { glob_attribute: true } %> 10 | 11 | 12 | 13 | <%= yield %> 14 | 15 | 16 | -------------------------------------------------------------------------------- /lib/propshaft.rb: -------------------------------------------------------------------------------- 1 | require "active_support" 2 | require "active_support/core_ext/module/attribute_accessors" 3 | require "active_support/core_ext/module/delegation" 4 | require "logger" 5 | 6 | module Propshaft 7 | mattr_accessor :logger, default: Logger.new(STDOUT) 8 | end 9 | 10 | require "propshaft/assembly" 11 | require "propshaft/errors" 12 | require "propshaft/helper" 13 | require "propshaft/railtie" if defined?(Rails::Railtie) 14 | -------------------------------------------------------------------------------- /lib/propshaft/errors.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Propshaft 4 | # Generic base class for all Propshaft exceptions. 5 | class Error < StandardError; end 6 | 7 | # Raised when LoadPath cannot find the requested asset 8 | class MissingAssetError < Error 9 | def initialize(path) 10 | super 11 | @path = path 12 | end 13 | 14 | def message 15 | "The asset '#{@path}' was not found in the load path." 16 | end 17 | end 18 | end 19 | -------------------------------------------------------------------------------- /bin/release: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | VERSION=$1 4 | 5 | if [ -z "$VERSION" ]; then 6 | echo "Usage: $0 " 7 | exit 1 8 | fi 9 | 10 | printf "module Propshaft\n VERSION = \"$VERSION\"\nend\n" > ./lib/propshaft/version.rb 11 | bundle 12 | git add Gemfile.lock lib/propshaft/version.rb 13 | git commit -m "Bump version for $VERSION" 14 | git push 15 | git tag v$VERSION 16 | git push --tags 17 | gem build propshaft.gemspec 18 | gem push "propshaft-$VERSION.gem" --host https://rubygems.org 19 | rm "propshaft-$VERSION.gem" 20 | -------------------------------------------------------------------------------- /lib/propshaft/compiler.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Base compiler from which other compilers can inherit 4 | class Propshaft::Compiler 5 | attr_reader :assembly 6 | delegate :config, :load_path, to: :assembly 7 | 8 | def initialize(assembly) 9 | @assembly = assembly 10 | end 11 | 12 | # Override this in a specific compiler 13 | def compile(asset, input) 14 | raise NotImplementedError 15 | end 16 | 17 | def referenced_by(asset) 18 | Set.new 19 | end 20 | 21 | private 22 | def url_prefix 23 | @url_prefix ||= File.join(config.relative_url_root.to_s, config.prefix.to_s).chomp("/") 24 | end 25 | end 26 | -------------------------------------------------------------------------------- /propshaft.gemspec: -------------------------------------------------------------------------------- 1 | require_relative "lib/propshaft/version" 2 | 3 | Gem::Specification.new do |s| 4 | s.name = "propshaft" 5 | s.version = Propshaft::VERSION 6 | s.authors = [ "David Heinemeier Hansson" ] 7 | s.email = "dhh@hey.com" 8 | s.summary = "Deliver assets for Rails." 9 | s.homepage = "https://github.com/rails/propshaft" 10 | s.license = "MIT" 11 | 12 | s.metadata = { 13 | "rubygems_mfa_required" => "true" 14 | } 15 | 16 | s.required_ruby_version = ">= 2.7.0" 17 | s.add_dependency "actionpack", ">= 7.0.0" 18 | s.add_dependency "activesupport", ">= 7.0.0" 19 | s.add_dependency "rack" 20 | 21 | s.files = Dir["lib/**/*", "MIT-LICENSE", "Rakefile", "README.md"] 22 | end 23 | -------------------------------------------------------------------------------- /test/test_helper.rb: -------------------------------------------------------------------------------- 1 | # Configure Rails Environment 2 | ENV["RAILS_ENV"] = "test" 3 | Warning[:deprecated] = true 4 | $VERBOSE = true 5 | 6 | require_relative "../test/dummy/config/environment" 7 | require "rails/test_help" 8 | 9 | require "rails/test_unit/reporter" 10 | Rails::TestUnitReporter.executable = "bin/test" 11 | 12 | class ActiveSupport::TestCase 13 | private 14 | def find_asset(logical_path, fixture_path:) 15 | root_path = Pathname.new("#{__dir__}/fixtures/assets/#{fixture_path}") 16 | path = root_path.join(logical_path) 17 | load_path = Propshaft::LoadPath.new([ root_path ], compilers: Propshaft::Compilers.new(nil)) 18 | 19 | Propshaft::Asset.new(path, logical_path: logical_path, load_path: load_path) 20 | end 21 | end 22 | -------------------------------------------------------------------------------- /benchmarks/dynamic_resolver: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | 3 | # Benchmark file for the Dynamic resolver 4 | 5 | require "active_support/ordered_options" 6 | require "benchmark/ips" 7 | require "open-uri" 8 | 9 | require_relative "./trackrod" 10 | require_relative "../lib/propshaft" 11 | 12 | trackrod = Trackrod.new(Dir.mktmpdir) 13 | trackrod.build 14 | 15 | assets = ActiveSupport::OrderedOptions.new 16 | assets.paths = [ trackrod.root ] 17 | assets.prefix = "/assets" 18 | assets.compilers = [ [ "text/css", Propshaft::Compiler::CssAssetUrls ] ] 19 | assets.output_path ||= Pathname.new(Dir.mktmpdir) 20 | 21 | assembly = Propshaft::Assembly.new(assets) 22 | 23 | Benchmark.ips do |x| 24 | x.config(time: 5, warmup: 2) 25 | x.report("compile") { trackrod.assets.images.each { assembly.resolver.resolve _1 } } 26 | end 27 | -------------------------------------------------------------------------------- /benchmarks/static_resolver: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | 3 | # Benchmark file for the static resolver 4 | 5 | require "active_support/core_ext/string/access" 6 | require "active_support/ordered_options" 7 | require "benchmark/ips" 8 | 9 | require_relative "./trackrod" 10 | require_relative "../lib/propshaft" 11 | 12 | trackrod = Trackrod.new(Dir.mktmpdir) 13 | trackrod.build 14 | 15 | assets = ActiveSupport::OrderedOptions.new 16 | assets.paths = [ trackrod.root ] 17 | assets.prefix = "/assets" 18 | assets.compilers = [ [ "text/css", Propshaft::Compiler::CssAssetUrls ] ] 19 | assets.output_path ||= Pathname.new(Dir.mktmpdir) 20 | 21 | assembly = Propshaft::Assembly.new(assets) 22 | assembly.processor.process 23 | 24 | Benchmark.ips do |x| 25 | x.config(time: 5, warmup: 2) 26 | x.report("compile") { trackrod.assets.images.each { assembly.resolver.resolve _1 } } 27 | end 28 | -------------------------------------------------------------------------------- /lib/propshaft/resolver/dynamic.rb: -------------------------------------------------------------------------------- 1 | module Propshaft::Resolver 2 | class Dynamic 3 | attr_reader :load_path, :prefix 4 | 5 | def initialize(load_path:, prefix:) 6 | @load_path, @prefix = load_path, prefix 7 | end 8 | 9 | def resolve(logical_path) 10 | if asset = find_asset(logical_path) 11 | File.join prefix, asset.digested_path 12 | end 13 | end 14 | 15 | def integrity(logical_path) 16 | hash_algorithm = load_path.integrity_hash_algorithm 17 | 18 | if hash_algorithm && (asset = find_asset(logical_path)) 19 | asset.integrity(hash_algorithm: hash_algorithm) 20 | end 21 | end 22 | 23 | def read(logical_path, options = {}) 24 | if asset = load_path.find(logical_path) 25 | asset.content(**options) 26 | end 27 | end 28 | 29 | private 30 | def find_asset(logical_path) 31 | load_path.find(logical_path) 32 | end 33 | end 34 | end 35 | -------------------------------------------------------------------------------- /benchmarks/css_asset_urls: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | 3 | # Benchmark file for the CssAssetUrls compiler 4 | 5 | require "active_support/ordered_options" 6 | require "benchmark/ips" 7 | require "open-uri" 8 | 9 | require_relative "./trackrod" 10 | require_relative "../lib/propshaft" 11 | require_relative "../lib/propshaft/compilers" 12 | require_relative "../lib/propshaft/compiler/css_asset_urls" 13 | 14 | trackrod = Trackrod.new(Dir.mktmpdir) 15 | trackrod.build 16 | 17 | assets = ActiveSupport::OrderedOptions.new 18 | assets.paths = [ trackrod.root ] 19 | assets.prefix = "/assets" 20 | assets.compilers = [ [ "text/css", Propshaft::Compiler::CssAssetUrls ] ] 21 | 22 | assembly = Propshaft::Assembly.new(assets) 23 | asset = assembly.load_path.find(trackrod.assets.css) 24 | compiler = Propshaft::Compiler::CssAssetUrls.new(assembly) 25 | 26 | Benchmark.ips do |x| 27 | x.config(time: 5, warmup: 2) 28 | x.report("compile") { compiler.compile(asset, asset.content) } 29 | end 30 | -------------------------------------------------------------------------------- /.github/workflows/ci.yml: -------------------------------------------------------------------------------- 1 | name: CI 2 | on: [push, pull_request] 3 | jobs: 4 | tests: 5 | strategy: 6 | matrix: 7 | ruby-version: 8 | - "3.1" 9 | - "3.2" 10 | - "3.3" 11 | - "3.4" 12 | rails-version: 13 | - "7.0" 14 | - "7.1" 15 | - "7.2" 16 | - "8.0" 17 | exclude: 18 | - ruby-version: "3.1" 19 | rails-version: "8.0" 20 | fail-fast: false 21 | env: 22 | BUNDLE_GEMFILE: gemfiles/Gemfile.rails-${{ matrix.rails-version }} 23 | 24 | name: ${{ format('Tests (Ruby {0}) (Rails {1})', matrix.ruby-version, matrix.rails-version) }} 25 | runs-on: ubuntu-latest 26 | 27 | steps: 28 | - uses: actions/checkout@v4 29 | 30 | - name: Install Ruby 31 | uses: ruby/setup-ruby@v1 32 | with: 33 | ruby-version: ${{ matrix.ruby-version }} 34 | bundler-cache: true 35 | 36 | - name: Run tests 37 | run: bin/test 38 | -------------------------------------------------------------------------------- /test/propshaft/quiet_assets_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/quiet_assets" 3 | 4 | class Propshaft::QuietAssetsTest < ActiveSupport::TestCase 5 | setup do 6 | Rails.logger.level = Logger::DEBUG 7 | end 8 | 9 | test "silences with default prefix" do 10 | assert_equal Logger::ERROR, middleware.call("PATH_INFO" => "/assets/stylesheets/application.css") 11 | end 12 | 13 | test "silences with custom prefix" do 14 | original = Rails.application.config.assets.prefix 15 | Rails.application.config.assets.prefix = "path/to" 16 | assert_equal Logger::ERROR, middleware.call("PATH_INFO" => "/path/to/thing") 17 | ensure 18 | Rails.application.config.assets.prefix = original 19 | end 20 | 21 | test "does not silence without match" do 22 | assert_equal Logger::DEBUG, middleware.call("PATH_INFO" => "/path/to/thing") 23 | end 24 | 25 | private 26 | 27 | def middleware 28 | @middleware ||= Propshaft::QuietAssets.new(->(env) { Rails.logger.level }) 29 | end 30 | end 31 | -------------------------------------------------------------------------------- /lib/propshaft/resolver/static.rb: -------------------------------------------------------------------------------- 1 | module Propshaft::Resolver 2 | class Static 3 | attr_reader :manifest_path, :prefix 4 | 5 | def initialize(manifest_path:, prefix:) 6 | @manifest_path, @prefix = manifest_path, prefix 7 | end 8 | 9 | def resolve(logical_path) 10 | if asset_path = digested_path(logical_path) 11 | File.join prefix, asset_path 12 | end 13 | end 14 | 15 | def integrity(logical_path) 16 | entry = manifest[logical_path] 17 | 18 | entry&.integrity 19 | end 20 | 21 | def read(logical_path, encoding: "ASCII-8BIT") 22 | if asset_path = digested_path(logical_path) 23 | File.read(manifest_path.dirname.join(asset_path), encoding: encoding) 24 | end 25 | end 26 | 27 | private 28 | def manifest 29 | @manifest ||= Propshaft::Manifest.from_path(manifest_path) 30 | end 31 | 32 | def digested_path(logical_path) 33 | entry = manifest[logical_path] 34 | 35 | entry&.digested_path 36 | end 37 | end 38 | end 39 | -------------------------------------------------------------------------------- /.devcontainer/devcontainer.json: -------------------------------------------------------------------------------- 1 | // For format details, see https://aka.ms/devcontainer.json. For config options, see the README at: 2 | // https://github.com/microsoft/vscode-dev-containers/tree/v0.205.2/containers/ruby 3 | { 4 | "name": "Propshaft", 5 | "build": { 6 | "dockerfile": "Dockerfile", 7 | "args": { 8 | // Update 'VARIANT' to pick a Ruby version: 3, 3.0, 2, 2.7, 2.6 9 | // Append -bullseye or -buster to pin to an OS version. 10 | // Use -bullseye variants on local on arm64/Apple Silicon. 11 | "VARIANT": "3.4.4", 12 | } 13 | }, 14 | 15 | // Configure tool-specific properties. 16 | // "customizations": {}, 17 | 18 | // Use 'forwardPorts' to make a list of ports inside the container available locally. 19 | // "forwardPorts": [], 20 | 21 | // Use 'postCreateCommand' to run commands after the container is created. 22 | // "postCreateCommand": "ruby --version", 23 | 24 | // Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root. 25 | "remoteUser": "vscode", 26 | "features": { 27 | "github-cli": "latest" 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /MIT-LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2021 Basecamp 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | -------------------------------------------------------------------------------- /test/propshaft/compilers_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/asset" 3 | require "propshaft/assembly" 4 | require "propshaft/compilers" 5 | 6 | class Propshaft::CompilersTest < ActiveSupport::TestCase 7 | setup do 8 | @assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 9 | config.paths = [ Pathname.new("#{__dir__}/../fixtures/assets/first_path") ] 10 | config.output_path = Pathname.new("#{__dir__}/../fixtures/output") 11 | config.prefix = "/assets" 12 | }) 13 | end 14 | 15 | test "replace asset-path function in css with digested url" do 16 | @assembly.compilers.register "text/css", Propshaft::Compiler::CssAssetUrls 17 | assert_match(/"\/assets\/archive-[a-z0-9]{8}.svg/, @assembly.compilers.compile(find_asset("another.css"))) 18 | end 19 | 20 | private 21 | def find_asset(logical_path) 22 | root_path = Pathname.new("#{__dir__}/../fixtures/assets/first_path") 23 | load_path = Propshaft::LoadPath.new([ root_path ], compilers: Propshaft::Compilers.new(nil)) 24 | Propshaft::Asset.new(root_path.join(logical_path), logical_path: logical_path, load_path: load_path) 25 | end 26 | end 27 | -------------------------------------------------------------------------------- /lib/propshaft/compilers.rb: -------------------------------------------------------------------------------- 1 | class Propshaft::Compilers 2 | attr_reader :registrations, :assembly 3 | 4 | def initialize(assembly) 5 | @assembly = assembly 6 | @registrations = Hash.new 7 | end 8 | 9 | def register(mime_type, klass) 10 | registrations[mime_type] ||= [] 11 | registrations[mime_type] << klass 12 | end 13 | 14 | def any? 15 | registrations.any? 16 | end 17 | 18 | def compilable?(asset) 19 | registrations[asset.content_type.to_s].present? 20 | end 21 | 22 | def compile(asset) 23 | if relevant_registrations = registrations[asset.content_type.to_s] 24 | asset.content.dup.tap do |input| 25 | relevant_registrations.each do |compiler| 26 | input.replace compiler.new(assembly).compile(asset, input) 27 | end 28 | end 29 | else 30 | asset.content 31 | end 32 | end 33 | 34 | def referenced_by(asset) 35 | Set.new.tap do |references| 36 | if relevant_registrations = registrations[asset.content_type.to_s] 37 | relevant_registrations.each do |compiler| 38 | references.merge compiler.new(assembly).referenced_by(asset) 39 | end 40 | end 41 | end 42 | end 43 | end 44 | -------------------------------------------------------------------------------- /lib/propshaft/compiler/source_mapping_urls.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "propshaft/compiler" 4 | 5 | class Propshaft::Compiler::SourceMappingUrls < Propshaft::Compiler 6 | SOURCE_MAPPING_PATTERN = %r{(//|/\*)# sourceMappingURL=(.+\.map)(\s*?\*\/)?\s*?\Z} 7 | 8 | def compile(asset, input) 9 | input.gsub(SOURCE_MAPPING_PATTERN) { source_mapping_url(asset.logical_path, asset_path($2, asset.logical_path), $1, $3) } 10 | end 11 | 12 | private 13 | def asset_path(source_mapping_url, logical_path) 14 | source_mapping_url.gsub!(/^(.+\/)?#{url_prefix}\//, "") 15 | 16 | if logical_path.dirname.to_s == "." 17 | source_mapping_url 18 | else 19 | logical_path.dirname.join(source_mapping_url).to_s 20 | end 21 | end 22 | 23 | def source_mapping_url(logical_path, resolved_path, comment_start, comment_end) 24 | if asset = load_path.find(resolved_path) 25 | "#{comment_start}# sourceMappingURL=#{url_prefix}/#{asset.digested_path}#{comment_end}" 26 | else 27 | Propshaft.logger.warn "Removed sourceMappingURL comment for missing asset '#{resolved_path}' from #{logical_path}" 28 | "#{comment_start}#{comment_end}" 29 | end 30 | end 31 | end 32 | -------------------------------------------------------------------------------- /lib/propshaft/railties/assets.rake: -------------------------------------------------------------------------------- 1 | namespace :assets do 2 | desc "Compile all the assets from config.assets.paths" 3 | task precompile: :environment do 4 | Rails.application.assets.processor.process 5 | if Rails.env.development? 6 | puts "Warning: You are precompiling assets in development. Rails will not " \ 7 | "serve any changed assets until you delete public#{Rails.application.config.assets.prefix}/.manifest.json" 8 | end 9 | end 10 | 11 | desc "Remove config.assets.output_path" 12 | task clobber: :environment do 13 | Rails.application.assets.processor.clobber 14 | end 15 | 16 | desc "Removes old files in config.assets.output_path" 17 | task :clean, [:count] => [:environment] do |_, args| 18 | count = args.fetch(:count, 2) 19 | Rails.application.assets.processor.clean(count.to_i) 20 | end 21 | 22 | desc "Print all the assets available in config.assets.paths" 23 | task reveal: :environment do 24 | puts Rails.application.assets.reveal(:logical_path).join("\n") 25 | end 26 | 27 | namespace :reveal do 28 | desc "Print the full path of assets available in config.assets.paths" 29 | task full: :environment do 30 | puts Rails.application.assets.reveal(:path).join("\n") 31 | end 32 | end 33 | end 34 | -------------------------------------------------------------------------------- /test/dummy/config/application.rb: -------------------------------------------------------------------------------- 1 | require_relative "boot" 2 | 3 | require "rails" 4 | # Pick the frameworks you want: 5 | # require "active_model/railtie" 6 | # require "active_job/railtie" 7 | # require "active_record/railtie" 8 | # require "active_storage/engine" 9 | require "action_controller/railtie" 10 | # require "action_mailer/railtie" 11 | # require "action_mailbox/engine" 12 | # require "action_text/engine" 13 | require "action_view/railtie" 14 | require "action_cable/engine" 15 | # require "sprockets/railtie" 16 | require "rails/test_unit/railtie" 17 | 18 | # Require the gems listed in Gemfile, including any gems 19 | # you've limited to :test, :development, or :production. 20 | Bundler.require(*Rails.groups) 21 | require "propshaft" 22 | 23 | module Dummy 24 | class Application < Rails::Application 25 | config.load_defaults Rails::VERSION::STRING.to_f 26 | 27 | # Configuration for the application, engines, and railties goes here. 28 | # 29 | # These settings can be overridden in specific environments using the files 30 | # in config/environments, which are processed later. 31 | # 32 | # config.time_zone = "Central Time (US & Canada)" 33 | # config.eager_load_paths << Rails.root.join("extras") 34 | 35 | config.assets.integrity_hash_algorithm = "sha384" 36 | end 37 | end 38 | -------------------------------------------------------------------------------- /lib/propshaft/compiler/css_asset_urls.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "propshaft/compiler" 4 | 5 | class Propshaft::Compiler::CssAssetUrls < Propshaft::Compiler 6 | ASSET_URL_PATTERN = /url\(\s*["']?(?!(?:\#|%23|data:|http:|https:|\/\/))([^"'\s?#)]+)([#?][^"')]+)?\s*["']?\)/ 7 | 8 | def compile(asset, input) 9 | input.gsub(ASSET_URL_PATTERN) { asset_url resolve_path(asset.logical_path.dirname, $1), asset.logical_path, $2, $1 } 10 | end 11 | 12 | def referenced_by(asset, references: Set.new) 13 | asset.content.scan(ASSET_URL_PATTERN).each do |referenced_asset_url, _| 14 | referenced_asset = load_path.find(resolve_path(asset.logical_path.dirname, referenced_asset_url)) 15 | 16 | if referenced_asset && references.exclude?(referenced_asset) 17 | references << referenced_asset 18 | references.merge referenced_by(referenced_asset, references: references) 19 | end 20 | end 21 | 22 | references 23 | end 24 | 25 | private 26 | def resolve_path(directory, filename) 27 | if filename.start_with?("../") 28 | Pathname.new(directory + filename).relative_path_from("").to_s 29 | elsif filename.start_with?("/") 30 | filename.delete_prefix("/").to_s 31 | else 32 | (directory + filename.delete_prefix("./")).to_s 33 | end 34 | end 35 | 36 | def asset_url(resolved_path, logical_path, fingerprint, pattern) 37 | if asset = load_path.find(resolved_path) 38 | %[url("#{url_prefix}/#{asset.digested_path}#{fingerprint}")] 39 | else 40 | Propshaft.logger.warn "Unable to resolve '#{pattern}' for missing asset '#{resolved_path}' in #{logical_path}" 41 | %[url("#{pattern}")] 42 | end 43 | end 44 | end 45 | -------------------------------------------------------------------------------- /lib/propshaft/compiler/js_asset_urls.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "propshaft/compiler" 4 | 5 | class Propshaft::Compiler::JsAssetUrls < Propshaft::Compiler 6 | ASSET_URL_PATTERN = %r{RAILS_ASSET_URL\(\s*["']?(?!(?:\#|%23|data|http|//))([^"'\s?#)]+)([#?][^"')]+)?\s*["']?\)} 7 | 8 | def compile(asset, input) 9 | input.gsub(ASSET_URL_PATTERN) { asset_url(resolve_path(asset.logical_path.dirname, $1), asset.logical_path, $2, $1) } 10 | end 11 | 12 | def referenced_by(asset, references: Set.new) 13 | asset.content.scan(ASSET_URL_PATTERN).each do |referenced_asset_url, _| 14 | referenced_asset = load_path.find(resolve_path(asset.logical_path.dirname, referenced_asset_url)) 15 | 16 | if referenced_asset && references.exclude?(referenced_asset) 17 | references << referenced_asset 18 | references.merge referenced_by(referenced_asset, references: references) 19 | end 20 | end 21 | 22 | references 23 | end 24 | 25 | private 26 | def resolve_path(directory, filename) 27 | if filename.start_with?("../") 28 | Pathname.new(directory + filename).relative_path_from("").to_s 29 | elsif filename.start_with?("/") 30 | filename.delete_prefix("/").to_s 31 | else 32 | (directory + filename.delete_prefix("./")).to_s 33 | end 34 | end 35 | 36 | def asset_url(resolved_path, logical_path, fingerprint, pattern) 37 | asset = load_path.find(resolved_path) 38 | if asset 39 | %["#{url_prefix}/#{asset.digested_path}#{fingerprint}"] 40 | else 41 | Propshaft.logger.warn("Unable to resolve '#{pattern}' for missing asset '#{resolved_path}' in #{logical_path}") 42 | %["#{pattern}"] 43 | end 44 | end 45 | end 46 | -------------------------------------------------------------------------------- /lib/propshaft/output_path.rb: -------------------------------------------------------------------------------- 1 | require "propshaft/asset" 2 | 3 | class Propshaft::OutputPath 4 | attr_reader :path, :manifest 5 | 6 | def initialize(path, manifest) 7 | @path, @manifest = path, manifest 8 | end 9 | 10 | def clean(count, age) 11 | asset_versions = files.group_by { |_, attrs| attrs[:logical_path] } 12 | asset_versions.each do |logical_path, versions| 13 | current = manifest[logical_path] 14 | 15 | versions 16 | .reject { |path, _| current && path == current } 17 | .sort_by { |_, attrs| attrs[:mtime] } 18 | .reverse 19 | .each_with_index 20 | .drop_while { |(_, attrs), index| fresh_version_within_limit(attrs[:mtime], count, expires_at: age, limit: index) } 21 | .each { |(path, _), _| remove(path) } 22 | end 23 | end 24 | 25 | def files 26 | Hash.new.tap do |files| 27 | all_files_from_tree(path).each do |file| 28 | digested_path = file.relative_path_from(path) 29 | logical_path, digest = Propshaft::Asset.extract_path_and_digest(digested_path.to_s) 30 | 31 | files[digested_path.to_s] = { 32 | logical_path: logical_path.to_s, 33 | digest: digest, 34 | mtime: File.mtime(file) 35 | } 36 | end 37 | end 38 | end 39 | 40 | private 41 | def fresh_version_within_limit(mtime, count, expires_at:, limit:) 42 | modified_at = [ 0, Time.now - mtime ].max 43 | modified_at < expires_at || limit < count 44 | end 45 | 46 | def remove(path) 47 | FileUtils.rm(@path.join(path)) 48 | Propshaft.logger.info "Removed #{path}" 49 | end 50 | 51 | def all_files_from_tree(path) 52 | path.children.flat_map { |child| child.directory? ? all_files_from_tree(child) : child } 53 | end 54 | end 55 | -------------------------------------------------------------------------------- /lib/propshaft/server.rb: -------------------------------------------------------------------------------- 1 | require "rack/utils" 2 | require "rack/version" 3 | 4 | class Propshaft::Server 5 | def initialize(app, assembly) 6 | @app = app 7 | @assembly = assembly 8 | end 9 | 10 | def call(env) 11 | execute_cache_sweeper_if_updated 12 | 13 | path = env["PATH_INFO"] 14 | method = env["REQUEST_METHOD"] 15 | 16 | if (method == "GET" || method == "HEAD") && path.start_with?(@assembly.prefix) 17 | path, digest = extract_path_and_digest(path) 18 | 19 | if (asset = @assembly.load_path.find(path)) && asset.fresh?(digest) 20 | compiled_content = asset.compiled_content 21 | 22 | [ 23 | 200, 24 | { 25 | Rack::CONTENT_LENGTH => compiled_content.length.to_s, 26 | Rack::CONTENT_TYPE => asset.content_type.to_s, 27 | VARY => "Accept-Encoding", 28 | Rack::ETAG => "\"#{asset.digest}\"", 29 | Rack::CACHE_CONTROL => "public, max-age=31536000, immutable" 30 | }, 31 | method == "HEAD" ? [] : [ compiled_content ] 32 | ] 33 | else 34 | [ 404, { Rack::CONTENT_TYPE => "text/plain", Rack::CONTENT_LENGTH => "9" }, [ "Not found" ] ] 35 | end 36 | else 37 | @app.call(env) 38 | end 39 | end 40 | 41 | def inspect 42 | self.class.inspect 43 | end 44 | 45 | private 46 | def extract_path_and_digest(path) 47 | path = path.delete_prefix(@assembly.prefix) 48 | path = Rack::Utils.unescape(path) 49 | 50 | Propshaft::Asset.extract_path_and_digest(path) 51 | end 52 | 53 | if Gem::Version.new(Rack::RELEASE) < Gem::Version.new("3") 54 | VARY = "Vary" 55 | else 56 | VARY = "vary" 57 | end 58 | 59 | def execute_cache_sweeper_if_updated 60 | if @assembly.config.sweep_cache 61 | @assembly.load_path.cache_sweeper.execute_if_updated 62 | end 63 | end 64 | end 65 | -------------------------------------------------------------------------------- /benchmarks/trackrod.rb: -------------------------------------------------------------------------------- 1 | require "active_support/ordered_options" 2 | require 'fileutils' 3 | 4 | class Trackrod 5 | attr_accessor :root, :images, :css, :javascript 6 | 7 | def initialize(dir = nil) 8 | @root = dir || "#{Dir.getwd}/trackrod" 9 | @images = @root + "/images" 10 | @css = @root + "/stylesheets" 11 | @javascript = @root + "/javascript" 12 | end 13 | 14 | def build 15 | puts "Building Trackrod assets" 16 | 17 | create_dir(root) 18 | create_dir(images) 19 | create_dir(css) 20 | create_dir(javascript) 21 | 22 | create_images 23 | create_css 24 | create_javascript 25 | end 26 | 27 | def assets 28 | @assets ||= ActiveSupport::InheritableOptions.new( 29 | css: "stylesheets/application.css", 30 | js: "javascript/application.js", 31 | images: (small_images + large_images).map { "images/#{_1}" } 32 | ) 33 | end 34 | 35 | private 36 | def create_css 37 | File.open("#{css}/application.css", "a") do |file| 38 | small_images.each_with_index { |img, idx| file.write(background(img, idx)) } 39 | large_images.each_with_index { |img, idx| file.write(background(img, idx)) } 40 | end 41 | end 42 | 43 | def create_javascript 44 | end 45 | 46 | def create_images 47 | small_images.each { |img| FileUtils.touch "#{images}/#{img}" } 48 | large_images.each { |img| File.open("#{images}/#{img}", "a") { |file| file.write("a" * 2 ** 23) } unless File.exist?(img) } 49 | 50 | nil 51 | end 52 | 53 | def small_images 54 | @small_images ||= (1..5000).map { "s#{_1}.jpg" } 55 | end 56 | 57 | def large_images 58 | @large_images ||= (1..100).map { "l#{_1}.jpg" } 59 | end 60 | 61 | def background(img, idx) 62 | ".background_#{idx} {\n background: url('../images/#{img}') \n}\n\n" 63 | end 64 | 65 | def create_dir(path) 66 | Dir.mkdir(path) unless File.exist?(path) 67 | end 68 | end 69 | -------------------------------------------------------------------------------- /test/propshaft/assembly_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/assembly" 3 | require "active_support/ordered_options" 4 | 5 | class Propshaft::AssemblyTest < ActiveSupport::TestCase 6 | test "uses static resolver when manifest is present" do 7 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 8 | config.output_path = Pathname.new("#{__dir__}/../fixtures/output") 9 | config.manifest_path = config.output_path.join(".manifest.json") 10 | config.prefix = "/assets" 11 | }) 12 | 13 | assert assembly.resolver.is_a?(Propshaft::Resolver::Static) 14 | end 15 | 16 | test "uses dynamic resolver when manifest is missing" do 17 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 18 | config.output_path = Pathname.new("#{__dir__}/../fixtures/assets") 19 | config.manifest_path = config.output_path.join(".manifest.json") 20 | config.prefix = "/assets" 21 | }) 22 | 23 | assert assembly.resolver.is_a?(Propshaft::Resolver::Dynamic) 24 | end 25 | 26 | test "costly methods are memoized" do 27 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 28 | config.output_path = Pathname.new("#{__dir__}/../fixtures/assets") 29 | config.manifest_path = config.output_path.join(".manifest.json") 30 | config.prefix = "/assets" 31 | }) 32 | 33 | assert_equal assembly.resolver.object_id, assembly.resolver.object_id 34 | assert_equal assembly.load_path.object_id, assembly.load_path.object_id 35 | end 36 | 37 | test "instantiates a valid processor" do 38 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 39 | config.output_path = Pathname.new("#{__dir__}/../fixtures/assets") 40 | config.manifest_path = config.output_path.join(".manifest.json") 41 | config.prefix = "/assets" 42 | }) 43 | 44 | assert assembly.processor.is_a?(Propshaft::Processor) 45 | end 46 | end 47 | -------------------------------------------------------------------------------- /test/propshaft_integration_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | 3 | class PropshaftIntegrationTest < ActionDispatch::IntegrationTest 4 | test "should be able to resolve real assets" do 5 | get sample_load_real_assets_url 6 | 7 | assert_response :success 8 | 9 | assert_select 'link[href="/assets/hello_world-4137140a.css"][data-custom-attribute="true"]' 10 | assert_select 'link[href="/assets/goodbye-b1dc9940.css"][data-custom-attribute="true"]' 11 | assert_select 'link[href="/assets/library-86a3b7a9.css"][data-custom-attribute="true"]' 12 | 13 | hello_css_link = css_select('link[href="/assets/hello_world-4137140a.css"][integrity]').first 14 | assert(hello_css_link) 15 | assert_equal "stylesheet", hello_css_link["rel"] 16 | assert_equal "sha384-ZSAt6UaTZ1OYvSB1fr2WXE8izMW4qnd17BZ1zaZ3TpAdIw3VEUmyupHd/k/cMCqM", hello_css_link["integrity"] 17 | 18 | hello_js_script = css_select('script[src="/assets/hello_world-888761f8.js"]').first 19 | assert(hello_js_script) 20 | assert_equal "sha384-BIr0kyMRq2sfytK/T0XlGjfav9ZZrWkSBC2yHVunCchnkpP83H28/UtHw+m9iNHO", hello_js_script["integrity"] 21 | end 22 | 23 | test "should prioritize app assets over engine assets" do 24 | get sample_load_real_assets_url 25 | 26 | assert_select 'script[src="/assets/actioncable-2e7de4f9.js"]' 27 | end 28 | 29 | test "should find app styles via glob" do 30 | get sample_load_real_assets_url 31 | 32 | assert_select 'link[href="/assets/hello_world-4137140a.css"][data-glob-attribute="true"]' 33 | assert_select 'link[href="/assets/goodbye-b1dc9940.css"][data-glob-attribute="true"]' 34 | assert_select('link[href="/assets/library-86a3b7a9.css"][data-glob-attribute="true"]', count: 0) 35 | end 36 | 37 | test "should raise an exception when resolving nonexistent assets" do 38 | exception = assert_raises ActionView::Template::Error do 39 | get sample_load_nonexistent_assets_url 40 | end 41 | assert_equal "The asset 'nonexistent.css' was not found in the load path.", exception.message 42 | end 43 | end 44 | -------------------------------------------------------------------------------- /lib/propshaft/assembly.rb: -------------------------------------------------------------------------------- 1 | require "propshaft/manifest" 2 | require "propshaft/load_path" 3 | require "propshaft/resolver/dynamic" 4 | require "propshaft/resolver/static" 5 | require "propshaft/server" 6 | require "propshaft/processor" 7 | require "propshaft/compilers" 8 | require "propshaft/compiler/css_asset_urls" 9 | require "propshaft/compiler/js_asset_urls" 10 | require "propshaft/compiler/source_mapping_urls" 11 | 12 | class Propshaft::Assembly 13 | attr_reader :config 14 | 15 | def initialize(config) 16 | @config = config 17 | end 18 | 19 | def load_path 20 | @load_path ||= Propshaft::LoadPath.new( 21 | config.paths, 22 | compilers: compilers, 23 | version: config.version, 24 | file_watcher: config.file_watcher, 25 | integrity_hash_algorithm: config.integrity_hash_algorithm 26 | ) 27 | end 28 | 29 | def resolver 30 | @resolver ||= if config.manifest_path.exist? 31 | Propshaft::Resolver::Static.new manifest_path: config.manifest_path, prefix: config.prefix 32 | else 33 | Propshaft::Resolver::Dynamic.new load_path: load_path, prefix: config.prefix 34 | end 35 | end 36 | 37 | def prefix 38 | @prefix ||= begin 39 | prefix = config.prefix || "/" 40 | prefix.end_with?("/") ? prefix : "#{prefix}/" 41 | end 42 | end 43 | 44 | def processor 45 | Propshaft::Processor.new \ 46 | load_path: load_path, output_path: config.output_path, compilers: compilers, manifest_path: config.manifest_path 47 | end 48 | 49 | def compilers 50 | @compilers ||= 51 | Propshaft::Compilers.new(self).tap do |compilers| 52 | Array(config.compilers).each do |(mime_type, klass)| 53 | compilers.register mime_type, klass 54 | end 55 | end 56 | end 57 | 58 | def reveal(path_type = :logical_path) 59 | path_type = path_type.presence_in(%i[ logical_path path ]) || raise(ArgumentError, "Unknown path_type: #{path_type}") 60 | 61 | load_path.assets.collect do |asset| 62 | asset.send(path_type) 63 | end 64 | end 65 | end 66 | -------------------------------------------------------------------------------- /lib/propshaft/processor.rb: -------------------------------------------------------------------------------- 1 | require "propshaft/output_path" 2 | 3 | class Propshaft::Processor 4 | attr_reader :load_path, :output_path, :compilers, :manifest_path 5 | 6 | def initialize(load_path:, output_path:, compilers:, manifest_path:) 7 | @load_path, @output_path = load_path, output_path 8 | @manifest_path = manifest_path 9 | @compilers = compilers 10 | end 11 | 12 | def process 13 | ensure_output_path_exists 14 | write_manifest 15 | output_assets 16 | end 17 | 18 | def clobber 19 | FileUtils.rm_r(output_path) if File.exist?(output_path) 20 | end 21 | 22 | def clean(count) 23 | Propshaft::OutputPath.new(output_path, load_path.manifest).clean(count, 1.hour) 24 | end 25 | 26 | private 27 | def ensure_output_path_exists 28 | FileUtils.mkdir_p output_path 29 | end 30 | 31 | 32 | def write_manifest 33 | FileUtils.mkdir_p(File.dirname(manifest_path)) 34 | File.open(manifest_path, "wb+") do |manifest| 35 | manifest.write load_path.manifest.to_json 36 | end 37 | end 38 | 39 | 40 | def output_assets 41 | load_path.assets.each do |asset| 42 | unless output_path.join(asset.digested_path).exist? 43 | Propshaft.logger.info "Writing #{asset.digested_path}" 44 | FileUtils.mkdir_p output_path.join(asset.digested_path.parent) 45 | output_asset(asset) 46 | end 47 | end 48 | end 49 | 50 | def output_asset(asset) 51 | compile_asset(asset) || copy_asset(asset) 52 | end 53 | 54 | def compile_asset(asset) 55 | File.open(output_path.join(asset.digested_path), "w+") do |file| 56 | begin 57 | file.write asset.compiled_content 58 | rescue Encoding::UndefinedConversionError 59 | # FIXME: Not sure if there's a better way here? 60 | file.write asset.compiled_content.force_encoding("UTF-8") 61 | end 62 | end if compilers.compilable?(asset) 63 | end 64 | 65 | def copy_asset(asset) 66 | FileUtils.copy asset.path, output_path.join(asset.digested_path) 67 | end 68 | end 69 | -------------------------------------------------------------------------------- /test/dummy/config/environments/development.rb: -------------------------------------------------------------------------------- 1 | require "active_support/core_ext/integer/time" 2 | 3 | Rails.application.configure do 4 | # Settings specified here will take precedence over those in config/application.rb. 5 | 6 | # In the development environment your application's code is reloaded any time 7 | # it changes. This slows down response time but is perfect for development 8 | # since you don't have to restart the web server when you make code changes. 9 | config.cache_classes = false 10 | 11 | # Do not eager load code on boot. 12 | config.eager_load = false 13 | 14 | # Show full error reports. 15 | config.consider_all_requests_local = true 16 | 17 | # Enable/disable caching. By default caching is disabled. 18 | # Run rails dev:cache to toggle caching. 19 | if Rails.root.join("tmp/caching-dev.txt").exist? 20 | config.action_controller.perform_caching = true 21 | config.action_controller.enable_fragment_cache_logging = true 22 | 23 | config.cache_store = :memory_store 24 | config.public_file_server.headers = { 25 | "Cache-Control" => "public, max-age=#{2.days.to_i}" 26 | } 27 | else 28 | config.action_controller.perform_caching = false 29 | 30 | config.cache_store = :null_store 31 | end 32 | 33 | # Print deprecation notices to the Rails logger. 34 | config.active_support.deprecation = :log 35 | 36 | # Raise exceptions for disallowed deprecations. 37 | config.active_support.disallowed_deprecation = :raise 38 | 39 | # Tell Active Support which deprecation messages to disallow. 40 | config.active_support.disallowed_deprecation_warnings = [] 41 | 42 | # Suppress logger output for asset requests. 43 | config.assets.quiet = true 44 | 45 | # Raises error for missing translations. 46 | # config.i18n.raise_on_missing_translations = true 47 | 48 | # Annotate rendered view with file names. 49 | # config.action_view.annotate_rendered_view_with_filenames = true 50 | 51 | # Uncomment if you wish to allow Action Cable access from any origin. 52 | # config.action_cable.disable_request_forgery_protection = true 53 | end 54 | -------------------------------------------------------------------------------- /test/propshaft/compiler/js_asset_urls_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "minitest/mock" 3 | require "propshaft/asset" 4 | require "propshaft/assembly" 5 | require "propshaft/compilers" 6 | 7 | require "propshaft/compiler/js_asset_urls" 8 | 9 | module Propshaft 10 | class Compiler 11 | class JsAssetUrlsTest < ActiveSupport::TestCase 12 | setup do 13 | @options = ActiveSupport::OrderedOptions.new.tap do |config| 14 | config.paths = [Pathname.new("#{__dir__}/../../fixtures/assets/vendor")] 15 | config.output_path = Pathname.new("#{__dir__}/../../fixtures/output") 16 | config.prefix = "/assets" 17 | end 18 | end 19 | 20 | test "the asset exists" do 21 | js_content = <<~JS 22 | export default class extends Controller { 23 | init() { 24 | this.img = RAILS_ASSET_URL("/foobar/source/file.svg"); 25 | } 26 | } 27 | JS 28 | 29 | compiled = compile_asset_with_content(js_content) 30 | 31 | assert_match(%r{this\.img = "/assets/foobar/source/file-[a-z0-9]{8}.svg"\;}, compiled) 32 | end 33 | 34 | test "the asset does not exist" do 35 | js_content = <<~JS 36 | export default class extends Controller { 37 | init() { 38 | this.img = RAILS_ASSET_URL("missing.svg"); 39 | } 40 | } 41 | JS 42 | 43 | compiled = compile_asset_with_content(js_content) 44 | 45 | assert_match(/this\.img = "missing.svg"\;/, compiled) 46 | end 47 | 48 | private 49 | 50 | def compile_asset_with_content(content) 51 | # This has one more set of .. than it would in the propshaft repo 52 | root_path = Pathname.new("#{__dir__}/../../fixtures/assets/vendor") 53 | logical_path = "foobar/source/test.js" 54 | 55 | assembly = Propshaft::Assembly.new(@options) 56 | assembly.compilers.register("text/javascript", Propshaft::Compiler::JsAssetUrls) 57 | 58 | asset = Propshaft::Asset.new(root_path.join(logical_path), logical_path: logical_path, load_path: assembly.load_path) 59 | asset.stub(:content, content) do 60 | assembly.compilers.compile(asset) 61 | end 62 | end 63 | end 64 | end 65 | end 66 | -------------------------------------------------------------------------------- /test/dummy/config/environments/test.rb: -------------------------------------------------------------------------------- 1 | require "active_support/core_ext/integer/time" 2 | 3 | # The test environment is used exclusively to run your application's 4 | # test suite. You never need to work with it otherwise. Remember that 5 | # your test database is "scratch space" for the test suite and is wiped 6 | # and recreated between test runs. Don't rely on the data there! 7 | 8 | Rails.application.configure do 9 | # Settings specified here will take precedence over those in config/application.rb. 10 | 11 | # Turn false under Spring and add config.action_view.cache_template_loading = true 12 | config.cache_classes = true 13 | 14 | # Do not eager load code on boot. This avoids loading your whole application 15 | # just for the purpose of running a single test. If you are using a tool that 16 | # preloads Rails for running tests, you may have to set it to true. 17 | config.eager_load = false 18 | 19 | # Configure public file server for tests with Cache-Control for performance. 20 | config.public_file_server.enabled = true 21 | config.public_file_server.headers = { 22 | "Cache-Control" => "public, max-age=#{1.hour.to_i}" 23 | } 24 | 25 | # Show full error reports and disable caching. 26 | config.consider_all_requests_local = true 27 | config.action_controller.perform_caching = false 28 | config.cache_store = :null_store 29 | 30 | # Raise exceptions instead of rendering exception templates. 31 | if Rails.version < "7.1" 32 | config.action_dispatch.show_exceptions = false 33 | else 34 | # For Rails 7.0 and earlier, we set this to :none to avoid 35 | config.action_dispatch.show_exceptions = :none 36 | end 37 | 38 | # Disable request forgery protection in test environment. 39 | config.action_controller.allow_forgery_protection = false 40 | 41 | # Print deprecation notices to the stderr. 42 | config.active_support.deprecation = :stderr 43 | 44 | # Raise exceptions for disallowed deprecations. 45 | config.active_support.disallowed_deprecation = :raise 46 | 47 | # Tell Active Support which deprecation messages to disallow. 48 | config.active_support.disallowed_deprecation_warnings = [] 49 | 50 | # Raises error for missing translations. 51 | # config.i18n.raise_on_missing_translations = true 52 | 53 | # Annotate rendered view with file names. 54 | # config.action_view.annotate_rendered_view_with_filenames = true 55 | end 56 | -------------------------------------------------------------------------------- /lib/propshaft/asset.rb: -------------------------------------------------------------------------------- 1 | require "digest/sha1" 2 | require "digest/sha2" 3 | require "action_dispatch/http/mime_type" 4 | 5 | class Propshaft::Asset 6 | attr_reader :path, :logical_path, :load_path 7 | 8 | class << self 9 | def extract_path_and_digest(digested_path) 10 | digest = digested_path[/-([0-9a-zA-Z]{7,128})\.(?!digested)([^.]|.map)+\z/, 1] 11 | path = digest ? digested_path.sub("-#{digest}", "") : digested_path 12 | 13 | [path, digest] 14 | end 15 | end 16 | 17 | def initialize(path, logical_path:, load_path:) 18 | @path, @logical_path, @load_path = path, Pathname.new(logical_path), load_path 19 | end 20 | 21 | def compiled_content 22 | @compiled_content ||= load_path.compilers.compile(self) 23 | end 24 | 25 | def content(encoding: "ASCII-8BIT") 26 | File.read(path, encoding: encoding, mode: "rb") 27 | end 28 | 29 | def content_type 30 | Mime::Type.lookup_by_extension(logical_path.extname.from(1)) 31 | end 32 | 33 | def length 34 | content.size 35 | end 36 | 37 | def digest 38 | @digest ||= Digest::SHA1.hexdigest("#{content_with_compile_references}#{load_path.version}").first(8) 39 | end 40 | 41 | def integrity(hash_algorithm:) 42 | # Following the Subresource Integrity spec draft 43 | # https://w3c.github.io/webappsec-subresource-integrity/ 44 | # allowing only sha256, sha384, and sha512 45 | bitlen = case hash_algorithm 46 | when "sha256" 47 | 256 48 | when "sha384" 49 | 384 50 | when "sha512" 51 | 512 52 | else 53 | raise(StandardError.new("Subresource Integrity hash algorithm must be one of SHA2 family (sha256, sha384, sha512)")) 54 | end 55 | 56 | [hash_algorithm, Digest::SHA2.new(bitlen).base64digest(compiled_content)].join("-") 57 | end 58 | 59 | def digested_path 60 | if already_digested? 61 | logical_path 62 | else 63 | logical_path.sub(/\.(\w+(\.map)?)$/) { |ext| "-#{digest}#{ext}" } 64 | end 65 | end 66 | 67 | def fresh?(digest) 68 | self.digest == digest || already_digested? 69 | end 70 | 71 | def ==(other_asset) 72 | logical_path.hash == other_asset.logical_path.hash 73 | end 74 | 75 | private 76 | def content_with_compile_references 77 | content + load_path.find_referenced_by(self).collect(&:content).join 78 | end 79 | 80 | def already_digested? 81 | logical_path.to_s =~ /-([0-9a-zA-Z_-]{7,128})\.digested/ 82 | end 83 | end 84 | -------------------------------------------------------------------------------- /test/propshaft/resolver/dynamic_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/resolver/dynamic" 3 | 4 | class Propshaft::Resolver::DynamicTest < ActiveSupport::TestCase 5 | setup do 6 | @resolver = create_resolver 7 | end 8 | 9 | test "resolving present asset returns uri path" do 10 | assert_equal "/assets/one-f2e1ec14.txt", 11 | @resolver.resolve("one.txt") 12 | end 13 | 14 | test "reading static asset" do 15 | assert_equal "ASCII-8BIT", @resolver.read("one.txt").encoding.to_s 16 | assert_equal "One from first path", @resolver.read("one.txt") 17 | end 18 | 19 | test "reading static asset with encoding option" do 20 | assert_equal "UTF-8", @resolver.read("one.txt", encoding: "UTF-8").encoding.to_s 21 | assert_equal "One from first path", @resolver.read("one.txt", encoding: "UTF-8") 22 | end 23 | 24 | test "resolving missing asset returns nil" do 25 | assert_nil @resolver.resolve("nowhere.txt") 26 | end 27 | 28 | test "integrity for asset returns value for configured hash format" do 29 | resolver = create_resolver(integrity_hash_algorithm: "sha384") 30 | assert_equal "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe", resolver.integrity("one.txt") 31 | end 32 | 33 | test "integrity for asset return the value for the compiled content instead of the source" do 34 | compilers = [["text/css", Propshaft::Compiler::CssAssetUrls ]] 35 | resolver = create_resolver(integrity_hash_algorithm: "sha384", compilers: compilers) 36 | assert_equal "sha384-jUiHGq2aPNACr4g68crM1I28TitXJKYhEgokcX6W5VYGwufEKQxfLpe4GakM84ex", resolver.integrity("another.css") 37 | end 38 | 39 | test "integrity for asset returns nil for no configured hash format" do 40 | assert_nil @resolver.integrity("one.txt") 41 | end 42 | 43 | test "integrity for missing asset returns nil" do 44 | resolver = create_resolver(integrity_hash_algorithm: "sha384") 45 | assert_nil resolver.integrity("nowhere.txt") 46 | end 47 | 48 | private 49 | def create_resolver(integrity_hash_algorithm: nil, compilers: []) 50 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 51 | config.paths = [ 52 | Pathname.new("#{__dir__}/../../fixtures/assets/first_path"), 53 | ] 54 | config.compilers = compilers 55 | config.integrity_hash_algorithm = integrity_hash_algorithm 56 | }) 57 | 58 | Propshaft::Resolver::Dynamic.new(load_path: assembly.load_path, prefix: "/assets") 59 | end 60 | end 61 | -------------------------------------------------------------------------------- /test/propshaft/processor_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/load_path" 3 | require "propshaft/processor" 4 | 5 | class Propshaft::ProcessorTest < ActiveSupport::TestCase 6 | setup do 7 | @assembly = create_assembly 8 | end 9 | 10 | test "manifest is written" do 11 | processed do |processor| 12 | manifest = JSON.load_file(processor.output_path.join(".manifest.json")) 13 | manifest_entry = manifest["one.txt"] 14 | 15 | assert_equal "one-f2e1ec14.txt", manifest_entry["digested_path"] 16 | assert_nil manifest_entry["integrity"], "Integrity should not be present by default" 17 | end 18 | end 19 | 20 | test "integrity is written in the manifest when configured" do 21 | assembly = create_assembly do |config| 22 | config.integrity_hash_algorithm = "sha384" 23 | end 24 | 25 | processed(assembly) do |processor| 26 | manifest = JSON.load_file(processor.output_path.join(".manifest.json")) 27 | manifest_entry = manifest["one.txt"] 28 | 29 | assert_equal manifest_entry["digested_path"], "one-f2e1ec14.txt" 30 | assert_equal manifest_entry["integrity"], "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe" 31 | end 32 | end 33 | 34 | test "assets are copied" do 35 | processed do |processor| 36 | digested_asset_name = "one-f2e1ec14.txt" 37 | assert processor.output_path.join(digested_asset_name).exist? 38 | 39 | nested_digested_asset_name = "nested/three-6c2b86a0.txt" 40 | assert processor.output_path.join(nested_digested_asset_name).exist? 41 | end 42 | end 43 | 44 | test "assets are clobbered" do 45 | processed do |processor| 46 | processor.clobber 47 | assert_not File.exist?(processor.output_path) 48 | FileUtils.mkdir_p processor.output_path 49 | end 50 | end 51 | 52 | private 53 | def create_assembly 54 | Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 55 | config.output_path = Pathname.new("#{__dir__}/../fixtures/output") 56 | config.prefix = "/assets" 57 | config.paths = [ 58 | Pathname.new("#{__dir__}/../fixtures/assets/first_path"), 59 | Pathname.new("#{__dir__}/../fixtures/assets/second_path") 60 | ] 61 | yield config if block_given? 62 | }) 63 | end 64 | 65 | def processed(assembly = @assembly) 66 | Dir.mktmpdir do |output_path| 67 | output_path = Pathname.new(output_path) 68 | processor = Propshaft::Processor.new( 69 | load_path: assembly.load_path, output_path: output_path, 70 | compilers: assembly.compilers, manifest_path: output_path.join(".manifest.json") 71 | ) 72 | 73 | processor.process 74 | 75 | yield processor 76 | end 77 | end 78 | end 79 | -------------------------------------------------------------------------------- /test/propshaft/output_path_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "minitest/mock" 3 | require "propshaft/asset" 4 | require "propshaft/load_path" 5 | require "propshaft/output_path" 6 | 7 | class Propshaft::OutputPathTest < ActiveSupport::TestCase 8 | setup do 9 | @manifest = { 10 | ".manifest.json": ".manifest.json", 11 | "one.txt": "one-f2e1ec14.txt", 12 | "one.txt.map": "one-f2e1ec15.txt.map" 13 | }.stringify_keys 14 | @output_path = Propshaft::OutputPath.new(Pathname.new("#{__dir__}/../fixtures/output"), @manifest) 15 | end 16 | 17 | test "files" do 18 | files = @output_path.files 19 | 20 | file = files["one-f2e1ec14.txt"] 21 | assert_equal "one.txt", file[:logical_path] 22 | assert_equal "f2e1ec14", file[:digest] 23 | assert file[:mtime].is_a?(Time) 24 | end 25 | 26 | test "clean always keeps most current versions" do 27 | @output_path.clean(0, 0) 28 | assert @output_path.path.join(@manifest["one.txt"]) 29 | assert @output_path.path.join(@manifest[".manifest.json"]) 30 | end 31 | 32 | test "clean keeps versions of assets that no longer exist" do 33 | removed = output_asset("no-longer-in-manifest.txt", "current") 34 | @output_path.clean(1, 0) 35 | assert File.exist?(removed) 36 | ensure 37 | FileUtils.rm(removed) if File.exist?(removed) 38 | end 39 | 40 | test "clean keeps the correct number of versions" do 41 | old = output_asset("by_count.txt", "old", created_at: Time.now - 300) 42 | current = output_asset("by_count.txt", "current", created_at: Time.now - 180) 43 | 44 | @output_path.clean(1, 0) 45 | 46 | assert File.exist?(current) 47 | assert_not File.exist?(old) 48 | ensure 49 | FileUtils.rm(old) if File.exist?(old) 50 | FileUtils.rm(current) if File.exist?(current) 51 | end 52 | 53 | test "clean keeps the correct number of versions regardless of the file extension" do 54 | old = output_asset("by_count.txt.map", "old", created_at: Time.now - 300) 55 | current = output_asset("by_count.txt.map", "current", created_at: Time.now - 180) 56 | 57 | assert File.exist?(current) 58 | assert File.exist?(old) 59 | 60 | @output_path.clean(1, 0) 61 | 62 | assert File.exist?(current) 63 | assert_not File.exist?(old), "#{old} should not exist" 64 | ensure 65 | FileUtils.rm(old) if File.exist?(old) 66 | FileUtils.rm(current) if File.exist?(current) 67 | end 68 | 69 | test "clean keeps all versions under a certain age" do 70 | old = output_asset("by_age.txt", "old") 71 | current = output_asset("by_age.txt", "current") 72 | 73 | @output_path.clean(0, 3600) 74 | 75 | assert File.exist?(current) 76 | assert File.exist?(old) 77 | ensure 78 | FileUtils.rm(old) if File.exist?(old) 79 | FileUtils.rm(current) if File.exist?(current) 80 | end 81 | 82 | private 83 | def output_asset(filename, content, created_at: Time.now) 84 | load_path = Propshaft::LoadPath.new([], compilers: Propshaft::Compilers.new(nil)) 85 | asset = Propshaft::Asset.new(nil, logical_path: filename, load_path: load_path) 86 | asset.stub :content, content do 87 | output_path = @output_path.path.join(asset.digested_path) 88 | `touch -mt #{created_at.strftime('%y%m%d%H%M')} #{output_path}` 89 | output_path 90 | end 91 | end 92 | end 93 | -------------------------------------------------------------------------------- /test/propshaft/server_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/assembly" 3 | require "propshaft/server" 4 | 5 | class Propshaft::ServerTest < ActiveSupport::TestCase 6 | include Rack::Test::Methods 7 | 8 | class RackApp 9 | attr_reader :calls 10 | 11 | def initialize 12 | @calls = [] 13 | end 14 | 15 | def call(env) 16 | @calls << env 17 | [200, {}, ["OK"]] 18 | end 19 | end 20 | 21 | setup do 22 | @assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 23 | config.paths = [Pathname.new("#{__dir__}/../fixtures/assets/vendor"), Pathname.new("#{__dir__}/../fixtures/assets/first_path")] 24 | config.output_path = Pathname.new("#{__dir__}../fixtures/output") 25 | config.prefix = "/assets" 26 | }) 27 | 28 | @rack_app = RackApp.new 29 | @assembly.compilers.register "text/css", Propshaft::Compiler::CssAssetUrls 30 | @server = Propshaft::Server.new(@rack_app, @assembly) 31 | end 32 | 33 | test "forward requests not under prefix" do 34 | get "/test" 35 | assert_not_empty @rack_app.calls 36 | end 37 | 38 | test "forward requests that aren't GET or HEAD" do 39 | asset = @assembly.load_path.find("foobar/source/test.css") 40 | post "/assets/#{asset.digested_path}" 41 | assert_not_empty @rack_app.calls 42 | end 43 | 44 | test "serve a compiled file" do 45 | asset = @assembly.load_path.find("foobar/source/test.css") 46 | get "/assets/#{asset.digested_path}" 47 | 48 | assert_equal 200, last_response.status 49 | assert_equal last_response.body.bytesize.to_s, last_response.headers['content-length'] 50 | assert_equal "text/css", last_response.headers['content-type'] 51 | assert_equal "Accept-Encoding", last_response.headers['vary'] 52 | assert_equal "\"#{asset.digest}\"", last_response.headers['etag'] 53 | assert_equal "public, max-age=31536000, immutable", last_response.headers['cache-control'] 54 | assert_equal ".hero { background: url(\"/assets/foobar/source/file-3e6a1297.jpg\") }\n", 55 | last_response.body 56 | end 57 | 58 | test "serve a predigested file" do 59 | asset = @assembly.load_path.find("file-already-abcdefVWXYZ0123456789_-.digested.css") 60 | get "/assets/#{asset.digested_path}" 61 | assert_equal 200, last_response.status 62 | end 63 | 64 | test "serve a sourcemap" do 65 | asset = @assembly.load_path.find("file-is-a-sourcemap.js.map") 66 | get "/assets/#{asset.digested_path}" 67 | assert_equal 200, last_response.status 68 | end 69 | 70 | test "not found" do 71 | get "/assets/not-found.js" 72 | 73 | assert_equal 404, last_response.status 74 | assert_equal "9", last_response.headers['content-length'] 75 | assert_equal "text/plain", last_response.headers['content-type'] 76 | assert_equal "Not found", last_response.body 77 | assert_not last_response.headers.key?('cache-control') 78 | assert_not last_response.headers.key?('etag') 79 | assert_not last_response.headers.key?('accept-encoding') 80 | end 81 | 82 | test "not found if digest does not match" do 83 | asset = @assembly.load_path.find("foobar/source/test.css") 84 | get "/assets/#{asset.logical_path}" 85 | assert_equal 404, last_response.status 86 | end 87 | 88 | private 89 | def app 90 | @app ||= Rack::Lint.new(@server) 91 | end 92 | end 93 | -------------------------------------------------------------------------------- /test/propshaft/load_path_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/load_path" 3 | 4 | class Propshaft::LoadPathTest < ActiveSupport::TestCase 5 | setup do 6 | @load_path = Propshaft::LoadPath.new [ 7 | Pathname.new("#{__dir__}/../fixtures/assets/first_path"), 8 | Pathname.new("#{__dir__}/../fixtures/assets/second_path").to_s 9 | ], compilers: Propshaft::Compilers.new(nil) 10 | end 11 | 12 | test "find asset that only appears once in the paths" do 13 | assert_equal "Two from second path", @load_path.find("two.txt").content 14 | end 15 | 16 | test "find asset from first path if it appears twice in the paths" do 17 | assert_equal "One from first path", @load_path.find("one.txt").content 18 | end 19 | 20 | test "find nested asset" do 21 | assert_equal "Three from first path", @load_path.find("nested/three.txt").content 22 | end 23 | 24 | test "assets" do 25 | assert_includes @load_path.assets, find_asset("one.txt") 26 | end 27 | 28 | test "assets dont include dot files" do 29 | assert_not_includes @load_path.assets, find_asset(".stuff") 30 | end 31 | 32 | test "manifest" do 33 | @load_path.manifest.tap do |manifest| 34 | assert_equal "one-f2e1ec14.txt", manifest["one.txt"].digested_path.to_s 35 | assert_equal "nested/three-6c2b86a0.txt", manifest["nested/three.txt"].digested_path.to_s 36 | end 37 | end 38 | 39 | test "manifest with version" do 40 | @load_path = Propshaft::LoadPath.new(@load_path.paths, version: "1", compilers: Propshaft::Compilers.new(nil)) 41 | @load_path.manifest.tap do |manifest| 42 | assert_equal "one-c9373b68.txt", manifest["one.txt"].digested_path.to_s 43 | assert_equal "nested/three-a41a5d38.txt", manifest["nested/three.txt"].digested_path.to_s 44 | end 45 | end 46 | 47 | test "missing load path directory" do 48 | assert_nil Propshaft::LoadPath.new(Pathname.new("#{__dir__}/../fixtures/assets/nowhere"), compilers: Propshaft::Compilers.new(nil)).find("missing") 49 | end 50 | 51 | test "deduplicate paths" do 52 | load_path = Propshaft::LoadPath.new [ 53 | "app/javascript", 54 | "app/javascript/packs", 55 | "app/assets/stylesheets", 56 | "app/assets/images", 57 | "app/assets" 58 | ], compilers: Propshaft::Compilers.new(nil) 59 | 60 | paths = load_path.paths 61 | assert_equal 2, paths.count 62 | assert_equal Pathname.new("app/javascript"), paths.first 63 | assert_equal Pathname.new("app/assets"), paths.last 64 | end 65 | 66 | test "asset paths by type" do 67 | assert_equal \ 68 | ["another.css", "dependent/a.css", "dependent/b.css", "dependent/c.css", "file-already-abcdefVWXYZ0123456789_-.digested.css", "file-already-abcdefVWXYZ0123456789_-.digested.debug.css", "file-not.digested.css"], 69 | @load_path.asset_paths_by_type("css") 70 | end 71 | 72 | test "asset paths by glob" do 73 | assert_equal \ 74 | ["dependent/a.css", "dependent/b.css", "dependent/c.css"], 75 | @load_path.asset_paths_by_glob("**/dependent/*.css") 76 | end 77 | 78 | private 79 | def find_asset(logical_path) 80 | root_path = Pathname.new("#{__dir__}/../fixtures/assets/first_path") 81 | load_path = Propshaft::LoadPath.new([ root_path ], compilers: Propshaft::Compilers.new(nil)) 82 | Propshaft::Asset.new(root_path.join(logical_path), logical_path: logical_path, load_path: load_path) 83 | end 84 | end 85 | -------------------------------------------------------------------------------- /test/propshaft/resolver/static_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "minitest/mock" 3 | require "propshaft/resolver/static" 4 | 5 | class Propshaft::Resolver::StaticTest < ActiveSupport::TestCase 6 | setup do 7 | @resolver = Propshaft::Resolver::Static.new( 8 | manifest_path: Pathname.new("#{__dir__}/../../fixtures/output/.manifest.json"), 9 | prefix: "/assets" 10 | ) 11 | end 12 | 13 | test "resolving present asset returns uri path" do 14 | assert_equal \ 15 | "/assets/one-f2e1ec14.txt", 16 | @resolver.resolve("one.txt") 17 | end 18 | 19 | test "reading static asset" do 20 | assert_equal "ASCII-8BIT", @resolver.read("one.txt").encoding.to_s 21 | assert_equal "One from first path", @resolver.read("one.txt") 22 | end 23 | 24 | test "reading static asset with encoding option" do 25 | assert_equal "UTF-8", @resolver.read("one.txt", encoding: "UTF-8").encoding.to_s 26 | assert_equal "One from first path", @resolver.read("one.txt", encoding: "UTF-8") 27 | end 28 | 29 | test "resolving missing asset returns nil" do 30 | assert_nil @resolver.resolve("nowhere.txt") 31 | end 32 | 33 | test "integrity for asset returns nil for simple manifest" do 34 | assert_nil @resolver.integrity("one.txt") 35 | end 36 | 37 | test "integrity for missing asset returns nil" do 38 | assert_nil @resolver.integrity("nowhere.txt") 39 | end 40 | 41 | test "resolver requests json optimizer gems to keep parsed manifest keys as strings" do 42 | stub = Proc.new do |_, opts| 43 | assert_equal false, opts[:symbolize_names] 44 | {} 45 | end 46 | 47 | JSON.stub :parse, stub do 48 | @resolver.resolve("one.txt") 49 | end 50 | end 51 | 52 | class Propshaft::Resolver::StaticTest::WithExtensibleManifest < ActiveSupport::TestCase 53 | setup do 54 | @resolver = Propshaft::Resolver::Static.new( 55 | manifest_path: Pathname.new("#{__dir__}/../../fixtures/new_manifest_format/.manifest.json"), 56 | prefix: "/assets" 57 | ) 58 | end 59 | 60 | test "resolving present asset returns uri path" do 61 | assert_equal \ 62 | "/assets/one-f2e1ec14.txt", 63 | @resolver.resolve("one.txt") 64 | end 65 | 66 | test "reading static asset" do 67 | assert_equal "ASCII-8BIT", @resolver.read("one.txt").encoding.to_s 68 | assert_equal "One from first path", @resolver.read("one.txt") 69 | end 70 | 71 | test "reading static asset with encoding option" do 72 | assert_equal "UTF-8", @resolver.read("one.txt", encoding: "UTF-8").encoding.to_s 73 | assert_equal "One from first path", @resolver.read("one.txt", encoding: "UTF-8") 74 | end 75 | 76 | test "resolving missing asset returns nil" do 77 | assert_nil @resolver.resolve("nowhere.txt") 78 | end 79 | 80 | test "integrity for asset returns value from extensible manifest" do 81 | assert_equal "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe", @resolver.integrity("one.txt") 82 | end 83 | 84 | test "integrity for missing asset returns nil" do 85 | assert_nil @resolver.integrity("nowhere.txt") 86 | end 87 | 88 | test "resolver requests json optimizer gems to keep parsed manifest keys as strings" do 89 | stub = Proc.new do |_, opts| 90 | assert_equal false, opts[:symbolize_names] 91 | {} 92 | end 93 | 94 | JSON.stub :parse, stub do 95 | @resolver.resolve("one.txt") 96 | end 97 | end 98 | end 99 | end 100 | -------------------------------------------------------------------------------- /lib/propshaft/railtie.rb: -------------------------------------------------------------------------------- 1 | require "rails" 2 | require "active_support/ordered_options" 3 | require "propshaft/quiet_assets" 4 | 5 | module Propshaft 6 | class Railtie < ::Rails::Railtie 7 | config.assets = ActiveSupport::OrderedOptions.new 8 | config.assets.paths = [] 9 | config.assets.excluded_paths = [] 10 | config.assets.version = "1" 11 | config.assets.prefix = "/assets" 12 | config.assets.quiet = false 13 | config.assets.compilers = [ 14 | [ "text/css", Propshaft::Compiler::CssAssetUrls ], 15 | [ "text/css", Propshaft::Compiler::SourceMappingUrls ], 16 | [ "text/javascript", Propshaft::Compiler::JsAssetUrls ], 17 | [ "text/javascript", Propshaft::Compiler::SourceMappingUrls ], 18 | ] 19 | config.assets.sweep_cache = Rails.env.development? 20 | config.assets.server = Rails.env.development? || Rails.env.test? 21 | config.assets.relative_url_root = nil 22 | 23 | # Register propshaft initializer to copy the assets path in all the Rails Engines. 24 | # This makes possible for us to keep all `assets` config in this Railtie, but still 25 | # allow engines to automatically register their own paths. 26 | Rails::Engine.initializer "propshaft.append_assets_path", group: :all do |app| 27 | app.config.assets.paths.unshift(*paths["vendor/assets"].existent_directories) 28 | app.config.assets.paths.unshift(*paths["lib/assets"].existent_directories) 29 | app.config.assets.paths.unshift(*paths["app/assets"].existent_directories) 30 | 31 | app.config.assets.paths = app.config.assets.paths.without(Array(app.config.assets.excluded_paths).collect(&:to_s)) 32 | end 33 | 34 | config.after_initialize do |app| 35 | # Prioritize assets from within the application over assets of the same path from engines/gems. 36 | config.assets.paths.sort_by!.with_index { |path, i| [path.to_s.start_with?(Rails.root.to_s) ? 0 : 1, i] } 37 | 38 | config.assets.file_watcher ||= app.config.file_watcher 39 | 40 | config.assets.relative_url_root ||= app.config.relative_url_root 41 | config.assets.output_path ||= 42 | Pathname.new(File.join(app.config.paths["public"].first, app.config.assets.prefix)) 43 | config.assets.manifest_path ||= config.assets.output_path.join(".manifest.json") 44 | 45 | ActiveSupport.on_load(:action_view) do 46 | include Propshaft::Helper 47 | end 48 | 49 | if config.assets.sweep_cache 50 | ActiveSupport.on_load(:action_controller_base) do 51 | before_action { Rails.application.assets.load_path.cache_sweeper.execute_if_updated } 52 | end 53 | end 54 | end 55 | 56 | initializer "propshaft.logger" do 57 | Propshaft.logger = config.assets.logger || Rails.logger 58 | end 59 | 60 | initializer :quiet_assets do |app| 61 | if app.config.assets.quiet 62 | app.middleware.insert_before ::Rails::Rack::Logger, Propshaft::QuietAssets 63 | end 64 | end 65 | 66 | initializer "propshaft.assets_middleware", group: :all do |app| 67 | app.assets = Propshaft::Assembly.new(app.config.assets) 68 | if config.assets.server 69 | app.middleware.insert_before ::ActionDispatch::Executor, Propshaft::Server, app.assets 70 | end 71 | end 72 | 73 | rake_tasks do 74 | load "propshaft/railties/assets.rake" 75 | end 76 | 77 | # Compatibility shiming (need to provide log warnings when used) 78 | config.assets.precompile = [] 79 | config.assets.debug = nil 80 | config.assets.compile = nil 81 | config.assets.css_compressor = nil 82 | config.assets.js_compressor = nil 83 | end 84 | end 85 | -------------------------------------------------------------------------------- /lib/propshaft/load_path.rb: -------------------------------------------------------------------------------- 1 | require "propshaft/manifest" 2 | require "propshaft/asset" 3 | 4 | class Propshaft::LoadPath 5 | class NullFileWatcher # :nodoc: 6 | def initialize(paths, files_to_watch, &block) 7 | @block = block 8 | end 9 | 10 | def execute_if_updated 11 | @block.call 12 | end 13 | end 14 | 15 | attr_reader :paths, :compilers, :version, :integrity_hash_algorithm 16 | 17 | def initialize(paths = [], compilers:, version: nil, file_watcher: nil, integrity_hash_algorithm: nil) 18 | @paths, @compilers, @version, @integrity_hash_algorithm = dedup(paths), compilers, version, integrity_hash_algorithm 19 | @file_watcher = file_watcher || NullFileWatcher 20 | end 21 | 22 | def find(asset_name) 23 | assets_by_path[asset_name] 24 | end 25 | 26 | def find_referenced_by(asset) 27 | compilers.referenced_by(asset).delete(self) 28 | end 29 | 30 | def assets 31 | assets_by_path.values 32 | end 33 | 34 | def asset_paths_by_type(content_type) 35 | (@cached_asset_paths_by_type ||= Hash.new)[content_type] ||= 36 | extract_logical_paths_from(assets.select { |a| a.content_type == Mime::EXTENSION_LOOKUP[content_type] }) 37 | end 38 | 39 | def asset_paths_by_glob(glob) 40 | (@cached_asset_paths_by_glob ||= Hash.new)[glob] ||= 41 | extract_logical_paths_from(assets.select { |a| a.path.fnmatch?(glob) }) 42 | end 43 | 44 | def manifest 45 | Propshaft::Manifest.new(integrity_hash_algorithm: integrity_hash_algorithm).tap do |manifest| 46 | assets.each { |asset| manifest.push_asset(asset) } 47 | end 48 | end 49 | 50 | # Returns a file watcher object configured to clear the cache of the load_path 51 | # when the directories passed during its initialization have changes. This is used in development 52 | # and test to ensure the map caches are reset when javascript files are changed. 53 | def cache_sweeper 54 | @cache_sweeper ||= begin 55 | exts_to_watch = Mime::EXTENSION_LOOKUP.map(&:first) 56 | files_to_watch = Array(paths).collect { |dir| [ dir.to_s, exts_to_watch ] }.to_h 57 | mutex = Mutex.new 58 | 59 | @file_watcher.new([], files_to_watch) do 60 | mutex.synchronize do 61 | clear_cache 62 | seed_cache 63 | end 64 | end 65 | end 66 | end 67 | 68 | private 69 | def assets_by_path 70 | @cached_assets_by_path ||= Hash.new.tap do |mapped| 71 | paths.each do |path| 72 | without_dotfiles(all_files_from_tree(path)).each do |file| 73 | logical_path = file.relative_path_from(path) 74 | mapped[logical_path.to_s] ||= Propshaft::Asset.new(file, logical_path: logical_path, load_path: self) 75 | end if path.exist? 76 | end 77 | end 78 | end 79 | 80 | def all_files_from_tree(path) 81 | path.children.flat_map { |child| child.directory? ? all_files_from_tree(child) : child } 82 | end 83 | 84 | def extract_logical_paths_from(assets) 85 | assets.collect { |asset| asset.logical_path.to_s }.sort 86 | end 87 | 88 | def without_dotfiles(files) 89 | files.reject { |file| file.basename.to_s.starts_with?(".") } 90 | end 91 | 92 | def clear_cache 93 | @cached_assets_by_path = nil 94 | @cached_asset_paths_by_type = nil 95 | @cached_asset_paths_by_glob = nil 96 | end 97 | 98 | def seed_cache 99 | assets_by_path 100 | end 101 | 102 | def dedup(paths) 103 | paths = Array(paths).map { |path| Pathname.new(path) } 104 | deduped = [].tap do |deduped| 105 | paths.sort.each { |path| deduped << path if deduped.blank? || !path.to_s.start_with?(deduped.last.to_s) } 106 | end 107 | 108 | paths & deduped 109 | end 110 | end 111 | -------------------------------------------------------------------------------- /test/propshaft/compiler/source_mapping_urls_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "minitest/mock" 3 | require "propshaft/asset" 4 | require "propshaft/assembly" 5 | require "propshaft/compilers" 6 | 7 | class Propshaft::Compiler::SourceMappingUrlsTest < ActiveSupport::TestCase 8 | setup do 9 | @options = ActiveSupport::OrderedOptions.new.tap { |config| 10 | config.paths = [ Pathname.new("#{__dir__}/../../fixtures/assets/mapped") ] 11 | config.output_path = Pathname.new("#{__dir__}/../../fixtures/output") 12 | config.prefix = "/assets" 13 | } 14 | end 15 | 16 | test "matching source map" do 17 | assert_match %r{//# sourceMappingURL=/assets/source-[a-z0-9]{8}\.js.map}, 18 | compile_asset(find_asset("source.js", fixture_path: "mapped")) 19 | assert_match %r{/\*# sourceMappingURL=/assets/source-[a-z0-9]{8}\.css.map}, 20 | compile_asset(find_asset("source.css", fixture_path: "mapped")) 21 | end 22 | 23 | test "matching nested source map" do 24 | assert_match %r{//# sourceMappingURL=/assets/nested/another-source-[a-z0-9]{8}\.js.map}, 25 | compile_asset(find_asset("nested/another-source.js", fixture_path: "mapped")) 26 | end 27 | 28 | test "missing source map" do 29 | assert_no_match %r{sourceMappingURL}, 30 | compile_asset(find_asset("sourceless.js", fixture_path: "mapped")) 31 | assert_no_match %r{sourceMappingURL}, 32 | compile_asset(find_asset("sourceless.css", fixture_path: "mapped")) 33 | end 34 | 35 | test "sourceMappingURL removal due to missing map does not damage /* ... */ comments" do 36 | assert_match %r{\A#{Regexp.escape ".failure { color: red; }\n/* */\n"}\Z}, 37 | compile_asset(find_asset("sourceless.css", fixture_path: "mapped")) 38 | end 39 | 40 | test "sourceMappingURL not at the beginning of the line, but at end of file, is processed" do 41 | assert_match %r{//# sourceMappingURL=/assets/sourceMappingURL-not-at-start-[a-z0-9]{8}\.js.map}, 42 | compile_asset(find_asset("sourceMappingURL-not-at-start.js", fixture_path: "mapped")) 43 | assert_match %r{/\*# sourceMappingURL=/assets/sourceMappingURL-not-at-start-[a-z0-9]{8}\.css.map \*/}, 44 | compile_asset(find_asset("sourceMappingURL-not-at-start.css", fixture_path: "mapped")) 45 | end 46 | 47 | test "sourceMappingURL not at end of file should be left alone" do 48 | assert_match %r{sourceMappingURL=sourceMappingURL-not-at-end.css.map}, 49 | compile_asset(find_asset("sourceMappingURL-not-at-end.css", fixture_path: "mapped")) 50 | end 51 | test "sourceMappingURL outside of a comment should be left alone" do 52 | assert_match %r{sourceMappingURL=sourceMappingURL-outside-comment.css.map}, 53 | compile_asset(find_asset("sourceMappingURL-outside-comment.css", fixture_path: "mapped")) 54 | end 55 | 56 | test "sourceMapURL is already prefixed with url_prefix" do 57 | assert_match %r{//# sourceMappingURL=/assets/sourceMappingURL-already-prefixed-[a-z0-9]{8}\.js\.map}, 58 | compile_asset(find_asset("sourceMappingURL-already-prefixed.js", fixture_path: "mapped")) 59 | assert_match %r{//# sourceMappingURL=/assets/nested/sourceMappingURL-already-prefixed-nested-[a-z0-9]{8}\.js\.map}, 60 | compile_asset(find_asset("nested/sourceMappingURL-already-prefixed-nested.js", fixture_path: "mapped")) 61 | end 62 | 63 | test "sourceMapURL is already prefixed with an incorrect url_prefix" do 64 | refute_match %r{//# sourceMappingURL=thisisinvalidassets/sourceMappingURL-already-prefixed-invalid.js-[a-z0-9]{8}\.map}, 65 | compile_asset(find_asset("sourceMappingURL-already-prefixed-invalid.js", fixture_path: "mapped")) 66 | end 67 | 68 | test "relative url root" do 69 | @options.relative_url_root = "/url-root" 70 | 71 | assert_match %r{//# sourceMappingURL=/url-root/assets/source-[a-z0-9]{8}\.js.map}, 72 | compile_asset(find_asset("source.js", fixture_path: "mapped")) 73 | end 74 | 75 | private 76 | def compile_asset(asset) 77 | 78 | assembly = Propshaft::Assembly.new(@options) 79 | assembly.compilers.register "text/javascript", Propshaft::Compiler::SourceMappingUrls 80 | assembly.compilers.register "text/css", Propshaft::Compiler::SourceMappingUrls 81 | 82 | assembly.compilers.compile(asset) 83 | end 84 | end 85 | 86 | # //# sourceMappingURL=/assets/sourceMappingURL-already-prefixed.js-[a-z0-9]{40}.map 87 | # //# sourceMappingURL=/assets/sourceMappingURL-already-prefixed.js-da39a3ee.map 88 | -------------------------------------------------------------------------------- /test/dummy/config/environments/production.rb: -------------------------------------------------------------------------------- 1 | require "active_support/core_ext/integer/time" 2 | 3 | Rails.application.configure do 4 | # Settings specified here will take precedence over those in config/application.rb. 5 | 6 | # Code is not reloaded between requests. 7 | config.cache_classes = true 8 | 9 | # Eager load code on boot. This eager loads most of Rails and 10 | # your application in memory, allowing both threaded web servers 11 | # and those relying on copy on write to perform better. 12 | # Rake tasks automatically ignore this option for performance. 13 | config.eager_load = true 14 | 15 | # Full error reports are disabled and caching is turned on. 16 | config.consider_all_requests_local = false 17 | config.action_controller.perform_caching = true 18 | 19 | # Ensures that a master key has been made available in either ENV["RAILS_MASTER_KEY"] 20 | # or in config/master.key. This key is used to decrypt credentials (and other encrypted files). 21 | # config.require_master_key = true 22 | 23 | # Disable serving static files from the `/public` folder by default since 24 | # Apache or NGINX already handles this. 25 | config.public_file_server.enabled = ENV["RAILS_SERVE_STATIC_FILES"].present? 26 | 27 | # Compress CSS using a preprocessor. 28 | # config.assets.css_compressor = :sass 29 | 30 | # Do not fallback to assets pipeline if a precompiled asset is missed. 31 | config.assets.compile = false 32 | 33 | # Enable serving of images, stylesheets, and JavaScripts from an asset server. 34 | # config.asset_host = "http://assets.example.com" 35 | 36 | # Specifies the header that your server uses for sending files. 37 | # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for Apache 38 | # config.action_dispatch.x_sendfile_header = "X-Accel-Redirect" # for NGINX 39 | 40 | # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies. 41 | # config.force_ssl = true 42 | 43 | # Include generic and useful information about system operation, but avoid logging too much 44 | # information to avoid inadvertent exposure of personally identifiable information (PII). 45 | config.log_level = :info 46 | 47 | # Prepend all log lines with the following tags. 48 | config.log_tags = [ :request_id ] 49 | 50 | # Use a different cache store in production. 51 | # config.cache_store = :mem_cache_store 52 | 53 | # Use a real queuing backend for Active Job (and separate queues per environment). 54 | # config.active_job.queue_adapter = :resque 55 | # config.active_job.queue_name_prefix = "dummy_production" 56 | 57 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to 58 | # the I18n.default_locale when a translation cannot be found). 59 | config.i18n.fallbacks = true 60 | 61 | # Don't log any deprecations. 62 | config.active_support.report_deprecations = false 63 | 64 | # Use default logging formatter so that PID and timestamp are not suppressed. 65 | config.log_formatter = ::Logger::Formatter.new 66 | 67 | # Use a different logger for distributed setups. 68 | # require "syslog/logger" 69 | # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new "app-name") 70 | 71 | if ENV["RAILS_LOG_TO_STDOUT"].present? 72 | logger = ActiveSupport::Logger.new(STDOUT) 73 | logger.formatter = config.log_formatter 74 | config.logger = ActiveSupport::TaggedLogging.new(logger) 75 | end 76 | 77 | # Inserts middleware to perform automatic connection switching. 78 | # The `database_selector` hash is used to pass options to the DatabaseSelector 79 | # middleware. The `delay` is used to determine how long to wait after a write 80 | # to send a subsequent read to the primary. 81 | # 82 | # The `database_resolver` class is used by the middleware to determine which 83 | # database is appropriate to use based on the time delay. 84 | # 85 | # The `database_resolver_context` class is used by the middleware to set 86 | # timestamps for the last write to the primary. The resolver uses the context 87 | # class timestamps to determine how long to wait before reading from the 88 | # replica. 89 | # 90 | # By default Rails will store a last write timestamp in the session. The 91 | # DatabaseSelector middleware is designed as such you can define your own 92 | # strategy for connection switching and pass that into the middleware through 93 | # these configuration options. 94 | # config.active_record.database_selector = { delay: 2.seconds } 95 | # config.active_record.database_resolver = ActiveRecord::Middleware::DatabaseSelector::Resolver 96 | # config.active_record.database_resolver_context = ActiveRecord::Middleware::DatabaseSelector::Resolver::Session 97 | end 98 | -------------------------------------------------------------------------------- /test/propshaft/asset_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/asset" 3 | require "propshaft/load_path" 4 | 5 | class Propshaft::AssetTest < ActiveSupport::TestCase 6 | test "content" do 7 | assert_equal "ASCII-8BIT", find_asset("one.txt").content.encoding.to_s 8 | assert_equal "One from first path", find_asset("one.txt").content 9 | end 10 | 11 | test "content with encoding" do 12 | assert_equal "UTF-8", find_asset("one.txt").content(encoding: "UTF-8").encoding.to_s 13 | assert_equal "One from first path", find_asset("one.txt").content(encoding: "UTF-8") 14 | end 15 | 16 | test "content type" do 17 | assert_equal "text/plain", find_asset("one.txt").content_type.to_s 18 | assert_equal "text/javascript", find_asset("again.js").content_type.to_s 19 | assert_equal "text/css", find_asset("another.css").content_type.to_s 20 | end 21 | 22 | test "length" do 23 | assert_equal 19, find_asset("one.txt").length 24 | end 25 | 26 | test "digest" do 27 | assert_equal "f2e1ec14", find_asset("one.txt").digest 28 | end 29 | 30 | test "fresh" do 31 | assert find_asset("one.txt").fresh?("f2e1ec14") 32 | assert_not find_asset("one.txt").fresh?("e206c34f") 33 | 34 | assert find_asset("file-already-abcdefVWXYZ0123456789_-.digested.css").fresh?(nil) 35 | end 36 | 37 | test "digested path" do 38 | assert_equal "one-f2e1ec14.txt", 39 | find_asset("one.txt").digested_path.to_s 40 | 41 | assert_equal "file-already-abcdefVWXYZ0123456789_-.digested.css", 42 | find_asset("file-already-abcdefVWXYZ0123456789_-.digested.css").digested_path.to_s 43 | 44 | assert_equal "file-already-abcdefVWXYZ0123456789_-.digested.debug.css", 45 | find_asset("file-already-abcdefVWXYZ0123456789_-.digested.debug.css").digested_path.to_s 46 | 47 | assert_equal "file-not.digested-e206c34f.css", 48 | find_asset("file-not.digested.css").digested_path.to_s 49 | 50 | assert_equal "file-is-a-sourcemap-da39a3ee.js.map", 51 | find_asset("file-is-a-sourcemap.js.map").digested_path.to_s 52 | end 53 | 54 | test "integrity" do 55 | assert_equal "sha256-+C/K/0dPvIdSC8rl/NDS8zqPp08R0VH+hKMM4D8tNJs=", 56 | find_asset("one.txt").integrity(hash_algorithm: "sha256").to_s 57 | 58 | assert_equal "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe", 59 | find_asset("one.txt").integrity(hash_algorithm: "sha384").to_s 60 | 61 | assert_equal "sha512-wzPP7om24750PjHXRlgiDOhILPd4V2AbLRxomBudQaTDI1eYZkM5j8pSH/ylSSUxiGqXR3F6lgVCbsmXkqKrEg==", 62 | find_asset("one.txt").integrity(hash_algorithm: "sha512").to_s 63 | 64 | exception = assert_raises StandardError do 65 | find_asset("one.txt").integrity(hash_algorithm: "md5") 66 | end 67 | assert_equal "Subresource Integrity hash algorithm must be one of SHA2 family (sha256, sha384, sha512)", exception.message 68 | end 69 | 70 | test "value object equality" do 71 | assert_equal find_asset("one.txt"), find_asset("one.txt") 72 | end 73 | 74 | test "compiled content for non-compilable asset" do 75 | asset = find_asset("one.txt") 76 | assert_equal "One from first path", asset.compiled_content 77 | assert_equal asset.content, asset.compiled_content 78 | end 79 | 80 | test "compiled content for css asset with url transformation" do 81 | asset = find_asset("another.css") 82 | compiled = asset.compiled_content 83 | 84 | assert_match(/url\("\/archive-[a-f0-9]+\.svg"\)/, compiled) 85 | assert_not_equal asset.content, asset.compiled_content 86 | end 87 | 88 | 89 | test "costly methods are memoized" do 90 | asset = find_asset("one.txt") 91 | assert_equal asset.digest.object_id, asset.digest.object_id 92 | end 93 | 94 | test "digest depends on first level of compiler dependency" do 95 | open_asset_with_reset("dependent/b.css") do |asset_file| 96 | digest_before_dependency_change = find_asset("dependent/a.css").digest 97 | 98 | asset_file.write "changes!" 99 | asset_file.flush 100 | 101 | digest_after_dependency_change = find_asset("dependent/a.css").digest 102 | 103 | assert_not_equal digest_before_dependency_change, digest_after_dependency_change 104 | end 105 | end 106 | 107 | test "digest depends on second level of compiler dependency" do 108 | open_asset_with_reset("dependent/c.css") do |asset_file| 109 | digest_before_dependency_change = find_asset("dependent/a.css").digest 110 | 111 | asset_file.write "changes!" 112 | asset_file.flush 113 | 114 | digest_after_dependency_change = find_asset("dependent/a.css").digest 115 | 116 | assert_not_equal digest_before_dependency_change, digest_after_dependency_change 117 | end 118 | end 119 | 120 | private 121 | def find_asset(logical_path) 122 | root_path = Pathname.new("#{__dir__}/../fixtures/assets/first_path") 123 | path = root_path.join(logical_path) 124 | 125 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 126 | config.paths = [ root_path ] 127 | config.compilers = [[ "text/css", Propshaft::Compiler::CssAssetUrls ]] 128 | }) 129 | 130 | Propshaft::Asset.new(path, logical_path: logical_path, load_path: assembly.load_path) 131 | end 132 | 133 | def open_asset_with_reset(logical_path) 134 | dependency_path = Pathname.new("#{__dir__}/../fixtures/assets/first_path/#{logical_path}") 135 | existing_dependency_content = File.read(dependency_path) 136 | 137 | File.open(dependency_path, "a") { |f| yield f } 138 | ensure 139 | File.write(dependency_path, existing_dependency_content) 140 | end 141 | end 142 | -------------------------------------------------------------------------------- /Gemfile.lock: -------------------------------------------------------------------------------- 1 | PATH 2 | remote: . 3 | specs: 4 | propshaft (1.3.1) 5 | actionpack (>= 7.0.0) 6 | activesupport (>= 7.0.0) 7 | rack 8 | 9 | GEM 10 | remote: https://rubygems.org/ 11 | specs: 12 | actioncable (8.0.2) 13 | actionpack (= 8.0.2) 14 | activesupport (= 8.0.2) 15 | nio4r (~> 2.0) 16 | websocket-driver (>= 0.6.1) 17 | zeitwerk (~> 2.6) 18 | actionmailbox (8.0.2) 19 | actionpack (= 8.0.2) 20 | activejob (= 8.0.2) 21 | activerecord (= 8.0.2) 22 | activestorage (= 8.0.2) 23 | activesupport (= 8.0.2) 24 | mail (>= 2.8.0) 25 | actionmailer (8.0.2) 26 | actionpack (= 8.0.2) 27 | actionview (= 8.0.2) 28 | activejob (= 8.0.2) 29 | activesupport (= 8.0.2) 30 | mail (>= 2.8.0) 31 | rails-dom-testing (~> 2.2) 32 | actionpack (8.0.2) 33 | actionview (= 8.0.2) 34 | activesupport (= 8.0.2) 35 | nokogiri (>= 1.8.5) 36 | rack (>= 2.2.4) 37 | rack-session (>= 1.0.1) 38 | rack-test (>= 0.6.3) 39 | rails-dom-testing (~> 2.2) 40 | rails-html-sanitizer (~> 1.6) 41 | useragent (~> 0.16) 42 | actiontext (8.0.2) 43 | actionpack (= 8.0.2) 44 | activerecord (= 8.0.2) 45 | activestorage (= 8.0.2) 46 | activesupport (= 8.0.2) 47 | globalid (>= 0.6.0) 48 | nokogiri (>= 1.8.5) 49 | actionview (8.0.2) 50 | activesupport (= 8.0.2) 51 | builder (~> 3.1) 52 | erubi (~> 1.11) 53 | rails-dom-testing (~> 2.2) 54 | rails-html-sanitizer (~> 1.6) 55 | activejob (8.0.2) 56 | activesupport (= 8.0.2) 57 | globalid (>= 0.3.6) 58 | activemodel (8.0.2) 59 | activesupport (= 8.0.2) 60 | activerecord (8.0.2) 61 | activemodel (= 8.0.2) 62 | activesupport (= 8.0.2) 63 | timeout (>= 0.4.0) 64 | activestorage (8.0.2) 65 | actionpack (= 8.0.2) 66 | activejob (= 8.0.2) 67 | activerecord (= 8.0.2) 68 | activesupport (= 8.0.2) 69 | marcel (~> 1.0) 70 | activesupport (8.0.2) 71 | base64 72 | benchmark (>= 0.3) 73 | bigdecimal 74 | concurrent-ruby (~> 1.0, >= 1.3.1) 75 | connection_pool (>= 2.2.5) 76 | drb 77 | i18n (>= 1.6, < 2) 78 | logger (>= 1.4.2) 79 | minitest (>= 5.1) 80 | securerandom (>= 0.3) 81 | tzinfo (~> 2.0, >= 2.0.5) 82 | uri (>= 0.13.1) 83 | base64 (0.3.0) 84 | benchmark (0.4.1) 85 | bigdecimal (3.2.2) 86 | builder (3.3.0) 87 | concurrent-ruby (1.3.5) 88 | connection_pool (2.5.3) 89 | crass (1.0.6) 90 | date (3.4.1) 91 | debug (1.11.0) 92 | irb (~> 1.10) 93 | reline (>= 0.3.8) 94 | drb (2.2.3) 95 | erb (5.0.1) 96 | erubi (1.13.1) 97 | globalid (1.2.1) 98 | activesupport (>= 6.1) 99 | i18n (1.14.7) 100 | concurrent-ruby (~> 1.0) 101 | io-console (0.8.0) 102 | irb (1.15.2) 103 | pp (>= 0.6.0) 104 | rdoc (>= 4.0.0) 105 | reline (>= 0.4.2) 106 | logger (1.7.0) 107 | loofah (2.24.1) 108 | crass (~> 1.0.2) 109 | nokogiri (>= 1.12.0) 110 | mail (2.8.1) 111 | mini_mime (>= 0.1.1) 112 | net-imap 113 | net-pop 114 | net-smtp 115 | marcel (1.0.4) 116 | mini_mime (1.1.5) 117 | mini_portile2 (2.8.9) 118 | minitest (5.25.5) 119 | net-imap (0.5.9) 120 | date 121 | net-protocol 122 | net-pop (0.1.2) 123 | net-protocol 124 | net-protocol (0.2.2) 125 | timeout 126 | net-smtp (0.5.1) 127 | net-protocol 128 | nio4r (2.7.4) 129 | nokogiri (1.18.8) 130 | mini_portile2 (~> 2.8.2) 131 | racc (~> 1.4) 132 | nokogiri (1.18.8-arm64-darwin) 133 | racc (~> 1.4) 134 | nokogiri (1.18.8-x86_64-darwin) 135 | racc (~> 1.4) 136 | nokogiri (1.18.8-x86_64-linux-gnu) 137 | racc (~> 1.4) 138 | pp (0.6.2) 139 | prettyprint 140 | prettyprint (0.2.0) 141 | psych (5.2.6) 142 | date 143 | stringio 144 | racc (1.8.1) 145 | rack (3.2.0) 146 | rack-session (2.1.1) 147 | base64 (>= 0.1.0) 148 | rack (>= 3.0.0) 149 | rack-test (2.2.0) 150 | rack (>= 1.3) 151 | rackup (2.2.1) 152 | rack (>= 3) 153 | rails (8.0.2) 154 | actioncable (= 8.0.2) 155 | actionmailbox (= 8.0.2) 156 | actionmailer (= 8.0.2) 157 | actionpack (= 8.0.2) 158 | actiontext (= 8.0.2) 159 | actionview (= 8.0.2) 160 | activejob (= 8.0.2) 161 | activemodel (= 8.0.2) 162 | activerecord (= 8.0.2) 163 | activestorage (= 8.0.2) 164 | activesupport (= 8.0.2) 165 | bundler (>= 1.15.0) 166 | railties (= 8.0.2) 167 | rails-dom-testing (2.3.0) 168 | activesupport (>= 5.0.0) 169 | minitest 170 | nokogiri (>= 1.6) 171 | rails-html-sanitizer (1.6.2) 172 | loofah (~> 2.21) 173 | nokogiri (>= 1.15.7, != 1.16.7, != 1.16.6, != 1.16.5, != 1.16.4, != 1.16.3, != 1.16.2, != 1.16.1, != 1.16.0.rc1, != 1.16.0) 174 | railties (8.0.2) 175 | actionpack (= 8.0.2) 176 | activesupport (= 8.0.2) 177 | irb (~> 1.13) 178 | rackup (>= 1.0.0) 179 | rake (>= 12.2) 180 | thor (~> 1.0, >= 1.2.2) 181 | zeitwerk (~> 2.6) 182 | rake (13.3.0) 183 | rdoc (6.14.2) 184 | erb 185 | psych (>= 4.0.0) 186 | reline (0.6.1) 187 | io-console (~> 0.5) 188 | securerandom (0.4.1) 189 | stringio (3.1.7) 190 | thor (1.3.2) 191 | timeout (0.4.3) 192 | tzinfo (2.0.6) 193 | concurrent-ruby (~> 1.0) 194 | uri (1.0.3) 195 | useragent (0.16.11) 196 | websocket-driver (0.8.0) 197 | base64 198 | websocket-extensions (>= 0.1.0) 199 | websocket-extensions (0.1.5) 200 | zeitwerk (2.7.3) 201 | 202 | PLATFORMS 203 | arm64-darwin-20 204 | ruby 205 | x86_64-darwin-20 206 | x86_64-linux 207 | 208 | DEPENDENCIES 209 | debug 210 | propshaft! 211 | rails (>= 7.0.1) 212 | rake 213 | 214 | BUNDLED WITH 215 | 2.7.1 216 | -------------------------------------------------------------------------------- /lib/propshaft/manifest.rb: -------------------------------------------------------------------------------- 1 | module Propshaft 2 | # Manages the manifest file that maps logical asset paths to their digested counterparts. 3 | # 4 | # The manifest is used to track assets that have been processed and digested, storing 5 | # their logical paths, digested paths, and optional integrity hashes. 6 | class Manifest 7 | # Represents a single entry in the asset manifest. 8 | # 9 | # Each entry contains information about an asset including its logical path 10 | # (the original path), digested path (the path with content hash), and 11 | # optional integrity hash for security verification. 12 | class ManifestEntry 13 | attr_reader :logical_path, :digested_path, :integrity 14 | 15 | # Creates a new manifest entry. 16 | # 17 | # ==== Parameters 18 | # 19 | # * +logical_path+ - The logical path of the asset 20 | # * +digested_path+ - The digested path of the asset 21 | # * +integrity+ - The integrity hash of the asset (optional) 22 | def initialize(logical_path:, digested_path:, integrity:) # :nodoc: 23 | @logical_path = logical_path 24 | @digested_path = digested_path 25 | @integrity = integrity 26 | end 27 | 28 | # Converts the manifest entry to a hash representation. 29 | # 30 | # Returns a hash containing the +digested_path+ and +integrity+ keys. 31 | def to_h 32 | { digested_path: digested_path, integrity: integrity} 33 | end 34 | end 35 | 36 | class << self 37 | # Creates a new Manifest instance from a manifest file. 38 | # 39 | # Reads and parses a manifest file, supporting both the current format 40 | # (with +digested_path+ and +integrity+ keys) and the legacy format 41 | # (simple string values for backwards compatibility). 42 | # 43 | # ==== Parameters 44 | # 45 | # * +manifest_path+ - The path to the manifest file 46 | # 47 | # ==== Returns 48 | # 49 | # A new manifest instance populated with entries from the file. 50 | def from_path(manifest_path) 51 | manifest = Manifest.new 52 | 53 | serialized_manifest = JSON.parse(manifest_path.read, symbolize_names: false) 54 | 55 | serialized_manifest.each_pair do |key, value| 56 | # Compatibility mode to be able to 57 | # read the old "simple manifest" format 58 | digested_path, integrity = if value.is_a?(String) 59 | [value, nil] 60 | else 61 | [value["digested_path"], value["integrity"]] 62 | end 63 | 64 | entry = ManifestEntry.new( 65 | logical_path: key, digested_path: digested_path, integrity: integrity 66 | ) 67 | 68 | manifest.push(entry) 69 | end 70 | 71 | manifest 72 | end 73 | end 74 | 75 | # Creates a new Manifest instance. 76 | # 77 | # ==== Parameters 78 | # 79 | # * +integrity_hash_algorithm+ - The algorithm to use for generating 80 | # integrity hashes (e.g., 'sha256', 'sha384', 'sha512'). If +nil+, integrity hashes 81 | # will not be generated. 82 | def initialize(integrity_hash_algorithm: nil) 83 | @integrity_hash_algorithm = integrity_hash_algorithm 84 | @entries = {} 85 | end 86 | 87 | # Adds an asset to the manifest. 88 | # 89 | # Creates a manifest entry from the given asset and adds it to the manifest. 90 | # The entry will include the asset's logical path, digested path, and optionally 91 | # an integrity hash if an integrity hash algorithm is configured. 92 | # 93 | # ==== Parameters 94 | # 95 | # * +asset+ - The asset to add to the manifest 96 | # 97 | # ==== Returns 98 | # 99 | # The manifest entry that was added. 100 | def push_asset(asset) 101 | entry = ManifestEntry.new( 102 | logical_path: asset.logical_path.to_s, 103 | digested_path: asset.digested_path.to_s, 104 | integrity: integrity_hash_algorithm && asset.integrity(hash_algorithm: integrity_hash_algorithm) 105 | ) 106 | 107 | push(entry) 108 | end 109 | 110 | # Adds a manifest entry to the manifest. 111 | # 112 | # ==== Parameters 113 | # 114 | # * +entry+ - The manifest entry to add 115 | # 116 | # ==== Returns 117 | # 118 | # The entry that was added. 119 | def push(entry) 120 | @entries[entry.logical_path] = entry 121 | end 122 | alias_method :<<, :push 123 | 124 | # Retrieves a manifest entry by its logical path. 125 | # 126 | # ==== Parameters 127 | # 128 | # * +logical_path+ - The logical path of the asset to retrieve 129 | # 130 | # ==== Returns 131 | # 132 | # The manifest entry, or +nil+ if not found. 133 | def [](logical_path) 134 | @entries[logical_path] 135 | end 136 | 137 | # Removes a manifest entry by its logical path. 138 | # 139 | # ==== Parameters 140 | # 141 | # * +logical_path+ - The logical path of the asset to remove 142 | # 143 | # ==== Returns 144 | # 145 | # The removed manifest entry, or +nil+ if not found. 146 | def delete(logical_path) 147 | @entries.delete(logical_path) 148 | end 149 | 150 | # Converts the manifest to JSON format. 151 | # 152 | # The JSON representation maps logical paths to hash representations of 153 | # manifest entries, containing +digested_path+ and +integrity+ information. 154 | # 155 | # ==== Returns 156 | # 157 | # The JSON representation of the manifest. 158 | def to_json 159 | @entries.transform_values do |manifest_entry| 160 | manifest_entry.to_h 161 | end.to_json 162 | end 163 | 164 | # Transforms the values of all manifest entries using the given block. 165 | # 166 | # This method is useful for applying transformations to all manifest entries 167 | # while preserving the logical path keys. 168 | # 169 | # ==== Parameters 170 | # 171 | # * +block+ - A block that will receive each manifest entry 172 | # 173 | # ==== Returns 174 | # 175 | # A new hash with the same keys but transformed values. 176 | def transform_values(&block) 177 | @entries.transform_values(&block) 178 | end 179 | 180 | private 181 | attr_reader :integrity_hash_algorithm 182 | end 183 | end 184 | -------------------------------------------------------------------------------- /lib/propshaft/helper.rb: -------------------------------------------------------------------------------- 1 | module Propshaft 2 | # Helper module that provides asset path resolution and integrity support for Rails applications. 3 | # 4 | # This module extends Rails' built-in asset helpers with additional functionality: 5 | # - Subresource Integrity (SRI) support for enhanced security 6 | # - Bulk stylesheet inclusion with :all and :app options 7 | # - Asset path resolution with proper error handling 8 | # 9 | # == Subresource Integrity (SRI) Support 10 | # 11 | # SRI helps protect against malicious modifications of assets by ensuring that 12 | # resources fetched from CDNs or other sources haven't been tampered with. 13 | # 14 | # SRI is automatically enabled in secure contexts (HTTPS or local development) 15 | # when the 'integrity' option is set to true: 16 | # 17 | # <%= stylesheet_link_tag "application", integrity: true %> 18 | # <%= javascript_include_tag "application", integrity: true %> 19 | # 20 | # This will generate integrity hashes and include them in the HTML: 21 | # 22 | # 24 | # 26 | # 27 | # == Bulk Stylesheet Inclusion 28 | # 29 | # The stylesheet_link_tag helper supports special symbols for bulk inclusion: 30 | # - :all - includes all CSS files found in the load path 31 | # - :app - includes only CSS files from app/assets/**/*.css 32 | # 33 | # <%= stylesheet_link_tag :all %> # All stylesheets 34 | # <%= stylesheet_link_tag :app %> # Only app stylesheets 35 | module Helper 36 | # Computes the Subresource Integrity (SRI) hash for the given asset path. 37 | # 38 | # This method generates a cryptographic hash of the asset content that can be used 39 | # to verify the integrity of the resource when it's loaded by the browser. 40 | # 41 | # asset_integrity("application.css") 42 | # # => "sha256-xyz789abcdef..." 43 | def asset_integrity(path, options = {}) 44 | path = _path_with_extname(path, options) 45 | Rails.application.assets.resolver.integrity(path) 46 | end 47 | 48 | # Resolves the full path for an asset, raising an error if not found. 49 | def compute_asset_path(path, options = {}) 50 | Rails.application.assets.resolver.resolve(path) || raise(MissingAssetError.new(path)) 51 | end 52 | 53 | # Enhanced +stylesheet_link_tag+ with integrity support and bulk inclusion options. 54 | # 55 | # In addition to the standard Rails functionality, this method supports: 56 | # * Automatic SRI (Subresource Integrity) hash generation in secure contexts 57 | # * Add an option to call +stylesheet_link_tag+ with +:all+ to include every css 58 | # file found on the load path or +:app+ to include css files found in 59 | # Rails.root("app/assets/**/*.css"), which will exclude lib/ and plugins. 60 | # 61 | # ==== Options 62 | # 63 | # * :integrity - Enable SRI hash generation 64 | # 65 | # ==== Examples 66 | # 67 | # stylesheet_link_tag "application", integrity: true 68 | # # => 70 | # 71 | # stylesheet_link_tag :all # All stylesheets in load path 72 | # stylesheet_link_tag :app # Only app/assets stylesheets 73 | def stylesheet_link_tag(*sources) 74 | options = sources.extract_options! 75 | 76 | case sources.first 77 | when :all 78 | sources = all_stylesheets_paths 79 | when :app 80 | sources = app_stylesheets_paths 81 | end 82 | 83 | _build_asset_tags(sources, options, :stylesheet) { |source, opts| super(source, opts) } 84 | end 85 | 86 | # Enhanced +javascript_include_tag+ with automatic SRI (Subresource Integrity) support. 87 | # 88 | # This method extends Rails' built-in +javascript_include_tag+ to automatically 89 | # generate and include integrity hashes when running in secure contexts. 90 | # 91 | # ==== Options 92 | # 93 | # * :integrity - Enable SRI hash generation 94 | # 95 | # ==== Examples 96 | # 97 | # javascript_include_tag "application", integrity: true 98 | # # => 100 | def javascript_include_tag(*sources) 101 | options = sources.extract_options! 102 | 103 | _build_asset_tags(sources, options, :javascript) { |source, opts| super(source, opts) } 104 | end 105 | 106 | # Returns a sorted and unique array of logical paths for all stylesheets in the load path. 107 | def all_stylesheets_paths 108 | Rails.application.assets.load_path.asset_paths_by_type("css") 109 | end 110 | 111 | # Returns a sorted and unique array of logical paths for all stylesheets in app/assets/**/*.css. 112 | def app_stylesheets_paths 113 | Rails.application.assets.load_path.asset_paths_by_glob("#{Rails.root.join("app/assets")}/**/*.css") 114 | end 115 | 116 | private 117 | # Core method that builds asset tags with optional integrity support. 118 | # 119 | # This method handles the common logic for both +stylesheet_link_tag+ and 120 | # +javascript_include_tag+, including SRI hash generation and HTML tag creation. 121 | def _build_asset_tags(sources, options, asset_type) 122 | options = options.stringify_keys 123 | integrity = _compute_integrity?(options) 124 | 125 | sources.map { |source| 126 | opts = integrity ? options.merge!('integrity' => asset_integrity(source, type: asset_type)) : options 127 | yield(source, opts) 128 | }.join("\n").html_safe 129 | end 130 | 131 | # Determines whether integrity hashes should be computed for assets. 132 | # 133 | # Integrity is only computed in secure contexts (HTTPS or local development) 134 | # and when explicitly requested via the +integrity+ option. 135 | def _compute_integrity?(options) 136 | if _secure_subresource_integrity_context? 137 | case options['integrity'] 138 | when nil, false, true 139 | options.delete('integrity') == true 140 | end 141 | else 142 | options.delete 'integrity' 143 | false 144 | end 145 | end 146 | 147 | # Checks if the current context is secure enough for Subresource Integrity. 148 | # 149 | # SRI is only beneficial in secure contexts. Returns true when: 150 | # * The request is made over HTTPS (SSL), OR 151 | # * The request is local (development environment) 152 | def _secure_subresource_integrity_context? 153 | respond_to?(:request) && self.request && (self.request.local? || self.request.ssl?) 154 | end 155 | 156 | # Ensures the asset path includes the appropriate file extension. 157 | def _path_with_extname(path, options) 158 | "#{path}#{compute_asset_extname(path, options)}" 159 | end 160 | end 161 | end 162 | -------------------------------------------------------------------------------- /test/propshaft/compiler/css_asset_urls_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "minitest/mock" 3 | require "propshaft/asset" 4 | require "propshaft/assembly" 5 | require "propshaft/compilers" 6 | 7 | class Propshaft::Compiler::CssAssetUrlsTest < ActiveSupport::TestCase 8 | setup do 9 | @options = ActiveSupport::OrderedOptions.new.tap { |config| 10 | config.paths = [ Pathname.new("#{__dir__}/../../fixtures/assets/vendor") ] 11 | config.output_path = Pathname.new("#{__dir__}/../../fixtures/output") 12 | config.prefix = "/assets" 13 | } 14 | end 15 | 16 | test "basic" do 17 | compiled = compile_asset_with_content(%({ background: url(file.jpg); })) 18 | assert_match(/{ background: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 19 | end 20 | 21 | test "blank spaces around name" do 22 | compiled = compile_asset_with_content(%({ background: url( file.jpg ); })) 23 | assert_match(/{ background: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 24 | end 25 | 26 | test "quotes around name" do 27 | compiled = compile_asset_with_content(%({ background: url("file.jpg"); })) 28 | assert_match(/{ background: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 29 | end 30 | 31 | test "single quotes around name" do 32 | compiled = compile_asset_with_content(%({ background: url('file.jpg'); })) 33 | assert_match(/{ background: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 34 | end 35 | 36 | test "root directory" do 37 | compiled = compile_asset_with_content(%({ background: url('/file.jpg'); })) 38 | assert_match(/{ background: url\("\/assets\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 39 | end 40 | 41 | test "same directory" do 42 | compiled = compile_asset_with_content(%({ background: url('./file.jpg'); })) 43 | assert_match(/{ background: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 44 | end 45 | 46 | test "subdirectory" do 47 | compiled = compile_asset_with_content(%({ background: url('./images/file.jpg'); })) 48 | assert_match(/{ background: url\("\/assets\/foobar\/source\/images\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 49 | end 50 | 51 | test "parent directory" do 52 | compiled = compile_asset_with_content(%({ background: url('../file.jpg'); })) 53 | assert_match(/{ background: url\("\/assets\/foobar\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 54 | end 55 | 56 | test "grandparent directory" do 57 | compiled = compile_asset_with_content(%({ background: url('../../file.jpg'); })) 58 | assert_match(/{ background: url\("\/assets\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 59 | end 60 | 61 | test "sibling directory" do 62 | compiled = compile_asset_with_content(%({ background: url('../sibling/file.jpg'); })) 63 | assert_match(/{ background: url\("\/assets\/foobar\/sibling\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 64 | end 65 | 66 | test "mixed" do 67 | compiled = compile_asset_with_content(%({ mask-image: image(url(file.jpg), skyblue, linear-gradient(rgba(0, 0, 0, 1.0), transparent)); })) 68 | assert_match(/{ mask-image: image\(url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\), skyblue, linear-gradient\(rgba\(0, 0, 0, 1.0\), transparent\)\); }/, compiled) 69 | end 70 | 71 | test "multiple" do 72 | compiled = compile_asset_with_content(%({ content: url(file.svg) url(file.svg); })) 73 | assert_match(/{ content: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.svg"\) url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.svg"\); }/, compiled) 74 | end 75 | 76 | test "url" do 77 | compiled = compile_asset_with_content(%({ background: url('https://rubyonrails.org/images/rails-logo.svg'); })) 78 | assert_match "{ background: url('https://rubyonrails.org/images/rails-logo.svg'); }", compiled 79 | 80 | compiled = compile_asset_with_content(%({ background: url(http-diagram.jpg); })) 81 | assert_match(/{ background: url\("\/assets\/foobar\/source\/http-diagram-[a-z0-9]{8}.jpg"\); }/, compiled) 82 | end 83 | 84 | test "relative protocol url" do 85 | compiled = compile_asset_with_content(%({ background: url('//rubyonrails.org/images/rails-logo.svg'); })) 86 | assert_match "{ background: url('//rubyonrails.org/images/rails-logo.svg'); }", compiled 87 | end 88 | 89 | test "data" do 90 | compiled = compile_asset_with_content(%({ background: url(); })) 91 | assert_match "{ background: url(); }", compiled 92 | 93 | compiled = compile_asset_with_content(%({ background: url(database.jpg); })) 94 | assert_match(/{ background: url\("\/assets\/foobar\/source\/database-[a-z0-9]{8}.jpg"\); }/, compiled) 95 | end 96 | 97 | test "anchor" do 98 | compiled = compile_asset_with_content(%({ background: url(#IDofSVGpath); })) 99 | assert_match "{ background: url(#IDofSVGpath); }", compiled 100 | end 101 | 102 | test "fingerprint" do 103 | compiled = compile_asset_with_content(%({ background: url('/file.jpg?30af91bf14e37666a085fb8a161ff36d'); })) 104 | assert_match(/{ background: url\("\/assets\/file-[a-z0-9]{8}.jpg\?30af91bf14e37666a085fb8a161ff36d"\); }/, compiled) 105 | end 106 | 107 | test "svg anchor" do 108 | compiled = compile_asset_with_content(%({ content: url(file.svg#rails); })) 109 | assert_match(/{ content: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.svg#rails"\); }/, compiled) 110 | end 111 | 112 | test "svg mask encoded anchor" do 113 | compiled = compile_asset_with_content(%({ background: url("data:image/svg+xml;charset=utf-8,%3Csvg mask='url(%23MyMask)'%3E%3C/svg%3E"); })) 114 | assert_match "{ background: url(\"data:image/svg+xml;charset=utf-8,%3Csvg mask='url(%23MyMask)'%3E%3C/svg%3E\"); }", compiled 115 | end 116 | 117 | test "non greedy anchors" do 118 | compiled = compile_asset_with_content(%({ content: url(file.svg#demo) url(file.svg#demo); })) 119 | assert_match(/{ content: url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.svg#demo"\) url\("\/assets\/foobar\/source\/file-[a-z0-9]{8}.svg#demo"\); }/, compiled) 120 | end 121 | 122 | test "missing asset" do 123 | compiled = compile_asset_with_content(%({ background: url("file-not-found.jpg"); })) 124 | assert_match(/{ background: url\("file-not-found.jpg"\); }/, compiled) 125 | end 126 | 127 | test "relative url root" do 128 | @options.relative_url_root = "/url-root" 129 | 130 | compiled = compile_asset_with_content(%({ background: url(file.jpg); })) 131 | assert_match(/{ background: url\("\/url-root\/assets\/foobar\/source\/file-[a-z0-9]{8}.jpg"\); }/, compiled) 132 | end 133 | 134 | private 135 | def compile_asset_with_content(content) 136 | root_path = Pathname.new("#{__dir__}/../../fixtures/assets/vendor") 137 | logical_path = "foobar/source/test.css" 138 | 139 | assembly = Propshaft::Assembly.new(@options) 140 | assembly.compilers.register "text/css", Propshaft::Compiler::CssAssetUrls 141 | 142 | asset = Propshaft::Asset.new(root_path.join(logical_path), logical_path: logical_path, load_path: assembly.load_path) 143 | asset.stub :content, content do 144 | assembly.compilers.compile(asset) 145 | end 146 | end 147 | end 148 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Propshaft 2 | 3 | Propshaft is an asset pipeline library for Rails. It's built for an era where bundling assets to save on HTTP connections is no longer urgent, where JavaScript and CSS are either compiled by dedicated Node.js bundlers or served directly to the browsers, and where increases in bandwidth have made the need for minification less pressing. These factors allow for a dramatically simpler and faster asset pipeline compared to previous options, like [Sprockets](https://github.com/rails/sprockets-rails). 4 | 5 | So that's what Propshaft doesn't do. Here's what it does provide: 6 | 7 | 1. **Configurable load path**: You can register directories from multiple places in your app and gems, and reference assets from all of these paths as though they were one. 8 | 1. **Digest stamping**: All assets in the load path will be copied (or compiled) in a precompilation step for production that also stamps all of them with a digest hash, so you can use long-expiry cache headers for better performance. The digested assets can be referred to through their logical path because the processing leaves a manifest file that provides a way to translate. 9 | 1. **Development server**: There's no need to precompile the assets in development. You can refer to them via the same asset_path helpers and they'll be served by a development server. 10 | 1. **Basic compilers**: Propshaft was explicitly not designed to provide full transpiler capabilities. You can get that better elsewhere. But it does offer a simple input->output compiler setup that by default is used to translate `url(asset)` function calls in CSS to `url(digested-asset)` instead and source mapping comments likewise. 11 | 12 | 13 | ## Installation 14 | 15 | With Rails 8, Propshaft is the default asset pipeline for new applications. With Rails 7, you can start a new application with propshaft using `rails new myapp -a propshaft`. For existing applications, check the [upgrade guide](https://github.com/rails/propshaft/blob/main/UPGRADING.md) which contains step-by-step instructions. 16 | 17 | ## Usage 18 | 19 | Propshaft makes all the assets from all the paths it's been configured with through `config.assets.paths` available for serving and will copy all of them into `public/assets` when precompiling. This is unlike Sprockets, which did not copy over assets that hadn't been explicitly included in one of the bundled assets. 20 | 21 | You can however exempt directories that have been added through the `config.assets.excluded_paths`. This is useful if you're for example using `app/assets/stylesheets` exclusively as a set of inputs to a compiler like Dart Sass for Rails, and you don't want these input files to be part of the load path. (Remember you need to add full paths, like `Rails.root.join("app/assets/stylesheets")`). 22 | 23 | These assets can be referenced through their logical path using the normal helpers like `asset_path`, `image_tag`, `javascript_include_tag`, and all the other asset helper tags. These logical references are automatically converted into digest-aware paths in production when `assets:precompile` has been run (through a JSON mapping file found in `public/assets/.manifest.json`). 24 | 25 | ## Referencing digested assets in CSS and JavaScript 26 | 27 | Propshaft will automatically convert asset references in CSS to use the digested file names. So `background: url("/bg/pattern.svg")` is converted to `background: url("/assets/bg/pattern-2169cbef.svg")` before the stylesheet is served. 28 | 29 | For JavaScript, you'll have to manually trigger this transformation by using the `RAILS_ASSET_URL` pseudo-method. It's used like this: 30 | 31 | ```javascript 32 | export default class extends Controller { 33 | init() { 34 | this.img = RAILS_ASSET_URL("/icons/trash.svg") 35 | } 36 | } 37 | ``` 38 | 39 | That'll turn into: 40 | 41 | ```javascript 42 | export default class extends Controller { 43 | init() { 44 | this.img = "/assets/icons/trash-54g9cbef.svg" 45 | } 46 | } 47 | ``` 48 | 49 | ## Bypassing the digest step 50 | 51 | If you need to put multiple files that refer to each other through Propshaft, like a JavaScript file and its source map, you have to digest these files in advance to retain stable file names. Propshaft looks for the specific pattern of `-[digest].digested.js` as the postfix to any asset file as an indication that the file has already been digested. 52 | 53 | ## Subresource Integrity (SRI) 54 | 55 | Propshaft supports Subresource Integrity (SRI) to help protect against malicious modifications of assets. SRI allows browsers to verify that resources fetched from CDNs or other sources haven't been tampered with by checking cryptographic hashes. 56 | 57 | ### Enabling SRI 58 | 59 | To enable SRI support, configure the hash algorithm in your Rails application: 60 | 61 | ```ruby 62 | config.assets.integrity_hash_algorithm = "sha384" 63 | ``` 64 | 65 | Valid hash algorithms include: 66 | - `"sha256"` - SHA-256 (most common) 67 | - `"sha384"` - SHA-384 (recommended for enhanced security) 68 | - `"sha512"` - SHA-512 (strongest) 69 | 70 | ### Using SRI in your views 71 | 72 | Once configured, you can enable SRI by passing the `integrity: true` option to asset helpers: 73 | 74 | ```erb 75 | <%= stylesheet_link_tag "application", integrity: true %> 76 | <%= javascript_include_tag "application", integrity: true %> 77 | ``` 78 | 79 | This generates HTML with integrity hashes: 80 | 81 | ```html 82 | 84 | 86 | ``` 87 | 88 | **Important**: SRI only works in secure contexts (HTTPS) or during local development. The integrity hashes are automatically omitted when serving over HTTP in production for security reasons. 89 | 90 | ### Bulk stylesheet inclusion with SRI 91 | 92 | Propshaft extends `stylesheet_link_tag` with special symbols for bulk inclusion: 93 | 94 | ```erb 95 | <%= stylesheet_link_tag :all, integrity: true %> 96 | <%= stylesheet_link_tag :app, integrity: true %> 97 | ``` 98 | 99 | ## Improving performance in development 100 | 101 | Before every request Propshaft checks if any asset was updated to decide if a cache sweep is needed. This verification is done using the application's configured file watcher which, by default, is `ActiveSupport::FileUpdateChecker`. 102 | 103 | If you have a lot of assets in your project, you can improve performance by adding the `listen` gem to the development group in your Gemfile, and this line to the `development.rb` environment file: 104 | 105 | ```ruby 106 | config.file_watcher = ActiveSupport::EventedFileUpdateChecker 107 | ``` 108 | 109 | 110 | ## Migrating from Sprockets 111 | 112 | Propshaft does a lot less than Sprockets, by design, so it might well be a fair bit of work to migrate if it's even desirable. This is particularly true if you rely on Sprockets to provide any form of transpiling, like CoffeeScript or Sass, or if you rely on any gems that do. You'll need to either stop transpiling or use a Node-based transpiler, like those in [`jsbundling-rails`](https://github.com/rails/jsbundling-rails) and [`cssbundling-rails`](https://github.com/rails/cssbundling-rails). 113 | 114 | On the other hand, if you're already bundling JavaScript and CSS through a Node-based setup, then Propshaft is going to slot in easily. Since you don't need another tool to bundle or transpile. Just to digest and serve. 115 | 116 | But for greenfield apps using the default import-map approach, Propshaft can also work well, if you're able to deal with vanilla CSS. 117 | 118 | 119 | ## License 120 | 121 | Propshaft is released under the [MIT License](https://opensource.org/licenses/MIT). 122 | -------------------------------------------------------------------------------- /test/propshaft/manifest_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | require "propshaft/manifest" 3 | 4 | class Propshaft::ManifestTest < ActiveSupport::TestCase 5 | test "serializes to the extensible manifest format with integrity hash value" do 6 | manifest = create_manifest("sha384") 7 | parsed_manifest = JSON.parse(manifest.to_json) 8 | 9 | manifest_entry = parsed_manifest["one.txt"] 10 | assert_equal "one-f2e1ec14.txt", manifest_entry["digested_path"] 11 | assert_equal "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe", manifest_entry["integrity"] 12 | 13 | manifest_entry = parsed_manifest["another.css"] 14 | assert_equal "another-c464b1ee.css", manifest_entry["digested_path"] 15 | assert_equal "sha384-jUiHGq2aPNACr4g68crM1I28TitXJKYhEgokcX6W5VYGwufEKQxfLpe4GakM84ex", manifest_entry["integrity"] 16 | end 17 | 18 | test "serializes to the extensible manifest format without integrity hash algorithm" do 19 | manifest = create_manifest 20 | parsed_manifest = JSON.parse(manifest.to_json) 21 | 22 | manifest_entry = parsed_manifest["one.txt"] 23 | assert_equal "one-f2e1ec14.txt", manifest_entry["digested_path"] 24 | assert_nil manifest_entry["integrity"] 25 | 26 | manifest_entry = parsed_manifest["another.css"] 27 | assert_equal "another-c464b1ee.css", manifest_entry["digested_path"] 28 | assert_nil manifest_entry["integrity"] 29 | end 30 | 31 | test "loads from new extensible manifest format" do 32 | manifest_path = Pathname.new("#{__dir__}/../fixtures/new_manifest_format/.manifest.json") 33 | manifest = Propshaft::Manifest.from_path(manifest_path) 34 | 35 | entry = manifest["one.txt"] 36 | assert_not_nil entry 37 | assert_equal "one.txt", entry.logical_path 38 | assert_equal "one-f2e1ec14.txt", entry.digested_path 39 | assert_equal "sha384-LdS8l2QTAF8bD8WPb8QSQv0skTWHhmcnS2XU5LBkVQneGzqIqnDRskQtJvi7ADMe", entry.integrity 40 | end 41 | 42 | test "loads from old simple manifest format" do 43 | manifest_path = Pathname.new("#{__dir__}/../fixtures/output/.manifest.json") 44 | manifest = Propshaft::Manifest.from_path(manifest_path) 45 | 46 | entry = manifest["one.txt"] 47 | assert_not_nil entry 48 | assert_equal "one.txt", entry.logical_path 49 | assert_equal "one-f2e1ec14.txt", entry.digested_path 50 | assert_nil entry.integrity 51 | end 52 | 53 | test "push method adds entry to manifest" do 54 | manifest = Propshaft::Manifest.new 55 | entry = Propshaft::Manifest::ManifestEntry.new( 56 | logical_path: "test.js", 57 | digested_path: "test-abc123.js", 58 | integrity: "sha384-test" 59 | ) 60 | 61 | manifest.push(entry) 62 | retrieved_entry = manifest["test.js"] 63 | 64 | assert_equal entry, retrieved_entry 65 | assert_equal "test.js", retrieved_entry.logical_path 66 | assert_equal "test-abc123.js", retrieved_entry.digested_path 67 | assert_equal "sha384-test", retrieved_entry.integrity 68 | end 69 | 70 | test "<< alias works for push method" do 71 | manifest = Propshaft::Manifest.new 72 | entry = Propshaft::Manifest::ManifestEntry.new( 73 | logical_path: "test.css", 74 | digested_path: "test-def456.css", 75 | integrity: nil 76 | ) 77 | 78 | manifest << entry 79 | retrieved_entry = manifest["test.css"] 80 | 81 | assert_equal entry, retrieved_entry 82 | assert_equal "test.css", retrieved_entry.logical_path 83 | assert_equal "test-def456.css", retrieved_entry.digested_path 84 | assert_nil retrieved_entry.integrity 85 | end 86 | 87 | test "[] accessor returns nil for missing entries" do 88 | manifest = Propshaft::Manifest.new 89 | assert_nil manifest["nonexistent.js"] 90 | end 91 | 92 | test "delete method removes entry and returns it" do 93 | manifest = Propshaft::Manifest.new 94 | entry = Propshaft::Manifest::ManifestEntry.new( 95 | logical_path: "test.js", 96 | digested_path: "test-abc123.js", 97 | integrity: "sha384-test" 98 | ) 99 | 100 | manifest.push(entry) 101 | assert_equal entry, manifest["test.js"] 102 | 103 | deleted_entry = manifest.delete("test.js") 104 | assert_equal entry, deleted_entry 105 | assert_nil manifest["test.js"] 106 | end 107 | 108 | test "delete method returns nil for missing entries" do 109 | manifest = Propshaft::Manifest.new 110 | assert_nil manifest.delete("nonexistent.js") 111 | end 112 | 113 | test "delete method with multiple entries" do 114 | manifest = Propshaft::Manifest.new 115 | 116 | entry1 = Propshaft::Manifest::ManifestEntry.new( 117 | logical_path: "app.js", 118 | digested_path: "app-abc123.js", 119 | integrity: "sha384-test1" 120 | ) 121 | 122 | entry2 = Propshaft::Manifest::ManifestEntry.new( 123 | logical_path: "style.css", 124 | digested_path: "style-def456.css", 125 | integrity: "sha384-test2" 126 | ) 127 | 128 | manifest.push(entry1) 129 | manifest.push(entry2) 130 | 131 | assert_equal entry1, manifest["app.js"] 132 | assert_equal entry2, manifest["style.css"] 133 | 134 | deleted_entry = manifest.delete("app.js") 135 | assert_equal entry1, deleted_entry 136 | assert_nil manifest["app.js"] 137 | assert_equal entry2, manifest["style.css"] 138 | end 139 | 140 | test "push_asset method creates entry from asset" do 141 | manifest = Propshaft::Manifest.new(integrity_hash_algorithm: "sha384") 142 | asset = find_asset("one.txt") 143 | 144 | manifest.push_asset(asset) 145 | entry = manifest["one.txt"] 146 | 147 | assert_not_nil entry 148 | assert_equal "one.txt", entry.logical_path 149 | assert_equal "one-f2e1ec14.txt", entry.digested_path 150 | assert_not_nil entry.integrity 151 | assert entry.integrity.start_with?("sha384-") 152 | end 153 | 154 | test "push_asset without integrity algorithm" do 155 | manifest = Propshaft::Manifest.new 156 | asset = find_asset("one.txt") 157 | 158 | manifest.push_asset(asset) 159 | entry = manifest["one.txt"] 160 | 161 | assert_not_nil entry 162 | assert_equal "one.txt", entry.logical_path 163 | assert_equal "one-f2e1ec14.txt", entry.digested_path 164 | assert_nil entry.integrity 165 | end 166 | 167 | test "transform_values applies block to all entries" do 168 | manifest = Propshaft::Manifest.new 169 | 170 | entry1 = Propshaft::Manifest::ManifestEntry.new( 171 | logical_path: "app.js", 172 | digested_path: "app-abc123.js", 173 | integrity: "sha384-test1" 174 | ) 175 | 176 | entry2 = Propshaft::Manifest::ManifestEntry.new( 177 | logical_path: "style.css", 178 | digested_path: "style-def456.css", 179 | integrity: nil 180 | ) 181 | 182 | manifest.push(entry1) 183 | manifest.push(entry2) 184 | 185 | # Transform to get digested_path 186 | hash = manifest.transform_values { |entry| entry.digested_path } 187 | assert_equal({ "app.js" => "app-abc123.js", "style.css" => "style-def456.css" }, hash) 188 | 189 | # Transform to get integrity 190 | hash = manifest.transform_values { |entry| entry.integrity } 191 | assert_equal({ "app.js" => "sha384-test1", "style.css" => nil }, hash) 192 | 193 | # Transform to get logical_path (for demonstration) 194 | hash = manifest.transform_values { |entry| entry.logical_path } 195 | assert_equal({ "app.js" => "app.js", "style.css" => "style.css" }, hash) 196 | end 197 | 198 | test "transform_values returns empty hash for empty manifest" do 199 | manifest = Propshaft::Manifest.new 200 | assert_equal({}, manifest.transform_values { |entry| entry.digested_path }) 201 | end 202 | 203 | private 204 | def create_manifest(integrity_hash_algorithm = nil) 205 | Propshaft::Manifest.new(integrity_hash_algorithm:).tap do |manifest| 206 | manifest.push_asset(find_asset("one.txt")) 207 | manifest.push_asset(find_asset("another.css")) 208 | end 209 | end 210 | 211 | def find_asset(logical_path) 212 | root_path = Pathname.new("#{__dir__}/../fixtures/assets/first_path") 213 | path = root_path.join(logical_path) 214 | 215 | assembly = Propshaft::Assembly.new(ActiveSupport::OrderedOptions.new.tap { |config| 216 | config.paths = [ root_path ] 217 | config.compilers = [[ "text/css", Propshaft::Compiler::CssAssetUrls ]] 218 | }) 219 | 220 | Propshaft::Asset.new(path, logical_path: logical_path, load_path: assembly.load_path) 221 | end 222 | end 223 | -------------------------------------------------------------------------------- /test/propshaft/helper_test.rb: -------------------------------------------------------------------------------- 1 | require "test_helper" 2 | 3 | class Propshaft::HelperTest < ActionView::TestCase 4 | test "asset_integrity returns SHA256 hash for existing asset" do 5 | integrity = asset_integrity("hello_world.js") 6 | assert_equal "sha384-BIr0kyMRq2sfytK/T0XlGjfav9ZZrWkSBC2yHVunCchnkpP83H28/UtHw+m9iNHO", integrity 7 | end 8 | 9 | test "asset_integrity with asset type option" do 10 | integrity = asset_integrity("hello_world", type: :stylesheet) 11 | assert_equal "sha384-ZSAt6UaTZ1OYvSB1fr2WXE8izMW4qnd17BZ1zaZ3TpAdIw3VEUmyupHd/k/cMCqM", integrity 12 | end 13 | 14 | test "compute_asset_path returns resolved path for existing asset" do 15 | path = compute_asset_path("hello_world.js") 16 | assert_equal "/assets/hello_world-888761f8.js", path 17 | end 18 | 19 | test "compute_asset_path raises MissingAssetError for nonexistent asset" do 20 | error = assert_raises(Propshaft::MissingAssetError) do 21 | compute_asset_path("nonexistent.txt") 22 | end 23 | assert_equal "The asset 'nonexistent.txt' was not found in the load path.", error.message 24 | end 25 | 26 | test "stylesheet_link_tag with integrity in secure context" do 27 | request.headers["HTTPS"] = "on" 28 | 29 | result = stylesheet_link_tag("hello_world", integrity: true) 30 | 31 | assert_dom_equal(<<~HTML, result) 32 | 37 | HTML 38 | end 39 | 40 | test "stylesheet_link_tag with integrity in local context" do 41 | request.remote_addr = "127.0.0.1" 42 | 43 | result = stylesheet_link_tag("hello_world", integrity: true) 44 | 45 | assert_dom_equal(<<~HTML, result) 46 | 51 | HTML 52 | end 53 | 54 | test "stylesheet_link_tag without integrity in insecure context" do 55 | result = stylesheet_link_tag("hello_world", integrity: true) 56 | 57 | assert_dom_equal(<<~HTML, result) 58 | 62 | HTML 63 | end 64 | 65 | test "stylesheet_link_tag without request context" do 66 | request.remote_addr = "127.0.0.1" 67 | @request = nil 68 | 69 | result = stylesheet_link_tag("hello_world", integrity: true) 70 | 71 | assert_dom_equal(<<~HTML, result) 72 | 76 | HTML 77 | end 78 | 79 | test "stylesheet_link_tag with multiple sources and integrity" do 80 | request.headers["HTTPS"] = "on" 81 | 82 | result = stylesheet_link_tag("hello_world", "goodbye", integrity: true) 83 | 84 | assert_dom_equal(<<~HTML, result) 85 | 90 | 95 | HTML 96 | end 97 | 98 | test "stylesheet_link_tag with :all option" do 99 | result = stylesheet_link_tag(:all) 100 | 101 | assert_dom_equal(<<~HTML, result) 102 | 103 | 104 | 105 | HTML 106 | end 107 | 108 | test "stylesheet_link_tag with :app option" do 109 | result = stylesheet_link_tag(:app) 110 | 111 | assert_dom_equal(<<~HTML, result) 112 | 113 | 114 | HTML 115 | end 116 | 117 | test "stylesheet_link_tag with additional options" do 118 | result = stylesheet_link_tag( 119 | "hello_world", 120 | media: "print", 121 | data: { turbo_track: "reload" } 122 | ) 123 | 124 | assert_dom_equal(<<~HTML, result) 125 | 131 | HTML 132 | end 133 | 134 | test "stylesheet_link_tag should extract options from the sources" do 135 | result = stylesheet_link_tag( 136 | "hello_world", 137 | { 138 | media: "print", 139 | data: { turbo_track: "reload" } 140 | } 141 | ) 142 | 143 | assert_dom_equal(<<~HTML, result) 144 | 150 | HTML 151 | end 152 | 153 | test "javascript_include_tag with integrity in secure context" do 154 | request.headers["HTTPS"] = "on" 155 | 156 | result = javascript_include_tag("hello_world", integrity: true) 157 | 158 | assert_dom_equal(<<~HTML, result) 159 | 163 | HTML 164 | end 165 | 166 | test "javascript_include_tag with integrity in local context" do 167 | request.remote_addr = "127.0.0.1" 168 | 169 | result = javascript_include_tag("hello_world", integrity: true) 170 | 171 | assert_dom_equal(<<~HTML, result) 172 | 176 | HTML 177 | end 178 | 179 | test "javascript_include_tag without integrity in insecure context" do 180 | result = javascript_include_tag("hello_world", integrity: true) 181 | 182 | assert_dom_equal(<<~HTML, result) 183 | 186 | HTML 187 | end 188 | 189 | test "javascript_include_tag with multiple sources and integrity" do 190 | request.headers["HTTPS"] = "on" 191 | 192 | result = javascript_include_tag("hello_world", "hello_world", integrity: true) 193 | 194 | assert_dom_equal(<<~HTML, result) 195 | 199 | 203 | HTML 204 | end 205 | 206 | test "javascript_include_tag with additional options" do 207 | result = javascript_include_tag( 208 | "hello_world", 209 | defer: true, 210 | data: { turbo_track: "reload" } 211 | ) 212 | 213 | assert_dom_equal(<<~HTML, result) 214 | 219 | HTML 220 | end 221 | 222 | test "javascript_include_tag should extract options from the sources" do 223 | result = javascript_include_tag( 224 | "hello_world", 225 | { 226 | defer: true, 227 | data: { turbo_track: "reload" } 228 | } 229 | ) 230 | 231 | assert_dom_equal(<<~HTML, result) 232 | 237 | HTML 238 | end 239 | 240 | test "all_stylesheets_paths returns array of CSS asset paths" do 241 | paths = all_stylesheets_paths 242 | 243 | assert_equal( 244 | [ 245 | "goodbye.css", 246 | "hello_world.css", 247 | "library.css" 248 | ], 249 | paths 250 | ) 251 | end 252 | 253 | test "app_stylesheets_paths returns array of app CSS asset paths" do 254 | paths = app_stylesheets_paths 255 | 256 | assert_equal( 257 | [ 258 | "goodbye.css", 259 | "hello_world.css" 260 | ], 261 | paths 262 | ) 263 | end 264 | 265 | test "asset_integrity handles file extensions correctly" do 266 | integrity1 = asset_integrity("hello_world.css") 267 | 268 | integrity2 = asset_integrity("hello_world", type: :stylesheet) 269 | 270 | assert_equal integrity1, integrity2 271 | end 272 | 273 | test "integrity option false explicitly disables integrity" do 274 | request.headers["HTTPS"] = "on" 275 | 276 | result = stylesheet_link_tag("hello_world", integrity: false) 277 | 278 | assert_dom_equal(<<~HTML, result) 279 | 283 | HTML 284 | end 285 | 286 | test "integrity option nil does not enable integrity" do 287 | request.headers["HTTPS"] = "on" 288 | 289 | result = stylesheet_link_tag("hello_world", integrity: nil) 290 | 291 | assert_dom_equal(<<~HTML, result) 292 | 296 | HTML 297 | end 298 | end 299 | -------------------------------------------------------------------------------- /UPGRADING.md: -------------------------------------------------------------------------------- 1 | # Upgrading from Sprockets to Propshaft 2 | 3 | Propshaft has a smaller scope than Sprockets, therefore migrating to it will also require you to adopt the [jsbundling-rails](https://github.com/rails/jsbundling-rails) and [cssbundling-rails](https://github.com/rails/cssbundling-rails) gems. This guide will assume your project follows Rails 6.1 conventions of using [webpacker](https://github.com/rails/webpacker) to bundle javascript, [sass-rails](https://github.com/rails/sass-rails) to bundle css and [sprockets](https://github.com/rails/sprockets) to digest assets. Finally, you will also need [npx](https://docs.npmjs.com/cli/v7/commands/npx) version 7.1.0 or later installed. 4 | 5 | Propshaft depends on Rails 7, so you will need to upgrade to Rails 7+ before starting the migration. 6 | 7 | ## 1. Migrate from Webpacker to jsbundling-rails 8 | 9 | Start by following these steps: 10 | 11 | 1. Replace `webpacker` with `jsbundling-rails` in your Gemfile; 12 | 2. Run `./bin/bundle install`; 13 | 3. Run `./bin/rails javascript:install:webpack`; 14 | 4. Remove the file `config/initializers/assets.rb`; 15 | 5. Remove the file `bin/webpack`; 16 | 6. Remove the file `bin/webpack-dev-server`; 17 | 7. Remove the folder `config/webpack` (note: any custom configuration should be migrated to the new `webpack.config.js` file); 18 | 8. Remove the file `config/webpacker.yml`; 19 | 9. Replace all instances of `javascript_pack_tag` with `javascript_include_tag` and add `defer: true` to them. 20 | 21 | After you are done you will notice that the install step added various files to your project and updated some of the existing ones. 22 | 23 | **The new 'bin/dev' and 'Procfile.dev' files** 24 | 25 | The `./bin/dev` file is a shell script that uses [foreman](https://github.com/ddollar/foreman) and `Procfile.dev` to start two processes in a single terminal: `rails s` and `yarn build`. The latter replaces `webpack-dev-server` for bundling and watching for changes in javascript files. 26 | 27 | **The 'build' attribute added to package.json** 28 | 29 | This is the command that `yarn build` will use to bundle javascript files. 30 | 31 | **The new 'webpack.config.js' file** 32 | 33 | In `webpacker` this file was hidden inside the gem, but now you can edit it directly. If you had custom configuration in `config/webpack` you can move them to here. Projects with multiple entrypoints will need to adjust the `entry` attribute: 34 | 35 | ```js 36 | module.exports = { 37 | entry: { 38 | application: "./app/javascript/application.js", 39 | admin: "./app/javascript/admin.js" 40 | } 41 | } 42 | ``` 43 | 44 | **The 'link_tree' directive added to 'app/assets/manifest.js'** 45 | 46 | This tells Sprockets to include the files in `app/assets/builds` during `assets:precompile`. This is the folder where `yarn build` will place the bundled files, so make sure you commit it to the repository and don't delete it when cleaning assets. 47 | 48 | **What about babel?** 49 | 50 | If you would like to continue using babel for transpiling, you will need to configure it manually. First, open `webpack.config.js` and add this: 51 | 52 | ```js 53 | module.exports = { 54 | module: { 55 | rules: [ 56 | { 57 | test: /\.(js)$/, 58 | exclude: /node_modules/, 59 | use: ['babel-loader'] 60 | } 61 | ] 62 | } 63 | } 64 | ``` 65 | 66 | Then open `package.json` and add this: 67 | ```json 68 | "babel": { 69 | "presets": [ 70 | "./webpack.babel.js" 71 | ] 72 | } 73 | ``` 74 | 75 | Finally, download [webpackers babel preset](https://github.com/rails/webpacker/blob/master/package/babel/preset.js) file and place it in the same directory as `package.json` with the name `webpack.babel.js`. 76 | 77 | **Module resolution** 78 | 79 | Webpacker included the `source_path` (default: `app/javascript/`) into module resolution, so a statement like `import 'channels'` imported `app/javascript/channels/`. After migrating to `jsbundling-rails` this is no longer the case. You will need to update your `webpack.config.js` to include the following if you wish to maintain that behavior: 80 | 81 | ```javascript 82 | module.exports = { 83 | // ... 84 | resolve: { 85 | modules: ["app/javascript", "node_modules"], 86 | }, 87 | //... 88 | } 89 | ``` 90 | 91 | Alternatively, you can change modules to use relative imports, for example: 92 | ```diff 93 | - import 'channels' 94 | + import './channels' 95 | ``` 96 | 97 | ### Extracting Sass/SCSS from JavaScript 98 | 99 | In webpacker it is possible to extract Sass/SCSS from JavaScript by enabling `extract_css` in `webpacker.yml`. This allows for including those source files in JavaScript, e.g. `import '../scss/application.scss` 100 | 101 | If you wish to keep this functionality follow these steps: 102 | 103 | 1. Run `yarn add mini-css-extract-plugin sass sass-loader css-loader`; 104 | 2. Update your `webpack.config.js` to require `mini-css-extract-plugin` and configure the loaders (see example below). 105 | 106 | Example `webpack.config.js`: 107 | 108 | ```javascript 109 | const path = require("path") 110 | const webpack = require("webpack") 111 | const MiniCssExtractPlugin = require("mini-css-extract-plugin") 112 | 113 | module.exports = { 114 | mode: "production", 115 | devtool: "source-map", 116 | entry: { 117 | application: "./app/javascript/application.js" 118 | }, 119 | resolve: { 120 | modules: ["app/javascript", "node_modules"], 121 | }, 122 | output: { 123 | filename: "[name].js", 124 | sourceMapFilename: "[file].map", 125 | path: path.resolve(__dirname, "app/assets/builds"), 126 | }, 127 | plugins: [ 128 | new MiniCssExtractPlugin(), 129 | new webpack.optimize.LimitChunkCountPlugin({ 130 | maxChunks: 1 131 | }) 132 | ], 133 | module: { 134 | rules: [ 135 | { 136 | test: /\.s[ac]ss$/i, 137 | use: [MiniCssExtractPlugin.loader, "css-loader", "sass-loader"], 138 | }, 139 | ], 140 | }, 141 | } 142 | ``` 143 | 144 | ## 2. Migrate from sass-rails to cssbundling-rails 145 | 146 | Note: if your application used Webpacker's `extract_css` to build your CSS and did not require `sass-rails`, you can skip this section. 147 | 148 | Start by following these steps: 149 | 150 | 1. Add `cssbundling-rails` to your Gemfile; 151 | 2. Run `./bin/bundle install`; 152 | 3. Run `./bin/rails css:install:sass`. 153 | 154 | After you are done you will notice that the install step updated some files. 155 | 156 | **The new process in 'Procfile.dev'** 157 | 158 | Just like the javascript process, this one will bundle and watch for changes in css files. 159 | 160 | **The 'build:css' attribute added to package.json** 161 | 162 | This is the command `yarn build` will use to bundle css files. 163 | 164 | **The 'link_tree' directive removed from 'app/assets/manifest.js'** 165 | 166 | Now that the CSS files will be placed into `app/assets/build`, Sprockets no longer needs to worry about the `app/assets/stylesheets` folder. If you have any other `link_tree` for css files, remove them too. 167 | 168 | ### Configuring multiple entrypoints 169 | 170 | Sprockets will only compile files in the root directories listed in `manifest.js`, but the sass package that `yarn build` uses will also check subfolders, which might cause compilation errors if your scss files are using features like `@import` and variables. This means that if you have multiple entry points in your app, you have some extra work ahead of you. 171 | 172 | Let's assume you have the following structure in your `app/asset/stylesheets` folder: 173 | 174 | ``` 175 | stylesheets/admin.scss 176 | stylesheets/admin/source_1.scss 177 | stylesheets/admin/source_2.scss 178 | stylesheets/application.scss 179 | stylesheets/application/source_1.scss 180 | stylesheets/application/source_2.scss 181 | ``` 182 | 183 | Start by your separating your entrypoints from your other files, and adjusting all `@import` for the new structure: 184 | 185 | ``` 186 | stylesheets/entrypoints/admin.scss 187 | stylesheets/entrypoints/application.scss 188 | stylesheets/sources/admin/source_1.scss 189 | stylesheets/sources/admin/source_2.scss 190 | stylesheets/sources/application/source_1.scss 191 | stylesheets/sources/application/source_2.scss 192 | ``` 193 | 194 | Then adjust the `build` attribute in `package.json`: 195 | ``` 196 | "build:css": "sass ./app/assets/stylesheets/entrypoints:./app/assets/builds --no-source-map --load-path=node_modules" 197 | ``` 198 | 199 | ### Deprecation warnings 200 | 201 | Sass might raise deprecation warnings depending on what features you are using (such as division), but the messages will explain how to fix them. If you are not sure, see more details in the [official documentation](https://sass-lang.com/documentation/breaking-changes). 202 | 203 | ## 3. Migrate from Sprockets to Propshaft 204 | 205 | Start by following these steps: 206 | 207 | 1. Remove `sprockets`, `sprockets-rails`, and `sass-rails` from the Gemfile and add `propshaft`; 208 | 2. Run `./bin/bundle install`; 209 | 3. Check your `Gemfile.lock`, repeat steps 1 and 2 for gems that list `sprockets` or `sprockets-rails` as a dependency; 210 | 4. Open `config/application.rb` and remove `config.assets.paths << Rails.root.join('app','assets')`; 211 | 5. Remove `app/assets/config/manifest.js`. 212 | 6. Replace all asset_helpers (`image_url`, `font_url`) in css files with standard `urls`. 213 | 7. If you are importing only the frameworks you need (instead of `rails/all`), remove `require "sprockets/railtie"`; 214 | 215 | ### Asset paths 216 | 217 | Propshaft will automatically include in its search paths the folders `vendor/assets`, `lib/assets` and `app/assets` of your project and of all the gems in your Gemfile. You can see all included files by using the `reveal` rake task: 218 | ``` 219 | rake assets:reveal 220 | ``` 221 | 222 | ### Asset helpers 223 | 224 | Propshaft does not rely on asset_helpers (`asset_path`, `asset_url`, `image_url`, etc.) like Sprockets did. Instead, it will search for every `url` function in your css files, and adjust them to include the digest of the assets they reference. 225 | 226 | Go through your css files, and make the necessary adjustments: 227 | ```diff 228 | - background: image_url('hero.jpg'); 229 | + background: url('/hero.jpg'); 230 | ``` 231 | 232 | Notice that Propshaft's version starts with an `/` and Sprockets' version does not? That's because the latter uses **absolute paths**, and the former uses **relative paths**. To better illustrate that difference, let's assume you have the following structure: 233 | 234 | ``` 235 | assets/stylesheets/theme/main.scss 236 | assets/images/hero.jpg 237 | ``` 238 | 239 | In Sprockets, `main.scss` can reference `hero.jpg` like this: 240 | ```css 241 | background: image_url('hero.jpg') 242 | ``` 243 | 244 | Using the same path with `url` in Propshaft will cause it to raise an error, saying it cannot locate `theme/hero.jpg`. That's because Propshaft assumes all paths are relative to the path of the file it's processing. Since it was processing a css file inside the `theme` folder, it will also look for `hero.jpg` in the same folder. 245 | 246 | By adding a `/` at the start of the path we are telling Propshaft to consider this path as an absolute path. While this change in behavior increases the work a bit when upgrading, it makes **external libraries like FontAwesome and Bootstrap themes work out-of-the-box**. 247 | 248 | ### Asset content 249 | 250 | It's a common pattern in apps to inline small SVG files and low resolution versions of images that need to be displayed as quickly as possible. In Propshaft, the same line of code works for all environments: 251 | ```ruby 252 | Rails.application.assets.load_path.find('logo.svg').content 253 | ``` 254 | 255 | As Rails escapes html tags in views by default, in order to output a rendered svg you will need to specify rails not to escape the string using [html_safe](https://api.rubyonrails.org/classes/String.html#method-i-html_safe) or [raw](https://api.rubyonrails.org/classes/ActionView/Helpers/OutputSafetyHelper.html#method-i-raw). 256 | ```ruby 257 | Rails.application.assets.load_path.find('logo.svg').content.html_safe 258 | raw Rails.application.assets.load_path.find('logo.svg').content 259 | ``` 260 | 261 | ### Precompilation in development 262 | 263 | Propshaft uses a dynamic assets resolver in development mode. However, when you run `assets:precompile` locally Propshaft will then switch to a static assets resolver. Therefore, changes to assets will not be observed anymore and you will have to precompile the assets each time changes are made. This is different to Sprockets. 264 | 265 | If you wish to have dynamic assets resolver enabled again, you need to clean your target folder (usually `public/assets`) and propshaft will start serving dynamic content from source. One way to do this is to run `rails assets:clobber`. 266 | 267 | Another way to watch changes in your CSS & JS assets is by running `bin/dev` command instead of `rails server` that not only runs the server but also keeps looking for any changes in the assets and once it detects any changes, it compiles them while the server is running. This is possible because of the `Procfile.dev`. 268 | --------------------------------------------------------------------------------