FIX: TopicSummarization workaround for Postgres' discrete range types (#23105)
Our code assumed the content_range interval was inclusive, but they are open-ended due to Postgres' [discrete range types](https://www.postgresql.org/docs/current/rangetypes.html#RANGETYPES-DISCRETE), meaning [1,2] will be represented as [1,3). It also fixes some flaky tests due to test data not being correctly setup and the registry not being resetted after each test.
This commit is contained in:
parent
eb4971cb06
commit
5683c90917
|
@ -8,6 +8,11 @@ class TopicSummarySerializer < ApplicationSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def new_posts_since_summary
|
def new_posts_since_summary
|
||||||
object.target.highest_post_number.to_i - object.content_range&.end.to_i
|
# Postgres uses discrete range types for int4range, which means
|
||||||
|
# an inclusive [1,2] ranges is stored as [1,3). To work around this
|
||||||
|
# an provide an accurate count, we do the following:
|
||||||
|
range_end = [object.content_range&.end.to_i - 1, 0].max
|
||||||
|
|
||||||
|
object.target.highest_post_number.to_i - range_end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -116,7 +116,10 @@ class TopicSummarization
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
main_summary
|
# Calling reload here ensures Postgres' discrete range type is applied.
|
||||||
|
# an inclusive [1,2] ranges is stored as [1,3).
|
||||||
|
# Read more about this here: https://www.postgresql.org/docs/current/rangetypes.html#RANGETYPES-DISCRETE
|
||||||
|
main_summary.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
def build_sha(ids)
|
def build_sha(ids)
|
||||||
|
|
|
@ -8,7 +8,7 @@ RSpec.describe Chat::Api::SummariesController do
|
||||||
before do
|
before do
|
||||||
group.add(current_user)
|
group.add(current_user)
|
||||||
|
|
||||||
strategy = DummyCustomSummarization.new("dummy")
|
strategy = DummyCustomSummarization.new({ summary: "dummy", chunks: [] })
|
||||||
plugin.register_summarization_strategy(strategy)
|
plugin.register_summarization_strategy(strategy)
|
||||||
SiteSetting.summarization_strategy = strategy.model
|
SiteSetting.summarization_strategy = strategy.model
|
||||||
SiteSetting.custom_summarization_allowed_groups = group.id
|
SiteSetting.custom_summarization_allowed_groups = group.id
|
||||||
|
@ -18,6 +18,8 @@ RSpec.describe Chat::Api::SummariesController do
|
||||||
sign_in(current_user)
|
sign_in(current_user)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
after { DiscoursePluginRegistry.reset_register!(:summarization_strategies) }
|
||||||
|
|
||||||
describe "#get_summary" do
|
describe "#get_summary" do
|
||||||
context "when the user is not allowed to join the channel" do
|
context "when the user is not allowed to join the channel" do
|
||||||
fab!(:channel) { Fabricate(:private_category_channel) }
|
fab!(:channel) { Fabricate(:private_category_channel) }
|
||||||
|
|
|
@ -4,7 +4,9 @@ RSpec.describe Jobs::StreamTopicSummary do
|
||||||
subject(:job) { described_class.new }
|
subject(:job) { described_class.new }
|
||||||
|
|
||||||
describe "#execute" do
|
describe "#execute" do
|
||||||
fab!(:topic) { Fabricate(:topic) }
|
fab!(:topic) { Fabricate(:topic, highest_post_number: 2) }
|
||||||
|
fab!(:post_1) { Fabricate(:post, topic: topic, post_number: 1) }
|
||||||
|
fab!(:post_2) { Fabricate(:post, topic: topic, post_number: 2) }
|
||||||
let(:plugin) { Plugin::Instance.new }
|
let(:plugin) { Plugin::Instance.new }
|
||||||
let(:strategy) { DummyCustomSummarization.new({ summary: "dummy", chunks: [] }) }
|
let(:strategy) { DummyCustomSummarization.new({ summary: "dummy", chunks: [] }) }
|
||||||
fab!(:user) { Fabricate(:leader) }
|
fab!(:user) { Fabricate(:leader) }
|
||||||
|
@ -16,6 +18,8 @@ RSpec.describe Jobs::StreamTopicSummary do
|
||||||
SiteSetting.summarization_strategy = strategy.model
|
SiteSetting.summarization_strategy = strategy.model
|
||||||
end
|
end
|
||||||
|
|
||||||
|
after { DiscoursePluginRegistry.reset_register!(:summarization_strategies) }
|
||||||
|
|
||||||
describe "validates params" do
|
describe "validates params" do
|
||||||
it "does nothing if there is no topic" do
|
it "does nothing if there is no topic" do
|
||||||
messages =
|
messages =
|
||||||
|
|
|
@ -10,11 +10,13 @@ describe Summarization::Base do
|
||||||
before do
|
before do
|
||||||
group.add(user)
|
group.add(user)
|
||||||
|
|
||||||
strategy = DummyCustomSummarization.new("dummy")
|
strategy = DummyCustomSummarization.new({ summary: "dummy", chunks: [] })
|
||||||
plugin.register_summarization_strategy(strategy)
|
plugin.register_summarization_strategy(strategy)
|
||||||
SiteSetting.summarization_strategy = strategy.model
|
SiteSetting.summarization_strategy = strategy.model
|
||||||
end
|
end
|
||||||
|
|
||||||
|
after { DiscoursePluginRegistry.reset_register!(:summarization_strategies) }
|
||||||
|
|
||||||
describe "#can_see_summary?" do
|
describe "#can_see_summary?" do
|
||||||
context "when the user cannot generate a summary" do
|
context "when the user cannot generate a summary" do
|
||||||
before { SiteSetting.custom_summarization_allowed_groups = "" }
|
before { SiteSetting.custom_summarization_allowed_groups = "" }
|
||||||
|
|
|
@ -5506,7 +5506,9 @@ RSpec.describe TopicsController do
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "#summary" do
|
describe "#summary" do
|
||||||
fab!(:topic) { Fabricate(:topic) }
|
fab!(:topic) { Fabricate(:topic, highest_post_number: 2) }
|
||||||
|
fab!(:post_1) { Fabricate(:post, topic: topic, post_number: 1) }
|
||||||
|
fab!(:post_2) { Fabricate(:post, topic: topic, post_number: 2) }
|
||||||
let(:plugin) { Plugin::Instance.new }
|
let(:plugin) { Plugin::Instance.new }
|
||||||
let(:strategy) { DummyCustomSummarization.new({ summary: "dummy", chunks: [] }) }
|
let(:strategy) { DummyCustomSummarization.new({ summary: "dummy", chunks: [] }) }
|
||||||
|
|
||||||
|
@ -5515,6 +5517,8 @@ RSpec.describe TopicsController do
|
||||||
SiteSetting.summarization_strategy = strategy.model
|
SiteSetting.summarization_strategy = strategy.model
|
||||||
end
|
end
|
||||||
|
|
||||||
|
after { DiscoursePluginRegistry.reset_register!(:summarization_strategies) }
|
||||||
|
|
||||||
context "for anons" do
|
context "for anons" do
|
||||||
it "returns a 404 if there is no cached summary" do
|
it "returns a 404 if there is no cached summary" do
|
||||||
get "/t/#{topic.id}/strategy-summary.json"
|
get "/t/#{topic.id}/strategy-summary.json"
|
||||||
|
@ -5577,6 +5581,20 @@ RSpec.describe TopicsController do
|
||||||
expect(summary["can_regenerate"]).to eq(true)
|
expect(summary["can_regenerate"]).to eq(true)
|
||||||
expect(summary["new_posts_since_summary"]).to be_zero
|
expect(summary["new_posts_since_summary"]).to be_zero
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it "signals the summary is outdated" do
|
||||||
|
get "/t/#{topic.id}/strategy-summary.json"
|
||||||
|
|
||||||
|
Fabricate(:post, topic: topic, post_number: 3)
|
||||||
|
topic.update!(highest_post_number: 3)
|
||||||
|
|
||||||
|
get "/t/#{topic.id}/strategy-summary.json"
|
||||||
|
expect(response.status).to eq(200)
|
||||||
|
summary = response.parsed_body["topic_summary"]
|
||||||
|
|
||||||
|
expect(summary["outdated"]).to eq(true)
|
||||||
|
expect(summary["new_posts_since_summary"]).to eq(1)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when the user is not a member of an allowlisted group" do
|
context "when the user is not a member of an allowlisted group" do
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
describe TopicSummarization do
|
describe TopicSummarization do
|
||||||
fab!(:topic) { Fabricate(:topic) }
|
|
||||||
fab!(:user) { Fabricate(:admin) }
|
fab!(:user) { Fabricate(:admin) }
|
||||||
fab!(:post_1) { Fabricate(:post, topic: topic) }
|
fab!(:topic) { Fabricate(:topic, highest_post_number: 2) }
|
||||||
fab!(:post_2) { Fabricate(:post, topic: topic) }
|
fab!(:post_1) { Fabricate(:post, topic: topic, post_number: 1) }
|
||||||
|
fab!(:post_2) { Fabricate(:post, topic: topic, post_number: 2) }
|
||||||
|
|
||||||
shared_examples "includes only public-visible topics" do
|
shared_examples "includes only public-visible topics" do
|
||||||
subject { described_class.new(DummyCustomSummarization.new({})) }
|
subject { described_class.new(DummyCustomSummarization.new({})) }
|
||||||
|
|
Loading…
Reference in New Issue