Chris@0: require 'active_record'
Chris@0:
Chris@0: module ActiveRecord
Chris@0: class Base
Chris@0: include Redmine::I18n
Chris@1115: # Translate attribute names for validation errors display
Chris@1115: def self.human_attribute_name(attr, *args)
Chris@1115: attr = attr.to_s.sub(/_id$/, '')
Chris@909:
Chris@1115: l("field_#{name.underscore.gsub('/', '_')}_#{attr}", :default => ["field_#{attr}".to_sym, attr])
Chris@0: end
Chris@0: end
Chris@0:
Chris@1115: # Undefines private Kernel#open method to allow using `open` scopes in models.
Chris@1115: # See Defect #11545 (http://www.redmine.org/issues/11545) for details.
Chris@1115: class Base
Chris@1115: class << self
Chris@1115: undef open
Chris@0: end
Chris@0: end
Chris@1115: class Relation ; undef open ; end
Chris@0: end
Chris@0:
Chris@0: module ActionView
Chris@0: module Helpers
Chris@0: module DateHelper
Chris@0: # distance_of_time_in_words breaks when difference is greater than 30 years
Chris@0: def distance_of_date_in_words(from_date, to_date = 0, options = {})
Chris@0: from_date = from_date.to_date if from_date.respond_to?(:to_date)
Chris@0: to_date = to_date.to_date if to_date.respond_to?(:to_date)
Chris@0: distance_in_days = (to_date - from_date).abs
Chris@0:
Chris@0: I18n.with_options :locale => options[:locale], :scope => :'datetime.distance_in_words' do |locale|
Chris@0: case distance_in_days
Chris@0: when 0..60 then locale.t :x_days, :count => distance_in_days.round
Chris@0: when 61..720 then locale.t :about_x_months, :count => (distance_in_days / 30).round
Chris@0: else locale.t :over_x_years, :count => (distance_in_days / 365).floor
Chris@0: end
Chris@0: end
Chris@0: end
Chris@0: end
Chris@0: end
Chris@1115:
Chris@1115: class Resolver
Chris@1115: def find_all(name, prefix=nil, partial=false, details={}, key=nil, locals=[])
Chris@1115: cached(key, [name, prefix, partial], details, locals) do
Chris@1115: if details[:formats] & [:xml, :json]
Chris@1115: details = details.dup
Chris@1115: details[:formats] = details[:formats].dup + [:api]
Chris@1115: end
Chris@1115: find_templates(name, prefix, partial, details)
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@0: end
Chris@0:
Chris@1115: # Do not HTML escape text templates
Chris@1115: module ActionView
Chris@1115: class Template
Chris@1115: module Handlers
Chris@1115: class ERB
Chris@1115: def call(template)
Chris@1115: if template.source.encoding_aware?
Chris@1115: # First, convert to BINARY, so in case the encoding is
Chris@1115: # wrong, we can still find an encoding tag
Chris@1115: # (<%# encoding %>) inside the String using a regular
Chris@1115: # expression
Chris@1115: template_source = template.source.dup.force_encoding("BINARY")
Chris@0:
Chris@1115: erb = template_source.gsub(ENCODING_TAG, '')
Chris@1115: encoding = $2
Chris@1115:
Chris@1115: erb.force_encoding valid_encoding(template.source.dup, encoding)
Chris@1115:
Chris@1115: # Always make sure we return a String in the default_internal
Chris@1115: erb.encode!
Chris@1115: else
Chris@1115: erb = template.source.dup
Chris@1115: end
Chris@1115:
Chris@1115: self.class.erb_implementation.new(
Chris@1115: erb,
Chris@1115: :trim => (self.class.erb_trim_mode == "-"),
Chris@1115: :escape => template.identifier =~ /\.text/ # only escape HTML templates
Chris@1115: ).src
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@1115:
Chris@1115: ActionView::Base.field_error_proc = Proc.new{ |html_tag, instance| html_tag || ''.html_safe }
Chris@1115:
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@1115: require 'mail'
Chris@1115:
Chris@1115: module DeliveryMethods
Chris@1115: class AsyncSMTP < ::Mail::SMTP
Chris@1115: def deliver!(*args)
Chris@0: Thread.start do
Chris@1115: super *args
Chris@909: end
Chris@0: end
Chris@0: end
Chris@909:
Chris@1115: class AsyncSendmail < ::Mail::Sendmail
Chris@1115: def deliver!(*args)
Chris@1115: Thread.start do
Chris@1115: super *args
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@1115:
Chris@1115: class TmpFile
Chris@1115: def initialize(*args); end
Chris@1115:
Chris@1115: def deliver!(mail)
Chris@1115: dest_dir = File.join(Rails.root, 'tmp', 'emails')
Chris@1115: Dir.mkdir(dest_dir) unless File.directory?(dest_dir)
Chris@1115: File.open(File.join(dest_dir, mail.message_id.gsub(/[<>]/, '') + '.eml'), 'wb') {|f| f.write(mail.encoded) }
Chris@1115: end
Chris@909: end
Chris@0: end
Chris@0:
Chris@1115: ActionMailer::Base.add_delivery_method :async_smtp, DeliveryMethods::AsyncSMTP
Chris@1115: ActionMailer::Base.add_delivery_method :async_sendmail, DeliveryMethods::AsyncSendmail
Chris@1115: ActionMailer::Base.add_delivery_method :tmp_file, DeliveryMethods::TmpFile
chris@37:
Chris@1115: # Changes how sent emails are logged
Chris@1115: # Rails doesn't log cc and bcc which is misleading when using bcc only (#12090)
Chris@1115: module ActionMailer
Chris@1115: class LogSubscriber < ActiveSupport::LogSubscriber
Chris@1115: def deliver(event)
Chris@1115: recipients = [:to, :cc, :bcc].inject("") do |s, header|
Chris@1115: r = Array.wrap(event.payload[header])
Chris@1115: if r.any?
Chris@1115: s << "\n #{header}: #{r.join(', ')}"
Chris@1115: end
Chris@1115: s
Chris@1115: end
Chris@1115: info("\nSent email \"#{event.payload[:subject]}\" (%1.fms)#{recipients}" % event.duration)
Chris@1115: debug(event.payload[:mail])
Chris@909: end
Chris@909: end
chris@37: end
Chris@119:
Chris@119: module ActionController
Chris@119: module MimeResponds
Chris@1115: class Collector
Chris@119: def api(&block)
Chris@119: any(:xml, :json, &block)
Chris@119: end
Chris@119: end
Chris@119: end
Chris@119: end
Chris@1115:
Chris@1115: module ActionController
Chris@1115: class Base
Chris@1115: # Displays an explicit message instead of a NoMethodError exception
Chris@1115: # when trying to start Redmine with an old session_store.rb
Chris@1115: # TODO: remove it in a later version
Chris@1115: def self.session=(*args)
Chris@1115: $stderr.puts "Please remove config/initializers/session_store.rb and run `rake generate_secret_token`.\n" +
Chris@1115: "Setting the session secret with ActionController.session= is no longer supported in Rails 3."
Chris@1115: exit 1
Chris@1115: end
Chris@1115: end
Chris@1115: end
Chris@1517:
Chris@1517: if Rails::VERSION::MAJOR < 4 && RUBY_VERSION >= "2.1"
Chris@1517: module ActiveSupport
Chris@1517: class HashWithIndifferentAccess
Chris@1517: def select(*args, &block)
Chris@1517: dup.tap { |hash| hash.select!(*args, &block) }
Chris@1517: end
Chris@1517:
Chris@1517: def reject(*args, &block)
Chris@1517: dup.tap { |hash| hash.reject!(*args, &block) }
Chris@1517: end
Chris@1517: end
Chris@1517:
Chris@1517: class OrderedHash
Chris@1517: def select(*args, &block)
Chris@1517: dup.tap { |hash| hash.select!(*args, &block) }
Chris@1517: end
Chris@1517:
Chris@1517: def reject(*args, &block)
Chris@1517: dup.tap { |hash| hash.reject!(*args, &block) }
Chris@1517: end
Chris@1517: end
Chris@1517: end
Chris@1517: end
Chris@1517:
Chris@1517: require 'awesome_nested_set/version'
Chris@1517:
Chris@1517: module CollectiveIdea
Chris@1517: module Acts
Chris@1517: module NestedSet
Chris@1517: module Model
Chris@1517: def leaf_with_new_record?
Chris@1517: new_record? || leaf_without_new_record?
Chris@1517: end
Chris@1517: alias_method_chain :leaf?, :new_record
Chris@1517: # Reload is needed because children may have updated
Chris@1517: # their parent (self) during deletion.
Chris@1517: if ::AwesomeNestedSet::VERSION > "2.1.6"
Chris@1517: module Prunable
Chris@1517: def destroy_descendants_with_reload
Chris@1517: destroy_descendants_without_reload
Chris@1517: reload
Chris@1517: end
Chris@1517: alias_method_chain :destroy_descendants, :reload
Chris@1517: end
Chris@1517: else
Chris@1517: def destroy_descendants_with_reload
Chris@1517: destroy_descendants_without_reload
Chris@1517: reload
Chris@1517: end
Chris@1517: alias_method_chain :destroy_descendants, :reload
Chris@1517: end
Chris@1517: end
Chris@1517: end
Chris@1517: end
Chris@1517: end