Chris@1464: require 'active_record' Chris@1464: Chris@1464: module ActiveRecord Chris@1464: class Base Chris@1464: include Redmine::I18n Chris@1464: # Translate attribute names for validation errors display Chris@1464: def self.human_attribute_name(attr, *args) Chris@1464: attr = attr.to_s.sub(/_id$/, '') Chris@1464: Chris@1464: l("field_#{name.underscore.gsub('/', '_')}_#{attr}", :default => ["field_#{attr}".to_sym, attr]) Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: # Undefines private Kernel#open method to allow using `open` scopes in models. Chris@1464: # See Defect #11545 (http://www.redmine.org/issues/11545) for details. Chris@1464: class Base Chris@1464: class << self Chris@1464: undef open Chris@1464: end Chris@1464: end Chris@1464: class Relation ; undef open ; end Chris@1464: end Chris@1464: Chris@1464: module ActionView Chris@1464: module Helpers Chris@1464: module DateHelper Chris@1464: # distance_of_time_in_words breaks when difference is greater than 30 years Chris@1464: def distance_of_date_in_words(from_date, to_date = 0, options = {}) Chris@1464: from_date = from_date.to_date if from_date.respond_to?(:to_date) Chris@1464: to_date = to_date.to_date if to_date.respond_to?(:to_date) Chris@1464: distance_in_days = (to_date - from_date).abs Chris@1464: Chris@1464: I18n.with_options :locale => options[:locale], :scope => :'datetime.distance_in_words' do |locale| Chris@1464: case distance_in_days Chris@1464: when 0..60 then locale.t :x_days, :count => distance_in_days.round Chris@1464: when 61..720 then locale.t :about_x_months, :count => (distance_in_days / 30).round Chris@1464: else locale.t :over_x_years, :count => (distance_in_days / 365).floor Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: class Resolver Chris@1464: def find_all(name, prefix=nil, partial=false, details={}, key=nil, locals=[]) Chris@1464: cached(key, [name, prefix, partial], details, locals) do Chris@1464: if details[:formats] & [:xml, :json] Chris@1464: details = details.dup Chris@1464: details[:formats] = details[:formats].dup + [:api] Chris@1464: end Chris@1464: find_templates(name, prefix, partial, details) Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: # Do not HTML escape text templates Chris@1464: module ActionView Chris@1464: class Template Chris@1464: module Handlers Chris@1464: class ERB Chris@1464: def call(template) Chris@1464: if template.source.encoding_aware? Chris@1464: # First, convert to BINARY, so in case the encoding is Chris@1464: # wrong, we can still find an encoding tag Chris@1464: # (<%# encoding %>) inside the String using a regular Chris@1464: # expression Chris@1464: template_source = template.source.dup.force_encoding("BINARY") Chris@1464: Chris@1464: erb = template_source.gsub(ENCODING_TAG, '') Chris@1464: encoding = $2 Chris@1464: Chris@1464: erb.force_encoding valid_encoding(template.source.dup, encoding) Chris@1464: Chris@1464: # Always make sure we return a String in the default_internal Chris@1464: erb.encode! Chris@1464: else Chris@1464: erb = template.source.dup Chris@1464: end Chris@1464: Chris@1464: self.class.erb_implementation.new( Chris@1464: erb, Chris@1464: :trim => (self.class.erb_trim_mode == "-"), Chris@1464: :escape => template.identifier =~ /\.text/ # only escape HTML templates Chris@1464: ).src Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: ActionView::Base.field_error_proc = Proc.new{ |html_tag, instance| html_tag || ''.html_safe } Chris@1464: Chris@1464: # HTML5: is invalid, use instead Chris@1464: module ActionView Chris@1464: module Helpers Chris@1464: class InstanceTag Chris@1464: private Chris@1464: def add_options_with_non_empty_blank_option(option_tags, options, value = nil) Chris@1464: if options[:include_blank] == true Chris@1464: options = options.dup Chris@1464: options[:include_blank] = ' '.html_safe Chris@1464: end Chris@1464: add_options_without_non_empty_blank_option(option_tags, options, value) Chris@1464: end Chris@1464: alias_method_chain :add_options, :non_empty_blank_option Chris@1464: end Chris@1464: Chris@1464: module FormTagHelper Chris@1464: def select_tag_with_non_empty_blank_option(name, option_tags = nil, options = {}) Chris@1464: if options.delete(:include_blank) Chris@1464: options[:prompt] = ' '.html_safe Chris@1464: end Chris@1464: select_tag_without_non_empty_blank_option(name, option_tags, options) Chris@1464: end Chris@1464: alias_method_chain :select_tag, :non_empty_blank_option Chris@1464: end Chris@1464: Chris@1464: module FormOptionsHelper Chris@1464: def options_for_select_with_non_empty_blank_option(container, selected = nil) Chris@1464: if container.is_a?(Array) Chris@1464: container = container.map {|element| element.blank? ? [" ".html_safe, ""] : element} Chris@1464: end Chris@1464: options_for_select_without_non_empty_blank_option(container, selected) Chris@1464: end Chris@1464: alias_method_chain :options_for_select, :non_empty_blank_option Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: require 'mail' Chris@1464: Chris@1464: module DeliveryMethods Chris@1464: class AsyncSMTP < ::Mail::SMTP Chris@1464: def deliver!(*args) Chris@1464: Thread.start do Chris@1464: super *args Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: class AsyncSendmail < ::Mail::Sendmail Chris@1464: def deliver!(*args) Chris@1464: Thread.start do Chris@1464: super *args Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: class TmpFile Chris@1464: def initialize(*args); end Chris@1464: Chris@1464: def deliver!(mail) Chris@1464: dest_dir = File.join(Rails.root, 'tmp', 'emails') Chris@1464: Dir.mkdir(dest_dir) unless File.directory?(dest_dir) Chris@1464: File.open(File.join(dest_dir, mail.message_id.gsub(/[<>]/, '') + '.eml'), 'wb') {|f| f.write(mail.encoded) } Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: ActionMailer::Base.add_delivery_method :async_smtp, DeliveryMethods::AsyncSMTP Chris@1464: ActionMailer::Base.add_delivery_method :async_sendmail, DeliveryMethods::AsyncSendmail Chris@1464: ActionMailer::Base.add_delivery_method :tmp_file, DeliveryMethods::TmpFile Chris@1464: Chris@1464: # Changes how sent emails are logged Chris@1464: # Rails doesn't log cc and bcc which is misleading when using bcc only (#12090) Chris@1464: module ActionMailer Chris@1464: class LogSubscriber < ActiveSupport::LogSubscriber Chris@1464: def deliver(event) Chris@1464: recipients = [:to, :cc, :bcc].inject("") do |s, header| Chris@1464: r = Array.wrap(event.payload[header]) Chris@1464: if r.any? Chris@1464: s << "\n #{header}: #{r.join(', ')}" Chris@1464: end Chris@1464: s Chris@1464: end Chris@1464: info("\nSent email \"#{event.payload[:subject]}\" (%1.fms)#{recipients}" % event.duration) Chris@1464: debug(event.payload[:mail]) Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: module ActionController Chris@1464: module MimeResponds Chris@1464: class Collector Chris@1464: def api(&block) Chris@1464: any(:xml, :json, &block) Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: end Chris@1464: Chris@1464: module ActionController Chris@1464: class Base Chris@1464: # Displays an explicit message instead of a NoMethodError exception Chris@1464: # when trying to start Redmine with an old session_store.rb Chris@1464: # TODO: remove it in a later version Chris@1464: def self.session=(*args) Chris@1464: $stderr.puts "Please remove config/initializers/session_store.rb and run `rake generate_secret_token`.\n" + Chris@1464: "Setting the session secret with ActionController.session= is no longer supported in Rails 3." Chris@1464: exit 1 Chris@1464: end Chris@1464: end Chris@1464: end