mirror of
https://github.com/discourse/discourse.git
synced 2025-02-25 18:55:32 -06:00
FEATURE: add more granular user option levels for email notifications (#7143)
Migrates email user options to a new data structure, where `email_always`, `email_direct` and `email_private_messages` are replace by * `email_messages_level`, with options: `always`, `only_when_away` and `never` (defaults to `always`) * `email_level`, with options: `always`, `only_when_away` and `never` (defaults to `only_when_away`)
This commit is contained in:
parent
93490fbfaf
commit
9334d2f4f7
@ -2,15 +2,29 @@ import PreferencesTabController from "discourse/mixins/preferences-tab-controlle
|
||||
import { default as computed } from "ember-addons/ember-computed-decorators";
|
||||
import { popupAjaxError } from "discourse/lib/ajax-error";
|
||||
|
||||
const EMAIL_LEVELS = {
|
||||
ALWAYS: 0,
|
||||
ONLY_WHEN_AWAY: 1,
|
||||
NEVER: 2
|
||||
};
|
||||
|
||||
export default Ember.Controller.extend(PreferencesTabController, {
|
||||
emailMessagesLevelAway: Ember.computed.equal(
|
||||
"model.user_option.email_messages_level",
|
||||
EMAIL_LEVELS.ONLY_WHEN_AWAY
|
||||
),
|
||||
emailLevelAway: Ember.computed.equal(
|
||||
"model.user_option.email_level",
|
||||
EMAIL_LEVELS.ONLY_WHEN_AWAY
|
||||
),
|
||||
|
||||
saveAttrNames: [
|
||||
"email_always",
|
||||
"email_level",
|
||||
"email_messages_level",
|
||||
"mailing_list_mode",
|
||||
"mailing_list_mode_frequency",
|
||||
"email_digests",
|
||||
"email_direct",
|
||||
"email_in_reply_to",
|
||||
"email_private_messages",
|
||||
"email_previous_replies",
|
||||
"digest_after_minutes",
|
||||
"include_tl0_in_digests"
|
||||
@ -42,6 +56,15 @@ export default Ember.Controller.extend(PreferencesTabController, {
|
||||
{ name: I18n.t("user.email_previous_replies.never"), value: 2 }
|
||||
],
|
||||
|
||||
emailLevelOptions: [
|
||||
{ name: I18n.t("user.email_level.always"), value: EMAIL_LEVELS.ALWAYS },
|
||||
{
|
||||
name: I18n.t("user.email_level.only_when_away"),
|
||||
value: EMAIL_LEVELS.ONLY_WHEN_AWAY
|
||||
},
|
||||
{ name: I18n.t("user.email_level.never"), value: EMAIL_LEVELS.NEVER }
|
||||
],
|
||||
|
||||
digestFrequencies: [
|
||||
{ name: I18n.t("user.email_digests.every_30_minutes"), value: 30 },
|
||||
{ name: I18n.t("user.email_digests.every_hour"), value: 60 },
|
||||
@ -51,6 +74,17 @@ export default Ember.Controller.extend(PreferencesTabController, {
|
||||
{ name: I18n.t("user.email_digests.every_two_weeks"), value: 20160 }
|
||||
],
|
||||
|
||||
@computed()
|
||||
emailFrequencyInstructions() {
|
||||
if (this.siteSettings.email_time_window_mins) {
|
||||
return I18n.t("user.email.frequency", {
|
||||
count: this.siteSettings.email_time_window_mins
|
||||
});
|
||||
} else {
|
||||
return I18n.t("user.email.frequency_immediately");
|
||||
}
|
||||
},
|
||||
|
||||
actions: {
|
||||
save() {
|
||||
this.set("saved", false);
|
||||
|
@ -264,14 +264,13 @@ const User = RestModel.extend({
|
||||
);
|
||||
|
||||
let userOptionFields = [
|
||||
"email_always",
|
||||
"mailing_list_mode",
|
||||
"mailing_list_mode_frequency",
|
||||
"external_links_in_new_tab",
|
||||
"email_digests",
|
||||
"email_direct",
|
||||
"email_in_reply_to",
|
||||
"email_private_messages",
|
||||
"email_messages_level",
|
||||
"email_level",
|
||||
"email_previous_replies",
|
||||
"dynamic_favicon",
|
||||
"enable_quoting",
|
||||
|
@ -7,23 +7,35 @@
|
||||
{{/unless}}
|
||||
<div class="control-group pref-email-settings">
|
||||
<label class="control-label">{{i18n 'user.email_settings'}}</label>
|
||||
|
||||
<div class='controls controls-dropdown'>
|
||||
<label for="user-email-messages-level">{{i18n 'user.email_messages_level'}}</label>
|
||||
{{combo-box valueAttribute="value"
|
||||
content=emailLevelOptions
|
||||
value=model.user_option.email_messages_level
|
||||
id="user-email-messages-level"}}
|
||||
{{#if emailMessagesLevelAway}}
|
||||
<div class='instructions'>{{emailFrequencyInstructions}}</div>
|
||||
{{/if}}
|
||||
</div>
|
||||
|
||||
<div class='controls controls-dropdown'>
|
||||
<label for="user-email-level">{{i18n 'user.email_level.title'}}</label>
|
||||
{{combo-box valueAttribute="value"
|
||||
content=emailLevelOptions
|
||||
value=model.user_option.email_level
|
||||
id="user-email-level"}}
|
||||
{{#if emailLevelAway}}
|
||||
<div class='instructions'>{{emailFrequencyInstructions}}</div>
|
||||
{{/if}}
|
||||
</div>
|
||||
|
||||
<div class='controls controls-dropdown'>
|
||||
<label>{{i18n 'user.email_previous_replies.title'}}</label>
|
||||
{{combo-box valueAttribute="value" content=previousRepliesOptions value=model.user_option.email_previous_replies}}
|
||||
</div>
|
||||
{{preference-checkbox labelKey="user.email_in_reply_to" checked=model.user_option.email_in_reply_to}}
|
||||
{{preference-checkbox labelKey="user.email_private_messages" checked=model.user_option.email_private_messages}}
|
||||
{{preference-checkbox labelKey="user.email_direct" checked=model.user_option.email_direct}}
|
||||
{{preference-checkbox labelKey="user.email_always" checked=model.user_option.email_always}}
|
||||
{{#unless model.user_option.email_always}}
|
||||
<div class='instructions'>
|
||||
{{#if siteSettings.email_time_window_mins}}
|
||||
{{i18n 'user.email.frequency' count=siteSettings.email_time_window_mins}}
|
||||
{{else}}
|
||||
{{i18n 'user.email.frequency_immediately'}}
|
||||
{{/if}}
|
||||
</div>
|
||||
{{/unless}}
|
||||
|
||||
{{plugin-outlet name="user-preferences-emails-pref-email-settings" args=(hash model=model save=(action "save"))}}
|
||||
</div>
|
||||
|
||||
|
@ -91,10 +91,9 @@ class EmailController < ApplicationController
|
||||
end
|
||||
|
||||
if params["unsubscribe_all"]
|
||||
user.user_option.update_columns(email_always: false,
|
||||
email_digests: false,
|
||||
email_direct: false,
|
||||
email_private_messages: false)
|
||||
user.user_option.update_columns(email_digests: false,
|
||||
email_level: UserOption.email_level_types[:never],
|
||||
email_messages_level: UserOption.email_level_types[:never])
|
||||
updated = true
|
||||
end
|
||||
|
||||
|
@ -102,7 +102,7 @@ module Jobs
|
||||
return if user.staged && type.to_s == "digest"
|
||||
|
||||
seen_recently = (user.last_seen_at.present? && user.last_seen_at > SiteSetting.email_time_window_mins.minutes.ago)
|
||||
seen_recently = false if user.user_option.email_always || user.staged
|
||||
seen_recently = false if always_email_regular?(user, type) || always_email_private_message?(user, type) || user.staged
|
||||
|
||||
email_args = {}
|
||||
|
||||
@ -133,7 +133,7 @@ module Jobs
|
||||
return [nil, nil]
|
||||
end
|
||||
|
||||
unless user.user_option.email_always?
|
||||
unless always_email_regular?(user, type) || always_email_private_message?(user, type)
|
||||
if (notification && notification.read?) || (post && post.seen?(user))
|
||||
return skip_message(SkippedEmailLog.reason_types[:user_email_notification_already_read])
|
||||
end
|
||||
@ -217,7 +217,7 @@ module Jobs
|
||||
return SkippedEmailLog.reason_types[:user_email_user_suspended]
|
||||
end
|
||||
|
||||
already_read = !user.user_option.email_always? && PostTiming.exists?(topic_id: post.topic_id, post_number: post.post_number, user_id: user.id)
|
||||
already_read = user.user_option.email_level != UserOption.email_level_types[:always] && PostTiming.exists?(topic_id: post.topic_id, post_number: post.post_number, user_id: user.id)
|
||||
if already_read
|
||||
return SkippedEmailLog.reason_types[:user_email_already_read]
|
||||
end
|
||||
@ -236,6 +236,13 @@ module Jobs
|
||||
)
|
||||
end
|
||||
|
||||
def always_email_private_message?(user, type)
|
||||
type == :user_private_message && user.user_option.email_messages_level == UserOption.email_level_types[:always]
|
||||
end
|
||||
|
||||
def always_email_regular?(user, type)
|
||||
type != :user_private_message && user.user_option.email_level == UserOption.email_level_types[:always]
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
22
app/models/email_level_site_setting.rb
Normal file
22
app/models/email_level_site_setting.rb
Normal file
@ -0,0 +1,22 @@
|
||||
require_dependency 'enum_site_setting'
|
||||
|
||||
class EmailLevelSiteSetting < EnumSiteSetting
|
||||
|
||||
def self.valid_value?(val)
|
||||
val.to_i.to_s == val.to_s &&
|
||||
values.any? { |v| v[:value] == val.to_i }
|
||||
end
|
||||
|
||||
def self.values
|
||||
@values ||= [
|
||||
{ name: 'user.email_level.always', value: 0 },
|
||||
{ name: 'user.email_level.only_when_away', value: 1 },
|
||||
{ name: 'user.email_level.never', value: 2 },
|
||||
]
|
||||
end
|
||||
|
||||
def self.translate_names?
|
||||
true
|
||||
end
|
||||
|
||||
end
|
@ -29,15 +29,20 @@ class UserOption < ActiveRecord::Base
|
||||
@text_sizes ||= Enum.new(normal: 0, larger: 1, largest: 2, smaller: 3)
|
||||
end
|
||||
|
||||
def self.email_level_types
|
||||
@email_level_type ||= Enum.new(always: 0, only_when_away: 1, never: 2)
|
||||
end
|
||||
|
||||
validates :text_size_key, inclusion: { in: UserOption.text_sizes.values }
|
||||
validates :email_level, inclusion: { in: UserOption.email_level_types.values }
|
||||
validates :email_messages_level, inclusion: { in: UserOption.email_level_types.values }
|
||||
|
||||
def set_defaults
|
||||
self.email_always = SiteSetting.default_email_always
|
||||
self.mailing_list_mode = SiteSetting.default_email_mailing_list_mode
|
||||
self.mailing_list_mode_frequency = SiteSetting.default_email_mailing_list_mode_frequency
|
||||
self.email_direct = SiteSetting.default_email_direct
|
||||
self.email_level = SiteSetting.default_email_level
|
||||
self.email_messages_level = SiteSetting.default_email_messages_level
|
||||
self.automatically_unpin_topics = SiteSetting.default_topics_automatic_unpin
|
||||
self.email_private_messages = SiteSetting.default_email_personal_messages
|
||||
self.email_previous_replies = SiteSetting.default_email_previous_replies
|
||||
self.email_in_reply_to = SiteSetting.default_email_in_reply_to
|
||||
|
||||
@ -173,11 +178,10 @@ end
|
||||
# Table name: user_options
|
||||
#
|
||||
# user_id :integer not null, primary key
|
||||
# email_always :boolean default(FALSE), not null
|
||||
# mailing_list_mode :boolean default(FALSE), not null
|
||||
# email_digests :boolean
|
||||
# email_direct :boolean default(TRUE), not null
|
||||
# email_private_messages :boolean default(TRUE), not null
|
||||
# email_level :integer default(1), not null
|
||||
# email_messages_level :integer default(0), not null
|
||||
# external_links_in_new_tab :boolean default(FALSE), not null
|
||||
# enable_quoting :boolean default(TRUE), not null
|
||||
# dynamic_favicon :boolean default(FALSE), not null
|
||||
|
@ -1,11 +1,10 @@
|
||||
class UserOptionSerializer < ApplicationSerializer
|
||||
attributes :user_id,
|
||||
:email_always,
|
||||
:mailing_list_mode,
|
||||
:mailing_list_mode_frequency,
|
||||
:email_digests,
|
||||
:email_private_messages,
|
||||
:email_direct,
|
||||
:email_level,
|
||||
:email_messages_level,
|
||||
:external_links_in_new_tab,
|
||||
:dynamic_favicon,
|
||||
:enable_quoting,
|
||||
|
@ -48,7 +48,7 @@ class AnonymousShadowCreator
|
||||
)
|
||||
|
||||
shadow.user_option.update_columns(
|
||||
email_private_messages: false,
|
||||
email_messages_level: UserOption.email_level_types[:never],
|
||||
email_digests: false
|
||||
)
|
||||
|
||||
|
@ -67,7 +67,7 @@ class NotificationEmailer
|
||||
EMAILABLE_POST_TYPES ||= Set.new [Post.types[:regular], Post.types[:whisper]]
|
||||
|
||||
def enqueue(type, delay = default_delay)
|
||||
return unless notification.user.user_option.email_direct?
|
||||
return if notification.user.user_option.email_level == UserOption.email_level_types[:never]
|
||||
perform_enqueue(type, delay)
|
||||
end
|
||||
|
||||
@ -80,7 +80,7 @@ class NotificationEmailer
|
||||
return
|
||||
end
|
||||
|
||||
return unless notification.user.user_option.email_private_messages?
|
||||
return if notification.user.user_option.email_messages_level == UserOption.email_level_types[:never]
|
||||
perform_enqueue(type, delay)
|
||||
end
|
||||
|
||||
|
@ -40,11 +40,10 @@ class UserAnonymizer
|
||||
@user.primary_email.update_attribute(:email, "#{@user.username}@anonymized.invalid")
|
||||
|
||||
options = @user.user_option
|
||||
options.email_always = false
|
||||
options.mailing_list_mode = false
|
||||
options.email_digests = false
|
||||
options.email_private_messages = false
|
||||
options.email_direct = false
|
||||
options.email_level = UserOption.email_level_types[:never]
|
||||
options.email_messages_level = UserOption.email_level_types[:never]
|
||||
options.save!
|
||||
|
||||
if profile = @user.user_profile
|
||||
|
@ -15,12 +15,11 @@ class UserUpdater
|
||||
}
|
||||
|
||||
OPTION_ATTR = [
|
||||
:email_always,
|
||||
:mailing_list_mode,
|
||||
:mailing_list_mode_frequency,
|
||||
:email_digests,
|
||||
:email_direct,
|
||||
:email_private_messages,
|
||||
:email_level,
|
||||
:email_messages_level,
|
||||
:external_links_in_new_tab,
|
||||
:enable_quoting,
|
||||
:dynamic_favicon,
|
||||
|
@ -950,12 +950,14 @@ en:
|
||||
every_three_days: "every three days"
|
||||
weekly: "weekly"
|
||||
every_two_weeks: "every two weeks"
|
||||
|
||||
email_level:
|
||||
title: "Send me an email when someone quotes me, replies to my post, mentions my @username, or invites me to a topic"
|
||||
always: "always"
|
||||
only_when_away: "only when away"
|
||||
never: "never"
|
||||
email_messages_level: "Send me an email when someone messages me"
|
||||
include_tl0_in_digests: "Include content from new users in summary emails"
|
||||
email_in_reply_to: "Include an excerpt of replied to post in emails"
|
||||
email_direct: "Send me an email when someone quotes me, replies to my post, mentions my @username, or invites me to a topic"
|
||||
email_private_messages: "Send me an email when someone messages me"
|
||||
email_always: "Send me email notifications even when I am active on the site"
|
||||
|
||||
other_settings: "Other"
|
||||
categories_settings: "Categories"
|
||||
|
@ -873,7 +873,7 @@ en:
|
||||
user_must_edit: "<p>This post was flagged by the community and is temporarily hidden.</p>"
|
||||
|
||||
ignored:
|
||||
hidden_content: '<p>Hidden content</p>'
|
||||
hidden_content: "<p>Hidden content</p>"
|
||||
|
||||
archetypes:
|
||||
regular:
|
||||
@ -1921,12 +1921,11 @@ en:
|
||||
|
||||
default_email_digest_frequency: "How often users receive summary emails by default."
|
||||
default_include_tl0_in_digests: "Include posts from new users in summary emails by default. Users can change this in their preferences."
|
||||
default_email_personal_messages: "Send an email when someone messages the user by default."
|
||||
default_email_direct: "Send an email when someone quotes/replies to/mentions or invites the user by default."
|
||||
default_email_level: "Set default email notification level when someone quotes/replies to/mentions or invites user."
|
||||
default_email_messages_level: "Set default email notification level when someone messages user."
|
||||
default_email_mailing_list_mode: "Send an email for every new post by default."
|
||||
default_email_mailing_list_mode_frequency: "Users who enable mailing list mode will receive emails this often by default."
|
||||
disable_mailing_list_mode: "Disallow users from enabling mailing list mode."
|
||||
default_email_always: "Send an email notification even when the user is active by default."
|
||||
default_email_previous_replies: "Include previous replies in emails by default."
|
||||
|
||||
default_email_in_reply_to: "Include excerpt of replied to post in emails by default."
|
||||
|
@ -1814,8 +1814,12 @@ user_preferences:
|
||||
enum: "DigestEmailSiteSetting"
|
||||
default: 10080
|
||||
default_include_tl0_in_digests: false
|
||||
default_email_personal_messages: true
|
||||
default_email_direct: true
|
||||
default_email_level:
|
||||
enum: "EmailLevelSiteSetting"
|
||||
default: 1
|
||||
default_email_messages_level:
|
||||
enum: "EmailLevelSiteSetting"
|
||||
default: 0
|
||||
default_email_mailing_list_mode: false
|
||||
default_email_mailing_list_mode_frequency:
|
||||
enum: "MailingListModeSiteSetting"
|
||||
@ -1823,7 +1827,6 @@ user_preferences:
|
||||
disable_mailing_list_mode:
|
||||
default: false
|
||||
client: true
|
||||
default_email_always: false
|
||||
default_email_previous_replies:
|
||||
enum: "PreviousRepliesSiteSetting"
|
||||
default: 2
|
||||
|
@ -27,8 +27,8 @@ User.seed do |u|
|
||||
end
|
||||
|
||||
UserOption.where(user_id: -1).update_all(
|
||||
email_private_messages: false,
|
||||
email_direct: false
|
||||
email_messages_level: UserOption.email_level_types[:never],
|
||||
email_level: UserOption.email_level_types[:never]
|
||||
)
|
||||
|
||||
Group.user_trust_level_change!(-1, TrustLevel[4])
|
||||
@ -55,9 +55,9 @@ if ENV["SMOKE"] == "1"
|
||||
end.first
|
||||
|
||||
UserOption.where(user_id: smoke_user.id).update_all(
|
||||
email_direct: false,
|
||||
email_digests: false,
|
||||
email_private_messages: false,
|
||||
email_messages_level: UserOption.email_level_types[:never],
|
||||
email_level: UserOption.email_level_types[:never]
|
||||
)
|
||||
|
||||
EmailToken.where(user_id: smoke_user.id).update_all(confirmed: true)
|
||||
|
29
db/migrate/20190312181641_migrate_email_user_options.rb
Normal file
29
db/migrate/20190312181641_migrate_email_user_options.rb
Normal file
@ -0,0 +1,29 @@
|
||||
class MigrateEmailUserOptions < ActiveRecord::Migration[5.2]
|
||||
def up
|
||||
# see UserOption.email_level_types
|
||||
# always = 0, only_while_away: 1, never: 2
|
||||
|
||||
add_column :user_options, :email_level, :integer, default: 1, null: false
|
||||
add_column :user_options, :email_messages_level, :integer, default: 0, null: false
|
||||
|
||||
execute <<~SQL
|
||||
UPDATE user_options
|
||||
SET email_level = CASE
|
||||
WHEN email_direct AND email_always
|
||||
THEN 0
|
||||
WHEN email_direct AND email_always IS NOT TRUE
|
||||
THEN 1
|
||||
ELSE 2
|
||||
END,
|
||||
email_messages_level = CASE
|
||||
WHEN email_private_messages
|
||||
THEN 0
|
||||
ELSE 2
|
||||
END
|
||||
SQL
|
||||
end
|
||||
|
||||
def down
|
||||
# See postmigration: 20190312194528_drop_email_user_options_columns.rb
|
||||
end
|
||||
end
|
@ -0,0 +1,34 @@
|
||||
class MigrateDefaultUserEmailOptions < ActiveRecord::Migration[5.2]
|
||||
def up
|
||||
# see UserOption.email_level_types
|
||||
# always = 0, only_while_away: 1, never: 2
|
||||
|
||||
email_always = DB.query_single("SELECT value FROM site_settings WHERE name = 'default_email_always'").first
|
||||
email_direct = DB.query_single("SELECT value FROM site_settings WHERE name = 'default_email_direct'").first
|
||||
email_personal_messages = DB.query_single("SELECT value FROM site_settings WHERE name = 'default_email_personal_messages'").first
|
||||
|
||||
default_email_level = nil
|
||||
default_email_level = 0 if email_direct != 'f' && email_always == 't'
|
||||
default_email_level = 2 if email_direct == 'f'
|
||||
|
||||
unless default_email_level.nil?
|
||||
execute "INSERT INTO site_settings (name, data_type, value, created_at, updated_at)
|
||||
VALUES ('default_email_level', 7, #{default_email_level}, now(), now())"
|
||||
end
|
||||
|
||||
default_email_messages_level = nil
|
||||
default_email_messages_level = 0 if email_personal_messages != 'f' && email_always == 't'
|
||||
default_email_messages_level = 2 if email_personal_messages == 'f'
|
||||
|
||||
unless default_email_messages_level.nil?
|
||||
execute "INSERT INTO site_settings (name, data_type, value, created_at, updated_at)
|
||||
VALUES ('default_email_messages_level', 7, #{default_email_messages_level}, now(), now())"
|
||||
end
|
||||
|
||||
execute "DELETE from site_settings where name in ('default_email_always', 'default_email_direct', 'default_email_personal_messages')"
|
||||
end
|
||||
|
||||
def down
|
||||
raise ActiveRecord::IrreversibleMigration
|
||||
end
|
||||
end
|
@ -0,0 +1,19 @@
|
||||
require 'migration/column_dropper'
|
||||
|
||||
class DropEmailUserOptionsColumns < ActiveRecord::Migration[5.2]
|
||||
def up
|
||||
{
|
||||
user_options: %i{
|
||||
email_direct
|
||||
email_private_messages
|
||||
email_always
|
||||
},
|
||||
}.each do |table, columns|
|
||||
Migration::ColumnDropper.execute_drop(table, columns)
|
||||
end
|
||||
end
|
||||
|
||||
def down
|
||||
raise ActiveRecord::IrreversibleMigration
|
||||
end
|
||||
end
|
@ -1154,8 +1154,8 @@ module Email
|
||||
|
||||
def enable_email_pm_setting(user)
|
||||
# ensure user PM emails are enabled (since user is posting via email)
|
||||
if !user.staged && !user.user_option.email_private_messages
|
||||
user.user_option.update!(email_private_messages: true)
|
||||
if !user.staged && user.user_option.email_messages_level == UserOption.email_level_types[:never]
|
||||
user.user_option.update!(email_messages_level: UserOption.email_level_types[:always])
|
||||
end
|
||||
end
|
||||
end
|
||||
|
@ -50,8 +50,8 @@ end
|
||||
bot.update!(admin: true, moderator: false)
|
||||
|
||||
bot.user_option.update!(
|
||||
email_private_messages: false,
|
||||
email_direct: false
|
||||
email_messages_level: UserOption.email_level_types[:never],
|
||||
email_level: UserOption.email_level_types[:never]
|
||||
)
|
||||
|
||||
if !bot.user_profile.bio_raw
|
||||
|
@ -109,7 +109,7 @@ describe DiscourseNarrativeBot::TrackSelector do
|
||||
|
||||
it 'should not enqueue any user email' do
|
||||
NotificationEmailer.enable
|
||||
user.user_option.update!(email_always: true)
|
||||
user.user_option.update!(email_level: UserOption.email_level_types[:always])
|
||||
|
||||
post.update!(
|
||||
raw: 'show me what you can do',
|
||||
|
@ -372,9 +372,8 @@ class ImportScripts::Base
|
||||
|
||||
user_option = u.user_option
|
||||
user_option.email_digests = false
|
||||
user_option.email_private_messages = false
|
||||
user_option.email_direct = false
|
||||
user_option.email_always = false
|
||||
user_option.email_level = UserOption.email_level_types[:never]
|
||||
user_option.email_messages_level = UserOption.email_level_types[:never]
|
||||
user_option.save!
|
||||
if u.save
|
||||
StaffActionLogger.new(Discourse.system_user).log_user_suspend(u, ban_reason)
|
||||
|
@ -246,9 +246,8 @@ EOM
|
||||
|
||||
user_option = user.user_option
|
||||
user_option.email_digests = false
|
||||
user_option.email_private_messages = false
|
||||
user_option.email_direct = false
|
||||
user_option.email_always = false
|
||||
user_option.email_level = UserOption.email_level_types[:never]
|
||||
user_option.email_messages_level = UserOption.email_level_types[:never]
|
||||
user_option.save!
|
||||
|
||||
if user.save
|
||||
|
@ -93,9 +93,8 @@ module ImportScripts::PhpBB3
|
||||
if disable_email
|
||||
user_option = user.user_option
|
||||
user_option.email_digests = false
|
||||
user_option.email_private_messages = false
|
||||
user_option.email_direct = false
|
||||
user_option.email_always = false
|
||||
user_option.email_level = UserOption.email_level_types[:never]
|
||||
user_option.email_messages_level = UserOption.email_level_types[:never]
|
||||
user_option.save!
|
||||
end
|
||||
|
||||
|
@ -631,15 +631,15 @@ describe Email::Receiver do
|
||||
expect { process(:reply_user_not_matching_but_known) }.to change { topic.posts.count }
|
||||
end
|
||||
|
||||
it "re-enables user's email_private_messages setting when user replies to a private topic" do
|
||||
it "re-enables user's PM email notifications when user replies to a private topic" do
|
||||
topic.update_columns(category_id: nil, archetype: Archetype.private_message)
|
||||
topic.allowed_users << user
|
||||
topic.save
|
||||
|
||||
user.user_option.update_columns(email_private_messages: false)
|
||||
user.user_option.update_columns(email_messages_level: UserOption.email_level_types[:never])
|
||||
expect { process(:reply_user_matching) }.to change { topic.posts.count }
|
||||
user.reload
|
||||
expect(user.user_option.email_private_messages).to eq(true)
|
||||
expect(user.user_option.email_messages_level).to eq(UserOption.email_level_types[:always])
|
||||
end
|
||||
|
||||
end
|
||||
@ -741,12 +741,12 @@ describe Email::Receiver do
|
||||
expect(Post.last.uploads.length).to eq 1
|
||||
end
|
||||
|
||||
it "enables user's email_private_messages setting when user emails new topic to group" do
|
||||
it "reenables user's PM email notifications when user emails new topic to group" do
|
||||
user = Fabricate(:user, email: "existing@bar.com")
|
||||
user.user_option.update_columns(email_private_messages: false)
|
||||
user.user_option.update_columns(email_messages_level: UserOption.email_level_types[:never])
|
||||
expect { process(:group_existing_user) }.to change(Topic, :count)
|
||||
user.reload
|
||||
expect(user.user_option.email_private_messages).to eq(true)
|
||||
expect(user.user_option.email_messages_level).to eq(UserOption.email_level_types[:always])
|
||||
end
|
||||
|
||||
context "with forwarded emails enabled" do
|
||||
|
@ -73,7 +73,7 @@ describe Jobs::EnqueueDigestEmails do
|
||||
|
||||
context 'visited the site this week' do
|
||||
let(:user_visited_this_week) { Fabricate(:active_user, last_seen_at: 6.days.ago) }
|
||||
let(:user_visited_this_week_email_always) { Fabricate(:active_user, last_seen_at: 6.days.ago, email_always: true) }
|
||||
let(:user_visited_this_week_email_always) { Fabricate(:active_user, last_seen_at: 6.days.ago, email_level: UserOption.email_level_types[:always]) }
|
||||
|
||||
it "doesn't return users who have been emailed recently" do
|
||||
user = user_visited_this_week
|
||||
|
@ -91,6 +91,71 @@ describe Jobs::UserEmail do
|
||||
end
|
||||
end
|
||||
|
||||
context "recently seen" do
|
||||
let(:post) { Fabricate(:post, user: user) }
|
||||
let(:notification) { Fabricate(
|
||||
:notification,
|
||||
user: user,
|
||||
topic: post.topic,
|
||||
post_number: post.post_number,
|
||||
data: { original_post_id: post.id }.to_json
|
||||
)
|
||||
}
|
||||
before do
|
||||
user.update_column(:last_seen_at, 9.minutes.ago)
|
||||
end
|
||||
|
||||
it "doesn't send an email to a user that's been recently seen" do
|
||||
Jobs::UserEmail.new.execute(type: :user_replied, user_id: user.id, post_id: post.id)
|
||||
expect(ActionMailer::Base.deliveries).to eq([])
|
||||
end
|
||||
|
||||
it "does send an email to a user that's been recently seen but has email_level set to always" do
|
||||
user.user_option.update_attributes(email_level: UserOption.email_level_types[:always])
|
||||
PostTiming.create!(topic_id: post.topic_id, post_number: post.post_number, user_id: user.id, msecs: 100)
|
||||
|
||||
Jobs::UserEmail.new.execute(
|
||||
type: :user_replied,
|
||||
user_id: user.id,
|
||||
post_id: post.id,
|
||||
notification_id: notification.id
|
||||
)
|
||||
|
||||
expect(ActionMailer::Base.deliveries.first.to).to contain_exactly(
|
||||
user.email
|
||||
)
|
||||
end
|
||||
|
||||
it "sends an email by default for a PM to a user that's been recently seen" do
|
||||
Jobs::UserEmail.new.execute(
|
||||
type: :user_private_message,
|
||||
user_id: user.id,
|
||||
post_id: post.id,
|
||||
notification_id: notification.id
|
||||
)
|
||||
|
||||
expect(ActionMailer::Base.deliveries.first.to).to contain_exactly(
|
||||
user.email
|
||||
)
|
||||
end
|
||||
|
||||
it "doesn't send a PM email to a user that's been recently seen and has email_messages_level set to never" do
|
||||
user.user_option.update_attributes(email_messages_level: UserOption.email_level_types[:never])
|
||||
user.user_option.update_attributes(email_level: UserOption.email_level_types[:always])
|
||||
Jobs::UserEmail.new.execute(type: :user_private_message, user_id: user.id, post_id: post.id)
|
||||
|
||||
expect(ActionMailer::Base.deliveries).to eq([])
|
||||
end
|
||||
|
||||
it "doesn't send a regular post email to a user that's been recently seen and has email_level set to never" do
|
||||
user.user_option.update_attributes(email_messages_level: UserOption.email_level_types[:always])
|
||||
user.user_option.update_attributes(email_level: UserOption.email_level_types[:never])
|
||||
Jobs::UserEmail.new.execute(type: :user_replied, user_id: user.id, post_id: post.id)
|
||||
|
||||
expect(ActionMailer::Base.deliveries).to eq([])
|
||||
end
|
||||
end
|
||||
|
||||
context "email_log" do
|
||||
let(:post) { Fabricate(:post) }
|
||||
|
||||
@ -256,9 +321,9 @@ describe Jobs::UserEmail do
|
||||
)).to eq(true)
|
||||
end
|
||||
|
||||
it "does send the email if the notification has been seen but the user is set for email_always" do
|
||||
it "does send the email if the notification has been seen but user has email_level set to always" do
|
||||
notification.update_column(:read, true)
|
||||
user.user_option.update_column(:email_always, true)
|
||||
user.user_option.update_column(:email_level, UserOption.email_level_types[:always])
|
||||
|
||||
Jobs::UserEmail.new.execute(
|
||||
type: :user_mentioned,
|
||||
@ -301,9 +366,9 @@ describe Jobs::UserEmail do
|
||||
expect(ActionMailer::Base.deliveries).to eq([])
|
||||
end
|
||||
|
||||
it "does send an email to a user that's been recently seen but has email_always set" do
|
||||
it "does send an email to a user that's been recently seen but has email_level set to always" do
|
||||
user.update!(last_seen_at: 9.minutes.ago)
|
||||
user.user_option.update!(email_always: true)
|
||||
user.user_option.update!(email_level: UserOption.email_level_types[:always])
|
||||
|
||||
Jobs::UserEmail.new.execute(
|
||||
type: :user_replied,
|
||||
|
@ -11,7 +11,8 @@ describe UserOption do
|
||||
|
||||
user.reload
|
||||
|
||||
expect(user.user_option.email_always).to eq(SiteSetting.default_email_always)
|
||||
expect(user.user_option.email_level).to eq(SiteSetting.default_email_level)
|
||||
expect(user.user_option.email_messages_level).to eq(SiteSetting.default_email_messages_level)
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -265,8 +265,8 @@ describe User do
|
||||
expect(subject.email_tokens).to be_present
|
||||
expect(subject.user_stat).to be_present
|
||||
expect(subject.user_profile).to be_present
|
||||
expect(subject.user_option.email_private_messages).to eq(true)
|
||||
expect(subject.user_option.email_direct).to eq(true)
|
||||
expect(subject.user_option.email_messages_level).to eq(UserOption.email_level_types[:always])
|
||||
expect(subject.user_option.email_level).to eq(UserOption.email_level_types[:only_when_away])
|
||||
end
|
||||
end
|
||||
|
||||
@ -1461,10 +1461,9 @@ describe User do
|
||||
|
||||
before do
|
||||
SiteSetting.default_email_digest_frequency = 1440 # daily
|
||||
SiteSetting.default_email_personal_messages = false
|
||||
SiteSetting.default_email_direct = false
|
||||
SiteSetting.default_email_level = UserOption.email_level_types[:never]
|
||||
SiteSetting.default_email_messages_level = UserOption.email_level_types[:never]
|
||||
SiteSetting.default_email_mailing_list_mode = true
|
||||
SiteSetting.default_email_always = true
|
||||
|
||||
SiteSetting.default_other_new_topic_duration_minutes = -1 # not viewed
|
||||
SiteSetting.default_other_auto_track_topics_after_msecs = 0 # immediately
|
||||
@ -1485,16 +1484,15 @@ describe User do
|
||||
it "has overriden preferences" do
|
||||
user = Fabricate(:user)
|
||||
options = user.user_option
|
||||
expect(options.email_always).to eq(true)
|
||||
expect(options.mailing_list_mode).to eq(true)
|
||||
expect(options.digest_after_minutes).to eq(1440)
|
||||
expect(options.email_private_messages).to eq(false)
|
||||
expect(options.email_level).to eq(UserOption.email_level_types[:never])
|
||||
expect(options.email_messages_level).to eq(UserOption.email_level_types[:never])
|
||||
expect(options.external_links_in_new_tab).to eq(true)
|
||||
expect(options.enable_quoting).to eq(false)
|
||||
expect(options.dynamic_favicon).to eq(true)
|
||||
expect(options.disable_jump_reply).to eq(true)
|
||||
expect(options.automatically_unpin_topics).to eq(false)
|
||||
expect(options.email_direct).to eq(false)
|
||||
expect(options.new_topic_duration_minutes).to eq(-1)
|
||||
expect(options.auto_track_topics_after_msecs).to eq(0)
|
||||
expect(options.notification_level_when_replying).to eq(3)
|
||||
@ -1518,7 +1516,7 @@ describe User do
|
||||
|
||||
it "Creates a UserOption row when a user record is created and destroys once done" do
|
||||
user = Fabricate(:user)
|
||||
expect(user.user_option.email_always).to eq(false)
|
||||
expect(user.user_option.email_level).to eq(UserOption.email_level_types[:only_when_away])
|
||||
|
||||
user_id = user.id
|
||||
user.destroy!
|
||||
|
@ -15,10 +15,9 @@ RSpec.describe EmailController do
|
||||
user = Fabricate(:user)
|
||||
key = UnsubscribeKey.create_key_for(user, "all")
|
||||
|
||||
user.user_option.update_columns(email_always: true,
|
||||
email_digests: true,
|
||||
email_direct: true,
|
||||
email_private_messages: true)
|
||||
user.user_option.update_columns(email_digests: true,
|
||||
email_level: UserOption.email_level_types[:never],
|
||||
email_messages_level: UserOption.email_level_types[:never])
|
||||
|
||||
post "/email/unsubscribe/#{key}.json",
|
||||
params: { unsubscribe_all: "1" }
|
||||
@ -32,11 +31,9 @@ RSpec.describe EmailController do
|
||||
|
||||
user.user_option.reload
|
||||
|
||||
expect(user.user_option.email_always).to eq(false)
|
||||
expect(user.user_option.email_digests).to eq(false)
|
||||
expect(user.user_option.email_direct).to eq(false)
|
||||
expect(user.user_option.email_private_messages).to eq(false)
|
||||
|
||||
expect(user.user_option.email_level).to eq(UserOption.email_level_types[:never])
|
||||
expect(user.user_option.email_messages_level).to eq(UserOption.email_level_types[:never])
|
||||
end
|
||||
|
||||
it 'can disable mailing list' do
|
||||
|
@ -1494,14 +1494,14 @@ describe UsersController do
|
||||
put "/u/#{user.username}.json", params: {
|
||||
muted_usernames: "",
|
||||
theme_ids: [theme.id],
|
||||
email_direct: false
|
||||
email_level: UserOption.email_level_types[:always]
|
||||
}
|
||||
|
||||
user.reload
|
||||
|
||||
expect(user.muted_users.pluck(:username).sort).to be_empty
|
||||
expect(user.user_option.theme_ids).to eq([theme.id])
|
||||
expect(user.user_option.email_direct).to eq(false)
|
||||
expect(user.user_option.email_level).to eq(UserOption.email_level_types[:always])
|
||||
end
|
||||
|
||||
context 'a locale is chosen that differs from I18n.locale' do
|
||||
|
@ -36,7 +36,7 @@ describe AnonymousShadowCreator do
|
||||
shadow3 = AnonymousShadowCreator.get(user)
|
||||
|
||||
expect(shadow3.user_option.email_digests).to eq(false)
|
||||
expect(shadow3.user_option.email_private_messages).to eq(false)
|
||||
expect(shadow3.user_option.email_messages_level).to eq(UserOption.email_level_types[:never])
|
||||
|
||||
expect(shadow2.id).not_to eq(shadow3.id)
|
||||
|
||||
|
@ -78,7 +78,7 @@ describe NotificationEmailer do
|
||||
include_examples "enqueue"
|
||||
|
||||
it "doesn't enqueue a job if the user has mention emails disabled" do
|
||||
notification.user.user_option.update_columns(email_direct: false)
|
||||
notification.user.user_option.update_columns(email_level: UserOption.email_level_types[:never])
|
||||
Jobs.expects(:enqueue_in).with(delay, :user_email, has_entry(type: type)).never
|
||||
NotificationEmailer.process_notification(notification)
|
||||
end
|
||||
@ -88,7 +88,7 @@ describe NotificationEmailer do
|
||||
include_examples "enqueue"
|
||||
|
||||
it "doesn't enqueue a job if the user has private message emails disabled" do
|
||||
notification.user.user_option.update_columns(email_private_messages: false)
|
||||
notification.user.user_option.update_columns(email_messages_level: UserOption.email_level_types[:never])
|
||||
Jobs.expects(:enqueue_in).with(delay, :user_email, has_entry(type: type)).never
|
||||
NotificationEmailer.process_notification(notification)
|
||||
end
|
||||
|
@ -54,15 +54,15 @@ describe UserAnonymizer do
|
||||
|
||||
it "turns off all notifications" do
|
||||
user.user_option.update_columns(
|
||||
email_always: true
|
||||
email_level: UserOption.email_level_types[:always],
|
||||
email_messages_level: UserOption.email_level_types[:always]
|
||||
)
|
||||
|
||||
make_anonymous
|
||||
user.reload
|
||||
expect(user.user_option.email_digests).to eq(false)
|
||||
expect(user.user_option.email_private_messages).to eq(false)
|
||||
expect(user.user_option.email_direct).to eq(false)
|
||||
expect(user.user_option.email_always).to eq(false)
|
||||
expect(user.user_option.email_level).to eq(UserOption.email_level_types[:never])
|
||||
expect(user.user_option.email_messages_level).to eq(UserOption.email_level_types[:never])
|
||||
expect(user.user_option.mailing_list_mode).to eq(false)
|
||||
end
|
||||
|
||||
|
@ -107,7 +107,7 @@ describe UserUpdater do
|
||||
seq = user.user_option.theme_key_seq
|
||||
|
||||
val = updater.update(bio_raw: 'my new bio',
|
||||
email_always: 'true',
|
||||
email_level: UserOption.email_level_types[:always],
|
||||
mailing_list_mode: true,
|
||||
digest_after_minutes: "45",
|
||||
new_topic_duration_minutes: 100,
|
||||
@ -123,7 +123,7 @@ describe UserUpdater do
|
||||
user.reload
|
||||
|
||||
expect(user.user_profile.bio_raw).to eq 'my new bio'
|
||||
expect(user.user_option.email_always).to eq true
|
||||
expect(user.user_option.email_level).to eq UserOption.email_level_types[:always]
|
||||
expect(user.user_option.mailing_list_mode).to eq true
|
||||
expect(user.user_option.digest_after_minutes).to eq 45
|
||||
expect(user.user_option.new_topic_duration_minutes).to eq 100
|
||||
|
@ -168,9 +168,8 @@ export default {
|
||||
can_delete_all_posts: false,
|
||||
locale: "",
|
||||
email_digests: true,
|
||||
email_private_messages: true,
|
||||
email_direct: true,
|
||||
email_always: true,
|
||||
email_messages_level: 0,
|
||||
email_level: 1,
|
||||
digest_after_minutes: 10080,
|
||||
mailing_list_mode: false,
|
||||
auto_track_topics_after_msecs: 60000,
|
||||
@ -2507,12 +2506,11 @@ export default {
|
||||
featured_user_badge_ids: [17],
|
||||
user_option: {
|
||||
user_id: 5,
|
||||
email_always: false,
|
||||
mailing_list_mode: false,
|
||||
mailing_list_mode_frequency: 1,
|
||||
email_digests: true,
|
||||
email_private_messages: true,
|
||||
email_direct: true,
|
||||
email_messages_level: 0,
|
||||
email_level: 1,
|
||||
external_links_in_new_tab: false,
|
||||
dynamic_favicon: false,
|
||||
enable_quoting: true,
|
||||
|
Loading…
Reference in New Issue
Block a user