caffeinate 0.12.0 → 0.14.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/models/caffeinate/campaign_subscription.rb +7 -4
- data/lib/caffeinate/action_mailer/interceptor.rb +0 -1
- data/lib/caffeinate/configuration.rb +10 -1
- data/lib/caffeinate/drip.rb +11 -3
- data/lib/caffeinate/dripper/callbacks.rb +25 -3
- data/lib/caffeinate/dripper/delivery.rb +6 -1
- data/lib/caffeinate/version.rb +1 -1
- data/lib/generators/caffeinate/install_generator.rb +13 -5
- data/lib/generators/caffeinate/templates/caffeinate.rb +11 -0
- data/{db/migrate/20201124183303_create_caffeinate_campaign_subscriptions.rb → lib/generators/caffeinate/templates/migrations/create_caffeinate_campaign_subscriptions.rb.tt} +1 -1
- data/{db/migrate/20201124183102_create_caffeinate_campaigns.rb → lib/generators/caffeinate/templates/migrations/create_caffeinate_campaigns.rb.tt} +1 -2
- data/{db/migrate/20201124183419_create_caffeinate_mailings.rb → lib/generators/caffeinate/templates/migrations/create_caffeinate_mailings.rb.tt} +1 -3
- metadata +5 -5
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a8ae5ac5375188ce5a7bc8960ac7d7b392ccc7dc7de71070e88a2285fa76e7af
|
4
|
+
data.tar.gz: dd06d837c2c1477bf8e48484e028016f00203066f0b2383c6f0a7a3cb5b5af1f
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 5c71b3ba2196a9a9f7ae66074b8eeb76ad6fe1a6b2ecff521e7f92c06da411c4722e0bc8b4a49fcfc9eaf0fde921c76fd4b13fbb5c7b59cfe1ec173ea69e5b29
|
7
|
+
data.tar.gz: 498848318785a2f8f4ab669cada385760f63663b1c6bf254235a7f640c189eaa170391aff58990c05c6a4d8ee6a6dc860399ea24745431dec7dfebaa8d760533
|
@@ -33,6 +33,9 @@ module Caffeinate
|
|
33
33
|
has_one :next_caffeinate_mailing, -> { joins(:caffeinate_campaign_subscription).where(caffeinate_campaign_subscriptions: { ended_at: nil, unsubscribed_at: nil }).upcoming.unsent.order(send_at: :asc) }, class_name: '::Caffeinate::Mailing', foreign_key: :caffeinate_campaign_subscription_id
|
34
34
|
has_one :next_mailing, -> { joins(:caffeinate_campaign_subscription).where(caffeinate_campaign_subscriptions: { ended_at: nil, unsubscribed_at: nil }).upcoming.unsent.order(send_at: :asc) }, class_name: '::Caffeinate::Mailing', foreign_key: :caffeinate_campaign_subscription_id
|
35
35
|
|
36
|
+
has_one :previous_caffeinate_mailing, -> { sent.order(sent_at: :desc) }, class_name: '::Caffeinate::Mailing', foreign_key: :caffeinate_campaign_subscription_id
|
37
|
+
has_one :previous_mailing, -> { sent.order(sent_at: :desc) }, class_name: '::Caffeinate::Mailing', foreign_key: :caffeinate_campaign_subscription_id
|
38
|
+
|
36
39
|
belongs_to :caffeinate_campaign, class_name: 'Caffeinate::Campaign', foreign_key: :caffeinate_campaign_id
|
37
40
|
alias_attribute :campaign, :caffeinate_campaign
|
38
41
|
|
@@ -77,7 +80,7 @@ module Caffeinate
|
|
77
80
|
end
|
78
81
|
|
79
82
|
# Updates `ended_at` and runs `on_complete` callbacks
|
80
|
-
def end!(reason =
|
83
|
+
def end!(reason = ::Caffeinate.config.default_ended_reason)
|
81
84
|
raise ::Caffeinate::InvalidState, 'CampaignSubscription is already unsubscribed.' if unsubscribed?
|
82
85
|
|
83
86
|
update!(ended_at: ::Caffeinate.config.time_now, ended_reason: reason)
|
@@ -87,7 +90,7 @@ module Caffeinate
|
|
87
90
|
end
|
88
91
|
|
89
92
|
# Updates `ended_at` and runs `on_complete` callbacks
|
90
|
-
def end(reason =
|
93
|
+
def end(reason = ::Caffeinate.config.default_ended_reason)
|
91
94
|
return false if unsubscribed?
|
92
95
|
|
93
96
|
result = update(ended_at: ::Caffeinate.config.time_now, ended_reason: reason)
|
@@ -97,7 +100,7 @@ module Caffeinate
|
|
97
100
|
end
|
98
101
|
|
99
102
|
# Updates `unsubscribed_at` and runs `on_subscribe` callbacks
|
100
|
-
def unsubscribe!(reason =
|
103
|
+
def unsubscribe!(reason = ::Caffeinate.config.default_unsubscribe_reason)
|
101
104
|
raise ::Caffeinate::InvalidState, 'CampaignSubscription is already ended.' if ended?
|
102
105
|
|
103
106
|
update!(unsubscribed_at: ::Caffeinate.config.time_now, unsubscribe_reason: reason)
|
@@ -107,7 +110,7 @@ module Caffeinate
|
|
107
110
|
end
|
108
111
|
|
109
112
|
# Updates `unsubscribed_at` and runs `on_subscribe` callbacks
|
110
|
-
def unsubscribe(reason =
|
113
|
+
def unsubscribe(reason = ::Caffeinate.config.default_unsubscribe_reason)
|
111
114
|
return false if ended?
|
112
115
|
|
113
116
|
result = update(unsubscribed_at: ::Caffeinate.config.time_now, unsubscribe_reason: reason)
|
@@ -5,7 +5,6 @@ module Caffeinate
|
|
5
5
|
# Handles the evaluation of a drip against a mailing to determine if it ultimately gets delivered.
|
6
6
|
# Also invokes the `before_send` callbacks.
|
7
7
|
class Interceptor
|
8
|
-
# Handles `before_send` callbacks for a `Caffeinate::Dripper`
|
9
8
|
def self.delivering_email(message)
|
10
9
|
mailing = message.caffeinate_mailing
|
11
10
|
return unless mailing
|
@@ -3,15 +3,19 @@
|
|
3
3
|
module Caffeinate
|
4
4
|
# Global configuration
|
5
5
|
class Configuration
|
6
|
-
attr_accessor :now, :async_delivery, :mailing_job, :batch_size, :drippers_path, :implicit_campaigns
|
6
|
+
attr_accessor :now, :async_delivery, :deliver_later,:mailing_job, :batch_size, :drippers_path, :implicit_campaigns,
|
7
|
+
:default_ended_reason, :default_unsubscribe_reason
|
7
8
|
|
8
9
|
def initialize
|
9
10
|
@now = -> { Time.current }
|
10
11
|
@async_delivery = false
|
12
|
+
@deliver_later = false
|
11
13
|
@mailing_job = nil
|
12
14
|
@batch_size = 1_000
|
13
15
|
@drippers_path = 'app/drippers'
|
14
16
|
@implicit_campaigns = true
|
17
|
+
@default_ended_reason = nil
|
18
|
+
@default_unsubscribe_reason = nil
|
15
19
|
end
|
16
20
|
|
17
21
|
def now=(val)
|
@@ -35,6 +39,11 @@ module Caffeinate
|
|
35
39
|
@async_delivery
|
36
40
|
end
|
37
41
|
|
42
|
+
# If we should use `#deliver_later` instead of `#deliver`
|
43
|
+
def deliver_later?
|
44
|
+
@deliver_later
|
45
|
+
end
|
46
|
+
|
38
47
|
# The @mailing_job constantized. Only used if `async_delivery = true`
|
39
48
|
def mailing_job_class
|
40
49
|
@mailing_job.constantize
|
data/lib/caffeinate/drip.rb
CHANGED
@@ -67,10 +67,18 @@ module Caffeinate
|
|
67
67
|
end
|
68
68
|
|
69
69
|
# Checks if the drip is enabled
|
70
|
+
#
|
71
|
+
# This is kind of messy and could use some love.
|
72
|
+
# todo: better.
|
70
73
|
def enabled?(mailing)
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
+
catch(:abort) do
|
75
|
+
if dripper.run_callbacks(:before_drip, self, mailing)
|
76
|
+
return DripEvaluator.new(mailing).call(&@block)
|
77
|
+
else
|
78
|
+
return false
|
79
|
+
end
|
80
|
+
end
|
81
|
+
false
|
74
82
|
end
|
75
83
|
end
|
76
84
|
end
|
@@ -13,12 +13,25 @@ module Caffeinate
|
|
13
13
|
self.class.run_callbacks(name, *args)
|
14
14
|
end
|
15
15
|
|
16
|
+
def callbacks_for(name)
|
17
|
+
self.class.callbacks_for(name)
|
18
|
+
end
|
19
|
+
|
16
20
|
module ClassMethods
|
17
21
|
# :nodoc:
|
18
22
|
def run_callbacks(name, *args)
|
19
|
-
|
20
|
-
|
23
|
+
catch(:abort) do
|
24
|
+
callbacks_for(name).each do |callback|
|
25
|
+
callback.call(*args)
|
26
|
+
end
|
27
|
+
return true
|
21
28
|
end
|
29
|
+
false
|
30
|
+
end
|
31
|
+
|
32
|
+
# :nodoc:
|
33
|
+
def callbacks_for(name)
|
34
|
+
send("#{name}_blocks")
|
22
35
|
end
|
23
36
|
|
24
37
|
def before_subscribe(&block)
|
@@ -114,10 +127,19 @@ module Caffeinate
|
|
114
127
|
|
115
128
|
# Callback before a Drip has called the mailer.
|
116
129
|
#
|
117
|
-
# before_drip do |
|
130
|
+
# before_drip do |drip, mailing|
|
118
131
|
# Slack.notify(:caffeinate, "#{drip.action_name} is starting")
|
119
132
|
# end
|
120
133
|
#
|
134
|
+
# Note: If you want to bail on the mailing for some reason, you need invoke `throw(:abort)`
|
135
|
+
#
|
136
|
+
# before_drip do |drip, mailing|
|
137
|
+
# if mailing.caffeinate_campaign_subscription.subscriber.trial_ended?
|
138
|
+
# unsubscribe!("Trial ended")
|
139
|
+
# throw(:abort)
|
140
|
+
# end
|
141
|
+
# end
|
142
|
+
#
|
121
143
|
# @yield Caffeinate::Drip current drip
|
122
144
|
# @yield Caffeinate::Mailing
|
123
145
|
def before_drip(&block)
|
@@ -20,7 +20,12 @@ module Caffeinate
|
|
20
20
|
mailing.mailer_class.constantize.send(mailing.mailer_action, mailing)
|
21
21
|
end
|
22
22
|
message.caffeinate_mailing = mailing
|
23
|
-
|
23
|
+
if ::Caffeinate.config.deliver_later?
|
24
|
+
message.deliver_later
|
25
|
+
else
|
26
|
+
message.deliver
|
27
|
+
end
|
28
|
+
|
24
29
|
end
|
25
30
|
end
|
26
31
|
end
|
data/lib/caffeinate/version.rb
CHANGED
@@ -5,7 +5,6 @@ module Caffeinate
|
|
5
5
|
# Installs Caffeinate
|
6
6
|
class InstallGenerator < Rails::Generators::Base
|
7
7
|
source_root File.expand_path('templates', __dir__)
|
8
|
-
include ::Rails::Generators::Migration
|
9
8
|
|
10
9
|
desc 'Creates a Caffeinate initializer and copies migrations to your application.'
|
11
10
|
|
@@ -33,12 +32,21 @@ module Caffeinate
|
|
33
32
|
@prev_migration_nr.to_s
|
34
33
|
end
|
35
34
|
|
35
|
+
def migration_version
|
36
|
+
if rails5_and_up?
|
37
|
+
"[#{Rails::VERSION::MAJOR}.#{Rails::VERSION::MINOR}]"
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def rails5_and_up?
|
42
|
+
Rails::VERSION::MAJOR >= 5
|
43
|
+
end
|
44
|
+
|
36
45
|
# :nodoc:
|
37
46
|
def copy_migrations
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
Rake::Task['caffeinate:install:migrations'].invoke
|
47
|
+
template 'migrations/create_caffeinate_campaigns.rb', "db/migrate/#{self.class.next_migration_number("")}_create_caffeinate_campaigns.rb"
|
48
|
+
template 'migrations/create_caffeinate_campaign_subscriptions.rb', "db/migrate/#{self.class.next_migration_number("")}_create_caffeinate_campaign_subscriptions.rb"
|
49
|
+
template 'migrations/create_caffeinate_mailings.rb', "db/migrate/#{self.class.next_migration_number("")}_create_caffeinate_mailings.rb"
|
42
50
|
end
|
43
51
|
end
|
44
52
|
end
|
@@ -41,4 +41,15 @@ Caffeinate.setup do |config|
|
|
41
41
|
# config.implicit_campaigns = true
|
42
42
|
#
|
43
43
|
# config.implicit_campaigns = false
|
44
|
+
#
|
45
|
+
# == Default reasons
|
46
|
+
#
|
47
|
+
# The default unsubscribe and end reasons.
|
48
|
+
#
|
49
|
+
# Default:
|
50
|
+
# config.default_unsubscribe_reason = nil
|
51
|
+
# config.default_ended_reason = nil
|
52
|
+
#
|
53
|
+
# config.default_unsubscribe_reason = "User unsubscribed"
|
54
|
+
# config.default_ended_reason = "User ended"
|
44
55
|
end
|
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
class CreateCaffeinateCampaignSubscriptions < ActiveRecord::Migration
|
3
|
+
class CreateCaffeinateCampaignSubscriptions < ActiveRecord::Migration<%= migration_version %>
|
4
4
|
def change
|
5
5
|
drop_table :caffeinate_campaign_subscriptions if table_exists?(:caffeinate_campaign_subscriptions)
|
6
6
|
|
@@ -1,8 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
class CreateCaffeinateCampaigns < ActiveRecord::Migration
|
3
|
+
class CreateCaffeinateCampaigns < ActiveRecord::Migration<%= migration_version %>
|
4
4
|
def change
|
5
|
-
drop_table :caffeinate_campaigns if table_exists?(:caffeinate_campaigns)
|
6
5
|
create_table :caffeinate_campaigns do |t|
|
7
6
|
t.string :name, null: false
|
8
7
|
t.string :slug, null: false
|
@@ -1,9 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
class CreateCaffeinateMailings < ActiveRecord::Migration
|
3
|
+
class CreateCaffeinateMailings < ActiveRecord::Migration<%= migration_version %>
|
4
4
|
def change
|
5
|
-
drop_table :caffeinate_mailings if table_exists?(:caffeinate_mailings)
|
6
|
-
|
7
5
|
create_table :caffeinate_mailings do |t|
|
8
6
|
t.references :caffeinate_campaign_subscription, null: false, foreign_key: true, index: { name: 'index_caffeinate_mailings_on_campaign_subscription' }
|
9
7
|
t.datetime :send_at
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: caffeinate
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.14.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Josh Brody
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2021-01-
|
11
|
+
date: 2021-01-18 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rails
|
@@ -143,9 +143,6 @@ files:
|
|
143
143
|
- app/views/layouts/_caffeinate.html.erb
|
144
144
|
- config/locales/en.yml
|
145
145
|
- config/routes.rb
|
146
|
-
- db/migrate/20201124183102_create_caffeinate_campaigns.rb
|
147
|
-
- db/migrate/20201124183303_create_caffeinate_campaign_subscriptions.rb
|
148
|
-
- db/migrate/20201124183419_create_caffeinate_mailings.rb
|
149
146
|
- lib/caffeinate.rb
|
150
147
|
- lib/caffeinate/action_mailer.rb
|
151
148
|
- lib/caffeinate/action_mailer/extension.rb
|
@@ -179,6 +176,9 @@ files:
|
|
179
176
|
- lib/generators/caffeinate/templates/application_dripper.rb
|
180
177
|
- lib/generators/caffeinate/templates/caffeinate.rb
|
181
178
|
- lib/generators/caffeinate/templates/mailer.rb.tt
|
179
|
+
- lib/generators/caffeinate/templates/migrations/create_caffeinate_campaign_subscriptions.rb.tt
|
180
|
+
- lib/generators/caffeinate/templates/migrations/create_caffeinate_campaigns.rb.tt
|
181
|
+
- lib/generators/caffeinate/templates/migrations/create_caffeinate_mailings.rb.tt
|
182
182
|
- lib/generators/caffeinate/views_generator.rb
|
183
183
|
homepage: https://github.com/joshmn/caffeinate
|
184
184
|
licenses:
|