forked from bterkuile/delete_softly
-
Notifications
You must be signed in to change notification settings - Fork 0
/
delete_softly.rb
84 lines (78 loc) · 2.96 KB
/
delete_softly.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
# DeleteSoftly
# This is a gem/plugin that adds soft delete functionality to ActiveRecord.
# Take a look at the
# DeleteSoftly::ArExtender
# DeleteSoftly::ClassMethods
# DeleteSoftly::InstanceMethods
# to get a feel of what is being done. This gem works through many relations
require 'active_record'
require 'meta_where'
require 'class_methods'
require 'instance_methods'
module DeleteSoftly
module ARExtender
# Always have Model.active available. It is a good practice to use it
# when you want the active records. Even use it when no logic is in
# place yet. Now it is an alias for scoped, but can be overwritten
# for custom behaviour, for example:
# class Post < ActiveRecord::Base
# delete_softly
# has_many :comments
# def self.active
# super.where(:disabled.ne => true)
# end
# end
# class Comment < ActiveRecord::Base
# belongs_to :post
# end
# will result in:
# Post.all #=> SELECT * FROM posts WHERE deleted_at IS NULL;
# Post.active #=> SELECT * FROM posts WHERE deleted_at IS NULL AND disabled != 't';
# Comment.all #=> SELECT * FROM comments;
# Comment.active #=> SELECT * FROM comments;
def active
scoped
end
# Make the model delete softly. A deleted_at:datetime column is required
# for this to work.
# The two most important differences are that it can be enforce on a model
# or be more free.
# class Post < ActiveRecord::Base
# delete_softly
# end
# will enforce soft delete on the post model. A deleted post will never appear,
# unless the explicit with_deleted is called. When the model is:
# class Post < ActiveRecord::Base
# delete_softly :enforce => false
# end
# An object will still be available after destroy is called,
def delete_softly(options = {:enforce => :active})
# Make destroy! the old destroy
alias_method :destroy!, :destroy
include DeleteSoftly::InstanceMethods
extend DeleteSoftly::ClassMethods
# Support single argument
# delete_softly :active # Same as :enforce => :active, default behaviour
# delete_softly :enforce=> :with_deleted # Same as without argument
options = {:enforce=> options } unless options.is_a?(Hash)
if options[:enforce]
if options[:enforce].is_a?(Symbol) && respond_to?(options[:enforce])
default_scope send(options[:enforce])
else
default_scope active
end
end
end
end
end
ActiveRecord::Base.send(:extend, DeleteSoftly::ARExtender)
# Overwrite ActiveRecord::Base#default_scope
class ActiveRecord::Base
# default_scope fix discussed in ticket:
# https://rails.lighthouseapp.com/projects/8994/tickets/4583-merge-default-scopes-by-default#ticket-4583-11
def self.default_scope(options = {})
key = :"#{self}_scoped_methods"
Thread.current[key] = nil
self.default_scoping << construct_finder_arel(options, default_scoping.pop)
end
end