2019-04-30 08:27:42 +08:00
# frozen_string_literal: true
2015-10-11 17:41:23 +08:00
require 'rails_helper'
2013-02-06 03:16:51 +08:00
describe Post do
2013-06-19 15:19:42 +08:00
before { Oneboxer . stubs :onebox }
2016-01-08 18:53:52 +08:00
describe '#hidden_reasons' do
context " verify enum sequence " do
before do
@hidden_reasons = Post . hidden_reasons
end
it " 'flag_threshold_reached' should be at 1st position " do
expect ( @hidden_reasons [ :flag_threshold_reached ] ) . to eq ( 1 )
end
it " 'flagged_by_tl3_user' should be at 4th position " do
expect ( @hidden_reasons [ :flagged_by_tl3_user ] ) . to eq ( 4 )
end
end
end
describe '#types' do
context " verify enum sequence " do
before do
@types = Post . types
end
it " 'regular' should be at 1st position " do
expect ( @types [ :regular ] ) . to eq ( 1 )
end
it " 'whisper' should be at 4th position " do
expect ( @types [ :whisper ] ) . to eq ( 4 )
end
end
end
describe '#cook_methods' do
context " verify enum sequence " do
before do
@cook_methods = Post . cook_methods
end
it " 'regular' should be at 1st position " do
expect ( @cook_methods [ :regular ] ) . to eq ( 1 )
end
it " 'email' should be at 3rd position " do
expect ( @cook_methods [ :email ] ) . to eq ( 3 )
end
end
end
2013-05-11 04:58:23 +08:00
# Help us build a post with a raw body
2017-07-28 09:20:09 +08:00
def post_with_body ( body , user = nil )
2013-05-11 04:58:23 +08:00
args = post_args . merge ( raw : body )
args [ :user ] = user if user . present?
2019-05-09 11:28:28 +08:00
Fabricate . build ( :post , args )
2013-05-11 04:58:23 +08:00
end
2015-01-06 00:04:23 +08:00
it { is_expected . to validate_presence_of :raw }
2013-02-06 03:16:51 +08:00
# Min/max body lengths, respecting padding
2015-01-06 00:04:23 +08:00
it { is_expected . not_to allow_value ( " x " ) . for ( :raw ) }
it { is_expected . not_to allow_value ( " x " * ( SiteSetting . max_post_length + 1 ) ) . for ( :raw ) }
it { is_expected . not_to allow_value ( ( " " * SiteSetting . min_post_length ) + " x " ) . for ( :raw ) }
2013-02-06 03:16:51 +08:00
2015-01-06 00:04:23 +08:00
it { is_expected . to rate_limit }
2013-02-06 03:16:51 +08:00
let ( :topic ) { Fabricate ( :topic ) }
let ( :post_args ) do
2013-12-12 10:41:34 +08:00
{ user : topic . user , topic : topic }
2013-02-06 03:16:51 +08:00
end
2013-02-22 02:20:00 +08:00
describe 'scopes' do
describe '#by_newest' do
it 'returns posts ordered by created_at desc' do
2013-08-21 03:45:58 +08:00
2 . times do | t |
2019-05-09 11:28:28 +08:00
Fabricate ( :post , created_at : t . seconds . from_now )
2013-08-21 03:45:58 +08:00
end
2015-01-06 00:04:23 +08:00
expect ( Post . by_newest . first . created_at ) . to be > Post . by_newest . last . created_at
2013-02-22 02:20:00 +08:00
end
end
describe '#with_user' do
it 'gives you a user' do
2019-05-09 11:28:28 +08:00
Fabricate ( :post , user : Fabricate . build ( :user ) )
2015-01-06 00:04:23 +08:00
expect ( Post . with_user . first . user ) . to be_a User
2013-02-22 02:20:00 +08:00
end
end
end
2013-12-12 10:41:34 +08:00
describe " revisions and deleting/recovery " do
2013-03-14 00:35:55 +08:00
2013-06-14 01:41:45 +08:00
context 'a post without links' do
2019-05-09 11:28:28 +08:00
let ( :post ) { Fabricate ( :post , post_args ) }
2013-06-14 01:41:45 +08:00
before do
post . trash!
post . reload
end
2013-03-14 00:35:55 +08:00
2013-12-12 10:41:34 +08:00
it " doesn't create a new revision when deleted " do
2015-01-06 00:04:23 +08:00
expect ( post . revisions . count ) . to eq ( 0 )
2013-06-14 01:41:45 +08:00
end
describe " recovery " do
before do
post . recover!
post . reload
end
2013-12-12 10:41:34 +08:00
it " doesn't create a new revision when recovered " do
2015-01-06 00:04:23 +08:00
expect ( post . revisions . count ) . to eq ( 0 )
2013-06-14 01:41:45 +08:00
end
end
2013-03-14 00:35:55 +08:00
end
2013-06-14 01:41:45 +08:00
context 'a post with links' do
let ( :post ) { Fabricate ( :post_with_external_links ) }
2013-03-14 00:35:55 +08:00
before do
2013-06-14 01:41:45 +08:00
post . trash!
2013-03-14 00:35:55 +08:00
post . reload
end
2013-06-14 01:41:45 +08:00
describe 'recovery' do
it 'recreates the topic_link records' do
TopicLink . expects ( :extract_from ) . with ( post )
post . recover!
end
2013-03-14 00:35:55 +08:00
end
end
2019-03-08 16:48:35 +08:00
context 'a post with notices' do
let ( :post ) {
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , post_args )
2019-04-19 22:53:58 +08:00
post . custom_fields [ " notice_type " ] = Post . notices [ :returning_user ]
post . custom_fields [ " notice_args " ] = 1 . day . ago
2019-03-11 11:00:47 +08:00
post . save_custom_fields
2019-03-08 16:48:35 +08:00
post
}
describe 'recovery' do
it 'deletes notices' do
2019-03-11 17:19:58 +08:00
expect { post . trash! }
2019-03-11 11:00:47 +08:00
. to change { post . custom_fields . length } . from ( 2 ) . to ( 0 )
2019-03-08 16:48:35 +08:00
end
end
end
2013-03-14 00:35:55 +08:00
end
2013-02-26 00:42:20 +08:00
describe 'flagging helpers' do
2019-05-09 11:28:28 +08:00
fab! ( :post ) { Fabricate ( :post ) }
2019-05-07 11:12:20 +08:00
fab! ( :user ) { Fabricate ( :coding_horror ) }
fab! ( :admin ) { Fabricate ( :admin ) }
2016-03-31 01:27:34 +08:00
2018-07-27 03:12:12 +08:00
it 'is_flagged? is accurate' do
2019-01-04 01:03:01 +08:00
PostActionCreator . off_topic ( user , post )
expect ( post . reload . is_flagged? ) . to eq ( true )
2013-02-26 00:42:20 +08:00
2019-01-04 01:03:01 +08:00
PostActionDestroyer . destroy ( user , post , :off_topic )
expect ( post . reload . is_flagged? ) . to eq ( false )
2013-02-07 12:15:48 +08:00
end
2016-03-31 01:27:34 +08:00
2018-07-27 03:12:12 +08:00
it 'is_flagged? is true if flag was deferred' do
2019-01-04 01:03:01 +08:00
result = PostActionCreator . off_topic ( user , post )
result . reviewable . perform ( admin , :ignore )
expect ( post . reload . is_flagged? ) . to eq ( true )
2018-07-27 03:12:12 +08:00
end
it 'is_flagged? is true if flag was cleared' do
2019-01-04 01:03:01 +08:00
result = PostActionCreator . off_topic ( user , post )
result . reviewable . perform ( admin , :disagree )
expect ( post . reload . is_flagged? ) . to eq ( true )
2018-07-27 03:12:12 +08:00
end
2019-01-04 01:03:01 +08:00
it 'reviewable_flag is nil when ignored' do
result = PostActionCreator . spam ( user , post )
expect ( post . reviewable_flag ) . to eq ( result . reviewable )
2016-03-31 01:27:34 +08:00
2019-01-04 01:03:01 +08:00
result . reviewable . perform ( admin , :ignore )
expect ( post . reviewable_flag ) . to be_nil
2016-03-31 01:27:34 +08:00
end
2018-07-27 03:12:12 +08:00
2019-01-04 01:03:01 +08:00
it 'reviewable_flag is nil when disagreed' do
result = PostActionCreator . spam ( user , post )
expect ( post . reviewable_flag ) . to eq ( result . reviewable )
result . reviewable . perform ( admin , :disagree )
expect ( post . reload . reviewable_flag ) . to be_nil
2018-07-27 03:12:12 +08:00
end
2013-02-07 12:15:48 +08:00
end
2013-02-06 03:16:51 +08:00
describe " maximum images " do
2019-05-07 11:12:20 +08:00
fab! ( :newuser ) { Fabricate ( :user , trust_level : TrustLevel [ 0 ] ) }
2019-05-09 11:28:28 +08:00
let ( :post_no_images ) { Fabricate . build ( :post , post_args . merge ( user : newuser ) ) }
2013-05-11 04:58:23 +08:00
let ( :post_one_image ) { post_with_body ( " ![sherlock](http://bbc.co.uk/sherlock.jpg) " , newuser ) }
let ( :post_two_images ) { post_with_body ( " <img src='http://discourse.org/logo.png'> <img src='http://bbc.co.uk/sherlock.jpg'> " , newuser ) }
let ( :post_with_avatars ) { post_with_body ( '<img alt="smiley" title=":smiley:" src="/assets/emoji/smiley.png" class="avatar"> <img alt="wink" title=":wink:" src="/assets/emoji/wink.png" class="avatar">' , newuser ) }
let ( :post_with_favicon ) { post_with_body ( '<img src="/assets/favicons/wikipedia.png" class="favicon">' , newuser ) }
2018-02-21 09:41:27 +08:00
let ( :post_image_within_quote ) { post_with_body ( '[quote]<img src="coolimage.png">[/quote]' , newuser ) }
2018-03-07 00:46:12 +08:00
let ( :post_image_within_code ) { post_with_body ( '<code><img src="coolimage.png"></code>' , newuser ) }
let ( :post_image_within_pre ) { post_with_body ( '<pre><img src="coolimage.png"></pre>' , newuser ) }
2013-05-11 04:58:23 +08:00
let ( :post_with_thumbnail ) { post_with_body ( '<img src="/assets/emoji/smiley.png" class="thumbnail">' , newuser ) }
let ( :post_with_two_classy_images ) { post_with_body ( " <img src='http://discourse.org/logo.png' class='classy'> <img src='http://bbc.co.uk/sherlock.jpg' class='classy'> " , newuser ) }
2013-02-06 03:16:51 +08:00
it " returns 0 images for an empty post " do
2019-05-09 11:28:28 +08:00
expect ( Fabricate . build ( :post ) . image_count ) . to eq ( 0 )
2013-02-06 03:16:51 +08:00
end
it " finds images from markdown " do
2015-01-06 00:04:23 +08:00
expect ( post_one_image . image_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
it " finds images from HTML " do
2015-01-06 00:04:23 +08:00
expect ( post_two_images . image_count ) . to eq ( 2 )
2013-02-06 03:16:51 +08:00
end
2013-02-12 15:43:48 +08:00
it " doesn't count avatars as images " do
2015-01-06 00:04:23 +08:00
expect ( post_with_avatars . image_count ) . to eq ( 0 )
2013-02-06 03:16:51 +08:00
end
2018-02-21 09:00:06 +08:00
it " allows images by default " do
expect ( post_one_image ) . to be_valid
end
it " doesn't allow more than `min_trust_to_post_images` " do
SiteSetting . min_trust_to_post_images = 4
post_one_image . user . trust_level = 3
expect ( post_one_image ) . not_to be_valid
end
2018-03-07 00:46:12 +08:00
it " doesn't allow more than `min_trust_to_post_images` in a quote " do
2018-02-21 09:41:27 +08:00
SiteSetting . min_trust_to_post_images = 4
post_one_image . user . trust_level = 3
expect ( post_image_within_quote ) . not_to be_valid
end
2018-03-07 00:46:12 +08:00
it " doesn't allow more than `min_trust_to_post_images` in code " do
SiteSetting . min_trust_to_post_images = 4
post_one_image . user . trust_level = 3
expect ( post_image_within_code ) . not_to be_valid
end
it " doesn't allow more than `min_trust_to_post_images` in pre " do
SiteSetting . min_trust_to_post_images = 4
post_one_image . user . trust_level = 3
expect ( post_image_within_pre ) . not_to be_valid
end
2018-02-21 09:00:06 +08:00
it " doesn't allow more than `min_trust_to_post_images` " do
SiteSetting . min_trust_to_post_images = 4
post_one_image . user . trust_level = 4
expect ( post_one_image ) . to be_valid
end
2013-04-12 00:36:45 +08:00
it " doesn't count favicons as images " do
2017-10-16 07:46:01 +08:00
PrettyText . stubs ( :cook ) . returns ( post_with_favicon . raw )
2015-01-06 00:04:23 +08:00
expect ( post_with_favicon . image_count ) . to eq ( 0 )
2013-04-12 00:36:45 +08:00
end
it " doesn't count thumbnails as images " do
2017-10-16 07:46:01 +08:00
PrettyText . stubs ( :cook ) . returns ( post_with_thumbnail . raw )
2015-01-06 00:04:23 +08:00
expect ( post_with_thumbnail . image_count ) . to eq ( 0 )
2013-04-12 00:36:45 +08:00
end
2013-02-12 15:43:48 +08:00
it " doesn't count whitelisted images " do
2013-02-26 00:42:20 +08:00
Post . stubs ( :white_listed_image_classes ) . returns ( [ " classy " ] )
2017-10-16 07:46:01 +08:00
# I dislike this, but passing in a custom whitelist is hard
PrettyText . stubs ( :cook ) . returns ( post_with_two_classy_images . raw )
2015-01-06 00:04:23 +08:00
expect ( post_with_two_classy_images . image_count ) . to eq ( 0 )
2013-02-12 15:43:48 +08:00
end
2013-02-06 03:16:51 +08:00
context " validation " do
2013-03-26 00:24:46 +08:00
before do
2017-07-07 14:09:14 +08:00
SiteSetting . newuser_max_images = 1
2013-02-06 03:16:51 +08:00
end
2013-04-18 07:11:13 +08:00
context 'newuser' do
it " allows a new user to post below the limit " do
2015-01-06 00:04:23 +08:00
expect ( post_one_image ) . to be_valid
2013-03-26 00:24:46 +08:00
end
it " doesn't allow more than the maximum " do
2015-01-06 00:04:23 +08:00
expect ( post_two_images ) . not_to be_valid
2013-03-26 00:24:46 +08:00
end
2013-04-18 07:11:13 +08:00
it " doesn't allow a new user to edit their post to insert an image " do
2014-09-05 13:20:39 +08:00
post_no_images . user . trust_level = TrustLevel [ 0 ]
2013-03-26 00:24:46 +08:00
post_no_images . save
2015-01-06 00:04:23 +08:00
expect {
2017-07-28 09:20:09 +08:00
post_no_images . revise ( post_no_images . user , raw : post_two_images . raw )
2013-03-26 00:24:46 +08:00
post_no_images . reload
2015-01-06 00:04:23 +08:00
} . not_to change ( post_no_images , :raw )
2013-03-26 00:24:46 +08:00
end
2013-02-06 03:16:51 +08:00
end
2013-04-18 07:11:13 +08:00
it " allows more images from a not-new account " do
2014-09-05 13:20:39 +08:00
post_two_images . user . trust_level = TrustLevel [ 1 ]
2015-01-06 00:04:23 +08:00
expect ( post_two_images ) . to be_valid
2013-02-06 03:16:51 +08:00
end
end
end
2013-07-22 08:39:17 +08:00
describe " maximum attachments " do
2019-05-07 11:12:20 +08:00
fab! ( :newuser ) { Fabricate ( :user , trust_level : TrustLevel [ 0 ] ) }
2019-05-09 11:28:28 +08:00
let ( :post_no_attachments ) { Fabricate . build ( :post , post_args . merge ( user : newuser ) ) }
2013-07-22 08:39:17 +08:00
let ( :post_one_attachment ) { post_with_body ( '<a class="attachment" href="/uploads/default/1/2082985.txt">file.txt</a>' , newuser ) }
let ( :post_two_attachments ) { post_with_body ( '<a class="attachment" href="/uploads/default/2/20947092.log">errors.log</a> <a class="attachment" href="/uploads/default/3/283572385.3ds">model.3ds</a>' , newuser ) }
it " returns 0 attachments for an empty post " do
2019-05-09 11:28:28 +08:00
expect ( Fabricate . build ( :post ) . attachment_count ) . to eq ( 0 )
2013-07-22 08:39:17 +08:00
end
it " finds attachments from HTML " do
2015-01-06 00:04:23 +08:00
expect ( post_two_attachments . attachment_count ) . to eq ( 2 )
2013-07-22 08:39:17 +08:00
end
context " validation " do
before do
2017-07-07 14:09:14 +08:00
SiteSetting . newuser_max_attachments = 1
2013-07-22 08:39:17 +08:00
end
context 'newuser' do
it " allows a new user to post below the limit " do
2015-01-06 00:04:23 +08:00
expect ( post_one_attachment ) . to be_valid
2013-07-22 08:39:17 +08:00
end
it " doesn't allow more than the maximum " do
2015-01-06 00:04:23 +08:00
expect ( post_two_attachments ) . not_to be_valid
2013-07-22 08:39:17 +08:00
end
it " doesn't allow a new user to edit their post to insert an attachment " do
2014-09-05 13:20:39 +08:00
post_no_attachments . user . trust_level = TrustLevel [ 0 ]
2013-07-22 08:39:17 +08:00
post_no_attachments . save
2015-01-06 00:04:23 +08:00
expect {
2017-07-28 09:20:09 +08:00
post_no_attachments . revise ( post_no_attachments . user , raw : post_two_attachments . raw )
2013-07-22 08:39:17 +08:00
post_no_attachments . reload
2015-01-06 00:04:23 +08:00
} . not_to change ( post_no_attachments , :raw )
2013-07-22 08:39:17 +08:00
end
end
it " allows more attachments from a not-new account " do
2014-09-05 13:20:39 +08:00
post_two_attachments . user . trust_level = TrustLevel [ 1 ]
2015-01-06 00:04:23 +08:00
expect ( post_two_attachments ) . to be_valid
2013-07-22 08:39:17 +08:00
end
end
end
2013-05-11 04:58:23 +08:00
context " links " do
2019-05-07 11:12:20 +08:00
fab! ( :newuser ) { Fabricate ( :user , trust_level : TrustLevel [ 0 ] ) }
2013-05-11 04:58:23 +08:00
let ( :no_links ) { post_with_body ( " hello world my name is evil trout " , newuser ) }
let ( :one_link ) { post_with_body ( " [jlawr](http://www.imdb.com/name/nm2225369) " , newuser ) }
2017-07-28 09:20:09 +08:00
let ( :two_links ) { post_with_body ( " <a href='http://disneyland.disney.go.com/'>disney</a> <a href='http://reddit.com'>reddit</a> " , newuser ) }
let ( :three_links ) { post_with_body ( " http://discourse.org and http://discourse.org/another_url and http://www.imdb.com/name/nm2225369 " , newuser ) }
2013-05-11 04:58:23 +08:00
describe " raw_links " do
it " returns a blank collection for a post with no links " do
2015-01-06 00:04:23 +08:00
expect ( no_links . raw_links ) . to be_blank
2013-05-11 04:58:23 +08:00
end
it " finds a link within markdown " do
2015-01-06 00:04:23 +08:00
expect ( one_link . raw_links ) . to eq ( [ " http://www.imdb.com/name/nm2225369 " ] )
2013-05-11 04:58:23 +08:00
end
it " can find two links from html " do
2015-01-06 00:04:23 +08:00
expect ( two_links . raw_links ) . to eq ( [ " http://disneyland.disney.go.com/ " , " http://reddit.com " ] )
2013-05-11 04:58:23 +08:00
end
it " can find three links without markup " do
2015-01-06 00:04:23 +08:00
expect ( three_links . raw_links ) . to eq ( [ " http://discourse.org " , " http://discourse.org/another_url " , " http://www.imdb.com/name/nm2225369 " ] )
2013-05-11 04:58:23 +08:00
end
end
describe " linked_hosts " do
it " returns blank with no links " do
2015-01-06 00:04:23 +08:00
expect ( no_links . linked_hosts ) . to be_blank
2013-05-11 04:58:23 +08:00
end
it " returns the host and a count for links " do
2017-07-28 09:20:09 +08:00
expect ( two_links . linked_hosts ) . to eq ( " disneyland.disney.go.com " = > 1 , " reddit.com " = > 1 )
2013-05-11 04:58:23 +08:00
end
it " it counts properly with more than one link on the same host " do
2017-07-28 09:20:09 +08:00
expect ( three_links . linked_hosts ) . to eq ( " discourse.org " = > 1 , " www.imdb.com " = > 1 )
2013-05-11 04:58:23 +08:00
end
end
describe " total host usage " do
it " has none for a regular post " do
2015-01-06 00:04:23 +08:00
expect ( no_links . total_hosts_usage ) . to be_blank
2013-05-11 04:58:23 +08:00
end
context " with a previous host " do
let ( :another_disney_link ) { post_with_body ( " [radiator springs](http://disneyland.disney.go.com/disney-california-adventure/radiator-springs-racers/) " , newuser ) }
before do
another_disney_link . save
TopicLink . extract_from ( another_disney_link )
end
it " contains the new post's links, PLUS the previous one " do
2017-07-28 09:20:09 +08:00
expect ( two_links . total_hosts_usage ) . to eq ( 'disneyland.disney.go.com' = > 2 , 'reddit.com' = > 1 )
2013-05-11 04:58:23 +08:00
end
end
end
end
2018-02-07 07:07:24 +08:00
describe " maximums " do
2019-05-07 11:12:20 +08:00
fab! ( :newuser ) { Fabricate ( :user , trust_level : TrustLevel [ 0 ] ) }
2013-05-11 04:58:23 +08:00
let ( :post_one_link ) { post_with_body ( " [sherlock](http://www.bbc.co.uk/programmes/b018ttws) " , newuser ) }
2018-02-09 07:26:56 +08:00
let ( :post_onebox ) { post_with_body ( " http://www.google.com " , newuser ) }
2018-03-07 00:46:12 +08:00
let ( :post_code_link ) { post_with_body ( " <code>http://www.google.com</code> " , newuser ) }
2013-05-11 04:58:23 +08:00
let ( :post_two_links ) { post_with_body ( " <a href='http://discourse.org'>discourse</a> <a href='http://twitter.com'>twitter</a> " , newuser ) }
let ( :post_with_mentions ) { post_with_body ( " hello @ #{ newuser . username } how are you doing? " , newuser ) }
2013-02-06 03:16:51 +08:00
2013-03-26 01:12:15 +08:00
it " returns 0 links for an empty post " do
2019-05-09 11:28:28 +08:00
expect ( Fabricate . build ( :post ) . link_count ) . to eq ( 0 )
2013-02-06 03:16:51 +08:00
end
2013-03-26 01:12:15 +08:00
it " returns 0 links for a post with mentions " do
2015-01-06 00:04:23 +08:00
expect ( post_with_mentions . link_count ) . to eq ( 0 )
2013-03-26 01:12:15 +08:00
end
it " finds links from markdown " do
2015-01-06 00:04:23 +08:00
expect ( post_one_link . link_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
2013-03-26 01:12:15 +08:00
it " finds links from HTML " do
2015-01-06 00:04:23 +08:00
expect ( post_two_links . link_count ) . to eq ( 2 )
2013-02-06 03:16:51 +08:00
end
context " validation " do
2013-03-26 00:24:46 +08:00
before do
2017-07-07 14:09:14 +08:00
SiteSetting . newuser_max_links = 1
2013-02-06 03:16:51 +08:00
end
2013-04-18 07:11:13 +08:00
context 'newuser' do
2013-03-26 00:24:46 +08:00
it " returns true when within the amount of links allowed " do
2015-01-06 00:04:23 +08:00
expect ( post_one_link ) . to be_valid
2013-03-26 00:24:46 +08:00
end
it " doesn't allow more links than allowed " do
2015-01-06 00:04:23 +08:00
expect ( post_two_links ) . not_to be_valid
2013-03-26 00:24:46 +08:00
end
2013-02-06 03:16:51 +08:00
end
2018-02-07 07:07:24 +08:00
it " allows multiple links for basic accounts " do
2014-09-05 13:20:39 +08:00
post_two_links . user . trust_level = TrustLevel [ 1 ]
2015-01-06 00:04:23 +08:00
expect ( post_two_links ) . to be_valid
2013-02-06 03:16:51 +08:00
end
2013-03-26 00:24:46 +08:00
2018-02-09 07:26:56 +08:00
context " min_trust_to_post_links " do
it " considers oneboxes links " do
SiteSetting . min_trust_to_post_links = 3
post_onebox . user . trust_level = TrustLevel [ 2 ]
expect ( post_onebox ) . not_to be_valid
end
2018-03-07 00:46:12 +08:00
it " considers links within code " do
SiteSetting . min_trust_to_post_links = 3
post_onebox . user . trust_level = TrustLevel [ 2 ]
expect ( post_code_link ) . not_to be_valid
end
2018-02-09 07:26:56 +08:00
it " doesn't allow allow links if `min_trust_to_post_links` is not met " do
SiteSetting . min_trust_to_post_links = 2
post_two_links . user . trust_level = TrustLevel [ 1 ]
expect ( post_one_link ) . not_to be_valid
end
2018-06-14 02:57:32 +08:00
it " will skip the check for whitelisted domains " do
SiteSetting . whitelisted_link_domains = 'www.bbc.co.uk'
SiteSetting . min_trust_to_post_links = 2
post_two_links . user . trust_level = TrustLevel [ 1 ]
expect ( post_one_link ) . to be_valid
end
2018-02-07 07:07:24 +08:00
end
2013-02-06 03:16:51 +08:00
end
end
2013-03-26 00:24:46 +08:00
describe " @mentions " do
2013-02-06 03:16:51 +08:00
context 'raw_mentions' do
it " returns an empty array with no matches " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " Hello Jake and Finn! " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ ] )
2013-02-06 03:16:51 +08:00
end
it " returns lowercase unique versions of the mentions " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " @Jake @Finn @Jake " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'jake' , 'finn' ] )
2013-02-06 03:16:51 +08:00
end
it " ignores pre " do
2017-07-14 20:27:28 +08:00
# we need to force an inline
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " p <pre>@Jake</pre> @Finn " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'finn' ] )
2013-02-06 03:16:51 +08:00
end
it " catches content between pre tags " do
2017-07-14 20:27:28 +08:00
# per common mark we need to force an inline
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " a <pre>hello</pre> @Finn <pre></pre> " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'finn' ] )
2013-02-06 03:16:51 +08:00
end
it " ignores code " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " @Jake `@Finn` " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'jake' ] )
2013-02-26 00:42:20 +08:00
end
2013-02-06 03:16:51 +08:00
it " ignores quotes " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " [quote= \" Evil Trout \" ] \n @Jake \n [/quote] \n @Finn " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'finn' ] )
2013-02-26 00:42:20 +08:00
end
2013-02-06 03:16:51 +08:00
2013-03-24 08:57:00 +08:00
it " handles underscore in username " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " @Jake @Finn @Jake_Old " ) )
2015-01-06 00:04:23 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'jake' , 'finn' , 'jake_old' ] )
2013-03-24 08:57:00 +08:00
end
2016-02-24 03:57:54 +08:00
it " handles hyphen in groupname " do
2019-05-09 11:28:28 +08:00
post = Fabricate . build ( :post , post_args . merge ( raw : " @org-board " ) )
2016-02-24 03:57:54 +08:00
expect ( post . raw_mentions ) . to eq ( [ 'org-board' ] )
end
2013-02-06 03:16:51 +08:00
end
2013-03-26 00:24:46 +08:00
context " max mentions " do
2019-05-07 11:12:20 +08:00
fab! ( :newuser ) { Fabricate ( :user , trust_level : TrustLevel [ 0 ] ) }
2013-05-11 04:58:23 +08:00
let ( :post_with_one_mention ) { post_with_body ( " @Jake is the person I'm mentioning " , newuser ) }
let ( :post_with_two_mentions ) { post_with_body ( " @Jake @Finn are the people I'm mentioning " , newuser ) }
2013-03-26 00:24:46 +08:00
2013-04-18 07:11:13 +08:00
context 'new user' do
2013-03-26 00:24:46 +08:00
before do
2017-07-07 14:09:14 +08:00
SiteSetting . newuser_max_mentions_per_post = 1
SiteSetting . max_mentions_per_post = 5
2013-03-26 00:24:46 +08:00
end
2013-04-18 07:11:13 +08:00
it " allows a new user to have newuser_max_mentions_per_post mentions " do
2015-01-06 00:04:23 +08:00
expect ( post_with_one_mention ) . to be_valid
2013-03-26 00:24:46 +08:00
end
2013-04-18 07:11:13 +08:00
it " doesn't allow a new user to have more than newuser_max_mentions_per_post mentions " do
2015-01-06 00:04:23 +08:00
expect ( post_with_two_mentions ) . not_to be_valid
2013-03-26 00:24:46 +08:00
end
2013-02-06 03:16:51 +08:00
end
2013-04-18 07:11:13 +08:00
context " not a new user " do
2013-03-26 00:24:46 +08:00
before do
2017-07-07 14:09:14 +08:00
SiteSetting . newuser_max_mentions_per_post = 0
SiteSetting . max_mentions_per_post = 1
2013-03-26 00:24:46 +08:00
end
it " allows vmax_mentions_per_post mentions " do
2014-09-05 13:20:39 +08:00
post_with_one_mention . user . trust_level = TrustLevel [ 1 ]
2015-01-06 00:04:23 +08:00
expect ( post_with_one_mention ) . to be_valid
2013-03-26 00:24:46 +08:00
end
it " doesn't allow to have more than max_mentions_per_post mentions " do
2014-09-05 13:20:39 +08:00
post_with_two_mentions . user . trust_level = TrustLevel [ 1 ]
2015-01-06 00:04:23 +08:00
expect ( post_with_two_mentions ) . not_to be_valid
2013-03-26 00:24:46 +08:00
end
2013-02-06 03:16:51 +08:00
end
2013-03-26 00:24:46 +08:00
2013-02-06 03:16:51 +08:00
end
end
2013-06-13 16:18:17 +08:00
context 'validation' do
it 'validates our default post' do
2019-05-09 11:28:28 +08:00
expect ( Fabricate . build ( :post , post_args ) ) . to be_valid
2013-06-13 16:18:17 +08:00
end
2017-01-16 13:24:47 +08:00
it 'create blank posts as invalid' do
2019-05-09 11:28:28 +08:00
expect ( Fabricate . build ( :post , raw : " " ) ) . not_to be_valid
2013-06-13 16:18:17 +08:00
end
2013-02-06 03:16:51 +08:00
end
context " raw_hash " do
2017-07-28 09:20:09 +08:00
let ( :raw ) { " this is our test post body " }
2013-05-11 04:58:23 +08:00
let ( :post ) { post_with_body ( raw ) }
2013-02-26 00:42:20 +08:00
2013-02-06 03:16:51 +08:00
it " returns a value " do
2015-01-06 00:04:23 +08:00
expect ( post . raw_hash ) . to be_present
2013-02-06 03:16:51 +08:00
end
it " returns blank for a nil body " do
post . raw = nil
2015-01-06 00:04:23 +08:00
expect ( post . raw_hash ) . to be_blank
2013-02-06 03:16:51 +08:00
end
it " returns the same value for the same raw " do
2015-01-06 00:04:23 +08:00
expect ( post . raw_hash ) . to eq ( post_with_body ( raw ) . raw_hash )
2013-02-06 03:16:51 +08:00
end
it " returns a different value for a different raw " do
2015-01-06 00:04:23 +08:00
expect ( post . raw_hash ) . not_to eq ( post_with_body ( " something else " ) . raw_hash )
2013-02-06 03:16:51 +08:00
end
2013-06-18 03:57:13 +08:00
it " returns a different value with different text case " do
2015-01-06 00:04:23 +08:00
expect ( post . raw_hash ) . not_to eq ( post_with_body ( " THIS is OUR TEST post BODy " ) . raw_hash )
2013-02-06 03:16:51 +08:00
end
end
context 'revise' do
2019-05-09 11:28:28 +08:00
let ( :post ) { Fabricate ( :post , post_args ) }
2013-02-06 03:16:51 +08:00
let ( :first_version_at ) { post . last_version_at }
2013-12-12 10:41:34 +08:00
it 'has no revision' do
2015-01-06 00:04:23 +08:00
expect ( post . revisions . size ) . to eq ( 0 )
expect ( first_version_at ) . to be_present
2017-07-28 09:20:09 +08:00
expect ( post . revise ( post . user , raw : post . raw ) ) . to eq ( false )
2013-02-06 03:16:51 +08:00
end
2013-04-22 15:45:03 +08:00
describe 'with the same body' do
2013-12-12 10:41:34 +08:00
it " doesn't change version " do
2017-07-28 09:20:09 +08:00
expect { post . revise ( post . user , raw : post . raw ) ; post . reload } . not_to change ( post , :version )
2013-02-06 03:16:51 +08:00
end
end
2014-10-28 05:06:43 +08:00
describe 'ninja editing & edit windows' do
2013-02-06 03:16:51 +08:00
2017-07-07 14:09:14 +08:00
before { SiteSetting . editing_grace_period = 1 . minute . to_i }
2013-04-30 20:54:30 +08:00
2014-10-28 05:06:43 +08:00
it 'works' do
revised_at = post . updated_at + 2 . minutes
new_revised_at = revised_at + 2 . minutes
2013-02-06 03:16:51 +08:00
2014-10-28 05:06:43 +08:00
# ninja edit
post . revise ( post . user , { raw : 'updated body' } , revised_at : post . updated_at + 10 . seconds )
2013-02-06 03:16:51 +08:00
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . version ) . to eq ( 1 )
expect ( post . public_version ) . to eq ( 1 )
expect ( post . revisions . size ) . to eq ( 0 )
expect ( post . last_version_at . to_i ) . to eq ( first_version_at . to_i )
2013-02-06 03:16:51 +08:00
2014-10-28 05:06:43 +08:00
# revision much later
post . revise ( post . user , { raw : 'another updated body' } , revised_at : revised_at )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . version ) . to eq ( 2 )
expect ( post . public_version ) . to eq ( 2 )
expect ( post . revisions . size ) . to eq ( 1 )
expect ( post . last_version_at . to_i ) . to eq ( revised_at . to_i )
2013-02-06 03:16:51 +08:00
2014-10-28 05:06:43 +08:00
# new edit window
post . revise ( post . user , { raw : 'yet another updated body' } , revised_at : revised_at + 10 . seconds )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . version ) . to eq ( 2 )
expect ( post . public_version ) . to eq ( 2 )
expect ( post . revisions . size ) . to eq ( 1 )
expect ( post . last_version_at . to_i ) . to eq ( revised_at . to_i )
2013-02-06 03:16:51 +08:00
2014-10-28 05:06:43 +08:00
# after second window
post . revise ( post . user , { raw : 'yet another, another updated body' } , revised_at : new_revised_at )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . version ) . to eq ( 3 )
expect ( post . public_version ) . to eq ( 3 )
expect ( post . revisions . size ) . to eq ( 2 )
expect ( post . last_version_at . to_i ) . to eq ( new_revised_at . to_i )
2013-02-06 03:16:51 +08:00
end
end
describe 'rate limiter' do
let ( :changed_by ) { Fabricate ( :coding_horror ) }
2013-02-26 00:42:20 +08:00
2013-02-06 03:16:51 +08:00
it " triggers a rate limiter " do
2013-02-09 23:33:07 +08:00
EditRateLimiter . any_instance . expects ( :performed! )
2017-07-28 09:20:09 +08:00
post . revise ( changed_by , raw : 'updated body' )
2013-02-06 03:16:51 +08:00
end
end
describe 'with a new body' do
let ( :changed_by ) { Fabricate ( :coding_horror ) }
2017-07-28 09:20:09 +08:00
let! ( :result ) { post . revise ( changed_by , raw : 'updated body' ) }
2013-02-06 03:16:51 +08:00
2013-04-22 15:45:03 +08:00
it 'acts correctly' do
2015-01-06 00:04:23 +08:00
expect ( result ) . to eq ( true )
expect ( post . raw ) . to eq ( 'updated body' )
expect ( post . invalidate_oneboxes ) . to eq ( true )
expect ( post . version ) . to eq ( 2 )
expect ( post . public_version ) . to eq ( 2 )
expect ( post . revisions . size ) . to eq ( 1 )
expect ( post . revisions . first . user ) . to be_present
2013-02-06 03:16:51 +08:00
end
context 'second poster posts again quickly' do
2014-10-28 05:06:43 +08:00
it 'is a ninja edit, because the second poster posted again quickly' do
2018-03-07 13:44:21 +08:00
SiteSetting . editing_grace_period = 1 . minute . to_i
2014-10-28 05:06:43 +08:00
post . revise ( changed_by , { raw : 'yet another updated body' } , revised_at : post . updated_at + 10 . seconds )
2013-02-06 03:16:51 +08:00
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . version ) . to eq ( 2 )
expect ( post . public_version ) . to eq ( 2 )
expect ( post . revisions . size ) . to eq ( 1 )
2013-02-26 00:42:20 +08:00
end
2013-02-06 03:16:51 +08:00
end
end
2014-10-28 05:06:43 +08:00
2013-02-06 03:16:51 +08:00
end
2016-10-24 12:02:38 +08:00
describe 'before save' do
let ( :cooked ) { " <p><div class= \" lightbox-wrapper \" ><a data-download-href= \" //localhost:3000/uploads/default/34784374092783e2fef84b8bc96d9b54c11ceea0 \" href= \" //localhost:3000/uploads/default/original/1X/34784374092783e2fef84b8bc96d9b54c11ceea0.gif \" class= \" lightbox \" title= \" Sword reworks.gif \" ><img src= \" //localhost:3000/uploads/default/optimized/1X/34784374092783e2fef84b8bc96d9b54c11ceea0_1_690x276.gif \" width= \" 690 \" height= \" 276 \" ><div class= \" meta \" > \n <span class= \" filename \" >Sword reworks.gif</span><span class= \" informations \" >1000x400 1000 KB</span><span class= \" expand \" ></span> \n </div></a></div></p> " }
let ( :post ) do
2019-05-09 11:28:28 +08:00
Fabricate ( :post ,
2016-10-24 12:02:38 +08:00
raw : " <img src= \" /uploads/default/original/1X/34784374092783e2fef84b8bc96d9b54c11ceea0.gif \" width= \" 690 \" height= \" 276 \" > " ,
cooked : cooked
)
end
it 'should not cook the post if raw has not been changed' do
post . save!
expect ( post . cooked ) . to eq ( cooked )
end
end
2013-02-06 03:16:51 +08:00
describe 'after save' do
2019-05-09 11:28:28 +08:00
let ( :post ) { Fabricate ( :post , post_args ) }
2013-02-06 03:16:51 +08:00
2013-04-22 15:45:03 +08:00
it " has correct info set " do
2015-01-06 00:04:23 +08:00
expect ( post . user_deleted? ) . to eq ( false )
expect ( post . post_number ) . to be_present
expect ( post . excerpt ) . to be_present
expect ( post . post_type ) . to eq ( Post . types [ :regular ] )
expect ( post . revisions ) . to be_blank
expect ( post . cooked ) . to be_present
expect ( post . external_id ) . to be_present
expect ( post . quote_count ) . to eq ( 0 )
expect ( post . replies ) . to be_blank
2013-02-06 03:16:51 +08:00
end
2013-03-19 03:54:08 +08:00
describe 'extract_quoted_post_numbers' do
2013-02-06 03:16:51 +08:00
2019-05-09 11:28:28 +08:00
let! ( :post ) { Fabricate ( :post , post_args ) }
let ( :reply ) { Fabricate . build ( :post , post_args ) }
2013-02-06 03:16:51 +08:00
it " finds the quote when in the same topic " do
reply . raw = " [quote= \" EvilTrout, post: #{ post . post_number } , topic: #{ post . topic_id } \" ]hello[/quote] "
reply . extract_quoted_post_numbers
2015-01-06 00:04:23 +08:00
expect ( reply . quoted_post_numbers ) . to eq ( [ post . post_number ] )
2013-02-06 03:16:51 +08:00
end
it " doesn't find the quote in a different topic " do
2017-07-28 09:20:09 +08:00
reply . raw = " [quote= \" EvilTrout, post: #{ post . post_number } , topic: #{ post . topic_id + 1 } \" ]hello[/quote] "
2013-02-06 03:16:51 +08:00
reply . extract_quoted_post_numbers
2015-01-06 00:04:23 +08:00
expect ( reply . quoted_post_numbers ) . to be_blank
2013-02-06 03:16:51 +08:00
end
2018-07-10 16:17:28 +08:00
it " doesn't find the quote in the same post " do
2019-05-09 11:28:28 +08:00
reply = Fabricate . build ( :post , post_args . merge ( post_number : 646 ) )
2018-07-10 16:17:28 +08:00
reply . raw = " [quote= \" EvilTrout, post: #{ reply . post_number } , topic: #{ post . topic_id } \" ]hello[/quote] "
reply . extract_quoted_post_numbers
expect ( reply . quoted_post_numbers ) . to be_blank
end
2013-02-06 03:16:51 +08:00
end
describe 'a new reply' do
2019-05-07 11:12:20 +08:00
fab! ( :topic ) { Fabricate ( :topic ) }
2013-03-19 03:54:08 +08:00
let ( :other_user ) { Fabricate ( :coding_horror ) }
2017-07-28 09:20:09 +08:00
let ( :reply_text ) { " [quote= \" Evil Trout, post:1 \" ] \n hello \n [/quote] \n Hmmm! " }
2019-05-09 11:28:28 +08:00
let! ( :post ) { PostCreator . new ( topic . user , raw : Fabricate . build ( :post ) . raw , topic_id : topic . id ) . create }
2017-07-28 09:20:09 +08:00
let! ( :reply ) { PostCreator . new ( other_user , raw : reply_text , topic_id : topic . id , reply_to_post_number : post . post_number ) . create }
2013-02-06 03:16:51 +08:00
it 'has a quote' do
2015-01-06 00:04:23 +08:00
expect ( reply . quote_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
it 'has a reply to the user of the original user' do
2015-01-06 00:04:23 +08:00
expect ( reply . reply_to_user ) . to eq ( post . user )
2013-02-06 03:16:51 +08:00
end
it 'increases the reply count of the parent' do
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . reply_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
it 'increases the reply count of the topic' do
topic . reload
2015-01-06 00:04:23 +08:00
expect ( topic . reply_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
it 'is the child of the parent post' do
2015-01-06 00:04:23 +08:00
expect ( post . replies ) . to eq ( [ reply ] )
2013-02-06 03:16:51 +08:00
end
it " doesn't change the post count when you edit the reply " do
reply . raw = 'updated raw'
reply . save
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . reply_count ) . to eq ( 1 )
2013-02-06 03:16:51 +08:00
end
context 'a multi-quote reply' do
2013-03-19 03:54:08 +08:00
let! ( :multi_reply ) do
raw = " [quote= \" Evil Trout, post:1 \" ]post1 quote[/quote] \n Aha! \n [quote= \" Evil Trout, post:2 \" ]post2 quote[/quote] \n Neat-o "
PostCreator . new ( other_user , raw : raw , topic_id : topic . id , reply_to_post_number : post . post_number ) . create
end
2013-02-06 03:16:51 +08:00
2013-04-22 15:45:03 +08:00
it 'has the correct info set' do
2015-01-06 00:04:23 +08:00
expect ( multi_reply . quote_count ) . to eq ( 2 )
expect ( post . replies . include? ( multi_reply ) ) . to eq ( true )
expect ( reply . replies . include? ( multi_reply ) ) . to eq ( true )
2013-02-06 03:16:51 +08:00
end
end
end
end
2013-11-19 01:48:26 +08:00
context 'summary' do
2019-05-09 11:28:28 +08:00
let! ( :p1 ) { Fabricate ( :post , post_args . merge ( score : 4 , percent_rank : 0 . 33 ) ) }
let! ( :p2 ) { Fabricate ( :post , post_args . merge ( score : 10 , percent_rank : 0 . 66 ) ) }
let! ( :p3 ) { Fabricate ( :post , post_args . merge ( score : 5 , percent_rank : 0 . 99 ) ) }
fab! ( :p4 ) { Fabricate ( :post , percent_rank : 0 . 99 ) }
2013-02-06 03:16:51 +08:00
2013-11-19 01:48:26 +08:00
it " returns the OP and posts above the threshold in summary mode " do
2017-07-07 14:09:14 +08:00
SiteSetting . summary_percent_filter = 66
2018-06-21 12:00:54 +08:00
expect ( Post . summary ( topic . id ) . order ( :post_number ) ) . to eq ( [ p1 , p2 ] )
expect ( Post . summary ( p4 . topic . id ) ) . to eq ( [ p4 ] )
2013-02-06 03:16:51 +08:00
end
end
context 'sort_order' do
context 'regular topic' do
2019-05-09 11:28:28 +08:00
let! ( :p1 ) { Fabricate ( :post , post_args ) }
let! ( :p2 ) { Fabricate ( :post , post_args ) }
let! ( :p3 ) { Fabricate ( :post , post_args ) }
2013-02-06 03:16:51 +08:00
it 'defaults to created order' do
2015-01-06 00:04:23 +08:00
expect ( Post . regular_order ) . to eq ( [ p1 , p2 , p3 ] )
2013-02-06 03:16:51 +08:00
end
end
2013-08-07 05:42:36 +08:00
end
context " reply_history " do
2019-05-09 11:28:28 +08:00
let! ( :p1 ) { Fabricate ( :post , post_args ) }
let! ( :p2 ) { Fabricate ( :post , post_args . merge ( reply_to_post_number : p1 . post_number ) ) }
let! ( :p3 ) { Fabricate ( :post , post_args ) }
let! ( :p4 ) { Fabricate ( :post , post_args . merge ( reply_to_post_number : p2 . post_number ) ) }
2013-08-07 05:42:36 +08:00
it " returns the posts in reply to this post " do
2015-01-06 00:04:23 +08:00
expect ( p4 . reply_history ) . to eq ( [ p1 , p2 ] )
expect ( p4 . reply_history ( 1 ) ) . to eq ( [ p2 ] )
expect ( p3 . reply_history ) . to be_blank
expect ( p2 . reply_history ) . to eq ( [ p1 ] )
2013-08-07 05:42:36 +08:00
end
2013-02-06 03:16:51 +08:00
end
2017-12-14 05:12:06 +08:00
context " reply_ids " do
2019-05-07 11:12:20 +08:00
fab! ( :topic ) { Fabricate ( :topic ) }
2019-05-09 11:28:28 +08:00
let! ( :p1 ) { Fabricate ( :post , topic : topic , post_number : 1 ) }
let! ( :p2 ) { Fabricate ( :post , topic : topic , post_number : 2 , reply_to_post_number : 1 ) }
let! ( :p3 ) { Fabricate ( :post , topic : topic , post_number : 3 ) }
let! ( :p4 ) { Fabricate ( :post , topic : topic , post_number : 4 , reply_to_post_number : 2 ) }
let! ( :p5 ) { Fabricate ( :post , topic : topic , post_number : 5 , reply_to_post_number : 4 ) }
let! ( :p6 ) { Fabricate ( :post , topic : topic , post_number : 6 ) }
2017-12-14 05:12:06 +08:00
before {
PostReply . create! ( post : p1 , reply : p2 )
PostReply . create! ( post : p2 , reply : p4 )
2017-12-14 07:43:48 +08:00
PostReply . create! ( post : p2 , reply : p6 ) # simulates p6 quoting p2
PostReply . create! ( post : p3 , reply : p5 ) # simulates p5 quoting p3
2017-12-14 05:12:06 +08:00
PostReply . create! ( post : p4 , reply : p5 )
2017-12-15 07:38:14 +08:00
PostReply . create! ( post : p6 , reply : p6 ) # https://meta.discourse.org/t/topic-quoting-itself-displays-reply-indicator/76085
2017-12-14 05:12:06 +08:00
}
it " returns the reply ids and their level " do
2017-12-14 07:43:48 +08:00
expect ( p1 . reply_ids ) . to eq ( [ { id : p2 . id , level : 1 } , { id : p4 . id , level : 2 } , { id : p6 . id , level : 2 } ] )
expect ( p2 . reply_ids ) . to eq ( [ { id : p4 . id , level : 1 } , { id : p6 . id , level : 1 } ] )
expect ( p3 . reply_ids ) . to be_empty # has no replies
expect ( p4 . reply_ids ) . to be_empty # p5 replies to 2 posts (p4 and p3)
expect ( p5 . reply_ids ) . to be_empty # has no replies
2017-12-15 07:38:14 +08:00
expect ( p6 . reply_ids ) . to be_empty # quotes itself
2017-12-14 05:12:06 +08:00
end
2018-04-21 05:05:51 +08:00
it " does not skip any replies " do
expect ( p1 . reply_ids ( only_replies_to_single_post : false ) ) . to eq ( [ { id : p2 . id , level : 1 } , { id : p4 . id , level : 2 } , { id : p5 . id , level : 3 } , { id : p6 . id , level : 2 } ] )
expect ( p2 . reply_ids ( only_replies_to_single_post : false ) ) . to eq ( [ { id : p4 . id , level : 1 } , { id : p5 . id , level : 2 } , { id : p6 . id , level : 1 } ] )
expect ( p3 . reply_ids ( only_replies_to_single_post : false ) ) . to eq ( [ { id : p5 . id , level : 1 } ] )
expect ( p4 . reply_ids ( only_replies_to_single_post : false ) ) . to eq ( [ { id : p5 . id , level : 1 } ] )
expect ( p5 . reply_ids ( only_replies_to_single_post : false ) ) . to be_empty # has no replies
expect ( p6 . reply_ids ( only_replies_to_single_post : false ) ) . to be_empty # quotes itself
end
2017-12-14 05:12:06 +08:00
end
2013-04-22 15:45:03 +08:00
describe 'urls' do
2013-05-11 04:58:23 +08:00
it 'no-ops for empty list' do
2015-01-06 00:04:23 +08:00
expect ( Post . urls ( [ ] ) ) . to eq ( { } )
2013-04-22 15:45:03 +08:00
end
2013-05-11 04:58:23 +08:00
# integration test -> should move to centralized integration test
it 'finds urls for posts presented' do
2019-05-09 11:28:28 +08:00
p1 = Fabricate ( :post )
p2 = Fabricate ( :post )
2017-07-28 09:20:09 +08:00
expect ( Post . urls ( [ p1 . id , p2 . id ] ) ) . to eq ( p1 . id = > p1 . url , p2 . id = > p2 . url )
2013-04-22 15:45:03 +08:00
end
end
2013-10-15 22:21:30 +08:00
describe " details " do
it " adds details " do
post = Fabricate . build ( :post )
post . add_detail ( " key " , " value " )
2015-01-06 00:04:23 +08:00
expect ( post . post_details . size ) . to eq ( 1 )
expect ( post . post_details . first . key ) . to eq ( " key " )
expect ( post . post_details . first . value ) . to eq ( " value " )
2013-10-15 22:21:30 +08:00
end
it " can find a post by a detail " do
detail = Fabricate ( :post_detail )
post = detail . post
2015-01-06 00:04:23 +08:00
expect ( Post . find_by_detail ( detail . key , detail . value ) . id ) . to eq ( post . id )
2013-10-15 22:21:30 +08:00
end
end
2014-01-16 00:34:17 +08:00
describe " cooking " do
let ( :post ) { Fabricate . build ( :post , post_args . merge ( raw : " please read my blog http://blog.example.com " ) ) }
2018-09-17 10:02:20 +08:00
it " should unconditionally follow links for staff " do
SiteSetting . tl3_links_no_follow = true
post . user . trust_level = 1
post . user . moderator = true
post . save
expect ( post . cooked ) . not_to match ( / nofollow / )
end
2014-01-16 00:34:17 +08:00
it " should add nofollow to links in the post for trust levels below 3 " do
post . user . trust_level = 2
post . save
2016-11-20 20:49:14 +08:00
expect ( post . cooked ) . to match ( / nofollow noopener / )
2014-01-16 00:34:17 +08:00
end
2014-09-05 04:16:46 +08:00
it " when tl3_links_no_follow is false, should not add nofollow for trust level 3 and higher " do
2017-07-07 14:09:14 +08:00
SiteSetting . tl3_links_no_follow = false
2014-01-16 00:34:17 +08:00
post . user . trust_level = 3
post . save
2015-01-06 00:04:23 +08:00
expect ( post . cooked ) . not_to match ( / nofollow / )
2014-01-16 00:34:17 +08:00
end
2014-07-15 01:34:21 +08:00
2014-09-05 04:16:46 +08:00
it " when tl3_links_no_follow is true, should add nofollow for trust level 3 and higher " do
2017-07-07 14:09:14 +08:00
SiteSetting . tl3_links_no_follow = true
2014-07-15 01:34:21 +08:00
post . user . trust_level = 3
post . save
2016-11-20 20:49:14 +08:00
expect ( post . cooked ) . to match ( / nofollow noopener / )
2014-07-15 01:34:21 +08:00
end
2018-11-22 16:01:03 +08:00
describe 'mentions' do
2019-05-07 11:12:20 +08:00
fab! ( :group ) do
2018-11-22 16:01:03 +08:00
Fabricate ( :group ,
mentionable_level : Group :: ALIAS_LEVELS [ :members_mods_and_admins ]
)
end
2018-11-26 12:57:07 +08:00
before do
2019-03-14 22:47:38 +08:00
Jobs . run_immediately!
2018-11-26 12:57:07 +08:00
end
2018-11-22 16:01:03 +08:00
describe 'when user can not mention a group' do
it " should not create the mention " do
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : " hello @ #{ group . name } " )
2018-11-26 12:57:07 +08:00
post . trigger_post_process
post . reload
2018-11-22 16:01:03 +08:00
expect ( post . cooked ) . to eq (
%Q|<p>hello <span class="mention">@#{group.name}</span></p>|
)
end
end
describe 'when user can mention a group' do
before do
group . add ( post . user )
end
it 'should create the mention' do
post . update! ( raw : " hello @ #{ group . name } " )
2018-11-26 12:57:07 +08:00
post . trigger_post_process
post . reload
2018-11-22 16:01:03 +08:00
expect ( post . cooked ) . to eq (
%Q|<p>hello <a class="mention-group" href="/groups/#{group.name}">@#{group.name}</a></p>|
)
end
end
end
2014-01-16 00:34:17 +08:00
end
2014-02-27 12:43:45 +08:00
describe " has_host_spam " do
2018-06-20 00:15:27 +08:00
let ( :raw ) { " hello from my site http://www.example.net http:// #{ GlobalSetting . hostname } http:// #{ RailsMultisite :: ConnectionManagement . current_hostname } " }
2017-08-11 05:18:57 +08:00
2014-02-27 12:43:45 +08:00
it " correctly detects host spam " do
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : raw )
2014-02-27 12:43:45 +08:00
2018-06-20 00:15:27 +08:00
expect ( post . total_hosts_usage ) . to eq ( " www.example.net " = > 1 )
2014-02-27 12:43:45 +08:00
post . acting_user . trust_level = 0
2015-01-06 00:04:23 +08:00
expect ( post . has_host_spam? ) . to eq ( false )
2014-02-27 12:43:45 +08:00
2014-05-28 10:30:43 +08:00
SiteSetting . newuser_spam_host_threshold = 1
2014-02-27 12:43:45 +08:00
2015-01-06 00:04:23 +08:00
expect ( post . has_host_spam? ) . to eq ( true )
2014-02-27 12:43:45 +08:00
2018-06-20 00:15:27 +08:00
SiteSetting . white_listed_spam_host_domains = " bla.com|boo.com | example.net "
2015-01-06 00:04:23 +08:00
expect ( post . has_host_spam? ) . to eq ( false )
2017-08-11 05:18:57 +08:00
end
it " doesn't punish staged users " do
SiteSetting . newuser_spam_host_threshold = 1
user = Fabricate ( :user , staged : true , trust_level : 0 )
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : raw , user : user )
2017-08-11 05:18:57 +08:00
expect ( post . has_host_spam? ) . to eq ( false )
end
2018-06-20 00:41:10 +08:00
it " punishes previously staged users that were created within 1 day " do
2018-06-16 01:45:53 +08:00
SiteSetting . newuser_spam_host_threshold = 1
2018-06-19 08:05:04 +08:00
SiteSetting . newuser_max_links = 3
2018-06-16 01:45:53 +08:00
user = Fabricate ( :user , staged : true , trust_level : 0 )
2018-06-19 08:05:04 +08:00
user . created_at = 1 . hour . ago
user . unstage
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : raw , user : user )
2018-06-19 08:05:04 +08:00
expect ( post . has_host_spam? ) . to eq ( true )
end
2018-06-20 00:41:10 +08:00
it " doesn't punish previously staged users over 1 day old " do
2018-06-19 08:05:04 +08:00
SiteSetting . newuser_spam_host_threshold = 1
SiteSetting . newuser_max_links = 3
user = Fabricate ( :user , staged : true , trust_level : 0 )
2018-06-20 00:41:10 +08:00
user . created_at = 1 . day . ago
2018-06-16 01:45:53 +08:00
user . unstage
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : raw , user : user )
2018-06-16 01:45:53 +08:00
expect ( post . has_host_spam? ) . to eq ( false )
end
2017-08-11 05:18:57 +08:00
it " ignores private messages " do
SiteSetting . newuser_spam_host_threshold = 1
user = Fabricate ( :user , trust_level : 0 )
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post , raw : raw , user : user , topic : Fabricate ( :private_message_topic , user : user ) )
2017-08-11 05:18:57 +08:00
expect ( post . has_host_spam? ) . to eq ( false )
2014-02-27 12:43:45 +08:00
end
end
2014-04-25 21:14:05 +08:00
it " has custom fields " do
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post )
2015-01-06 00:04:23 +08:00
expect ( post . custom_fields [ " a " ] ) . to eq ( nil )
2014-04-25 21:14:05 +08:00
post . custom_fields [ " Tommy " ] = " Hanks "
post . custom_fields [ " Vincent " ] = " Vega "
post . save
post = Post . find ( post . id )
2017-07-28 09:20:09 +08:00
expect ( post . custom_fields ) . to eq ( " Tommy " = > " Hanks " , " Vincent " = > " Vega " )
2014-04-25 21:14:05 +08:00
end
2014-05-28 10:30:43 +08:00
describe " # rebake! " do
it " will rebake a post correctly " do
post = create_post
2015-01-06 00:04:23 +08:00
expect ( post . baked_at ) . not_to eq ( nil )
2014-05-28 10:30:43 +08:00
first_baked = post . baked_at
first_cooked = post . cooked
2018-06-19 14:13:14 +08:00
DB . exec ( " UPDATE posts SET cooked = 'frogs' WHERE id = ? " , [ post . id ] )
2014-05-28 10:30:43 +08:00
post . reload
2014-09-23 00:55:13 +08:00
post . expects ( :publish_change_to_clients! ) . with ( :rebaked )
2014-05-28 10:30:43 +08:00
result = post . rebake!
2015-01-06 00:04:23 +08:00
expect ( post . baked_at ) . not_to eq ( first_baked )
expect ( post . cooked ) . to eq ( first_cooked )
expect ( result ) . to eq ( true )
2014-05-28 10:30:43 +08:00
end
end
2016-08-20 03:27:12 +08:00
describe " # set_owner " do
2019-05-09 11:28:28 +08:00
fab! ( :post ) { Fabricate ( :post ) }
2019-05-07 11:12:20 +08:00
fab! ( :coding_horror ) { Fabricate ( :coding_horror ) }
2016-08-20 03:27:12 +08:00
it " will change owner of a post correctly " do
post . set_owner ( coding_horror , Discourse . system_user )
post . reload
expect ( post . user ) . to eq ( coding_horror )
expect ( post . revisions . size ) . to eq ( 1 )
end
it " skips creating new post revision if skip_revision is true " do
post . set_owner ( coding_horror , Discourse . system_user , true )
post . reload
expect ( post . user ) . to eq ( coding_horror )
expect ( post . revisions . size ) . to eq ( 0 )
end
2017-09-14 22:15:07 +08:00
it " uses default locale for edit reason " do
I18n . locale = 'de'
post . set_owner ( coding_horror , Discourse . system_user )
post . reload
expected_reason = I18n . with_locale ( SiteSetting . default_locale ) do
2018-08-20 18:26:19 +08:00
I18n . t ( 'change_owner.post_revision_text' )
2017-09-14 22:15:07 +08:00
end
expect ( post . edit_reason ) . to eq ( expected_reason )
end
2016-08-20 03:27:12 +08:00
end
2014-05-28 10:30:43 +08:00
describe " .rebake_old " do
it " will catch posts it needs to rebake " do
post = create_post
2017-07-28 09:20:09 +08:00
post . update_columns ( baked_at : Time . new ( 2000 , 1 , 1 ) , baked_version : - 1 )
2014-05-28 10:30:43 +08:00
Post . rebake_old ( 100 )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . baked_at ) . to be > 1 . day . ago
2014-05-30 12:45:39 +08:00
baked = post . baked_at
Post . rebake_old ( 100 )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . baked_at ) . to eq ( baked )
2014-05-28 10:30:43 +08:00
end
2019-01-04 06:24:46 +08:00
it " will rate limit globally " do
post1 = create_post
post2 = create_post
post3 = create_post
Post . where ( id : [ post1 . id , post2 . id , post3 . id ] ) . update_all ( baked_version : - 1 )
global_setting :max_old_rebakes_per_15_minutes , 2
RateLimiter . clear_all_global!
RateLimiter . enable
Post . rebake_old ( 100 )
expect ( post3 . reload . baked_version ) . not_to eq ( - 1 )
expect ( post2 . reload . baked_version ) . not_to eq ( - 1 )
expect ( post1 . reload . baked_version ) . to eq ( - 1 )
end
2014-05-28 10:30:43 +08:00
end
2014-08-11 16:48:00 +08:00
describe " .unhide! " do
2017-07-07 14:09:14 +08:00
before { SiteSetting . unique_posts_mins = 5 }
2014-08-11 16:48:00 +08:00
2014-11-12 23:34:30 +08:00
it " will unhide the first post & make the topic visible " do
hidden_topic = Fabricate ( :topic , visible : false )
post = create_post ( topic : hidden_topic )
2014-08-11 16:48:00 +08:00
post . update_columns ( hidden : true , hidden_at : Time . now , hidden_reason_id : 1 )
post . reload
2015-01-06 00:04:23 +08:00
expect ( post . hidden ) . to eq ( true )
2014-08-11 16:48:00 +08:00
2014-09-23 00:55:13 +08:00
post . expects ( :publish_change_to_clients! ) . with ( :acted )
2014-08-11 16:48:00 +08:00
post . unhide!
2014-11-12 23:34:30 +08:00
2014-08-11 16:48:00 +08:00
post . reload
2014-11-12 23:34:30 +08:00
hidden_topic . reload
2014-08-11 16:48:00 +08:00
2015-01-06 00:04:23 +08:00
expect ( post . hidden ) . to eq ( false )
expect ( hidden_topic . visible ) . to eq ( true )
2014-08-11 16:48:00 +08:00
end
end
2014-11-12 23:34:30 +08:00
it " will unhide the post but will keep the topic invisible/unlisted " do
hidden_topic = Fabricate ( :topic , visible : false )
2017-03-30 00:24:46 +08:00
create_post ( topic : hidden_topic )
2014-11-12 23:34:30 +08:00
second_post = create_post ( topic : hidden_topic )
second_post . update_columns ( hidden : true , hidden_at : Time . now , hidden_reason_id : 1 )
second_post . expects ( :publish_change_to_clients! ) . with ( :acted )
second_post . unhide!
second_post . reload
hidden_topic . reload
2015-01-06 00:04:23 +08:00
expect ( second_post . hidden ) . to eq ( false )
expect ( hidden_topic . visible ) . to eq ( false )
2014-11-12 23:34:30 +08:00
end
2018-05-29 07:34:12 +08:00
it " automatically orders post revisions by number ascending " do
2019-05-09 11:28:28 +08:00
post = Fabricate ( :post )
2018-05-29 07:34:12 +08:00
post . revisions . create! ( user_id : 1 , post_id : post . id , number : 2 )
post . revisions . create! ( user_id : 1 , post_id : post . id , number : 1 )
expect ( post . revisions . pluck ( :number ) ) . to eq ( [ 1 , 2 ] )
end
2018-09-06 09:58:01 +08:00
describe '#link_post_uploads' do
2019-05-29 09:00:25 +08:00
fab! ( :video_upload ) { Fabricate ( :upload , extension : " mp4 " ) }
fab! ( :image_upload ) { Fabricate ( :upload ) }
fab! ( :audio_upload ) { Fabricate ( :upload , extension : " ogg " ) }
fab! ( :attachment_upload ) { Fabricate ( :upload , extension : " csv " ) }
fab! ( :attachment_upload_2 ) { Fabricate ( :upload ) }
2018-09-06 09:58:01 +08:00
2019-05-04 03:46:20 +08:00
let ( :base_url ) { " #{ Discourse . base_url_no_prefix } #{ Discourse . base_uri } " }
let ( :video_url ) { " #{ base_url } #{ video_upload . url } " }
let ( :audio_url ) { " #{ base_url } #{ audio_upload . url } " }
2018-09-06 09:58:01 +08:00
let ( :raw ) do
<< ~ RAW
< a href = " #{ attachment_upload . url } " > Link < / a>
2019-05-29 09:00:25 +08:00
[ test | attachment ] ( #{attachment_upload_2.short_url})
2018-09-06 09:58:01 +08:00
< img src = " #{ image_upload . url } " >
< video width = " 100% " height = " 100% " controls >
2019-05-04 03:46:20 +08:00
< source src = " #{ video_url } " >
< a href = " #{ video_url } " > #{video_url}</a>
2018-09-06 09:58:01 +08:00
< / video>
< audio controls >
2019-05-04 03:46:20 +08:00
< source src = " #{ audio_url } " >
< a href = " #{ audio_url } " > #{audio_url}</a>
2018-09-06 09:58:01 +08:00
< / audio>
RAW
end
2019-05-09 11:28:28 +08:00
let ( :post ) { Fabricate ( :post , raw : raw ) }
2018-09-06 09:58:01 +08:00
it " finds all the uploads in the post " do
post . custom_fields [ Post :: DOWNLOADED_IMAGES ] = {
" /uploads/default/original/1X/1/1234567890123456.csv " : attachment_upload . id
}
post . save_custom_fields
post . link_post_uploads
expect ( PostUpload . where ( post : post ) . pluck ( :upload_id ) ) . to contain_exactly (
2019-05-29 09:00:25 +08:00
video_upload . id ,
image_upload . id ,
audio_upload . id ,
attachment_upload . id ,
attachment_upload_2 . id
2018-09-06 09:58:01 +08:00
)
end
it " cleans the reverse index up for the current post " do
post . link_post_uploads
post_uploads_ids = post . post_uploads . pluck ( :id )
post . link_post_uploads
expect ( post . reload . post_uploads . pluck ( :id ) ) . to_not contain_exactly (
post_uploads_ids
)
end
end
2019-03-28 14:28:01 +08:00
context 'topic updated_at' do
let :topic do
create_post . topic
end
def updates_topic_updated_at
freeze_time 1 . day . from_now
time = Time . now
result = yield
topic . reload
expect ( topic . updated_at ) . to eq_time ( time )
result
end
it " will update topic updated_at for all topic related events " do
SiteSetting . enable_whispers = true
post = updates_topic_updated_at do
create_post ( topic_id : topic . id , post_type : Post . types [ :whisper ] )
end
updates_topic_updated_at do
PostDestroyer . new ( Discourse . system_user , post ) . destroy
end
updates_topic_updated_at do
PostDestroyer . new ( Discourse . system_user , post ) . recover
end
end
end
2019-04-15 04:24:55 +08:00
context " have_uploads " do
it " should find all posts with the upload " do
ids = [ ]
2019-05-09 11:28:28 +08:00
ids << Fabricate ( :post , cooked : " A post with upload <img src='/uploads/default/1/defghijklmno.png'> " ) . id
ids << Fabricate ( :post , cooked : " A post with optimized image <img src='/uploads/default/_optimized/601/961/defghijklmno.png'> " ) . id
Fabricate ( :post )
ids << Fabricate ( :post , cooked : " A post with upload <img src='/uploads/default/original/1X/abc/defghijklmno.png'> " ) . id
ids << Fabricate ( :post , cooked : " A post with upload link <a href='https://cdn.example.com/original/1X/abc/defghijklmno.png'> " ) . id
ids << Fabricate ( :post , cooked : " A post with optimized image <img src='https://cdn.example.com/bucket/optimized/1X/abc/defghijklmno.png'> " ) . id
Fabricate ( :post , cooked : " A post with external link <a href='https://example.com/wp-content/uploads/abcdef.gif'> " )
2019-05-16 19:09:38 +08:00
ids << Fabricate ( :post , cooked : 'A post with missing upload <img src="https://cdn.example.com/images/transparent.png" data-orig-src="upload://defghijklmno.png">' ) . id
2019-04-15 04:24:55 +08:00
expect ( Post . have_uploads . order ( :id ) . pluck ( :id ) ) . to eq ( ids )
end
end
2019-06-11 22:25:02 +08:00
describe '#each_upload_url' do
let ( :upload ) { Fabricate ( :upload_s3 ) }
it " correctly identifies all upload urls " do
urls = [ ]
upload1 = Fabricate ( :upload )
upload2 = Fabricate ( :upload )
2019-06-11 22:45:40 +08:00
post = Fabricate ( :post , raw : " A post with image and link upload. \n \n ![]( #{ upload1 . short_url } ) \n \n <a href=' #{ upload2 . url } '>Link to upload</a> \n ![](http://example.com/external.png) " )
2019-06-11 22:25:02 +08:00
post . each_upload_url { | src , _ , _ | urls << src }
expect ( urls ) . to eq ( [ upload1 . url , upload2 . url ] )
end
it " should skip external urls with upload url in query string " do
SiteSetting . enable_s3_uploads = true
SiteSetting . s3_upload_bucket = " s3-upload-bucket "
SiteSetting . s3_access_key_id = " some key "
SiteSetting . s3_secret_access_key = " some secret key "
SiteSetting . s3_cdn_url = " https://cdn.s3.amazonaws.com "
urls = [ ]
upload = Fabricate ( :upload_s3 )
post = Fabricate ( :post , raw : " <a href='https://link.example.com/redirect?url= #{ Discourse . store . cdn_url ( upload . url ) } '>Link to upload</a> " )
post . each_upload_url { | src , _ , _ | urls << src }
expect ( urls ) . to be_empty
end
end
2013-02-06 03:16:51 +08:00
end