2013-12-18 20:33:17 -05:00
|
|
|
task 'assets:precompile:before' do
|
2013-11-03 17:58:34 -05:00
|
|
|
|
2014-05-15 01:52:09 -04:00
|
|
|
require 'uglifier'
|
2017-11-24 10:40:49 -05:00
|
|
|
require 'open3'
|
2014-05-15 01:52:09 -04:00
|
|
|
|
2013-12-18 20:33:17 -05:00
|
|
|
unless %w{profile production}.include? Rails.env
|
|
|
|
raise "rake assets:precompile should only be run in RAILS_ENV=production, you are risking unminified assets"
|
2013-11-03 17:58:34 -05:00
|
|
|
end
|
2013-12-18 20:33:17 -05:00
|
|
|
|
2014-02-06 19:36:44 -05:00
|
|
|
# Ensure we ALWAYS do a clean build
|
|
|
|
# We use many .erbs that get out of date quickly, especially with plugins
|
2014-02-07 05:43:05 -05:00
|
|
|
puts "Purging temp files"
|
2014-02-06 19:36:44 -05:00
|
|
|
`rm -fr #{Rails.root}/tmp/cache`
|
|
|
|
|
2017-07-05 04:36:06 -04:00
|
|
|
# Ensure we clear emoji cache before pretty-text/emoji/data.js.es6.erb
|
|
|
|
# is recompiled
|
|
|
|
Emoji.clear_cache
|
|
|
|
|
2019-05-08 01:54:21 -04:00
|
|
|
if Rails.configuration.assets.js_compressor == :uglifier && !`which uglifyjs`.empty? && !ENV['SKIP_NODE_UGLIFY']
|
2014-12-17 12:14:12 -05:00
|
|
|
$node_uglify = true
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
|
2017-03-09 16:44:50 -05:00
|
|
|
unless ENV['USE_SPROCKETS_UGLIFY']
|
|
|
|
$bypass_sprockets_uglify = true
|
|
|
|
end
|
|
|
|
|
2014-12-12 02:53:26 -05:00
|
|
|
puts "Bundling assets"
|
|
|
|
|
2013-12-18 20:33:17 -05:00
|
|
|
# in the past we applied a patch that removed asset postfixes, but it is terrible practice
|
|
|
|
# leaving very complicated build issues
|
|
|
|
# https://github.com/rails/sprockets-rails/issues/49
|
2014-02-06 00:55:53 -05:00
|
|
|
|
|
|
|
require 'sprockets'
|
|
|
|
require 'digest/sha1'
|
|
|
|
|
2015-07-21 14:52:54 -04:00
|
|
|
# Needed for proper source maps with a CDN
|
|
|
|
load "#{Rails.root}/lib/global_path.rb"
|
|
|
|
include GlobalPath
|
2014-03-04 00:54:58 -05:00
|
|
|
|
2017-03-09 16:44:50 -05:00
|
|
|
if $bypass_sprockets_uglify
|
2016-04-19 02:28:10 -04:00
|
|
|
Rails.configuration.assets.js_compressor = nil
|
2016-08-23 00:04:53 -04:00
|
|
|
Rails.configuration.assets.gzip = false
|
2014-02-06 00:55:53 -05:00
|
|
|
end
|
|
|
|
|
2013-11-03 17:58:34 -05:00
|
|
|
end
|
2013-12-18 20:33:17 -05:00
|
|
|
|
2014-05-02 17:46:03 -04:00
|
|
|
task 'assets:precompile:css' => 'environment' do
|
2015-10-12 19:48:21 -04:00
|
|
|
if ENV["DONT_PRECOMPILE_CSS"] == "1"
|
|
|
|
STDERR.puts "Skipping CSS precompilation, ensure CSS lives in a shared directory across hosts"
|
|
|
|
else
|
|
|
|
STDERR.puts "Start compiling CSS: #{Time.zone.now}"
|
|
|
|
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection do |db|
|
|
|
|
# Heroku precompiles assets before db migration, so tables may not exist.
|
|
|
|
# css will get precompiled during first request instead in that case.
|
|
|
|
|
2017-04-14 10:33:35 -04:00
|
|
|
if ActiveRecord::Base.connection.table_exists?(Theme.table_name)
|
2017-04-12 10:52:52 -04:00
|
|
|
STDERR.puts "Compiling css for #{db} #{Time.zone.now}"
|
2017-05-05 13:50:48 -04:00
|
|
|
begin
|
|
|
|
Stylesheet::Manager.precompile_css
|
2019-03-07 09:03:05 -05:00
|
|
|
rescue PG::UndefinedColumn, ActiveModel::MissingAttributeError => e
|
2019-02-07 09:27:42 -05:00
|
|
|
STDERR.puts "#{e.class} #{e.message}: #{e.backtrace.join("\n")}"
|
2017-05-05 13:50:48 -04:00
|
|
|
STDERR.puts "Skipping precompilation of CSS cause schema is old, you are precompiling prior to running migrations."
|
|
|
|
end
|
2014-06-12 14:41:37 -04:00
|
|
|
end
|
2014-05-02 17:46:03 -04:00
|
|
|
end
|
2015-10-12 02:31:37 -04:00
|
|
|
|
2015-10-12 19:48:21 -04:00
|
|
|
STDERR.puts "Done compiling CSS: #{Time.zone.now}"
|
|
|
|
end
|
2014-05-02 17:46:03 -04:00
|
|
|
end
|
|
|
|
|
2014-12-12 02:53:26 -05:00
|
|
|
def assets_path
|
|
|
|
"#{Rails.root}/public/assets"
|
|
|
|
end
|
|
|
|
|
2017-07-27 21:20:09 -04:00
|
|
|
def compress_node(from, to)
|
2014-12-12 02:53:26 -05:00
|
|
|
to_path = "#{assets_path}/#{to}"
|
2016-02-04 21:05:47 -05:00
|
|
|
assets = cdn_relative_path("/assets")
|
2017-07-27 21:20:09 -04:00
|
|
|
source_map_root = assets + ((d = File.dirname(from)) == "." ? "" : "/#{d}")
|
2015-07-21 14:52:54 -04:00
|
|
|
source_map_url = cdn_path "/assets/#{to}.map"
|
2014-12-12 02:53:26 -05:00
|
|
|
|
2019-05-08 01:54:21 -04:00
|
|
|
cmd = "uglifyjs '#{assets_path}/#{from}' -p relative -c -m -o '#{to_path}' --source-map-root '#{source_map_root}' --source-map '#{assets_path}/#{to}.map' --source-map-url '#{source_map_url}'"
|
2014-12-17 12:14:12 -05:00
|
|
|
|
|
|
|
STDERR.puts cmd
|
2015-09-21 21:28:15 -04:00
|
|
|
result = `#{cmd} 2>&1`
|
|
|
|
unless $?.success?
|
|
|
|
STDERR.puts result
|
|
|
|
exit 1
|
|
|
|
end
|
2014-12-17 12:14:12 -05:00
|
|
|
|
2015-09-21 21:28:15 -04:00
|
|
|
result
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
|
2017-07-27 21:20:09 -04:00
|
|
|
def compress_ruby(from, to)
|
2014-12-12 02:53:26 -05:00
|
|
|
data = File.read("#{assets_path}/#{from}")
|
|
|
|
|
|
|
|
uglified, map = Uglifier.new(comments: :none,
|
2017-03-09 16:44:50 -05:00
|
|
|
source_map: {
|
|
|
|
filename: File.basename(from),
|
|
|
|
output_filename: File.basename(to)
|
|
|
|
}
|
2014-12-12 02:53:26 -05:00
|
|
|
)
|
2017-07-27 21:20:09 -04:00
|
|
|
.compile_with_map(data)
|
2014-12-12 02:53:26 -05:00
|
|
|
dest = "#{assets_path}/#{to}"
|
|
|
|
|
2015-07-21 14:52:54 -04:00
|
|
|
File.write(dest, uglified << "\n//# sourceMappingURL=#{cdn_path "/assets/#{to}.map"}")
|
2014-12-12 02:53:26 -05:00
|
|
|
File.write(dest + ".map", map)
|
2017-03-10 11:35:54 -05:00
|
|
|
|
|
|
|
GC.start
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def gzip(path)
|
2017-11-24 09:52:08 -05:00
|
|
|
STDERR.puts "gzip -f -c -9 #{path} > #{path}.gz"
|
2018-07-03 19:42:21 -04:00
|
|
|
STDERR.puts `gzip -f -c -9 #{path} > #{path}.gz`.strip
|
2017-11-24 09:52:08 -05:00
|
|
|
raise "gzip compression failed: exit code #{$?.exitstatus}" if $?.exitstatus != 0
|
2016-06-07 02:55:57 -04:00
|
|
|
end
|
|
|
|
|
2019-04-10 22:36:02 -04:00
|
|
|
# different brotli versions use different parameters
|
2019-04-11 02:53:13 -04:00
|
|
|
def brotli_command(path)
|
|
|
|
"brotli -f --quality=11 #{path} --output=#{path}.br"
|
2017-11-24 10:40:49 -05:00
|
|
|
end
|
|
|
|
|
2016-06-07 02:55:57 -04:00
|
|
|
def brotli(path)
|
2019-04-10 22:36:02 -04:00
|
|
|
STDERR.puts brotli_command(path)
|
|
|
|
STDERR.puts `#{brotli_command(path)}`
|
|
|
|
raise "brotli compression failed: exit code #{$?.exitstatus}" if $?.exitstatus != 0
|
|
|
|
STDERR.puts `chmod +r #{path}.br`.strip
|
|
|
|
raise "chmod failed: exit code #{$?.exitstatus}" if $?.exitstatus != 0
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
|
2017-07-27 21:20:09 -04:00
|
|
|
def compress(from, to)
|
2017-03-09 16:44:50 -05:00
|
|
|
if $node_uglify
|
2017-07-27 21:20:09 -04:00
|
|
|
compress_node(from, to)
|
2014-12-12 02:53:26 -05:00
|
|
|
else
|
2017-07-27 21:20:09 -04:00
|
|
|
compress_ruby(from, to)
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-04-19 21:34:30 -04:00
|
|
|
def concurrent?
|
2018-10-03 04:43:18 -04:00
|
|
|
executor = Concurrent::FixedThreadPool.new(Concurrent.processor_count)
|
|
|
|
|
2016-06-07 03:03:05 -04:00
|
|
|
if ENV["SPROCKETS_CONCURRENT"] == "1"
|
2016-04-19 21:34:30 -04:00
|
|
|
concurrent_compressors = []
|
2018-10-03 04:43:18 -04:00
|
|
|
yield(Proc.new { |&block| concurrent_compressors << Concurrent::Future.execute(executor: executor) { block.call } })
|
2016-04-19 21:34:30 -04:00
|
|
|
concurrent_compressors.each(&:wait!)
|
|
|
|
else
|
|
|
|
yield(Proc.new { |&block| block.call })
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-05-02 17:46:03 -04:00
|
|
|
task 'assets:precompile' => 'assets:precompile:before' do
|
2019-04-10 05:37:29 -04:00
|
|
|
if refresh_days = SiteSetting.refresh_maxmind_db_during_precompile_days
|
2019-04-10 06:44:58 -04:00
|
|
|
mmdb_path = DiscourseIpInfo.mmdb_path('GeoLite2-City')
|
|
|
|
mmdb_time = File.exist?(mmdb_path) && File.mtime(mmdb_path)
|
|
|
|
if !mmdb_time || mmdb_time < refresh_days.days.ago
|
2019-04-10 05:37:29 -04:00
|
|
|
puts "Downloading MaxMindDB..."
|
|
|
|
mmdb_thread = Thread.new do
|
|
|
|
DiscourseIpInfo.mmdb_download('GeoLite2-City')
|
|
|
|
DiscourseIpInfo.mmdb_download('GeoLite2-ASN')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2014-12-12 02:53:26 -05:00
|
|
|
|
2017-03-09 16:44:50 -05:00
|
|
|
if $bypass_sprockets_uglify
|
2014-12-12 02:53:26 -05:00
|
|
|
puts "Compressing Javascript and Generating Source Maps"
|
|
|
|
manifest = Sprockets::Manifest.new(assets_path)
|
2015-02-20 15:48:45 -05:00
|
|
|
|
2016-04-19 21:34:30 -04:00
|
|
|
concurrent? do |proc|
|
2019-05-08 01:54:21 -04:00
|
|
|
to_skip = Rails.configuration.assets.skip_minification || []
|
2016-04-19 21:34:30 -04:00
|
|
|
manifest.files
|
2017-07-27 21:20:09 -04:00
|
|
|
.select { |k, v| k =~ /\.js$/ }
|
|
|
|
.each do |file, info|
|
2016-04-19 21:34:30 -04:00
|
|
|
|
2017-07-27 21:20:09 -04:00
|
|
|
path = "#{assets_path}/#{file}"
|
2016-04-19 21:34:30 -04:00
|
|
|
_file = (d = File.dirname(file)) == "." ? "_#{file}" : "#{d}/_#{File.basename(file)}"
|
|
|
|
_path = "#{assets_path}/#{_file}"
|
|
|
|
|
|
|
|
if File.exists?(_path)
|
|
|
|
STDERR.puts "Skipping: #{file} already compressed"
|
|
|
|
else
|
|
|
|
proc.call do
|
2018-10-04 22:25:52 -04:00
|
|
|
start = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
2018-07-03 19:42:21 -04:00
|
|
|
STDERR.puts "#{start} Compressing: #{file}"
|
|
|
|
|
2019-05-08 01:54:21 -04:00
|
|
|
# We can specify some files to never minify
|
|
|
|
unless (ENV["DONT_MINIFY"] == "1") || to_skip.include?(info['logical_path'])
|
2016-04-19 21:34:30 -04:00
|
|
|
FileUtils.mv(path, _path)
|
2017-07-27 21:20:09 -04:00
|
|
|
compress(_file, file)
|
2016-04-19 21:34:30 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
info["size"] = File.size(path)
|
|
|
|
info["mtime"] = File.mtime(path).iso8601
|
|
|
|
gzip(path)
|
2019-05-08 01:54:21 -04:00
|
|
|
brotli(path)
|
2018-07-03 19:42:21 -04:00
|
|
|
|
2018-10-04 22:25:52 -04:00
|
|
|
STDERR.puts "Done compressing #{file} : #{(Process.clock_gettime(Process::CLOCK_MONOTONIC) - start).round(2)} secs"
|
2018-07-03 19:42:21 -04:00
|
|
|
STDERR.puts
|
2016-04-19 21:34:30 -04:00
|
|
|
end
|
2015-02-20 15:48:45 -05:00
|
|
|
end
|
2016-04-19 21:34:30 -04:00
|
|
|
end
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
# protected
|
|
|
|
manifest.send :save
|
2017-03-20 13:05:39 -04:00
|
|
|
|
2017-03-20 15:59:06 -04:00
|
|
|
if GlobalSetting.fallback_assets_path.present?
|
2017-03-20 13:43:59 -04:00
|
|
|
begin
|
2017-03-20 15:59:06 -04:00
|
|
|
FileUtils.cp_r("#{Rails.root}/public/assets/.", GlobalSetting.fallback_assets_path)
|
2017-03-20 13:43:59 -04:00
|
|
|
rescue => e
|
2017-03-20 15:59:06 -04:00
|
|
|
STDERR.puts "Failed to backup assets to #{GlobalSetting.fallback_assets_path}"
|
2017-03-20 13:43:59 -04:00
|
|
|
STDERR.puts e
|
|
|
|
STDERR.puts e.backtrace
|
|
|
|
end
|
2017-03-20 13:05:39 -04:00
|
|
|
end
|
2017-04-14 10:30:19 -04:00
|
|
|
end
|
2017-03-20 13:05:39 -04:00
|
|
|
|
2019-04-10 05:37:29 -04:00
|
|
|
mmdb_thread.join if mmdb_thread
|
2017-04-14 10:30:19 -04:00
|
|
|
end
|
2017-03-20 13:05:39 -04:00
|
|
|
|
2017-04-14 10:30:19 -04:00
|
|
|
Rake::Task["assets:precompile"].enhance do
|
|
|
|
class Sprockets::Manifest
|
|
|
|
def reload
|
2017-04-14 15:06:52 -04:00
|
|
|
@filename = find_directory_manifest(@directory)
|
2017-04-14 10:30:19 -04:00
|
|
|
@data = json_decode(File.read(@filename))
|
|
|
|
end
|
2014-12-12 02:53:26 -05:00
|
|
|
end
|
2017-04-14 15:06:52 -04:00
|
|
|
|
2017-04-14 10:30:19 -04:00
|
|
|
# cause on boot we loaded a blank manifest,
|
|
|
|
# we need to know where all the assets are to precompile CSS
|
|
|
|
# cause CSS uses asset_path
|
|
|
|
Rails.application.assets_manifest.reload
|
|
|
|
Rake::Task["assets:precompile:css"].invoke
|
2014-05-02 17:46:03 -04:00
|
|
|
end
|