Add ability to email announcements to all users (#33928)
parent
d2ce9a6064
commit
5a100bf38f
|
@ -0,0 +1,18 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class Admin::Announcements::DistributionsController < Admin::BaseController
|
||||||
|
before_action :set_announcement
|
||||||
|
|
||||||
|
def create
|
||||||
|
authorize @announcement, :distribute?
|
||||||
|
@announcement.touch(:notification_sent_at)
|
||||||
|
Admin::DistributeAnnouncementNotificationWorker.perform_async(@announcement.id)
|
||||||
|
redirect_to admin_announcements_path
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_announcement
|
||||||
|
@announcement = Announcement.find(params[:announcement_id])
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,16 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class Admin::Announcements::PreviewsController < Admin::BaseController
|
||||||
|
before_action :set_announcement
|
||||||
|
|
||||||
|
def show
|
||||||
|
authorize @announcement, :distribute?
|
||||||
|
@user_count = @announcement.scope_for_notification.count
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_announcement
|
||||||
|
@announcement = Announcement.find(params[:announcement_id])
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,17 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class Admin::Announcements::TestsController < Admin::BaseController
|
||||||
|
before_action :set_announcement
|
||||||
|
|
||||||
|
def create
|
||||||
|
authorize @announcement, :distribute?
|
||||||
|
UserMailer.announcement_published(current_user, @announcement).deliver_later!
|
||||||
|
redirect_to admin_announcements_path
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_announcement
|
||||||
|
@announcement = Announcement.find(params[:announcement_id])
|
||||||
|
end
|
||||||
|
end
|
|
@ -219,6 +219,15 @@ class UserMailer < Devise::Mailer
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def announcement_published(user, announcement)
|
||||||
|
@resource = user
|
||||||
|
@announcement = announcement
|
||||||
|
|
||||||
|
I18n.with_locale(locale) do
|
||||||
|
mail subject: default_i18n_subject
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def default_devise_subject
|
def default_devise_subject
|
||||||
|
|
|
@ -5,16 +5,17 @@
|
||||||
# Table name: announcements
|
# Table name: announcements
|
||||||
#
|
#
|
||||||
# id :bigint(8) not null, primary key
|
# id :bigint(8) not null, primary key
|
||||||
# text :text default(""), not null
|
|
||||||
# published :boolean default(FALSE), not null
|
|
||||||
# all_day :boolean default(FALSE), not null
|
# all_day :boolean default(FALSE), not null
|
||||||
|
# ends_at :datetime
|
||||||
|
# notification_sent_at :datetime
|
||||||
|
# published :boolean default(FALSE), not null
|
||||||
|
# published_at :datetime
|
||||||
# scheduled_at :datetime
|
# scheduled_at :datetime
|
||||||
# starts_at :datetime
|
# starts_at :datetime
|
||||||
# ends_at :datetime
|
# status_ids :bigint(8) is an Array
|
||||||
|
# text :text default(""), not null
|
||||||
# created_at :datetime not null
|
# created_at :datetime not null
|
||||||
# updated_at :datetime not null
|
# updated_at :datetime not null
|
||||||
# published_at :datetime
|
|
||||||
# status_ids :bigint(8) is an Array
|
|
||||||
#
|
#
|
||||||
|
|
||||||
class Announcement < ApplicationRecord
|
class Announcement < ApplicationRecord
|
||||||
|
@ -54,6 +55,10 @@ class Announcement < ApplicationRecord
|
||||||
update!(published: false, scheduled_at: nil)
|
update!(published: false, scheduled_at: nil)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def notification_sent?
|
||||||
|
notification_sent_at.present?
|
||||||
|
end
|
||||||
|
|
||||||
def mentions
|
def mentions
|
||||||
@mentions ||= Account.from_text(text)
|
@mentions ||= Account.from_text(text)
|
||||||
end
|
end
|
||||||
|
@ -86,6 +91,10 @@ class Announcement < ApplicationRecord
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def scope_for_notification
|
||||||
|
User.confirmed.joins(:account).merge(Account.without_suspended)
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def grouped_ordered_announcement_reactions
|
def grouped_ordered_announcement_reactions
|
||||||
|
|
|
@ -16,4 +16,8 @@ class AnnouncementPolicy < ApplicationPolicy
|
||||||
def destroy?
|
def destroy?
|
||||||
role.can?(:manage_announcements)
|
role.can?(:manage_announcements)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def distribute?
|
||||||
|
record.published? && !record.notification_sent? && role.can?(:manage_settings)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -10,6 +10,8 @@
|
||||||
= l(announcement.created_at)
|
= l(announcement.created_at)
|
||||||
|
|
||||||
%div
|
%div
|
||||||
|
- if can?(:distribute, announcement)
|
||||||
|
= table_link_to 'mail', t('admin.terms_of_service.notify_users'), admin_announcement_preview_path(announcement)
|
||||||
- if can?(:update, announcement)
|
- if can?(:update, announcement)
|
||||||
- if announcement.published?
|
- if announcement.published?
|
||||||
= table_link_to 'toggle_off', t('admin.announcements.unpublish'), unpublish_admin_announcement_path(announcement), method: :post, data: { confirm: t('admin.accounts.are_you_sure') }
|
= table_link_to 'toggle_off', t('admin.announcements.unpublish'), unpublish_admin_announcement_path(announcement), method: :post, data: { confirm: t('admin.accounts.are_you_sure') }
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
- content_for :page_title do
|
||||||
|
= t('admin.announcements.preview.title')
|
||||||
|
|
||||||
|
- content_for :heading_actions do
|
||||||
|
.back-link
|
||||||
|
= link_to admin_announcements_path do
|
||||||
|
= material_symbol 'chevron_left'
|
||||||
|
= t('admin.announcements.back')
|
||||||
|
|
||||||
|
%p.lead
|
||||||
|
= t('admin.announcements.preview.explanation_html', count: @user_count, display_count: number_with_delimiter(@user_count))
|
||||||
|
|
||||||
|
.prose
|
||||||
|
= linkify(@announcement.text)
|
||||||
|
|
||||||
|
%hr.spacer/
|
||||||
|
|
||||||
|
.content__heading__actions
|
||||||
|
= link_to t('admin.terms_of_service.preview.send_preview', email: current_user.email), admin_announcement_test_path(@announcement), method: :post, class: 'button button-secondary'
|
||||||
|
= link_to t('admin.terms_of_service.preview.send_to_all', count: @user_count, display_count: number_with_delimiter(@user_count)), admin_announcement_distribution_path(@announcement), method: :post, class: 'button', data: { confirm: t('admin.reports.are_you_sure') }
|
|
@ -0,0 +1,12 @@
|
||||||
|
= content_for :heading do
|
||||||
|
= render 'application/mailer/heading',
|
||||||
|
image_url: frontend_asset_url('images/mailer-new/heading/user.png'),
|
||||||
|
title: t('user_mailer.announcement_published.title', domain: site_hostname)
|
||||||
|
%table.email-w-full{ cellspacing: 0, cellpadding: 0, border: 0, role: 'presentation' }
|
||||||
|
%tr
|
||||||
|
%td.email-body-padding-td
|
||||||
|
%table.email-inner-card-table{ cellspacing: 0, cellpadding: 0, border: 0, role: 'presentation' }
|
||||||
|
%tr
|
||||||
|
%td.email-inner-card-td.email-prose
|
||||||
|
%p= t('user_mailer.announcement_published.description', domain: site_hostname)
|
||||||
|
= linkify(@announcement.text)
|
|
@ -0,0 +1,7 @@
|
||||||
|
<%= t('user_mailer.announcement_published.title') %>
|
||||||
|
|
||||||
|
===
|
||||||
|
|
||||||
|
<%= t('user_mailer.announcement_published.description', domain: site_hostname) %>
|
||||||
|
|
||||||
|
<%= @announcement.text %>
|
|
@ -0,0 +1,15 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class Admin::DistributeAnnouncementNotificationWorker
|
||||||
|
include Sidekiq::Worker
|
||||||
|
|
||||||
|
def perform(announcement_id)
|
||||||
|
announcement = Announcement.find(announcement_id)
|
||||||
|
|
||||||
|
announcement.scope_for_notification.find_each do |user|
|
||||||
|
UserMailer.announcement_published(user, announcement).deliver_later!
|
||||||
|
end
|
||||||
|
rescue ActiveRecord::RecordNotFound
|
||||||
|
true
|
||||||
|
end
|
||||||
|
end
|
|
@ -309,6 +309,7 @@ en:
|
||||||
title: Audit log
|
title: Audit log
|
||||||
unavailable_instance: "(domain name unavailable)"
|
unavailable_instance: "(domain name unavailable)"
|
||||||
announcements:
|
announcements:
|
||||||
|
back: Back to announcements
|
||||||
destroyed_msg: Announcement successfully deleted!
|
destroyed_msg: Announcement successfully deleted!
|
||||||
edit:
|
edit:
|
||||||
title: Edit announcement
|
title: Edit announcement
|
||||||
|
@ -317,6 +318,9 @@ en:
|
||||||
new:
|
new:
|
||||||
create: Create announcement
|
create: Create announcement
|
||||||
title: New announcement
|
title: New announcement
|
||||||
|
preview:
|
||||||
|
explanation_html: 'The email will be sent to <strong>%{display_count} users</strong>. The following text will be included in the e-mail:'
|
||||||
|
title: Preview announcement notification
|
||||||
publish: Publish
|
publish: Publish
|
||||||
published_msg: Announcement successfully published!
|
published_msg: Announcement successfully published!
|
||||||
scheduled_for: Scheduled for %{time}
|
scheduled_for: Scheduled for %{time}
|
||||||
|
@ -1906,6 +1910,10 @@ en:
|
||||||
recovery_instructions_html: If you ever lose access to your phone, you can use one of the recovery codes below to regain access to your account. <strong>Keep the recovery codes safe</strong>. For example, you may print them and store them with other important documents.
|
recovery_instructions_html: If you ever lose access to your phone, you can use one of the recovery codes below to regain access to your account. <strong>Keep the recovery codes safe</strong>. For example, you may print them and store them with other important documents.
|
||||||
webauthn: Security keys
|
webauthn: Security keys
|
||||||
user_mailer:
|
user_mailer:
|
||||||
|
announcement_published:
|
||||||
|
description: 'The administrators of %{domain} are making an announcement:'
|
||||||
|
subject: Service announcement
|
||||||
|
title: "%{domain} service announcement"
|
||||||
appeal_approved:
|
appeal_approved:
|
||||||
action: Account Settings
|
action: Account Settings
|
||||||
explanation: The appeal of the strike against your account on %{strike_date} that you submitted on %{appeal_date} has been approved. Your account is once again in good standing.
|
explanation: The appeal of the strike against your account on %{strike_date} that you submitted on %{appeal_date} has been approved. Your account is once again in good standing.
|
||||||
|
|
|
@ -50,6 +50,10 @@ namespace :admin do
|
||||||
post :publish
|
post :publish
|
||||||
post :unpublish
|
post :unpublish
|
||||||
end
|
end
|
||||||
|
|
||||||
|
resource :preview, only: [:show], module: :announcements
|
||||||
|
resource :test, only: [:create], module: :announcements
|
||||||
|
resource :distribution, only: [:create], module: :announcements
|
||||||
end
|
end
|
||||||
|
|
||||||
with_options to: redirect('/admin/settings/branding') do
|
with_options to: redirect('/admin/settings/branding') do
|
||||||
|
|
|
@ -0,0 +1,7 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class AddNotificationSentAtToAnnouncements < ActiveRecord::Migration[8.0]
|
||||||
|
def change
|
||||||
|
add_column :announcements, :notification_sent_at, :datetime
|
||||||
|
end
|
||||||
|
end
|
|
@ -258,6 +258,7 @@ ActiveRecord::Schema[8.0].define(version: 2025_03_05_074104) do
|
||||||
t.datetime "updated_at", precision: nil, null: false
|
t.datetime "updated_at", precision: nil, null: false
|
||||||
t.datetime "published_at", precision: nil
|
t.datetime "published_at", precision: nil
|
||||||
t.bigint "status_ids", array: true
|
t.bigint "status_ids", array: true
|
||||||
|
t.datetime "notification_sent_at"
|
||||||
end
|
end
|
||||||
|
|
||||||
create_table "annual_report_statuses_per_account_counts", force: :cascade do |t|
|
create_table "annual_report_statuses_per_account_counts", force: :cascade do |t|
|
||||||
|
|
|
@ -317,4 +317,16 @@ RSpec.describe UserMailer do
|
||||||
.and(have_body_text(I18n.t('user_mailer.terms_of_service_changed.changelog')))
|
.and(have_body_text(I18n.t('user_mailer.terms_of_service_changed.changelog')))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe '#announcement_published' do
|
||||||
|
let(:announcement) { Fabricate :announcement }
|
||||||
|
let(:mail) { described_class.announcement_published(receiver, announcement) }
|
||||||
|
|
||||||
|
it 'renders announcement_published mail' do
|
||||||
|
expect(mail)
|
||||||
|
.to be_present
|
||||||
|
.and(have_subject(I18n.t('user_mailer.announcement_published.subject')))
|
||||||
|
.and(have_body_text(I18n.t('user_mailer.announcement_published.description', domain: Rails.configuration.x.local_domain)))
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe 'Admin Announcement Mail Distributions' do
|
||||||
|
let(:user) { Fabricate(:admin_user) }
|
||||||
|
let(:announcement) { Fabricate(:announcement, notification_sent_at: nil) }
|
||||||
|
|
||||||
|
before { sign_in(user) }
|
||||||
|
|
||||||
|
describe 'Sending an announcement notification', :inline_jobs do
|
||||||
|
it 'marks the announcement as notified and sends the email' do
|
||||||
|
visit admin_announcement_preview_path(announcement)
|
||||||
|
expect(page)
|
||||||
|
.to have_title(I18n.t('admin.announcements.preview.title'))
|
||||||
|
|
||||||
|
emails = capture_emails do
|
||||||
|
expect { click_on I18n.t('admin.terms_of_service.preview.send_to_all', count: 1, display_count: 1) }
|
||||||
|
.to(change { announcement.reload.notification_sent_at })
|
||||||
|
end
|
||||||
|
expect(emails.first)
|
||||||
|
.to be_present
|
||||||
|
.and(deliver_to(user.email))
|
||||||
|
expect(page)
|
||||||
|
.to have_title(I18n.t('admin.announcements.title'))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,18 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe 'Admin Announcements Mail Previews' do
|
||||||
|
let(:announcement) { Fabricate(:announcement, notification_sent_at: nil) }
|
||||||
|
|
||||||
|
before { sign_in(admin_user) }
|
||||||
|
|
||||||
|
describe 'Viewing Announcements Mail previews' do
|
||||||
|
it 'shows the Announcement Mail preview page' do
|
||||||
|
visit admin_announcement_preview_path(announcement)
|
||||||
|
|
||||||
|
expect(page)
|
||||||
|
.to have_title(I18n.t('admin.announcements.preview.title'))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,25 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe 'Admin TermsOfService Tests' do
|
||||||
|
let(:user) { Fabricate(:admin_user) }
|
||||||
|
let(:announcement) { Fabricate(:announcement, notification_sent_at: nil) }
|
||||||
|
|
||||||
|
before { sign_in(user) }
|
||||||
|
|
||||||
|
describe 'Sending test Announcement email', :inline_jobs do
|
||||||
|
it 'generates the test email' do
|
||||||
|
visit admin_announcement_preview_path(announcement)
|
||||||
|
expect(page)
|
||||||
|
.to have_title(I18n.t('admin.announcements.preview.title'))
|
||||||
|
|
||||||
|
emails = capture_emails { click_on I18n.t('admin.terms_of_service.preview.send_preview', email: user.email) }
|
||||||
|
expect(emails.first)
|
||||||
|
.to be_present
|
||||||
|
.and(deliver_to(user.email))
|
||||||
|
expect(page)
|
||||||
|
.to have_title(I18n.t('admin.announcements.title'))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,32 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe Admin::DistributeAnnouncementNotificationWorker do
|
||||||
|
let(:worker) { described_class.new }
|
||||||
|
|
||||||
|
describe '#perform' do
|
||||||
|
context 'with missing record' do
|
||||||
|
it 'runs without error' do
|
||||||
|
expect { worker.perform(nil) }.to_not raise_error
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'with valid announcement' do
|
||||||
|
let(:announcement) { Fabricate(:announcement) }
|
||||||
|
let!(:user) { Fabricate :user, confirmed_at: 3.days.ago }
|
||||||
|
|
||||||
|
it 'sends the announcement via email', :inline_jobs do
|
||||||
|
emails = capture_emails { worker.perform(announcement.id) }
|
||||||
|
|
||||||
|
expect(emails.size)
|
||||||
|
.to eq(1)
|
||||||
|
expect(emails.first)
|
||||||
|
.to have_attributes(
|
||||||
|
to: [user.email],
|
||||||
|
subject: I18n.t('user_mailer.announcement_published.subject')
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue