mirror of
https://gh.wpcy.net/https://github.com/discourse/discourse.git
synced 2026-05-04 13:44:17 +08:00
Previously, `DiscourseUpdates.has_unseen_features?` ran the full new features pipeline on every request for staff users — parsing JSON, shelling out to `git merge-base` per entry via `GitUtils.has_commit?`, merging with upcoming changes, and sorting — taking 300-400ms to produce a single boolean. In this update, the max `created_at` timestamp across all valid merged features is computed once and cached in Redis, reducing `has_unseen_features?` to two Redis GETs and an integer comparison; the cache is invalidated when the daily job fetches new features, on deploy/restart, and when an upcoming change transitions to permanent.
683 lines
21 KiB
Ruby
683 lines
21 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
RSpec.describe UpcomingChanges do
|
|
let(:setting_name) { :enable_upload_debug_mode }
|
|
|
|
before do
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :experimental,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
conceptual_setting: {
|
|
status: :conceptual,
|
|
},
|
|
alpha_setting: {
|
|
status: :alpha,
|
|
},
|
|
beta_setting: {
|
|
status: :beta,
|
|
},
|
|
stable_setting: {
|
|
status: :stable,
|
|
},
|
|
permanent_setting: {
|
|
status: :permanent,
|
|
},
|
|
},
|
|
)
|
|
|
|
# There is a fixture image at spec/fixtures/images/upcoming_changes/enable_upload_debug_mode.png,
|
|
# but normally upcoming change images are at Rails.public_path + /images/upcoming_changes/
|
|
Rails.stubs(:public_path).returns(File.join(Rails.root, "spec", "fixtures"))
|
|
end
|
|
|
|
describe ".image_exists?" do
|
|
it "returns true when the image file exists" do
|
|
expect(described_class.image_exists?(setting_name)).to eq(true)
|
|
end
|
|
|
|
it "returns false when the image file does not exist" do
|
|
expect(described_class.image_exists?("nonexistent_setting")).to eq(false)
|
|
end
|
|
end
|
|
|
|
describe ".image_path" do
|
|
it "returns the correct path for the image" do
|
|
expect(described_class.image_path(setting_name)).to eq(
|
|
"images/upcoming_changes/#{setting_name}.png",
|
|
)
|
|
end
|
|
|
|
it "returns the correct path for plugin images" do
|
|
plugin_setting = :enable_experimental_sample_plugin_feature
|
|
|
|
expect(described_class.image_path(plugin_setting)).to eq(
|
|
"plugins/discourse-sample-plugin/images/upcoming_changes/#{plugin_setting}.png",
|
|
)
|
|
end
|
|
end
|
|
|
|
describe ".image_data" do
|
|
it "returns image URL, width, and height" do
|
|
result = described_class.image_data(setting_name)
|
|
|
|
expect(result[:url]).to eq(
|
|
"#{Discourse.base_url}/images/upcoming_changes/#{setting_name}.png",
|
|
)
|
|
expect(result[:width]).to eq(244)
|
|
expect(result[:height]).to eq(66)
|
|
end
|
|
|
|
context "when include_file_path is true" do
|
|
it "returns the image URL, width, height, and file path" do
|
|
result = described_class.image_data(setting_name, include_file_path: true)
|
|
|
|
expect(result[:file_path]).to eq(
|
|
File.join(
|
|
Rails.root,
|
|
"spec",
|
|
"fixtures",
|
|
"images",
|
|
"upcoming_changes",
|
|
"#{setting_name}.png",
|
|
),
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe ".change_metadata" do
|
|
it "returns the metadata hash for a setting with metadata" do
|
|
metadata = described_class.change_metadata(setting_name)
|
|
|
|
expect(metadata).to eq(
|
|
{
|
|
impact: "other,developers",
|
|
status: :experimental,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
)
|
|
end
|
|
|
|
it "returns an empty hash for a setting without metadata" do
|
|
metadata = described_class.change_metadata("nonexistent_setting")
|
|
|
|
expect(metadata).to eq({})
|
|
end
|
|
|
|
it "accepts string setting names" do
|
|
metadata = described_class.change_metadata(setting_name)
|
|
|
|
expect(metadata[:status]).to eq(:experimental)
|
|
end
|
|
|
|
it "accepts symbol setting names" do
|
|
metadata = described_class.change_metadata(setting_name.to_sym)
|
|
|
|
expect(metadata[:status]).to eq(:experimental)
|
|
end
|
|
end
|
|
|
|
describe ".not_yet_stable?" do
|
|
it "returns true for conceptual status" do
|
|
expect(described_class.not_yet_stable?("conceptual_setting")).to eq(true)
|
|
end
|
|
|
|
it "returns true for experimental status" do
|
|
expect(described_class.not_yet_stable?(setting_name)).to eq(true)
|
|
end
|
|
|
|
it "returns true for alpha status" do
|
|
expect(described_class.not_yet_stable?("alpha_setting")).to eq(true)
|
|
end
|
|
|
|
it "returns true for beta status" do
|
|
expect(described_class.not_yet_stable?("beta_setting")).to eq(true)
|
|
end
|
|
|
|
it "returns false for stable status" do
|
|
expect(described_class.not_yet_stable?("stable_setting")).to eq(false)
|
|
end
|
|
|
|
it "returns false for permanent status" do
|
|
expect(described_class.not_yet_stable?("permanent_setting")).to eq(false)
|
|
end
|
|
end
|
|
|
|
describe ".stable_or_permanent?" do
|
|
it "returns false for conceptual status" do
|
|
expect(described_class.stable_or_permanent?("conceptual_setting")).to eq(false)
|
|
end
|
|
|
|
it "returns false for experimental status" do
|
|
expect(described_class.stable_or_permanent?(setting_name)).to eq(false)
|
|
end
|
|
|
|
it "returns false for alpha status" do
|
|
expect(described_class.stable_or_permanent?("alpha_setting")).to eq(false)
|
|
end
|
|
|
|
it "returns false for beta status" do
|
|
expect(described_class.stable_or_permanent?("beta_setting")).to eq(false)
|
|
end
|
|
|
|
it "returns true for stable status" do
|
|
expect(described_class.stable_or_permanent?("stable_setting")).to eq(true)
|
|
end
|
|
|
|
it "returns true for permanent status" do
|
|
expect(described_class.stable_or_permanent?("permanent_setting")).to eq(true)
|
|
end
|
|
end
|
|
|
|
describe ".change_status_value" do
|
|
it "returns -100 for conceptual status" do
|
|
expect(described_class.change_status_value("conceptual_setting")).to eq(-100)
|
|
end
|
|
|
|
it "returns 0 for experimental status" do
|
|
expect(described_class.change_status_value(setting_name)).to eq(0)
|
|
end
|
|
|
|
it "returns 100 for alpha status" do
|
|
expect(described_class.change_status_value("alpha_setting")).to eq(100)
|
|
end
|
|
|
|
it "returns 200 for beta status" do
|
|
expect(described_class.change_status_value("beta_setting")).to eq(200)
|
|
end
|
|
|
|
it "returns 300 for stable status" do
|
|
expect(described_class.change_status_value("stable_setting")).to eq(300)
|
|
end
|
|
|
|
it "returns 500 for permanent status" do
|
|
expect(described_class.change_status_value("permanent_setting")).to eq(500)
|
|
end
|
|
end
|
|
|
|
describe ".change_status" do
|
|
it "returns :conceptual for conceptual status" do
|
|
expect(described_class.change_status("conceptual_setting")).to eq(:conceptual)
|
|
end
|
|
|
|
it "returns :experimental for experimental status" do
|
|
expect(described_class.change_status(setting_name)).to eq(:experimental)
|
|
end
|
|
|
|
it "returns :alpha for alpha status" do
|
|
expect(described_class.change_status("alpha_setting")).to eq(:alpha)
|
|
end
|
|
|
|
it "returns :beta for beta status" do
|
|
expect(described_class.change_status("beta_setting")).to eq(:beta)
|
|
end
|
|
|
|
it "returns :stable for stable status" do
|
|
expect(described_class.change_status("stable_setting")).to eq(:stable)
|
|
end
|
|
|
|
it "returns :permanent for permanent status" do
|
|
expect(described_class.change_status("permanent_setting")).to eq(:permanent)
|
|
end
|
|
end
|
|
|
|
describe ".meets_or_exceeds_status?" do
|
|
it "returns true when the change meets the required status" do
|
|
expect(described_class.meets_or_exceeds_status?("stable_setting", :beta)).to eq(true)
|
|
expect(described_class.meets_or_exceeds_status?("permanent_setting", :stable)).to eq(true)
|
|
end
|
|
|
|
it "returns false when the change does not meet the required status" do
|
|
expect(described_class.meets_or_exceeds_status?("alpha_setting", :beta)).to eq(false)
|
|
expect(described_class.meets_or_exceeds_status?("beta_setting", :stable)).to eq(false)
|
|
end
|
|
end
|
|
|
|
describe ".previous_status_value" do
|
|
it "returns -100 for conceptual (lowest status)" do
|
|
expect(described_class.previous_status_value(:conceptual)).to eq(-100)
|
|
end
|
|
|
|
it "returns -100 for experimental" do
|
|
expect(described_class.previous_status_value(:experimental)).to eq(-100)
|
|
end
|
|
|
|
it "returns 0 for alpha" do
|
|
expect(described_class.previous_status_value(:alpha)).to eq(0)
|
|
end
|
|
|
|
it "returns 100 for beta" do
|
|
expect(described_class.previous_status_value(:beta)).to eq(100)
|
|
end
|
|
|
|
it "returns 200 for stable" do
|
|
expect(described_class.previous_status_value(:stable)).to eq(200)
|
|
end
|
|
|
|
it "returns 300 for permanent" do
|
|
expect(described_class.previous_status_value(:permanent)).to eq(300)
|
|
end
|
|
|
|
it "accepts string status names" do
|
|
expect(described_class.previous_status_value("stable")).to eq(200)
|
|
end
|
|
end
|
|
|
|
describe ".previous_status" do
|
|
it "returns :conceptual for conceptual (lowest status)" do
|
|
expect(described_class.previous_status(:conceptual)).to eq(:conceptual)
|
|
end
|
|
|
|
it "returns :conceptual for experimental" do
|
|
expect(described_class.previous_status(:experimental)).to eq(:conceptual)
|
|
end
|
|
|
|
it "returns :experimental for alpha" do
|
|
expect(described_class.previous_status(:alpha)).to eq(:experimental)
|
|
end
|
|
|
|
it "returns :alpha for beta" do
|
|
expect(described_class.previous_status(:beta)).to eq(:alpha)
|
|
end
|
|
|
|
it "returns :beta for stable" do
|
|
expect(described_class.previous_status(:stable)).to eq(:beta)
|
|
end
|
|
|
|
it "returns :stable for permanent" do
|
|
expect(described_class.previous_status(:permanent)).to eq(:stable)
|
|
end
|
|
|
|
it "accepts string status names" do
|
|
expect(described_class.previous_status("stable")).to eq(:beta)
|
|
end
|
|
end
|
|
|
|
describe ".history_for" do
|
|
fab!(:admin)
|
|
|
|
it "returns UserHistory records for the given setting" do
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
)
|
|
|
|
history = described_class.history_for(setting_name)
|
|
|
|
expect(history.count).to eq(1)
|
|
expect(history.first.subject).to eq(setting_name.to_s)
|
|
expect(history.first.action).to eq(UserHistory.actions[:upcoming_change_toggled])
|
|
end
|
|
|
|
it "returns records ordered by created_at descending" do
|
|
first_history =
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
created_at: 2.days.ago,
|
|
)
|
|
|
|
second_history =
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
created_at: 1.day.ago,
|
|
)
|
|
|
|
history = described_class.history_for(setting_name)
|
|
|
|
expect(history.first.id).to eq(second_history.id)
|
|
expect(history.last.id).to eq(first_history.id)
|
|
end
|
|
|
|
it "returns only records matching the setting name" do
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
)
|
|
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: "different_setting",
|
|
acting_user_id: admin.id,
|
|
)
|
|
|
|
history = described_class.history_for(setting_name)
|
|
|
|
expect(history.count).to eq(1)
|
|
expect(history.first.subject).to eq(setting_name.to_s)
|
|
end
|
|
|
|
it "returns only records with upcoming_change_toggled action" do
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:upcoming_change_toggled],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
)
|
|
|
|
UserHistory.create!(
|
|
action: UserHistory.actions[:change_site_setting],
|
|
subject: setting_name,
|
|
acting_user_id: admin.id,
|
|
)
|
|
|
|
history = described_class.history_for(setting_name)
|
|
|
|
expect(history.count).to eq(1)
|
|
expect(history.first.action).to eq(UserHistory.actions[:upcoming_change_toggled])
|
|
end
|
|
|
|
it "returns an empty relation when no history exists" do
|
|
history = described_class.history_for("nonexistent_setting")
|
|
|
|
expect(history.count).to eq(0)
|
|
expect(history).to be_a(ActiveRecord::Relation)
|
|
end
|
|
end
|
|
|
|
describe ".enabled?" do
|
|
after do
|
|
SiteSetting.remove_override!(setting_name)
|
|
SiteSetting.promote_upcoming_changes_on_status = :stable
|
|
UpcomingChanges.clear_caches!
|
|
end
|
|
|
|
context "when the setting has no row in the database (admin has not saved it)" do
|
|
before { SiteSetting.remove_override!(setting_name) }
|
|
|
|
it "returns the yaml default when the change is below promote_upcoming_changes_on_status" do
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :experimental,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
},
|
|
)
|
|
SiteSetting.promote_upcoming_changes_on_status = :stable
|
|
|
|
expect(described_class.enabled?(setting_name)).to eq(SiteSetting.defaults[setting_name])
|
|
end
|
|
|
|
it "returns true when the change meets or exceeds promote_upcoming_changes_on_status" do
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :stable,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
},
|
|
)
|
|
SiteSetting.promote_upcoming_changes_on_status = :stable
|
|
|
|
expect(described_class.enabled?(setting_name)).to eq(true)
|
|
end
|
|
end
|
|
|
|
context "when an admin has saved a value to the database" do
|
|
it "returns the stored value when true" do
|
|
SiteSetting.enable_upload_debug_mode = true
|
|
|
|
expect(described_class.enabled?(setting_name)).to eq(true)
|
|
end
|
|
|
|
it "returns the stored value when false even when the change meets promote_upcoming_changes_on_status" do
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :beta,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
},
|
|
)
|
|
SiteSetting.promote_upcoming_changes_on_status = :beta
|
|
SiteSetting.enable_upload_debug_mode = false
|
|
|
|
expect(described_class.enabled?(setting_name)).to eq(false)
|
|
end
|
|
end
|
|
|
|
context "when the change is permanent" do
|
|
before do
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :permanent,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
},
|
|
)
|
|
end
|
|
|
|
it "returns true even when the database value is false" do
|
|
SiteSetting.enable_upload_debug_mode = false
|
|
|
|
expect(described_class.enabled?(setting_name)).to eq(true)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe ".current_statuses" do
|
|
include ActiveSupport::Testing::TimeHelpers
|
|
|
|
before { described_class.clear_caches! }
|
|
|
|
after do
|
|
described_class.clear_caches!
|
|
UpcomingChangeEvent.where(upcoming_change_name: "timeline_status_setting").delete_all
|
|
end
|
|
|
|
it "returns an empty hash when there are no status_changed events" do
|
|
expect(described_class.current_statuses).to eq({})
|
|
end
|
|
|
|
it "maps each upcoming change to the latest status_changed new_value and timestamp" do
|
|
travel_to Time.zone.parse("2024-06-01 12:00:00") do
|
|
UpcomingChangeEvent.create!(
|
|
event_type: :status_changed,
|
|
upcoming_change_name: "timeline_status_setting",
|
|
event_data: {
|
|
"previous_value" => "alpha",
|
|
"new_value" => "beta",
|
|
},
|
|
)
|
|
end
|
|
|
|
latest_event =
|
|
travel_to(Time.zone.parse("2024-06-15 12:00:00")) do
|
|
UpcomingChangeEvent.create!(
|
|
event_type: :status_changed,
|
|
upcoming_change_name: "timeline_status_setting",
|
|
event_data: {
|
|
"previous_value" => "beta",
|
|
"new_value" => "stable",
|
|
},
|
|
)
|
|
end
|
|
|
|
result = described_class.current_statuses
|
|
|
|
expect(result["timeline_status_setting"]).to eq(
|
|
{ status: "stable", changed_at: latest_event.created_at },
|
|
)
|
|
end
|
|
|
|
it "caches the result so the SQL runs only once until the cache key is deleted" do
|
|
UpcomingChangeEvent.create!(
|
|
event_type: :status_changed,
|
|
upcoming_change_name: "timeline_status_setting",
|
|
event_data: {
|
|
"previous_value" => nil,
|
|
"new_value" => "experimental",
|
|
},
|
|
)
|
|
|
|
allow(DB).to receive(:query).and_call_original
|
|
|
|
2.times { described_class.current_statuses }
|
|
expect(DB).to have_received(:query).once
|
|
|
|
described_class.clear_caches!
|
|
|
|
described_class.current_statuses
|
|
expect(DB).to have_received(:query).twice
|
|
end
|
|
end
|
|
|
|
describe ".permanent_upcoming_changes" do
|
|
before do
|
|
described_class.clear_caches!
|
|
mock_upcoming_change_metadata(
|
|
{
|
|
enable_upload_debug_mode: {
|
|
impact: "other,developers",
|
|
status: :permanent,
|
|
impact_type: "other",
|
|
impact_role: "developers",
|
|
},
|
|
},
|
|
)
|
|
end
|
|
|
|
after { described_class.clear_caches! }
|
|
|
|
it "returns only changes whose metadata status is permanent" do
|
|
list = described_class.permanent_upcoming_changes
|
|
|
|
expect(list.all? { |c| described_class.change_status(c[:setting]) == :permanent }).to eq(true)
|
|
expect(list.map { |c| c[:setting] }).to include(:enable_upload_debug_mode)
|
|
end
|
|
|
|
it "caches the list so UpcomingChanges::List runs only once until the cache key is deleted" do
|
|
allow(UpcomingChanges::List).to receive(:call).and_call_original
|
|
|
|
2.times { described_class.permanent_upcoming_changes }
|
|
expect(UpcomingChanges::List).to have_received(:call).once
|
|
|
|
described_class.clear_caches!
|
|
|
|
described_class.permanent_upcoming_changes
|
|
expect(UpcomingChanges::List).to have_received(:call).twice
|
|
end
|
|
end
|
|
|
|
describe ".clear_caches!" do
|
|
it "clears the latest new feature created_at cache" do
|
|
Discourse.redis.set("latest_new_feature_created_at", Time.zone.now.iso8601)
|
|
described_class.clear_caches!
|
|
expect(Discourse.redis.get("latest_new_feature_created_at")).to be_nil
|
|
end
|
|
end
|
|
|
|
describe ".enabled_for_user?" do
|
|
context "for logged-in user" do
|
|
fab!(:user)
|
|
|
|
context "when the upcoming change is disabled" do
|
|
before { SiteSetting.enable_upload_debug_mode = false }
|
|
|
|
it "returns false" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(false)
|
|
end
|
|
end
|
|
|
|
context "when the upcoming change is enabled for everyone" do
|
|
before { SiteSetting.enable_upload_debug_mode = true }
|
|
|
|
it "returns true" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(true)
|
|
end
|
|
end
|
|
|
|
context "when the upcoming change is only enabled for certain groups" do
|
|
before do
|
|
SiteSetting.enable_upload_debug_mode = true
|
|
Fabricate(
|
|
:site_setting_group,
|
|
name: setting_name,
|
|
group_ids: Group::AUTO_GROUPS[:trust_level_4].to_s,
|
|
)
|
|
end
|
|
|
|
it "returns false" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(false)
|
|
end
|
|
|
|
context "when the user is in that group" do
|
|
before do
|
|
trust_level_4_group = Group.find_by(id: Group::AUTO_GROUPS[:trust_level_4])
|
|
trust_level_4_group.add(user)
|
|
end
|
|
|
|
it "returns true" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(true)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
context "for anonymous user" do
|
|
let(:user) { nil }
|
|
|
|
context "when the upcoming change is disabled" do
|
|
before { SiteSetting.enable_upload_debug_mode = false }
|
|
|
|
it "returns false" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(false)
|
|
end
|
|
end
|
|
|
|
context "when the upcoming change is enabled for everyone" do
|
|
before { SiteSetting.enable_upload_debug_mode = true }
|
|
|
|
it "returns true" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(true)
|
|
end
|
|
end
|
|
|
|
context "when the upcoming change is only enabled for certain groups" do
|
|
before do
|
|
SiteSetting.enable_upload_debug_mode = true
|
|
Fabricate(
|
|
:site_setting_group,
|
|
name: setting_name,
|
|
group_ids: Group::AUTO_GROUPS[:trust_level_4].to_s,
|
|
)
|
|
end
|
|
|
|
it "returns false" do
|
|
expect(UpcomingChanges.enabled_for_user?(setting_name, user)).to eq(false)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "conceptual status filtering" do
|
|
it "excludes conceptual changes from all_settings with only_upcoming_changes" do
|
|
settings =
|
|
SiteSetting
|
|
.all_settings(only_upcoming_changes: true, include_hidden: true)
|
|
.map { |s| s[:setting] }
|
|
expect(settings).not_to include(:conceptual_setting)
|
|
expect(settings).to include(:enable_upload_debug_mode)
|
|
end
|
|
end
|
|
end
|