2019-04-29 20:27:42 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-10-02 00:01:53 -04:00
|
|
|
describe PostValidator do
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:topic) { Fabricate(:topic) }
|
2019-03-13 22:39:51 -04:00
|
|
|
let(:post) { build(:post, topic: topic) }
|
2019-10-02 00:01:53 -04:00
|
|
|
let(:validator) { PostValidator.new({}) }
|
2013-06-13 04:18:17 -04:00
|
|
|
|
2017-04-26 23:53:53 -04:00
|
|
|
context "#post_body_validator" do
|
|
|
|
it 'should not allow a post with an empty raw' do
|
2016-12-05 07:31:43 -05:00
|
|
|
post.raw = ""
|
|
|
|
validator.post_body_validator(post)
|
2017-04-26 23:53:53 -04:00
|
|
|
expect(post.errors).to_not be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
context "when empty raw can bypass validation" do
|
2019-10-02 00:01:53 -04:00
|
|
|
let(:validator) { PostValidator.new(skip_post_body: true) }
|
2017-04-26 23:53:53 -04:00
|
|
|
|
|
|
|
it "should be allowed for empty raw based on site setting" do
|
|
|
|
post.raw = ""
|
|
|
|
validator.post_body_validator(post)
|
|
|
|
expect(post.errors).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "when post's topic is a PM between a human and a non human user" do
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:robot) { Fabricate(:user, id: -3) }
|
|
|
|
fab!(:user) { Fabricate(:user) }
|
2017-04-26 23:53:53 -04:00
|
|
|
|
|
|
|
let(:topic) do
|
|
|
|
Fabricate(:private_message_topic, topic_allowed_users: [
|
|
|
|
Fabricate.build(:topic_allowed_user, user: robot),
|
|
|
|
Fabricate.build(:topic_allowed_user, user: user)
|
|
|
|
])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should allow a post with an empty raw' do
|
|
|
|
post = Fabricate.build(:post, topic: topic)
|
|
|
|
post.raw = ""
|
|
|
|
validator.post_body_validator(post)
|
|
|
|
|
|
|
|
expect(post.errors).to be_empty
|
|
|
|
end
|
2016-12-05 07:31:43 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-06-13 04:18:17 -04:00
|
|
|
context "stripped_length" do
|
|
|
|
it "adds an error for short raw" do
|
|
|
|
post.raw = "abc"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
end
|
|
|
|
|
2021-03-24 10:47:35 -04:00
|
|
|
it "counts emoji as a single character" do
|
|
|
|
post.raw = ":smiling_face_with_three_hearts:" * (SiteSetting.min_post_length - 1)
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
|
|
|
|
post = build(:post, topic: topic)
|
|
|
|
post.raw = ":smiling_face_with_three_hearts:" * SiteSetting.min_post_length
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "counts multiple characters as a single character" do
|
|
|
|
post.raw = "." * SiteSetting.min_post_length
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
|
|
|
|
post = build(:post, topic: topic)
|
|
|
|
post.raw = "," * SiteSetting.min_post_length
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
|
|
|
|
post = build(:post, topic: topic)
|
|
|
|
post.raw = "<!-- #{'very long comment' * SiteSetting.min_post_length} -->"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
end
|
|
|
|
|
2013-06-13 04:18:17 -04:00
|
|
|
it "adds no error for long raw" do
|
|
|
|
post.raw = "this is a long topic body testing 123"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(0)
|
|
|
|
end
|
2021-01-07 13:44:17 -05:00
|
|
|
|
|
|
|
it "ignores an html comment" do
|
|
|
|
post.raw = "<!-- an html comment -->abc"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "ignores multiple html comments" do
|
|
|
|
post.raw = "<!-- an html comment -->\n abc \n<!-- a comment -->"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "ignores nested html comments" do
|
|
|
|
post.raw = "<!-- <!-- an html comment --> -->"
|
|
|
|
validator.stripped_length(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
|
|
|
end
|
2013-06-13 04:18:17 -04:00
|
|
|
end
|
|
|
|
|
2013-12-19 13:45:55 -05:00
|
|
|
context "too_many_posts" do
|
|
|
|
it "should be invalid when the user has posted too much" do
|
|
|
|
post.user.expects(:posted_too_much_in_topic?).returns(true)
|
|
|
|
validator.max_posts_validator(post)
|
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
|
2013-12-20 11:29:44 -05:00
|
|
|
it "should be allowed to edit when the user has posted too much" do
|
|
|
|
post.user.stubs(:posted_too_much_in_topic?).returns(true)
|
|
|
|
post.expects(:new_record?).returns(false)
|
|
|
|
validator.max_posts_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
2013-12-19 13:45:55 -05:00
|
|
|
it "should be valid when the user hasn't posted too much" do
|
|
|
|
post.user.expects(:posted_too_much_in_topic?).returns(false)
|
|
|
|
validator.max_posts_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-02-01 10:37:49 -05:00
|
|
|
context "too_many_mentions" do
|
|
|
|
before do
|
|
|
|
SiteSetting.newuser_max_mentions_per_post = 2
|
|
|
|
SiteSetting.max_mentions_per_post = 3
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be invalid when new user exceeds max mentions limit" do
|
|
|
|
post.acting_user = build(:newuser)
|
|
|
|
post.expects(:raw_mentions).returns(['jake', 'finn', 'jake_old'])
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
|
2017-02-26 19:40:42 -05:00
|
|
|
it "should be invalid when leader user exceeds max mentions limit" do
|
2016-02-01 10:37:49 -05:00
|
|
|
post.acting_user = build(:trust_level_4)
|
|
|
|
post.expects(:raw_mentions).returns(['jake', 'finn', 'jake_old', 'jake_new'])
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid when new user does not exceed max mentions limit" do
|
|
|
|
post.acting_user = build(:newuser)
|
|
|
|
post.expects(:raw_mentions).returns(['jake', 'finn'])
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
2016-04-18 16:08:42 -04:00
|
|
|
it "should be valid when new user exceeds max mentions limit in PM" do
|
|
|
|
post.acting_user = build(:newuser)
|
|
|
|
post.topic.expects(:private_message?).returns(true)
|
|
|
|
post.expects(:raw_mentions).returns(['jake', 'finn', 'jake_old'])
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
2017-02-26 19:40:42 -05:00
|
|
|
it "should be valid when leader user does not exceed max mentions limit" do
|
2016-02-01 10:37:49 -05:00
|
|
|
post.acting_user = build(:trust_level_4)
|
|
|
|
post.expects(:raw_mentions).returns(['jake', 'finn', 'jake_old'])
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid for moderator in all cases" do
|
|
|
|
post.acting_user = build(:moderator)
|
|
|
|
post.expects(:raw_mentions).never
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid for admin in all cases" do
|
|
|
|
post.acting_user = build(:admin)
|
|
|
|
post.expects(:raw_mentions).never
|
|
|
|
validator.max_mention_validator(post)
|
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-08-07 12:08:59 -04:00
|
|
|
context "too_many_embedded_media" do
|
2018-04-05 00:21:03 -04:00
|
|
|
before do
|
2020-08-07 12:08:59 -04:00
|
|
|
SiteSetting.min_trust_to_post_embedded_media = 0
|
|
|
|
SiteSetting.newuser_max_embedded_media = 2
|
2018-04-05 00:21:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "should be invalid when new user exceeds max mentions limit" do
|
|
|
|
post.acting_user = build(:newuser)
|
2020-08-07 12:08:59 -04:00
|
|
|
post.expects(:embedded_media_count).returns(3)
|
|
|
|
validator.max_embedded_media_validator(post)
|
2018-04-05 00:21:03 -04:00
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid when new user does not exceed max mentions limit" do
|
|
|
|
post.acting_user = build(:newuser)
|
2020-08-07 12:08:59 -04:00
|
|
|
post.expects(:embedded_media_count).returns(2)
|
|
|
|
validator.max_embedded_media_validator(post)
|
2018-04-05 00:21:03 -04:00
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be invalid when user trust level is not sufficient" do
|
2020-08-07 12:08:59 -04:00
|
|
|
SiteSetting.min_trust_to_post_embedded_media = 4
|
2018-04-05 00:21:03 -04:00
|
|
|
post.acting_user = build(:leader)
|
2020-08-07 12:08:59 -04:00
|
|
|
post.expects(:embedded_media_count).returns(2)
|
|
|
|
validator.max_embedded_media_validator(post)
|
2018-04-05 00:21:03 -04:00
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid for moderator in all cases" do
|
|
|
|
post.acting_user = build(:moderator)
|
2020-08-07 12:08:59 -04:00
|
|
|
post.expects(:embedded_media_count).never
|
|
|
|
validator.max_embedded_media_validator(post)
|
2018-04-05 00:21:03 -04:00
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should be valid for admin in all cases" do
|
|
|
|
post.acting_user = build(:admin)
|
2020-08-07 12:08:59 -04:00
|
|
|
post.expects(:embedded_media_count).never
|
|
|
|
validator.max_embedded_media_validator(post)
|
2018-04-05 00:21:03 -04:00
|
|
|
expect(post.errors.count).to be(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-06-13 04:18:17 -04:00
|
|
|
context "invalid post" do
|
|
|
|
it "should be invalid" do
|
|
|
|
validator.validate(post)
|
|
|
|
expect(post.errors.count).to be > 0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-09-06 11:50:05 -04:00
|
|
|
describe "unique_post_validator" do
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:user) { Fabricate(:user, id: 999) }
|
|
|
|
fab!(:post) { Fabricate(:post, user: user, topic: topic) }
|
2019-03-13 22:39:51 -04:00
|
|
|
|
2013-09-06 11:50:05 -04:00
|
|
|
before do
|
2017-07-07 02:09:14 -04:00
|
|
|
SiteSetting.unique_posts_mins = 5
|
2019-03-13 22:39:51 -04:00
|
|
|
post.store_unique_post_key
|
|
|
|
@key = post.unique_post_key
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
2019-12-03 04:05:53 -05:00
|
|
|
Discourse.redis.del(@key)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context "post is unique" do
|
2019-03-13 22:39:51 -04:00
|
|
|
let(:new_post) do
|
|
|
|
Fabricate.build(:post, user: user, raw: "unique content", topic: topic)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "should not add an error" do
|
2019-03-13 22:39:51 -04:00
|
|
|
validator.unique_post_validator(new_post)
|
|
|
|
expect(new_post.errors.count).to eq(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should not add an error when changing an existing post' do
|
|
|
|
post.raw = "changing raw"
|
|
|
|
|
2013-09-06 11:50:05 -04:00
|
|
|
validator.unique_post_validator(post)
|
2015-01-09 11:34:37 -05:00
|
|
|
expect(post.errors.count).to eq(0)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "post is not unique" do
|
2019-03-13 22:39:51 -04:00
|
|
|
let(:new_post) do
|
|
|
|
Fabricate.build(:post, user: user, raw: post.raw, topic: topic)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "should add an error" do
|
2019-03-13 22:39:51 -04:00
|
|
|
validator.unique_post_validator(new_post)
|
2019-04-30 02:58:18 -04:00
|
|
|
expect(new_post.errors.to_hash.keys).to contain_exactly(:raw)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "should not add an error if post.skip_unique_check is true" do
|
2019-03-13 22:39:51 -04:00
|
|
|
new_post.skip_unique_check = true
|
|
|
|
validator.unique_post_validator(new_post)
|
|
|
|
expect(new_post.errors.count).to eq(0)
|
2013-09-06 11:50:05 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-14 09:48:16 -05:00
|
|
|
context "force_edit_last_validator" do
|
|
|
|
|
2019-05-06 23:12:20 -04:00
|
|
|
fab!(:user) { Fabricate(:user) }
|
|
|
|
fab!(:other_user) { Fabricate(:user) }
|
|
|
|
fab!(:topic) { Fabricate(:topic) }
|
2018-11-14 09:48:16 -05:00
|
|
|
|
|
|
|
before do
|
|
|
|
SiteSetting.max_consecutive_replies = 2
|
|
|
|
end
|
|
|
|
|
2018-12-03 11:03:11 -05:00
|
|
|
it "should always allow original poster to post" do
|
2018-12-03 05:32:29 -05:00
|
|
|
[user, user, user, other_user, user, user, user].each_with_index do |u, i|
|
2018-12-03 11:03:11 -05:00
|
|
|
post = Post.new(user: u, topic: topic, raw: "post number #{i}")
|
2018-12-03 05:32:29 -05:00
|
|
|
validator.force_edit_last_validator(post)
|
|
|
|
expect(post.errors.count).to eq(0)
|
2018-12-03 11:03:11 -05:00
|
|
|
post.save!
|
2018-12-03 05:32:29 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-14 09:48:16 -05:00
|
|
|
it "should not allow posting more than 2 consecutive replies" do
|
2019-01-17 21:18:20 -05:00
|
|
|
Post.create!(user: user, topic: topic, raw: "post number 2", post_number: 2)
|
|
|
|
Post.create!(user: user, topic: topic, raw: "post number 3", post_number: 3)
|
|
|
|
Post.create!(user: other_user, topic: topic, raw: "post number 1", post_number: 1)
|
2018-12-03 05:32:29 -05:00
|
|
|
|
2019-01-17 21:18:20 -05:00
|
|
|
post = Post.new(user: user, topic: topic, raw: "post number 4", post_number: 4)
|
2018-12-03 11:03:11 -05:00
|
|
|
validator.force_edit_last_validator(post)
|
|
|
|
expect(post.errors.count).to eq(1)
|
2018-11-14 09:48:16 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "should always allow editing" do
|
|
|
|
post = Fabricate(:post, user: user, topic: topic)
|
|
|
|
post = Fabricate(:post, user: user, topic: topic)
|
|
|
|
|
|
|
|
revisor = PostRevisor.new(post)
|
|
|
|
revisor.revise!(post.user, raw: 'hello world123456789')
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should allow posting more than 2 replies" do
|
|
|
|
3.times do
|
|
|
|
post = Fabricate(:post, user: user, topic: topic)
|
|
|
|
Fabricate(:post, user: other_user, topic: topic)
|
|
|
|
validator.force_edit_last_validator(post)
|
|
|
|
expect(post.errors.count).to eq(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-12-01 04:40:23 -05:00
|
|
|
shared_examples "almost no validations" do
|
2014-07-28 16:40:06 -04:00
|
|
|
it "skips most validations" do
|
2015-12-01 04:40:23 -05:00
|
|
|
validator.expects(:stripped_length).never
|
|
|
|
validator.expects(:raw_quality).never
|
|
|
|
validator.expects(:max_posts_validator).never
|
|
|
|
validator.expects(:max_mention_validator).never
|
2020-08-07 12:08:59 -04:00
|
|
|
validator.expects(:max_embedded_media_validator).never
|
2015-12-01 04:40:23 -05:00
|
|
|
validator.expects(:max_attachments_validator).never
|
2018-02-06 18:07:24 -05:00
|
|
|
validator.expects(:newuser_links_validator).never
|
2015-12-01 04:40:23 -05:00
|
|
|
validator.expects(:unique_post_validator).never
|
2018-11-14 09:48:16 -05:00
|
|
|
validator.expects(:force_edit_last_validator).never
|
2015-12-01 04:40:23 -05:00
|
|
|
validator.validate(post)
|
2014-07-28 16:40:06 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-12-01 04:40:23 -05:00
|
|
|
context "admin editing a static page" do
|
|
|
|
before do
|
|
|
|
post.acting_user = build(:admin)
|
2017-07-07 02:09:14 -04:00
|
|
|
SiteSetting.tos_topic_id = post.topic_id
|
2015-12-01 04:40:23 -05:00
|
|
|
end
|
2015-11-30 13:08:35 -05:00
|
|
|
|
2015-12-01 04:40:23 -05:00
|
|
|
include_examples "almost no validations"
|
|
|
|
end
|
2015-11-30 13:08:35 -05:00
|
|
|
|
2015-12-01 04:40:23 -05:00
|
|
|
context "staged user" do
|
|
|
|
before { post.acting_user = build(:user, staged: true) }
|
|
|
|
include_examples "almost no validations"
|
2015-11-30 13:08:35 -05:00
|
|
|
end
|
|
|
|
|
2013-06-13 04:18:17 -04:00
|
|
|
end
|