mirror of
https://gh.wpcy.net/https://github.com/discourse/discourse.git
synced 2026-05-03 00:20:04 +08:00
This reverts commit 7633f0d0ac, cleans up
the tests workflow, and works around the Chrome crash issue. (see:
https://issues.chromium.org/issues/504886986)
Closes #39050
Closes #39051
Closes #39070
Closes #39100
Closes #39158
Closes #39195
Closes #39255
Closes #39308
182 lines
5.2 KiB
Ruby
Executable file
182 lines
5.2 KiB
Ruby
Executable file
#!/usr/bin/env ruby
|
|
# frozen_string_literal: true
|
|
# rubocop:disable Discourse/NoChdir
|
|
|
|
require "bundler/setup"
|
|
require "fileutils"
|
|
require "tempfile"
|
|
require "open3"
|
|
require "json"
|
|
require "time"
|
|
require "parallel"
|
|
require "etc"
|
|
|
|
require_relative "../lib/version"
|
|
|
|
DOWNLOAD_PRE_BUILT_ASSETS = ENV["DISCOURSE_DOWNLOAD_PRE_BUILT_ASSETS"] != "0"
|
|
DOWNLOAD_TEMP_FILE = "#{__dir__}/../tmp/assets.tar.gz"
|
|
DOWNLOAD_EXTRACT_DIR = "#{__dir__}/../tmp/extracted_assets"
|
|
|
|
PRE_BUILD_ROOT = "https://get.discourse.org/discourse-assets"
|
|
|
|
JS_SOURCE_PATHS = %w[frontend package.json pnpm-lock.yaml]
|
|
|
|
EMBER_APP_DIR = "frontend/discourse"
|
|
BUILD_INFO_FILE = "#{EMBER_APP_DIR}/dist/BUILD_INFO.json"
|
|
|
|
Dir.chdir("#{__dir__}/..")
|
|
|
|
def capture(*args)
|
|
output, status = Open3.capture2(*args)
|
|
raise "Command failed: #{args.inspect}" if status != 0
|
|
output
|
|
end
|
|
|
|
def log(message)
|
|
STDERR.puts "[assemble_ember_build] #{message}"
|
|
end
|
|
|
|
# Returns a git tree-hash representing the current state of Discourse core JS source paths.
|
|
# Only files in JS_SOURCE_PATHS are included in the tree hash.
|
|
def core_tree_hash
|
|
Tempfile.create do |f|
|
|
f.close
|
|
|
|
git_dir = capture("git", "rev-parse", "--git-dir").strip
|
|
FileUtils.cp "#{git_dir}/index", f.path
|
|
|
|
env = { "GIT_INDEX_FILE" => f.path }
|
|
|
|
# Remove all files from the index, then add only JS_SOURCE_PATHS
|
|
system(env, "git", "rm", "-r", "--cached", ".", "--quiet", exception: true)
|
|
system(env, "git", "add", *JS_SOURCE_PATHS, exception: true)
|
|
|
|
capture(env, "git", "write-tree").strip
|
|
end
|
|
end
|
|
|
|
def node_heap_size_limit
|
|
capture("node", "-e", "console.log(v8.getHeapStatistics().heap_size_limit/1024/1024)").to_f
|
|
end
|
|
|
|
def low_memory_environment?
|
|
node_heap_size_limit < 2048
|
|
end
|
|
|
|
def resolved_ember_env
|
|
ENV["EMBER_ENV"] || "development"
|
|
end
|
|
|
|
def build_info
|
|
{ "ember_env" => resolved_ember_env, "core_tree_hash" => core_tree_hash }
|
|
end
|
|
|
|
def existing_core_build_usable?
|
|
if !File.exist?(BUILD_INFO_FILE)
|
|
log "No existing build info file found."
|
|
return false
|
|
end
|
|
|
|
existing = JSON.parse(File.read(BUILD_INFO_FILE))
|
|
expected = build_info
|
|
|
|
if existing == expected
|
|
true
|
|
else
|
|
log <<~MSG
|
|
Existing build is not reusable.
|
|
- Existing: #{existing.inspect}
|
|
- Current: #{expected.inspect}
|
|
MSG
|
|
false
|
|
end
|
|
end
|
|
|
|
def download_prebuild_assets!
|
|
return false if !DOWNLOAD_PRE_BUILT_ASSETS
|
|
|
|
status_output = capture("git", "status", "--porcelain", *JS_SOURCE_PATHS).strip
|
|
git_is_clean = status_output.empty?
|
|
if !git_is_clean
|
|
log "JS-related files in the git working directory have been changed. Skipping download of prebuilt assets."
|
|
return false
|
|
end
|
|
|
|
core_commit_hash = capture("git", "rev-parse", "HEAD").strip
|
|
version_string = "#{Discourse::VERSION::STRING}-#{core_commit_hash.slice(0, 8)}"
|
|
|
|
url = "#{PRE_BUILD_ROOT}/#{version_string}/#{resolved_ember_env}.tar.gz"
|
|
puts "Fetching and extracting #{url}..."
|
|
|
|
begin
|
|
system("curl", "--fail-with-body", "-L", url, "-o", DOWNLOAD_TEMP_FILE, exception: true)
|
|
rescue RuntimeError => e
|
|
log "Failed to download prebuilt assets: #{e.message}"
|
|
return false
|
|
end
|
|
|
|
FileUtils.mkdir_p(DOWNLOAD_EXTRACT_DIR)
|
|
begin
|
|
system("tar", "-xzf", DOWNLOAD_TEMP_FILE, "-C", DOWNLOAD_EXTRACT_DIR, exception: true)
|
|
rescue RuntimeError => e
|
|
log "Failed to extract prebuilt assets: #{e.message}"
|
|
return false
|
|
end
|
|
|
|
FileUtils.rm_rf("#{EMBER_APP_DIR}/dist")
|
|
FileUtils.mv("#{DOWNLOAD_EXTRACT_DIR}/core", "#{EMBER_APP_DIR}/dist")
|
|
|
|
FileUtils.rm_rf("app/assets/generated")
|
|
FileUtils.mv("#{DOWNLOAD_EXTRACT_DIR}/plugins", "./app/assets/generated")
|
|
|
|
puts "Prebuilt assets downloaded and extracted successfully."
|
|
true
|
|
ensure
|
|
FileUtils.rm_f(DOWNLOAD_TEMP_FILE) if File.exist?(DOWNLOAD_TEMP_FILE)
|
|
FileUtils.rm_rf(DOWNLOAD_EXTRACT_DIR) if File.exist?(DOWNLOAD_EXTRACT_DIR)
|
|
end
|
|
|
|
build_cmd = %w[pnpm ember build]
|
|
build_env = { "CI" => "1" }
|
|
|
|
if Etc.nprocessors > 2
|
|
# Anything more than 2 doesn't seem to improve build times
|
|
build_env["JOBS"] ||= "2"
|
|
end
|
|
|
|
if low_memory_environment?
|
|
log "Node.js heap_size_limit is less than 2048MB. Setting --max-old-space-size=2048 and CHEAP_SOURCE_MAPS=1"
|
|
build_env["NODE_OPTIONS"] = "--max_old_space_size=2048"
|
|
build_env["CHEAP_SOURCE_MAPS"] = "1"
|
|
build_env["JOBS"] = "1"
|
|
end
|
|
|
|
build_cmd << "-prod" if resolved_ember_env == "production"
|
|
|
|
core_build_reusable =
|
|
existing_core_build_usable? || (download_prebuild_assets! && existing_core_build_usable?)
|
|
|
|
if core_build_reusable
|
|
log "Reusing existing core ember build. All done."
|
|
else
|
|
log "Running full core build..."
|
|
system(build_env, *build_cmd, exception: true, chdir: EMBER_APP_DIR)
|
|
File.write(BUILD_INFO_FILE, JSON.pretty_generate(build_info))
|
|
end
|
|
|
|
system("bin/rake", "assets:precompile:build_plugins", exception: true)
|
|
|
|
if ARGV.include?("--compress")
|
|
files = [*Dir.glob("#{EMBER_APP_DIR}/dist/**/*.js"), *Dir.glob("app/assets/generated/**/*.js")]
|
|
Parallel.map(files, in_threads: 4) do |file|
|
|
next if File.exist?("#{file}.gz") && File.exist?("#{file}.br")
|
|
|
|
start = Time.now
|
|
|
|
system("brotli", "-f", "--quality=11", "-o", "#{file}.br", file, exception: true)
|
|
IO.popen(["gzip", "-f", "-9", "-c", file], "rb") { |io| File.write("#{file}.gz", io.read) }
|
|
raise "gzip failed for #{file}" unless $?.success?
|
|
|
|
puts "Compressed #{file} in #{(Time.now - start).round(2)}s"
|
|
end
|
|
end
|