aboutsummaryrefslogblamecommitdiffstats
path: root/actionmailer/lib/action_mailer/inline_preview_interceptor.rb
blob: ca4cffc7b58b960171141574f6ad91cbdab520e8 (plain) (tree)
1
2
3
4
5
6
7
8






                                                                                  
                                                                 




















































                                                                                            
require 'base64'

module ActionMailer
  # Implements a mailer preview interceptor that converts image tag src attributes
  # that use inline cid: style urls to data: style urls so that they are visible
  # when previewing a HTML email in a web browser.
  #
  # This interceptor is enabled by default, to remove it from the
  # <tt>ActionMailer::Base.preview_interceptors</tt> array:
  #
  #   ActionMailer::Base.preview_interceptors.delete(ActionMailer::InlinePreviewInterceptor)
  #
  class InlinePreviewInterceptor
    PATTERN  = /src=(?:"cid:[^"]+"|'cid:[^']+')/i

    include Base64

    def self.previewing_email(message) #:nodoc:
      new(message).transform!
    end

    def initialize(message) #:nodoc:
      @message = message
    end

    def transform! #:nodoc:
      return message if html_part.blank?

      html_source.gsub!(PATTERN) do |match|
        if part = find_part(match[9..-2])
          %[src="#{data_url(part)}"]
        else
          match
        end
      end

      message
    end

    private
      def message
        @message
      end

      def html_part
        @html_part ||= message.html_part
      end

      def html_source
        html_part.body.raw_source
      end

      def data_url(part)
        "data:#{part.mime_type};base64,#{strict_encode64(part.body.raw_source)}"
      end

      def find_part(cid)
        message.all_parts.find{ |p| p.attachment? && p.cid == cid }
      end
  end
end