DEV: Run QUnit tests for official Discourse themes (#24405)
Why this change? As the number of themes which the Discourse team supports officially grows, we want to ensure that changes made to Discourse core do not break the plugins. As such, we are adding a step to our Github actions test job to run the QUnit tests for all official themes. What does this change do? This change adds a new job to our tests Github actions workflow to run the QUnit tests for all official plugins. This is achieved with the following changes: 1. Update `testem.js` to rely on the `THEME_TEST_PAGES` env variable to set the `test_page` option when running theme QUnit tests with testem. The `test_page` option [allows an array to be specified](https://github.com/testem/testem#multiple-test-pages) such that tests for multiple pages can be run at the same time. We are relying on a ENV variable because the `testem` CLI does not support passing a list of pages to the `--test_page` option. 2. Support a `/testem-theme-qunit/:testem_id/theme-qunit` Rails route in the development environment. This is done because testem prefixes the path with a unique ID to the configured `test_page` URL. This is problematic for us because we proxy all testem requests to the Rails server and testem's proxy configuration option does not allow us to easily rewrite the URL to remove the prefix. Therefore, we configure a proxy in testem to prefix `theme-qunit` requests with `/testem-theme-qunit` which can then be easily identified by the Rails server and routed accordingly. 3. Update `qunit:test` to support a `THEME_IDS` environment variable which will allow it to run QUnit tests for multiple themes at the same time. 4. Support `bin/rake themes:qunit[ids,"<theme_id>|<theme_id>"]` to run the QUnit tests for multiple themes at the same time. 5. Adds a `themes:qunit_all_official` Rake task which runs the QUnit tests for all the official themes.
This commit is contained in:
parent
a1644d9d8d
commit
e0ef88abca
|
@ -48,8 +48,6 @@ jobs:
|
||||||
target: plugins
|
target: plugins
|
||||||
- build_type: annotations
|
- build_type: annotations
|
||||||
target: themes
|
target: themes
|
||||||
- build_type: frontend
|
|
||||||
target: themes
|
|
||||||
- build_type: backend
|
- build_type: backend
|
||||||
target: themes
|
target: themes
|
||||||
- build_type: frontend
|
- build_type: frontend
|
||||||
|
@ -120,7 +118,7 @@ jobs:
|
||||||
run: bin/rake plugin:pull_compatible_all
|
run: bin/rake plugin:pull_compatible_all
|
||||||
|
|
||||||
- name: Checkout official themes
|
- name: Checkout official themes
|
||||||
if: matrix.target == 'themes'
|
if: matrix.target == 'themes' && matrix.build_type == 'system'
|
||||||
run: bin/rake themes:clone_all_official
|
run: bin/rake themes:clone_all_official
|
||||||
|
|
||||||
- name: Add hosts to /etc/hosts, otherwise Chrome cannot reach minio
|
- name: Add hosts to /etc/hosts, otherwise Chrome cannot reach minio
|
||||||
|
@ -222,6 +220,11 @@ jobs:
|
||||||
run: QUNIT_WRITE_EXECUTION_FILE=1 QUNIT_PARALLEL=3 bin/rake plugin:qunit['*','1200000']
|
run: QUNIT_WRITE_EXECUTION_FILE=1 QUNIT_PARALLEL=3 bin/rake plugin:qunit['*','1200000']
|
||||||
timeout-minutes: 30
|
timeout-minutes: 30
|
||||||
|
|
||||||
|
- name: Theme QUnit
|
||||||
|
if: matrix.build_type == 'frontend' && matrix.target == 'themes'
|
||||||
|
run: DISCOURSE_DEV_DB=discourse_test QUNIT_PARALLEL=3 bin/rake themes:qunit_all_official
|
||||||
|
timeout-minutes: 15
|
||||||
|
|
||||||
- uses: actions/upload-artifact@v3
|
- uses: actions/upload-artifact@v3
|
||||||
if: always() && matrix.build_type == 'frontend' && matrix.target == 'plugins'
|
if: always() && matrix.build_type == 'frontend' && matrix.target == 'plugins'
|
||||||
with:
|
with:
|
||||||
|
|
|
@ -156,17 +156,21 @@ if (process.env.TESTEM_FIREFOX_PATH) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const target = `http://127.0.0.1:${process.env.UNICORN_PORT || "3000"}`;
|
const target = `http://127.0.0.1:${process.env.UNICORN_PORT || "3000"}`;
|
||||||
|
const themeTestPages = process.env.THEME_TEST_PAGES;
|
||||||
|
|
||||||
if (process.argv.includes("-t")) {
|
if (themeTestPages) {
|
||||||
// Running testem without ember cli. Probably for theme-qunit
|
module.exports.test_page = themeTestPages.split(",");
|
||||||
const testPage = process.argv[process.argv.indexOf("-t") + 1];
|
|
||||||
|
|
||||||
module.exports.proxies = {};
|
module.exports.proxies = {};
|
||||||
|
|
||||||
|
// Prepend a prefix to the path of the route such that the server handling the request can easily identify `/theme-qunit`
|
||||||
|
// requests. This is required because testem prepends a string to the path of the `test_page` option when it makes
|
||||||
|
// the request and there is no easy way for us to strip the string from the path through the proxy. As such, we let the
|
||||||
|
// destination server handle the request base on the prefix instead.
|
||||||
module.exports.proxies[`/*/theme-qunit`] = {
|
module.exports.proxies[`/*/theme-qunit`] = {
|
||||||
target: `${target}${testPage}`,
|
target: `${target}/testem-theme-qunit`,
|
||||||
ignorePath: true,
|
|
||||||
xfwd: true,
|
xfwd: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports.proxies["/*/*"] = { target, xfwd: true };
|
module.exports.proxies["/*/*"] = { target, xfwd: true };
|
||||||
|
|
||||||
module.exports.middleware = [
|
module.exports.middleware = [
|
||||||
|
|
|
@ -80,8 +80,8 @@ class RemoteTheme < ActiveRecord::Base
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
# This is only used in the tests environment and is currently not supported for other environments
|
# This is only used in the development and test environment and is currently not supported for other environments
|
||||||
if Rails.env.test?
|
if Rails.env.test? || Rails.env.development?
|
||||||
def self.import_theme_from_directory(directory)
|
def self.import_theme_from_directory(directory)
|
||||||
update_theme(ThemeStore::DirectoryImporter.new(directory))
|
update_theme(ThemeStore::DirectoryImporter.new(directory))
|
||||||
end
|
end
|
||||||
|
|
|
@ -54,8 +54,7 @@ if defined?(Rack::MiniProfiler) && defined?(Rack::MiniProfiler::Config)
|
||||||
/topics/timings
|
/topics/timings
|
||||||
/uploads/
|
/uploads/
|
||||||
/user_avatar/
|
/user_avatar/
|
||||||
/theme-qunit
|
].map { |path| "#{Discourse.base_path}#{path}" }.concat([/.*theme-qunit/])
|
||||||
].map { |path| "#{Discourse.base_path}#{path}" }
|
|
||||||
|
|
||||||
# we DO NOT WANT mini-profiler loading on anything but real desktops and laptops
|
# we DO NOT WANT mini-profiler loading on anything but real desktops and laptops
|
||||||
# so let's rule out all handheld, tablet, and mobile devices
|
# so let's rule out all handheld, tablet, and mobile devices
|
||||||
|
|
|
@ -1583,6 +1583,16 @@ Discourse::Application.routes.draw do
|
||||||
|
|
||||||
get "/theme-qunit" => "qunit#theme"
|
get "/theme-qunit" => "qunit#theme"
|
||||||
|
|
||||||
|
# This is a special route that is used when theme QUnit tests are run through testem which appends a testem_id to the
|
||||||
|
# path. Unfortunately, testem's proxy support does not allow us to easily remove this from the path, so we have to
|
||||||
|
# handle it here.
|
||||||
|
if Rails.env.development?
|
||||||
|
get "/testem-theme-qunit/:testem_id/theme-qunit" => "qunit#theme",
|
||||||
|
:constraints => {
|
||||||
|
testem_id: /\d+/,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
post "/push_notifications/subscribe" => "push_notification#subscribe"
|
post "/push_notifications/subscribe" => "push_notification#subscribe"
|
||||||
post "/push_notifications/unsubscribe" => "push_notification#unsubscribe"
|
post "/push_notifications/unsubscribe" => "push_notification#unsubscribe"
|
||||||
|
|
||||||
|
|
|
@ -100,24 +100,37 @@ task "qunit:test", %i[timeout qunit_path filter] do |_, args|
|
||||||
end
|
end
|
||||||
puts "Rails server is warmed up"
|
puts "Rails server is warmed up"
|
||||||
|
|
||||||
cmd = ["env", "UNICORN_PORT=#{unicorn_port}"]
|
env = { "UNICORN_PORT" => unicorn_port.to_s }
|
||||||
|
cmd = []
|
||||||
|
|
||||||
|
parallel = ENV["QUNIT_PARALLEL"]
|
||||||
|
|
||||||
if qunit_path
|
if qunit_path
|
||||||
# Bypass `ember test` - it only works properly for the `/tests` path.
|
# Bypass `ember test` - it only works properly for the `/tests` path.
|
||||||
# We have to trigger a `build` manually so that JS is available for rails to serve.
|
# We have to trigger a `build` manually so that JS is available for rails to serve.
|
||||||
system("yarn", "ember", "build", chdir: "#{Rails.root}/app/assets/javascripts/discourse")
|
system("yarn", "ember", "build", chdir: "#{Rails.root}/app/assets/javascripts/discourse")
|
||||||
test_page = "#{qunit_path}?#{query}&testem=1"
|
|
||||||
cmd += ["yarn", "testem", "ci", "-f", "testem.js", "-t", test_page]
|
env["THEME_TEST_PAGES"] = if ENV["THEME_IDS"]
|
||||||
|
ENV["THEME_IDS"]
|
||||||
|
.split("|")
|
||||||
|
.map { |theme_id| "#{qunit_path}?#{query}&testem=1&id=#{theme_id}" }
|
||||||
|
.join(",")
|
||||||
|
else
|
||||||
|
"#{qunit_path}?#{query}&testem=1"
|
||||||
|
end
|
||||||
|
|
||||||
|
cmd += %w[yarn testem ci -f testem.js]
|
||||||
|
cmd += ["--parallel", parallel] if parallel
|
||||||
else
|
else
|
||||||
cmd += ["yarn", "ember", "exam", "--query", query]
|
cmd += ["yarn", "ember", "exam", "--query", query]
|
||||||
if parallel = ENV["QUNIT_PARALLEL"]
|
cmd += ["--load-balance", "--parallel", parallel] if parallel
|
||||||
cmd += ["--load-balance", "--parallel", parallel]
|
|
||||||
end
|
|
||||||
cmd += ["--filter", filter] if filter
|
cmd += ["--filter", filter] if filter
|
||||||
cmd << "--write-execution-file" if ENV["QUNIT_WRITE_EXECUTION_FILE"]
|
cmd << "--write-execution-file" if ENV["QUNIT_WRITE_EXECUTION_FILE"]
|
||||||
end
|
end
|
||||||
|
|
||||||
system(*cmd, chdir: "#{Rails.root}/app/assets/javascripts/discourse")
|
# Print out all env for debugging purposes
|
||||||
|
p env
|
||||||
|
system(env, *cmd, chdir: "#{Rails.root}/app/assets/javascripts/discourse")
|
||||||
|
|
||||||
success &&= $?.success?
|
success &&= $?.success?
|
||||||
ensure
|
ensure
|
||||||
|
|
|
@ -124,7 +124,7 @@ desc "Run QUnit tests of a theme/component"
|
||||||
task "themes:qunit", :type, :value do |t, args|
|
task "themes:qunit", :type, :value do |t, args|
|
||||||
type = args[:type]
|
type = args[:type]
|
||||||
value = args[:value]
|
value = args[:value]
|
||||||
raise <<~TEXT if !%w[name url id].include?(type) || value.blank?
|
raise <<~TEXT if !%w[name url id ids].include?(type) || value.blank?
|
||||||
Wrong arguments type:#{type.inspect}, value:#{value.inspect}"
|
Wrong arguments type:#{type.inspect}, value:#{value.inspect}"
|
||||||
Usage:
|
Usage:
|
||||||
`bundle exec rake "themes:qunit[url,<theme_url>]"`
|
`bundle exec rake "themes:qunit[url,<theme_url>]"`
|
||||||
|
@ -132,7 +132,10 @@ task "themes:qunit", :type, :value do |t, args|
|
||||||
`bundle exec rake "themes:qunit[name,<theme_name>]"`
|
`bundle exec rake "themes:qunit[name,<theme_name>]"`
|
||||||
OR
|
OR
|
||||||
`bundle exec rake "themes:qunit[id,<theme_id>]"`
|
`bundle exec rake "themes:qunit[id,<theme_id>]"`
|
||||||
|
OR
|
||||||
|
`bundle exec rake "themes:qunit[ids,<theme_id|theme_id|theme_id>]
|
||||||
TEXT
|
TEXT
|
||||||
|
|
||||||
ENV["THEME_#{type.upcase}"] = value.to_s
|
ENV["THEME_#{type.upcase}"] = value.to_s
|
||||||
ENV["QUNIT_RAILS_ENV"] ||= "development" # qunit:test will switch to `test` by default
|
ENV["QUNIT_RAILS_ENV"] ||= "development" # qunit:test will switch to `test` by default
|
||||||
Rake::Task["qunit:test"].reenable
|
Rake::Task["qunit:test"].reenable
|
||||||
|
@ -195,7 +198,7 @@ ensure
|
||||||
redis&.remove
|
redis&.remove
|
||||||
end
|
end
|
||||||
|
|
||||||
desc "Clones all official themes."
|
desc "Clones all official themes"
|
||||||
task "themes:clone_all_official" do |task, args|
|
task "themes:clone_all_official" do |task, args|
|
||||||
require "theme_metadata"
|
require "theme_metadata"
|
||||||
FileUtils.rm_rf("tmp/themes")
|
FileUtils.rm_rf("tmp/themes")
|
||||||
|
@ -203,7 +206,7 @@ task "themes:clone_all_official" do |task, args|
|
||||||
official_themes =
|
official_themes =
|
||||||
ThemeMetadata::OFFICIAL_THEMES.each do |theme_name|
|
ThemeMetadata::OFFICIAL_THEMES.each do |theme_name|
|
||||||
repo = "https://github.com/discourse/#{theme_name}"
|
repo = "https://github.com/discourse/#{theme_name}"
|
||||||
path = File.expand_path("tmp/themes/#{theme_name}")
|
path = File.join(Rails.root, "tmp/themes/#{theme_name}")
|
||||||
|
|
||||||
attempts = 0
|
attempts = 0
|
||||||
|
|
||||||
|
@ -217,3 +220,24 @@ task "themes:clone_all_official" do |task, args|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Note that this should only be used in CI where it is safe to mutate the database without rolling back since running
|
||||||
|
# the themes QUnit tests requires the themes to be installed in the database.
|
||||||
|
desc "Runs qunit tests for all official themes"
|
||||||
|
task "themes:qunit_all_official" => ["themes:clone_all_official", :environment] do |task, args|
|
||||||
|
theme_ids_with_qunit_tests = []
|
||||||
|
|
||||||
|
ThemeMetadata::OFFICIAL_THEMES.each do |theme_name|
|
||||||
|
path = File.join(Rails.root, "tmp/themes/#{theme_name}")
|
||||||
|
|
||||||
|
if Dir.glob("#{File.join(path, "test")}/**/*.{js,es6}").any?
|
||||||
|
theme = RemoteTheme.import_theme_from_directory(path)
|
||||||
|
theme_ids_with_qunit_tests << theme.id
|
||||||
|
else
|
||||||
|
puts "Skipping #{theme_name} as no QUnit tests have been detected"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
Rake::Task["themes:qunit"].reenable
|
||||||
|
Rake::Task["themes:qunit"].invoke("ids", theme_ids_with_qunit_tests.join("|"))
|
||||||
|
end
|
||||||
|
|
Loading…
Reference in New Issue