OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/app/mailers/user_mailer.rb

477 lines
15 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is a project management system.
# Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See doc/COPYRIGHT.rdoc for more details.
#++
class UserMailer < BaseMailer
helper :application, # for format_text
:work_packages, # for css classes
:custom_fields # for show_value
helper IssuesHelper
include OpenProject::LocaleHelper
# wrap in a lambda to allow changing at run-time
default from: Proc.new { Setting.mail_from }
def test_mail(user)
@welcome_url = url_for(controller: '/welcome')
headers['X-OpenProject-Type'] = 'Test'
with_locale_for(user) do
mail to: "\"#{user.name}\" <#{user.mail}>", subject: 'OpenProject Test'
end
end
def work_package_added(user, journal, author)
User.execute_as user do
work_package = journal.journable.reload
@issue = work_package # instance variable is used in the view
@journal = journal
set_work_package_headers(work_package)
message_id work_package, user
with_locale_for(user) do
mail_for_author author, to: user.mail, subject: subject_for_work_package(work_package)
end
end
end
def work_package_updated(user, journal, author = User.current)
User.execute_as user do
work_package = journal.journable.reload
# instance variables are used in the view
@issue = work_package
@journal = journal
set_work_package_headers(work_package)
message_id journal, user
references work_package, user
with_locale_for(user) do
mail_for_author author, to: user.mail, subject: subject_for_work_package(work_package)
end
end
end
def work_package_watcher_added(work_package, user, watcher_setter)
User.execute_as user do
@issue = work_package
@watcher_setter = watcher_setter
set_work_package_headers(work_package)
message_id work_package, user
references work_package, user
with_locale_for(user) do
mail to: user.mail, subject: subject_for_work_package(work_package)
end
end
end
def password_lost(token)
return unless token.user # token's can have no user
@token = token
@reset_password_url = url_for(controller: '/account',
action: :lost_password,
token: @token.value)
open_project_headers 'Type' => 'Account'
user = @token.user
with_locale_for(user) do
subject = t(:mail_subject_lost_password, value: Setting.app_title)
mail to: user.mail, subject: subject
end
end
def copy_project_failed(user, source_project, target_project_name, errors)
@source_project = source_project
@target_project_name = target_project_name
@errors = errors
open_project_headers 'Source-Project' => source_project.identifier,
'Author' => user.login
message_id source_project, user
with_locale_for(user) do
subject = I18n.t('copy_project.failed', source_project_name: source_project.name)
mail to: user.mail, subject: subject
end
end
def copy_project_succeeded(user, source_project, target_project, errors)
@source_project = source_project
@target_project = target_project
@errors = errors
open_project_headers 'Source-Project' => source_project.identifier,
'Target-Project' => target_project.identifier,
'Author' => user.login
message_id target_project, user
with_locale_for(user) do
subject = I18n.t('copy_project.succeeded', target_project_name: target_project.name)
mail to: user.mail, subject: subject
end
end
def news_added(user, news, author)
@news = news
open_project_headers 'Type' => 'News'
open_project_headers 'Project' => @news.project.identifier if @news.project
message_id @news, user
with_locale_for(user) do
subject = "#{News.model_name.human}: #{@news.title}"
subject = "[#{@news.project.name}] #{subject}" if @news.project
mail_for_author author, to: user.mail, subject: subject
end
end
def user_signed_up(token)
return unless token.user
@token = token
@activation_url = url_for(controller: '/account',
action: :activate,
token: @token.value)
open_project_headers 'Type' => 'Account'
user = token.user
with_locale_for(user) do
subject = t(:mail_subject_register, value: Setting.app_title)
mail to: user.mail, subject: subject
end
end
def news_comment_added(user, comment, author)
@comment = comment
@news = @comment.commented
open_project_headers 'Project' => @news.project.identifier if @news.project
message_id @comment, user
references @news, user
with_locale_for(user) do
subject = "#{News.model_name.human}: #{@news.title}"
subject = "Re: [#{@news.project.name}] #{subject}" if @news.project
mail_for_author author, to: user.mail, subject: subject
end
end
def wiki_content_added(user, wiki_content, author)
@wiki_content = wiki_content
open_project_headers 'Project' => @wiki_content.project.identifier,
'Wiki-Page-Id' => @wiki_content.page.id,
'Type' => 'Wiki'
message_id @wiki_content, user
with_locale_for(user) do
subject = "[#{@wiki_content.project.name}] #{t(:mail_subject_wiki_content_added, id: @wiki_content.page.title)}"
mail_for_author author, to: user.mail, subject: subject
end
end
def wiki_content_updated(user, wiki_content, author)
@wiki_content = wiki_content
@wiki_diff_url = url_for(controller: '/wiki',
action: :diff,
project_id: wiki_content.project,
id: wiki_content.page.title,
# using wiki_content.version + 1 because at this point the journal is not saved yet
version: wiki_content.version + 1)
open_project_headers 'Project' => @wiki_content.project.identifier,
'Wiki-Page-Id' => @wiki_content.page.id,
'Type' => 'Wiki'
message_id @wiki_content, user
with_locale_for(user) do
subject = "[#{@wiki_content.project.name}] #{t(:mail_subject_wiki_content_updated, id: @wiki_content.page.title)}"
mail_for_author author, to: user.mail, subject: subject
end
end
def message_posted(user, message, author)
@message = message
@message_url = topic_url(@message.root, r: @message.id, anchor: "message-#{@message.id}")
open_project_headers 'Project' => @message.project.identifier,
'Wiki-Page-Id' => @message.parent_id || @message.id,
'Type' => 'Forum'
message_id @message, user
references @message.parent, user if @message.parent
with_locale_for(user) do
subject = "[#{@message.board.project.name} - #{@message.board.name} - msg#{@message.root.id}] #{@message.subject}"
mail_for_author author, to: user.mail, subject: subject
end
end
def account_activated(user)
@user = user
open_project_headers 'Type' => 'Account'
with_locale_for(user) do
subject = t(:mail_subject_register, value: Setting.app_title)
mail to: user.mail, subject: subject
end
end
def account_information(user, password)
@user = user
@password = password
open_project_headers 'Type' => 'Account'
with_locale_for(user) do
subject = t(:mail_subject_register, value: Setting.app_title)
mail to: user.mail, subject: subject
end
end
def account_activation_requested(admin, user)
@user = user
@activation_url = url_for(controller: '/users',
action: :index,
status: User::STATUSES[:registered],
sort: 'created_at:desc')
open_project_headers 'Type' => 'Account'
with_locale_for(admin) do
subject = t(:mail_subject_account_activation_request, value: Setting.app_title)
mail to: admin.mail, subject: subject
end
end
def reminder_mail(user, issues, days)
@issues = issues
@days = days
@assigned_issues_url = url_for(controller: :work_packages,
action: :index,
set_filter: 1,
assigned_to_id: user.id,
sort: 'due_date:asc')
open_project_headers 'Type' => 'Issue'
with_locale_for(user) do
subject = t(:mail_subject_reminder, count: @issues.size, days: @days)
mail to: user.mail, subject: subject
end
end
# Activates/deactivates email deliveries during +block+
def self.with_deliveries(temporary_state = true, &_block)
old_state = ActionMailer::Base.perform_deliveries
ActionMailer::Base.perform_deliveries = temporary_state
yield
ensure
ActionMailer::Base.perform_deliveries = old_state
end
def self.generate_message_id(object, user)
# id + timestamp should reduce the odds of a collision
# as far as we don't send multiple emails for the same object
journable = (object.is_a? Journal) ? object.journable : object
timestamp = mail_timestamp(object)
hash = 'openproject'\
'.'\
"#{journable.class.name.demodulize.underscore}"\
'-'\
"#{user.id}"\
'-'\
"#{journable.id}"\
'.'\
"#{timestamp.strftime('%Y%m%d%H%M%S')}"
host = Setting.mail_from.to_s.gsub(%r{\A.*@}, '')
host = "#{::Socket.gethostname}.openproject" if host.empty?
"#{hash}@#{host}"
end
private
def subject_for_work_package(work_package)
subject = "[#{work_package.project.name} - #{work_package.type.name} ##{work_package.id}] "
subject << "(#{work_package.status.name}) " << work_package.subject
end
# like #mail, but contains special author based filters
# currently only:
# - remove_self_notifications
# might be refactored at a later time to be as generic as Interceptors
def mail_for_author(author, headers = {}, &block)
message = mail headers, &block
self.class.remove_self_notifications(message, author)
message
end
def self.remove_self_notifications(message, author)
if author.pref && author.pref[:no_self_notified]
message.to = message.to.reject { |address| address == author.mail } if message.to.present?
end
end
def self.mail_timestamp(object)
if object.respond_to? :created_at
timestamp = object.send(object.respond_to?(:created_at) ? :created_at : :updated_at)
else
timestamp = object.send(object.respond_to?(:created_on) ? :created_on : :updated_on)
end
end
def self.host
if OpenProject::Configuration.rails_relative_url_root.blank?
Setting.host_name
else
Setting.host_name.to_s.gsub(%r{\/.*\z}, '')
end
end
def self.protocol
Setting.protocol
end
def self.default_url_options
options = super.merge host: host, protocol: protocol
unless OpenProject::Configuration.rails_relative_url_root.blank?
options[:script_name] = OpenProject::Configuration.rails_relative_url_root
end
options
end
def message_id(object, user)
headers['Message-ID'] = "<#{self.class.generate_message_id(object, user)}>"
end
def references(object, user)
headers['References'] = "<#{self.class.generate_message_id(object, user)}>"
end
def set_work_package_headers(work_package)
open_project_headers 'Project' => work_package.project.identifier,
'Issue-Id' => work_package.id,
'Issue-Author' => work_package.author.login,
'Type' => 'WorkPackage'
if work_package.assigned_to
open_project_headers 'Issue-Assignee' => work_package.assigned_to.login
end
end
# Prepends given fields with 'X-OpenProject-' to save some duplication
def open_project_headers(hash)
hash.each { |key, value| headers["X-OpenProject-#{key}"] = value.to_s }
end
end
##
# Interceptors
#
# These are registered in config/initializers/register_mail_interceptors.rb
#
# Unfortunately, this results in changes on the interceptor classes during development mode
# not being reflected until a server restart.
class DefaultHeadersInterceptor
def self.delivering_email(mail)
mail.headers(default_headers)
end
def self.default_headers
{
'X-Mailer' => 'OpenProject',
'X-OpenProject-Host' => Setting.host_name,
'X-OpenProject-Site' => Setting.app_title,
'Precedence' => 'bulk',
'Auto-Submitted' => 'auto-generated'
}
end
end
class DoNotSendMailsWithoutReceiverInterceptor
def self.delivering_email(mail)
receivers = [mail.to, mail.cc, mail.bcc]
# the above fields might be empty arrays (if entries have been removed
# by another interceptor) or nil, therefore checking for blank?
mail.perform_deliveries = false if receivers.all?(&:blank?)
end
end
# helper object for `rake redmine:send_reminders`
class DueIssuesReminder
def initialize(days = nil, project_id = nil, type_id = nil, user_ids = [])
@days = days ? days.to_i : 7
@project = Project.find_by(id: project_id)
@type = ::Type.find_by(id: type_id)
@user_ids = user_ids
end
def remind_users
s = ARCondition.new ["#{Status.table_name}.is_closed = ? AND #{WorkPackage.table_name}.due_date <= ?", false, @days.days.from_now.to_date]
s << "#{WorkPackage.table_name}.assigned_to_id IS NOT NULL"
s << ["#{WorkPackage.table_name}.assigned_to_id IN (?)", @user_ids] if @user_ids.any?
s << "#{Project.table_name}.status = #{Project::STATUS_ACTIVE}"
s << "#{WorkPackage.table_name}.project_id = #{@project.id}" if @project
s << "#{WorkPackage.table_name}.type_id = #{@type.id}" if @type
issues_by_assignee = WorkPackage.includes(:status, :assigned_to, :project, :type)
.where(s.conditions)
.references(:projects)
.group_by(&:assigned_to)
issues_by_assignee.each do |assignee, issues|
UserMailer.reminder_mail(assignee, issues, @days).deliver_now if assignee && assignee.active?
end
end
end