Commit 17538ffe authored by Nicolas Dular's avatar Nicolas Dular Committed by Bob Van Landuyt

Record cta click for in-product marketing emails

parent bb89a15b
...@@ -17,16 +17,19 @@ class Groups::EmailCampaignsController < Groups::ApplicationController ...@@ -17,16 +17,19 @@ class Groups::EmailCampaignsController < Groups::ApplicationController
private private
def track_click def track_click
data = { if Gitlab.com?
namespace_id: group.id, data = {
track: @track.to_s, namespace_id: group.id,
series: @series, track: @track.to_s,
subject_line: subject_line(@track, @series) series: @series,
} subject_line: subject_line(@track, @series)
}
context = SnowplowTracker::SelfDescribingJson.new(EMAIL_CAMPAIGNS_SCHEMA_URL, data)
context = SnowplowTracker::SelfDescribingJson.new(EMAIL_CAMPAIGNS_SCHEMA_URL, data) ::Gitlab::Tracking.event(self.class.name, 'click', context: [context])
else
::Gitlab::Tracking.event(self.class.name, 'click', context: [context]) ::Users::InProductMarketingEmail.save_cta_click(current_user, @track, @series)
end
end end
def redirect_link def redirect_link
......
...@@ -35,5 +35,15 @@ module Users ...@@ -35,5 +35,15 @@ module Users
.where(in_product_marketing_emails: { id: nil }) .where(in_product_marketing_emails: { id: nil })
.select(Arel.sql("DISTINCT ON(#{users.table_name}.id) #{users.table_name}.*")) .select(Arel.sql("DISTINCT ON(#{users.table_name}.id) #{users.table_name}.*"))
end end
scope :for_user_with_track_and_series, -> (user, track, series) do
where(user: user, track: track, series: series)
end
def self.save_cta_click(user, track, series)
email = for_user_with_track_and_series(user, track, series).take
email.update(cta_clicked_at: Time.zone.now) if email && email.cta_clicked_at.blank?
end
end end
end end
...@@ -3,6 +3,9 @@ ...@@ -3,6 +3,9 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe Users::InProductMarketingEmail, type: :model do RSpec.describe Users::InProductMarketingEmail, type: :model do
let(:track) { :create }
let(:series) { 0 }
describe 'associations' do describe 'associations' do
it { is_expected.to belong_to(:user) } it { is_expected.to belong_to(:user) }
end end
...@@ -17,9 +20,6 @@ RSpec.describe Users::InProductMarketingEmail, type: :model do ...@@ -17,9 +20,6 @@ RSpec.describe Users::InProductMarketingEmail, type: :model do
end end
describe '.without_track_and_series' do describe '.without_track_and_series' do
let(:track) { :create }
let(:series) { 0 }
let_it_be(:user) { create(:user) } let_it_be(:user) { create(:user) }
subject(:without_track_and_series) { User.merge(described_class.without_track_and_series(track, series)) } subject(:without_track_and_series) { User.merge(described_class.without_track_and_series(track, series)) }
...@@ -57,4 +57,75 @@ RSpec.describe Users::InProductMarketingEmail, type: :model do ...@@ -57,4 +57,75 @@ RSpec.describe Users::InProductMarketingEmail, type: :model do
it { expect(without_track_and_series).to eq [@other_user] } it { expect(without_track_and_series).to eq [@other_user] }
end end
end end
describe '.for_user_with_track_and_series' do
let_it_be(:user) { create(:user) }
let_it_be(:in_product_marketing_email) { create(:in_product_marketing_email, series: 0, track: 0, user: user) }
subject(:for_user_with_track_and_series) { described_class.for_user_with_track_and_series(user, track, series).first }
context 'when record for user with given track and series exists' do
it { is_expected.to eq(in_product_marketing_email) }
end
context 'when user is different' do
let(:user) { build_stubbed(:user) }
it { is_expected.to be_nil }
end
context 'when track is different' do
let(:track) { 1 }
it { is_expected.to be_nil }
end
context 'when series is different' do
let(:series) { 1 }
it { is_expected.to be_nil }
end
end
describe '.save_cta_click' do
let(:user) { create(:user) }
subject(:save_cta_click) { described_class.save_cta_click(user, track, series) }
context 'when there is no record' do
it 'does not error' do
expect { save_cta_click }.not_to raise_error
end
end
context 'when there is no record for the track and series' do
it 'does not perform an update' do
other_email = create(:in_product_marketing_email, user: user, track: :verify, series: 2, cta_clicked_at: nil)
expect { save_cta_click }.not_to change { other_email.reload }
end
end
context 'when there is a record for the track and series' do
it 'saves the cta click date' do
email = create(:in_product_marketing_email, user: user, track: track, series: series, cta_clicked_at: nil)
freeze_time do
expect { save_cta_click }.to change { email.reload.cta_clicked_at }.from(nil).to(Time.zone.now)
end
end
context 'cta_clicked_at is already set' do
it 'does not update' do
create(:in_product_marketing_email, user: user, track: track, series: series, cta_clicked_at: Time.zone.now)
expect_next_found_instance_of(described_class) do |record|
expect(record).not_to receive(:update)
end
save_cta_click
end
end
end
end
end end
...@@ -38,17 +38,43 @@ RSpec.describe Groups::EmailCampaignsController do ...@@ -38,17 +38,43 @@ RSpec.describe Groups::EmailCampaignsController do
expect(subject).to have_gitlab_http_status(:redirect) expect(subject).to have_gitlab_http_status(:redirect)
end end
it 'emits a snowplow event', :snowplow do context 'on .com' do
subject before do
allow(Gitlab).to receive(:com?).and_return(true)
end
it 'emits a snowplow event', :snowplow do
subject
expect_snowplow_event(
category: described_class.name,
action: 'click',
context: [{
schema: described_class::EMAIL_CAMPAIGNS_SCHEMA_URL,
data: { namespace_id: group.id, series: series.to_i, subject_line: subject_line_text, track: track.to_s }
}]
)
end
expect_snowplow_event( it 'does not save the cta_click' do
category: described_class.name, expect(Users::InProductMarketingEmail).not_to receive(:save_cta_click)
action: 'click',
context: [{ subject
schema: described_class::EMAIL_CAMPAIGNS_SCHEMA_URL, end
data: { namespace_id: group.id, series: series.to_i, subject_line: subject_line_text, track: track.to_s } end
}]
) context 'when not on.com' do
it 'saves the cta_click' do
expect(Users::InProductMarketingEmail).to receive(:save_cta_click)
subject
end
it 'does not track snowplow events' do
subject
expect_no_snowplow_event
end
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment