I have a problem with changing the ActionMailer::Base.default_url_options = {:host => host} during runtime in one of my projects.
Setup: I have multiple subdomains which are using the same Rails-Application in the backend. I want to send my Devise mails over a Sidekiq queue to the users. The Devise mails (confirmation, reset-password) contain links and these links need the specific subdomain to be correct.
My environment
rails (4.2.0)
sidekiq (3.3.1)
devise (3.4.1)
devise-async (0.9.0)
I have a before_action in my application_controller
class ApplicationController < ActionController::Base
before_action :set_action_mailer_default_url_options
private
def set_action_mailer_default_url_options
host = "my-logic-to-get-the-correct-host"
ActionMailer::Base.default_url_options = {:host => host}
end
end
Now if I want to reset my password I always get the default url options, which I have specified in the environments file. If I remove the default_url_options from my environments I get the default Devise error in my sidekiq logs.
ActionView::Template::Error: Missing host to link to! Please provide the :host parameter, set default_url_options[:host], or set :only_path to true
The host is always set correctly in the controller. I already debugged that. It seems that the host is not passed to sidekiq.. any idea how I can fix this?
I can't save the subdomain somewhere, because an user can trigger the emails from different subdomains and not always the same. I need a way to tell Devise which host should be used to send a specific email. Can I override the Devise mailer and pass the host or something similar?
Following solution is not possible in my case: Dynamic domain name in Rails action mailer while using sidekiq
Btw: the complete workflow with devise, devise-async and sidekiq itself works (in development, staging and production). Only the host of the links is not correct -> and that is my big problem :-)..
I ended up in following solution. This solution contains the case that you have multiple customers and these customers could have own email-settings and own domains / subdomains for the urls in the mails.
I used an own mailer class and hooked in possible custom email settings of a customer:
config/initializers/devise.rb:
config/secrets.yml:
app/mailers/devise_mailer.rb:
Change the urls in the email view templates:
app/views/devise/mailer/confirmation_instructions.html.erb
Simple change the link with the new url-options..
app/views/devise/reset_password_instructions.html.erb
This was my solution.. if you have any question, just let me know.
EDIT: Don't forget to set default url-options in your environment. For example in your development.rb - config/environments/development.rb: