2013-04-12 04:04:20 +08:00
|
|
|
require 'spec_helper'
|
|
|
|
require_dependency 'user_destroyer'
|
|
|
|
|
|
|
|
describe UserDestroyer do
|
2013-04-16 02:52:07 +08:00
|
|
|
|
|
|
|
before do
|
|
|
|
RestClient.stubs(:delete).returns( {success: 'OK'}.to_json )
|
|
|
|
end
|
|
|
|
|
2013-04-12 04:04:20 +08:00
|
|
|
describe 'new' do
|
|
|
|
it 'raises an error when user is nil' do
|
|
|
|
expect { UserDestroyer.new(nil) }.to raise_error(Discourse::InvalidParameters)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises an error when user is not a User' do
|
|
|
|
expect { UserDestroyer.new(5) }.to raise_error(Discourse::InvalidParameters)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'destroy' do
|
|
|
|
before do
|
|
|
|
@admin = Fabricate(:admin)
|
|
|
|
@user = Fabricate(:user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises an error when user is nil' do
|
|
|
|
expect { UserDestroyer.new(@admin).destroy(nil) }.to raise_error(Discourse::InvalidParameters)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises an error when user is not a User' do
|
|
|
|
expect { UserDestroyer.new(@admin).destroy('nothing') }.to raise_error(Discourse::InvalidParameters)
|
|
|
|
end
|
|
|
|
|
2014-02-14 00:42:35 +08:00
|
|
|
it 'raises an error when regular user tries to delete another user' do
|
|
|
|
expect { UserDestroyer.new(@user).destroy(Fabricate(:user)) }.to raise_error(Discourse::InvalidAccess)
|
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
shared_examples "successfully destroy a user" do
|
|
|
|
it 'should delete the user' do
|
|
|
|
expect { destroy }.to change { User.count }.by(-1)
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
it 'should return the deleted user record' do
|
|
|
|
return_value = destroy
|
|
|
|
return_value.should == @user
|
|
|
|
return_value.should be_destroyed
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
it 'should log the action' do
|
2013-07-26 06:04:51 +08:00
|
|
|
StaffActionLogger.any_instance.expects(:log_user_deletion).with(@user, anything).once
|
2013-07-25 01:48:55 +08:00
|
|
|
destroy
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-26 03:30:03 +08:00
|
|
|
shared_examples "email block list" do
|
|
|
|
it "doesn't add email to block list by default" do
|
2013-08-14 23:05:53 +08:00
|
|
|
ScreenedEmail.expects(:block).never
|
2013-07-26 03:30:03 +08:00
|
|
|
destroy
|
|
|
|
end
|
|
|
|
|
|
|
|
it "adds email to block list if block_email is true" do
|
2013-08-14 23:05:53 +08:00
|
|
|
b = Fabricate.build(:screened_email, email: @user.email)
|
2013-08-23 07:04:17 +08:00
|
|
|
ScreenedEmail.expects(:block).with(@user.email, has_key(:ip_address)).returns(b)
|
2013-07-26 03:30:03 +08:00
|
|
|
b.expects(:record_match!).once.returns(true)
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, destroy_opts.merge({block_email: true}))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-02-14 00:42:35 +08:00
|
|
|
context 'user deletes self' do
|
|
|
|
let(:destroy_opts) { {delete_posts: true} }
|
|
|
|
subject(:destroy) { UserDestroyer.new(@user).destroy(@user, destroy_opts) }
|
|
|
|
|
|
|
|
include_examples "successfully destroy a user"
|
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
context 'user has posts' do
|
2013-09-05 03:35:10 +08:00
|
|
|
let!(:topic_starter) { Fabricate(:user) }
|
|
|
|
let!(:topic) { Fabricate(:topic, user: topic_starter) }
|
|
|
|
let!(:first_post) { Fabricate(:post, user: topic_starter, topic: topic) }
|
|
|
|
let!(:post) { Fabricate(:post, user: @user, topic: topic) }
|
2013-04-12 04:04:20 +08:00
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
context "delete_posts is false" do
|
|
|
|
subject(:destroy) { UserDestroyer.new(@admin).destroy(@user) }
|
2014-07-29 01:17:37 +08:00
|
|
|
before do
|
|
|
|
@user.stubs(:post_count).returns(1)
|
|
|
|
@user.stubs(:first_post_created_at).returns(Time.zone.now)
|
|
|
|
end
|
2013-07-25 01:48:55 +08:00
|
|
|
|
|
|
|
it 'should not delete the user' do
|
|
|
|
expect { destroy rescue nil }.to_not change { User.count }
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
it 'should raise an error' do
|
|
|
|
expect { destroy }.to raise_error( UserDestroyer::PostsExistError )
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
2013-04-16 02:52:07 +08:00
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
it 'should not log the action' do
|
|
|
|
StaffActionLogger.any_instance.expects(:log_user_deletion).never
|
|
|
|
destroy rescue nil
|
2013-04-16 02:52:07 +08:00
|
|
|
end
|
2013-07-25 01:48:55 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
context "delete_posts is true" do
|
2013-07-26 03:30:03 +08:00
|
|
|
let(:destroy_opts) { {delete_posts: true} }
|
2013-07-25 01:48:55 +08:00
|
|
|
|
2014-02-14 00:42:35 +08:00
|
|
|
context "staff deletes user" do
|
|
|
|
subject(:destroy) { UserDestroyer.new(@admin).destroy(@user, destroy_opts) }
|
|
|
|
|
|
|
|
include_examples "successfully destroy a user"
|
|
|
|
include_examples "email block list"
|
|
|
|
|
|
|
|
it "deletes the posts" do
|
|
|
|
destroy
|
2014-09-25 23:44:48 +08:00
|
|
|
post.reload.deleted_at.should_not == nil
|
|
|
|
post.user_id.should == nil
|
2014-02-14 00:42:35 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
it "does not delete topics started by others in which the user has replies" do
|
|
|
|
destroy
|
2014-09-25 23:44:48 +08:00
|
|
|
topic.reload.deleted_at.should == nil
|
|
|
|
topic.user_id.should_not == nil
|
2014-02-14 00:42:35 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
it "deletes topics started by the deleted user" do
|
|
|
|
spammer_topic = Fabricate(:topic, user: @user)
|
|
|
|
spammer_post = Fabricate(:post, user: @user, topic: spammer_topic)
|
|
|
|
destroy
|
2014-09-25 23:44:48 +08:00
|
|
|
spammer_topic.reload.deleted_at.should_not == nil
|
|
|
|
spammer_topic.user_id.should == nil
|
2014-02-14 00:42:35 +08:00
|
|
|
end
|
2014-10-20 22:59:06 +08:00
|
|
|
|
|
|
|
context "delete_as_spammer is true" do
|
|
|
|
|
|
|
|
before { destroy_opts[:delete_as_spammer] = true }
|
|
|
|
|
|
|
|
it "agrees with flags on user's posts" do
|
|
|
|
spammer_post = Fabricate(:post, user: @user)
|
|
|
|
flag = PostAction.act(@admin, spammer_post, PostActionType.types[:inappropriate])
|
|
|
|
flag.agreed_at.should == nil
|
|
|
|
|
|
|
|
destroy
|
|
|
|
|
|
|
|
flag.reload
|
|
|
|
flag.agreed_at.should_not == nil
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
2013-04-29 23:38:43 +08:00
|
|
|
end
|
2013-09-05 03:35:10 +08:00
|
|
|
|
2014-02-14 00:42:35 +08:00
|
|
|
context "users deletes self" do
|
|
|
|
subject(:destroy) { UserDestroyer.new(@user).destroy(@user, destroy_opts) }
|
|
|
|
|
|
|
|
include_examples "successfully destroy a user"
|
|
|
|
include_examples "email block list"
|
2013-09-05 03:35:10 +08:00
|
|
|
|
2014-02-14 00:42:35 +08:00
|
|
|
it "deletes the posts" do
|
|
|
|
destroy
|
2014-09-25 23:44:48 +08:00
|
|
|
post.reload.deleted_at.should_not == nil
|
|
|
|
post.user_id.should == nil
|
2014-02-14 00:42:35 +08:00
|
|
|
end
|
2013-09-05 03:35:10 +08:00
|
|
|
end
|
2013-07-25 01:48:55 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-08-19 00:07:21 +08:00
|
|
|
context 'user has no posts, but user_stats table has post_count > 0' do
|
|
|
|
before do
|
|
|
|
# out of sync user_stat data shouldn't break UserDestroyer
|
|
|
|
@user.user_stat.update_attribute(:post_count, 1)
|
|
|
|
end
|
|
|
|
subject(:destroy) { UserDestroyer.new(@user).destroy(@user, {delete_posts: false}) }
|
|
|
|
|
|
|
|
include_examples "successfully destroy a user"
|
|
|
|
end
|
|
|
|
|
2013-09-04 05:19:29 +08:00
|
|
|
context 'user has deleted posts' do
|
|
|
|
let!(:deleted_post) { Fabricate(:post, user: @user, deleted_at: 1.hour.ago) }
|
|
|
|
it "should mark the user's deleted posts as belonging to a nuked user" do
|
|
|
|
expect { UserDestroyer.new(@admin).destroy(@user) }.to change { User.count }.by(-1)
|
2014-09-25 23:44:48 +08:00
|
|
|
deleted_post.reload.user_id.should == nil
|
2013-09-04 05:19:29 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-25 01:48:55 +08:00
|
|
|
context 'user has no posts' do
|
|
|
|
context 'and destroy succeeds' do
|
2013-07-26 03:30:03 +08:00
|
|
|
let(:destroy_opts) { {} }
|
2013-07-25 01:48:55 +08:00
|
|
|
subject(:destroy) { UserDestroyer.new(@admin).destroy(@user) }
|
|
|
|
|
|
|
|
include_examples "successfully destroy a user"
|
2013-07-26 03:30:03 +08:00
|
|
|
include_examples "email block list"
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'and destroy fails' do
|
2013-07-25 01:48:55 +08:00
|
|
|
subject(:destroy) { UserDestroyer.new(@admin).destroy(@user) }
|
|
|
|
|
2013-04-12 04:04:20 +08:00
|
|
|
before do
|
|
|
|
@user.stubs(:destroy).returns(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should return false' do
|
|
|
|
destroy.should == false
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should not log the action' do
|
2013-07-24 05:58:26 +08:00
|
|
|
StaffActionLogger.any_instance.expects(:log_user_deletion).never
|
2013-04-12 04:04:20 +08:00
|
|
|
destroy
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-08-14 23:05:53 +08:00
|
|
|
|
|
|
|
context 'user has posts with links' do
|
|
|
|
context 'external links' do
|
|
|
|
before do
|
|
|
|
@post = Fabricate(:post_with_external_links, user: @user)
|
|
|
|
TopicLink.extract_from(@post)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "doesn't add ScreenedUrl records by default" do
|
|
|
|
ScreenedUrl.expects(:watch).never
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true})
|
|
|
|
end
|
|
|
|
|
|
|
|
it "adds ScreenedUrl records when :block_urls is true" do
|
2013-08-23 07:04:17 +08:00
|
|
|
ScreenedUrl.expects(:watch).with(anything, anything, has_key(:ip_address)).at_least_once
|
2013-08-14 23:05:53 +08:00
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true, block_urls: true})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'internal links' do
|
|
|
|
before do
|
|
|
|
@post = Fabricate(:post_with_external_links, user: @user)
|
|
|
|
TopicLink.extract_from(@post)
|
|
|
|
TopicLink.any_instance.stubs(:internal).returns(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "doesn't add ScreenedUrl records" do
|
|
|
|
ScreenedUrl.expects(:watch).never
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true, block_urls: true})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with oneboxed links' do
|
|
|
|
before do
|
|
|
|
@post = Fabricate(:post_with_youtube, user: @user)
|
|
|
|
TopicLink.extract_from(@post)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "doesn't add ScreenedUrl records" do
|
|
|
|
ScreenedUrl.expects(:watch).never
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true, block_urls: true})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-10-22 02:49:51 +08:00
|
|
|
|
|
|
|
context 'ip address screening' do
|
|
|
|
it "doesn't create screened_ip_address records by default" do
|
|
|
|
ScreenedIpAddress.expects(:watch).never
|
|
|
|
UserDestroyer.new(@admin).destroy(@user)
|
|
|
|
end
|
|
|
|
|
2014-04-30 02:37:56 +08:00
|
|
|
context "block_ip is true" do
|
|
|
|
it "creates a new screened_ip_address record" do
|
|
|
|
ScreenedIpAddress.expects(:watch).with(@user.ip_address).returns(stub_everything)
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {block_ip: true})
|
|
|
|
end
|
|
|
|
|
|
|
|
it "creates two new screened_ip_address records when registration_ip_address is different than last ip_address" do
|
|
|
|
@user.registration_ip_address = '12.12.12.12'
|
|
|
|
ScreenedIpAddress.expects(:watch).with(@user.ip_address).returns(stub_everything)
|
|
|
|
ScreenedIpAddress.expects(:watch).with(@user.registration_ip_address).returns(stub_everything)
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {block_ip: true})
|
|
|
|
end
|
2013-10-22 02:49:51 +08:00
|
|
|
end
|
|
|
|
end
|
2013-11-02 04:55:56 +08:00
|
|
|
|
|
|
|
context 'user created a category' do
|
|
|
|
let!(:category) { Fabricate(:category, user: @user) }
|
|
|
|
|
|
|
|
it "assigns the system user to the categories" do
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true})
|
|
|
|
category.reload.user_id.should == Discourse.system_user.id
|
|
|
|
category.topic.should be_present
|
|
|
|
category.topic.user_id.should == Discourse.system_user.id
|
|
|
|
end
|
|
|
|
end
|
2014-04-01 02:06:25 +08:00
|
|
|
|
|
|
|
context 'user liked things' do
|
|
|
|
before do
|
|
|
|
@topic = Fabricate(:topic, user: Fabricate(:user))
|
|
|
|
@post = Fabricate(:post, user: @topic.user, topic: @topic)
|
|
|
|
@like = PostAction.act(@user, @post, PostActionType.types[:like])
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'should destroy the like' do
|
|
|
|
expect {
|
|
|
|
UserDestroyer.new(@admin).destroy(@user, {delete_posts: true})
|
|
|
|
}.to change { PostAction.count }.by(-1)
|
|
|
|
@post.reload.like_count.should == 0
|
|
|
|
end
|
|
|
|
end
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|