discourse/lib/temporary_db.rb
Osama Sayegh 7217dcb67a
FEATURE: Allow theme tests to be run in production (#12815)
This commit allows site admins to run theme tests in production via a new `/theme-qunit` route. When you visit `/theme-qunit`, you'll see a list of the themes/components installed on your site that have tests, and from there you can select a theme or component that you run its tests.

We also have a new rake task `themes:install_and_test` that can be used to install a list of themes/components on a temporary database and run the tests of the themes/components that are installed. This rake task can be useful when upgrading/deploying a Discourse instance to make sure that the installed themes/components are compatible with the new Discourse version being deployed, and if the tests fail you can abort the build/deploy process so you don't end up with a broken site.
2021-04-26 12:56:45 +03:00

132 lines
3.1 KiB
Ruby

# frozen_string_literal: true
class TemporaryDb
PG_TEMP_PATH = "/tmp/pg_schema_tmp"
PG_CONF = "#{PG_TEMP_PATH}/postgresql.conf"
PG_SOCK_PATH = "#{PG_TEMP_PATH}/sockets"
def port_available?(port)
TCPServer.open(port).close
true
rescue Errno::EADDRINUSE
false
end
def pg_bin_path
return @pg_bin_path if @pg_bin_path
["13", "12", "11", "10"].each do |v|
bin_path = "/usr/lib/postgresql/#{v}/bin"
if File.exist?("#{bin_path}/pg_ctl")
@pg_bin_path = bin_path
break
end
end
if !@pg_bin_path
bin_path = "/Applications/Postgres.app/Contents/Versions/latest/bin"
if File.exists?("#{bin_path}/pg_ctl")
@pg_bin_path = bin_path
end
end
if !@pg_bin_path
puts "Can not find postgres bin path"
exit 1
end
@pg_bin_path
end
def initdb_path
return @initdb_path if @initdb_path
@initdb_path = `which initdb 2> /dev/null`.strip
if @initdb_path.length == 0
@initdb_path = "#{pg_bin_path}/initdb"
end
@initdb_path
end
def find_free_port(range)
range.each do |port|
return port if port_available?(port)
end
end
def pg_port
@pg_port ||= find_free_port(11000..11900)
end
def pg_ctl_path
return @pg_ctl_path if @pg_ctl_path
@pg_ctl_path = `which pg_ctl 2> /dev/null`.strip
if @pg_ctl_path.length == 0
@pg_ctl_path = "#{pg_bin_path}/pg_ctl"
end
@pg_ctl_path
end
def start
FileUtils.rm_rf PG_TEMP_PATH
`#{initdb_path} -D '#{PG_TEMP_PATH}' --auth-host=trust --locale=en_US.UTF-8 -E UTF8 2> /dev/null`
FileUtils.mkdir PG_SOCK_PATH
conf = File.read(PG_CONF)
File.write(PG_CONF, conf + "\nport = #{pg_port}\nunix_socket_directories = '#{PG_SOCK_PATH}'")
puts "Starting postgres on port: #{pg_port}"
ENV['DISCOURSE_PG_PORT'] = pg_port.to_s
Thread.new do
`#{pg_ctl_path} -D '#{PG_TEMP_PATH}' start`
end
puts "Waiting for PG server to start..."
while !`#{pg_ctl_path} -D '#{PG_TEMP_PATH}' status`.include?('server is running')
sleep 0.1
end
@started = true
`createuser -h localhost -p #{pg_port} -s -D -w discourse 2> /dev/null`
`createdb -h localhost -p #{pg_port} discourse`
puts "PG server is ready and DB is loaded"
end
def stop
@started = false
`#{pg_ctl_path} -D '#{PG_TEMP_PATH}' stop`
end
def remove
raise "Error: the database must be stopped before it can be removed" if @started
FileUtils.rm_rf PG_TEMP_PATH
end
def migrate
if !@started
raise "Error: the database must be started before it can be migrated."
end
ActiveRecord::Base.establish_connection(
adapter: 'postgresql',
database: 'discourse',
port: pg_port,
host: 'localhost'
)
puts "Running migrations on blank database!"
old_stdout = $stdout.clone
old_stderr = $stderr.clone
$stdout.reopen(File.new('/dev/null', 'w'))
$stderr.reopen(File.new('/dev/null', 'w'))
SeedFu.quiet = true
Rake::Task["db:migrate"].invoke
ensure
$stdout.reopen(old_stdout) if old_stdout
$stderr.reopen(old_stderr) if old_stderr
end
end