h2. Ruby on Rails 4.0 Release Notes
Highlights in Rails 4.0:
These release notes cover the major changes, but do not include each bug-fix and changes. If you want to see everything, check out the "list of commits":https://github.com/rails/rails/commits/master in the main Rails repository on GitHub.
endprologue.
h3. Upgrading to Rails 4.0
TODO. This is a WIP guide.
If you're upgrading an existing application, it's a great idea to have good test coverage before going in. You should also first upgrade to Rails 3.2 in case you haven't and make sure your application still runs as expected before attempting an update to Rails 4.0. Then take heed of the following changes:
h4. Rails 4.0 requires at least Ruby 1.9.3
Rails 4.0 requires Ruby 1.9.3 or higher. Support for all of the previous Ruby versions has been dropped officially and you should upgrade as early as possible.
h4. What to update in your apps
* Update your Gemfile to depend on
** rails = 4.0.0
** sass-rails ~> 3.2.3
** coffee-rails ~> 3.2.1
** uglifier >= 1.0.3
TODO: Update the versions above.
* Rails 4.0 removes vendor/plugins completely. You have to replace these plugins by extracting them as gems and adding them in your Gemfile. If you choose not to make them gems, you can move them into, say, lib/my_plugin/* and add an appropriate initializer in config/initializers/my_plugin.rb.
TODO: Configuration changes in environment files
h3. Creating a Rails 4.0 application
# You should have the 'rails' rubygem installed
$ rails new myapp
$ cd myapp
h4. Vendoring Gems
Rails now uses a +Gemfile+ in the application root to determine the gems you require for your application to start. This +Gemfile+ is processed by the "Bundler":https://github.com/carlhuda/bundler gem, which then installs all your dependencies. It can even install all the dependencies locally to your application so that it doesn't depend on the system gems.
More information: "Bundler homepage":http://gembundler.com
h4. Living on the Edge
+Bundler+ and +Gemfile+ makes freezing your Rails application easy as pie with the new dedicated +bundle+ command. If you want to bundle straight from the Git repository, you can pass the +--edge+ flag:
$ rails new myapp --edge
If you have a local checkout of the Rails repository and want to generate an application using that, you can pass the +--dev+ flag:
$ ruby /path/to/rails/railties/bin/rails new myapp --dev
h3. Major Features
h3. Documentation
h3. Railties
* Allow scaffold/model/migration generators to accept a polymorphic modifier for references/belongs_to, for instance
rails g model Product supplier:references{polymorphic}
will generate the model with belongs_to :supplier, polymorphic: true association and appropriate migration.
* Set config.active_record.migration_error to :page_load for development.
* Add runner to Rails::Railtie as a hook called just after runner starts.
* Add /rails/info/routes path which displays the same information as +rake routes+.
* Improved +rake routes+ output for redirects.
* Load all environments available in config.paths["config/environments"].
* The application generator generates public/humans.txt with some basic data.
* Add config.queue_consumer to allow the default consumer to be configurable.
* Add Rails.queue as an interface with a default implementation that consumes jobs in a separate thread.
* Remove Rack::SSL in favour of ActionDispatch::SSL.
* Allow to set class that will be used to run as a console, other than IRB, with Rails.application.config.console=. It's best to add it to console block.
# it can be added to config/application.rb
console do
# this block is called only when running console,
# so we can safely require pry here
require "pry"
config.console = Pry
end
* Add a convenience method hide! to Rails generators to hide the current generator namespace from showing when running rails generate.
* Scaffold now uses +content_tag_for+ in index.html.erb.
* Rails::Plugin is removed. Instead of adding plugins to vendor/plugins, use gems or bundler with path or git dependencies.
h4(#railties_deprecations). Deprecations
h3. Action Mailer
* Raise an ActionView::MissingTemplate exception when no implicit template could be found.
* Asynchronously send messages via the Rails Queue.
h3. Action Pack
h4. Action Controller
* Remove Active Model dependency from Action Pack.
* Support unicode characters in routes. Route will be automatically escaped, so instead of manually escaping:
get Rack::Utils.escape('こんにちは') => 'home#index'
You just have to write the unicode route:
get 'こんにちは' => 'home#index'
* Return proper format on exceptions.
* Extracted redirect logic from ActionController::ForceSSL::ClassMethods.force_ssl into ActionController::ForceSSL#force_ssl_redirect.
* URL path parameters with invalid encoding now raise ActionController::BadRequest.
* Malformed query and request parameter hashes now raise ActionController::BadRequest.
* +respond_to+ and +respond_with+ now raise ActionController::UnknownFormat instead of directly returning head 406. The exception is rescued and converted to 406 in the exception handling middleware.
* JSONP now uses application/javascript instead of application/json as the MIME type.
* Session arguments passed to process calls in functional tests are now merged into the existing session, whereas previously they would replace the existing session. This change may break some existing tests if they are asserting the exact contents of the session but should not break existing tests that only assert individual keys.
* Forms of persisted records use always PATCH (via the +_method+ hack).
* For resources, both PATCH and PUT are routed to the +update+ action.
* Don't ignore +force_ssl+ in development. This is a change of behavior - use an :if condition to recreate the old behavior.
class AccountsController < ApplicationController
force_ssl :if => :ssl_configured?
def ssl_configured?
!Rails.env.development?
end
end
h5(#actioncontroller_deprecations). Deprecations
* Deprecated ActionController::Integration in favour of ActionDispatch::Integration.
* Deprecated ActionController::IntegrationTest in favour of ActionDispatch::IntegrationTest.
* Deprecated ActionController::PerformanceTest in favour of ActionDispatch::PerformanceTest.
* Deprecated ActionController::AbstractRequest in favour of ActionDispatch::Request.
* Deprecated ActionController::Request in favour of ActionDispatch::Request.
* Deprecated ActionController::AbstractResponse in favour of ActionDispatch::Response.
* Deprecated ActionController::Response in favour of ActionDispatch::Response.
* Deprecated ActionController::Routing in favour of ActionDispatch::Routing.
h4. Action Dispatch
* Include mounted_helpers (helpers for accessing mounted engines) in ActionDispatch::IntegrationTest by default.
* Added ActionDispatch::SSL middleware that when included force all the requests to be under HTTPS protocol.
* Copy literal route constraints to defaults so that url generation know about them. The copied constraints are :protocol, :subdomain, :domain, :host and :port.
* Allows +assert_redirected_to+ to match against a regular expression.
* Adds a backtrace to the routing error page in development.
* +assert_generates+, +assert_recognizes+, and +assert_routing+ all raise +Assertion+ instead of +RoutingError+.
* Allows the route helper root to take a string argument. For example, root 'pages#main' as a shortcut for root to: 'pages#main'.
* Adds support for the PATCH verb: Request objects respond to patch?. Routes now have a new +patch+ method, and understand +:patch+ in the existing places where a verb is configured, like :via. Functional tests have a new method +patch+ and integration tests have a new method +patch_via_redirect+.
If :patch is the default verb for updates, edits are tunneled as PATCH rather than as PUT and routing acts accordingly.
* Integration tests support the OPTIONS method.
* +expires_in+ accepts a +must_revalidate+ flag. If true, "must-revalidate" is added to the Cache-Control header.
* Default responder will now always use your overridden block in respond_with to render your response.
* Turn off verbose mode of rack-cache, we still have X-Rack-Cache to check that info.
h5(#actiondispatch_deprecations). Deprecations
h4. Action View
* Remove Active Model dependency from Action Pack.
* Allow to use mounted_helpers (helpers for accessing mounted engines) in ActionView::TestCase.
* Make current object and counter (when it applies) variables accessible when rendering templates with :object or :collection.
* Allow to lazy load +default_form_builder+ by passing a string instead of a constant.
* Add index method to +FormBuilder+ class.
* Adds support for layouts when rendering a partial with a given collection.
* Remove :disable_with in favor of data-disable-with option from +submit_tag+, +button_tag+ and +button_to+ helpers.
* Remove :mouseover option from +image_tag+ helper.
* Templates without a handler extension now raises a deprecation warning but still defaults to +ERb+. In future releases, it will simply return the template content.
* Add a +divider+ option to +grouped_options_for_select+ to generate a separator optgroup automatically, and deprecate prompt as third argument, in favor of using an options hash.
* Add +time_field+ and +time_field_tag+ helpers which render an input[type="time"] tag.
* Removed old +text_helper+ apis for +highlight+, +excerpt+ and +word_wrap+.
* Remove the leading \n added by textarea on +assert_select+.
* Changed default value for config.action_view.embed_authenticity_token_in_remote_forms to false. This change breaks remote forms that need to work also without JavaScript, so if you need such behavior, you can either set it to true or explicitly pass :authenticity_token => true in form options.
* Make possible to use a block in +button_to+ helper if button text is hard to fit into the name parameter:
<%= button_to [:make_happy, @user] do %>
Make happy <%= @user.name %>
<% end %>
# => ""
* Replace +include_seconds+ boolean argument with :include_seconds => true option in +distance_of_time_in_words+ and +time_ago_in_words+ signature.
* Remove +button_to_function+ and +link_to_function+ helpers.
* +truncate+ now always returns an escaped HTML-safe string. The option :escape can be used as +false+ to not escape the result.
* +truncate+ now accepts a block to show extra content when the text is truncated.
* Add +week_field+, +week_field_tag+, +month_field+, +month_field_tag+, +datetime_local_field+, +datetime_local_field_tag+, +datetime_field+ and +datetime_field_tag+ helpers.
* Add +color_field+ and +color_field_tag+ helpers.
* Add +include_hidden+ option to select tag. With :include_hidden => false select with multiple attribute doesn't generate hidden input with blank value.
* Removed default size option from the +text_field+, +search_field+, +telephone_field+, +url_field+, +email_field+ helpers.
* Removed default cols and rows options from the +text_area+ helper.
* Adds +image_url+, +javascript_url+, +stylesheet_url+, +audio_url+, +video_url+, and +font_url+ to assets tag helper. These URL helpers will return the full path to your assets. This is useful when you are going to reference this asset from external host.
* Allow +value_method+ and +text_method+ arguments from +collection_select+ and +options_from_collection_for_select+ to receive an object that responds to :call such as a proc, to evaluate the option in the current element context. This works the same way with +collection_radio_buttons+ and +collection_check_boxes+.
* Add +date_field+ and +date_field_tag+ helpers which render an input[type="date"] tag.
* Add +collection_check_boxes+ form helper, similar to +collection_select+:
collection_check_boxes :post, :author_ids, Author.all, :id, :name
# Outputs something like:
The label/check_box pairs can be customized with a block.
* Add +collection_radio_buttons+ form helper, similar to +collection_select+:
collection_radio_buttons :post, :author_id, Author.all, :id, :name
# Outputs something like:
The label/radio_button pairs can be customized with a block.
* +check_box+ with an HTML5 attribute +:form+ will now replicate the +:form+ attribute to the hidden field as well.
* label form helper accepts :for => nil to not generate the attribute.
* Add :format option to +number_to_percentage+.
* Add config.action_view.logger to configure logger for +Action View+.
* +check_box+ helper with :disabled => true will generate a +disabled+ hidden field to conform with the HTML convention where disabled fields are not submitted with the form. This is a behavior change, previously the hidden tag had a value of the disabled checkbox.
* +favicon_link_tag+ helper will now use the favicon in app/assets by default.
* ActionView::Helpers::TextHelper#highlight now defaults to the HTML5 +mark+ element.
h5(#actionview_deprecations). Deprecations
h4. Sprockets
Moved into a separate gem sprockets-rails.
h3. Active Record
* Add add_reference and remove_reference schema statements. Aliases, add_belongs_to and remove_belongs_to are acceptable. References are reversible.
# Create a user_id column
add_reference(:products, :user)
# Create a supplier_id, supplier_type columns and appropriate index
add_reference(:products, :supplier, polymorphic: true, index: true)
# Remove polymorphic reference
remove_reference(:products, :supplier, polymorphic: true)
* Add :default and :null options to column_exists?.
column_exists?(:testings, :taggable_id, :integer, null: false)
column_exists?(:testings, :taggable_type, :string, default: 'Photo')
* ActiveRelation#inspect no longer calls #to_a. This means that in places where #inspect is implied (such as in the console), creating a relation will not execute it anymore, you'll have to call #to_a when necessary:
User.where(:age => 30) # => returns the relation
User.where(:age => 30).to_a # => executes the query and returns the loaded objects, as before
* Add :collation and :ctype support to PostgreSQL. These are available for PostgreSQL 8.4 or later.
development:
adapter: postgresql
host: localhost
database: rails_development
username: foo
password: bar
encoding: UTF8
collation: ja_JP.UTF8
ctype: ja_JP.UTF8
* FinderMethods#exists? now returns false with the false argument.
* Added support for specifying the precision of a timestamp in the postgresql adapter. So, instead of having to incorrectly specify the precision using the :limit option, you may use :precision, as intended. For example, in a migration:
def change
create_table :foobars do |t|
t.timestamps :precision => 0
end
end
* Allow ActiveRecord::Relation#pluck to accept multiple columns. Returns an array of arrays containing the typecasted values:
Person.pluck(:id, :name)
# SELECT people.id, people.name FROM people
# => [[1, 'David'], [2, 'Jeremy'], [3, 'Jose']]
* Improve the derivation of HABTM join table name to take account of nesting. It now takes the table names of the two models, sorts them lexically and then joins them, stripping any common prefix from the second table name. Some examples:
Top level models (Category <=> Product)
Old: categories_products
New: categories_products
Top level models with a global table_name_prefix (Category <=> Product)
Old: site_categories_products
New: site_categories_products
Nested models in a module without a table_name_prefix method (Admin::Category <=> Admin::Product)
Old: categories_products
New: categories_products
Nested models in a module with a table_name_prefix method (Admin::Category <=> Admin::Product)
Old: categories_products
New: admin_categories_products
Nested models in a parent model (Catalog::Category <=> Catalog::Product)
Old: categories_products
New: catalog_categories_products
Nested models in different parent models (Catalog::Category <=> Content::Page)
Old: categories_pages
New: catalog_categories_content_pages
* Move HABTM validity checks to ActiveRecord::Reflection. One side effect of this is to move when the exceptions are raised from the point of declaration to when the association is built. This is consistant with other association validity checks.
* Added stored_attributes hash which contains the attributes stored using ActiveRecord::Store. This allows you to retrieve the list of attributes you've defined.
class User < ActiveRecord::Base
store :settings, accessors: [:color, :homepage]
end
User.stored_attributes[:settings] # [:color, :homepage]
* composed_of was removed. You'll have to write your own accessor and mutator methods if you'd like to use value objects to represent some portion of your models. So, instead of:
class Person < ActiveRecord::Base
composed_of :address, :mapping => [ %w(address_street street), %w(address_city city) ]
end
you could write something like this:
def address
@address ||= Address.new(address_street, address_city)
end
def address=(address)
self[:address_street] = @address.street
self[:address_city] = @address.city
@address = address
end
* PostgreSQL default log level is now 'warning', to bypass the noisy notice messages. You can change the log level using the min_messages option available in your config/database.yml.
* Add uuid datatype support to PostgreSQL adapter.
* update_attribute has been removed. Use update_column if you want to bypass mass-assignment protection, validations, callbacks, and touching of updated_at. Otherwise please use update_attributes.
* Added ActiveRecord::Migration.check_pending! that raises an error if migrations are pending.
* Added #destroy! which acts like #destroy but will raise an ActiveRecord::RecordNotDestroyed exception instead of returning false.
* Allow blocks for count with ActiveRecord::Relation, to work similar as Array#count: Person.where("age > 26").count { |person| person.gender == 'female' }
* Added support to CollectionAssociation#delete for passing fixnum or string values as record ids. This finds the records responding to the ids and deletes them.
class Person < ActiveRecord::Base
has_many :pets
end
person.pets.delete("1") # => [#]
person.pets.delete(2, 3) # => [#, #]
* It's not possible anymore to destroy a model marked as read only.
* Added ability to ActiveRecord::Relation#from to accept other ActiveRecord::Relation objects.
* Added custom coders support for ActiveRecord::Store. Now you can set your custom coder like this:
store :settings, accessors: [ :color, :homepage ], coder: JSON
* +mysql+ and +mysql2+ connections will set SQL_MODE=STRICT_ALL_TABLES by default to avoid silent data loss. This can be disabled by specifying strict: false in config/database.yml.
* Added default order to ActiveRecord::Base#first to assure consistent results among diferent database engines. Introduced ActiveRecord::Base#take as a replacement to the old behavior.
* Added an :index option to automatically create indexes for +references+ and +belongs_to+ statements in migrations. This can be either a boolean or a hash that is identical to options available to the +add_index+ method:
create_table :messages do |t|
t.references :person, :index => true
end
Is the same as:
create_table :messages do |t|
t.references :person
end
add_index :messages, :person_id
Generators have also been updated to use the new syntax.
* Added bang methods for mutating ActiveRecord::Relation objects. For example, while foo.where(:bar) will return a new object leaving foo unchanged, foo.where!(:bar) will mutate the foo object.
* Added #find_by and #find_by! to mirror the functionality provided by dynamic finders in a way that allows dynamic input more easily:
Post.find_by name: 'Spartacus', rating: 4
Post.find_by "published_at < ?", 2.weeks.ago
Post.find_by! name: 'Spartacus'
* Added ActiveRecord::Base#slice to return a hash of the given methods with their names as keys and returned values as values.
* Remove IdentityMap - IdentityMap has never graduated to be an "enabled-by-default" feature, due to some inconsistencies with associations, as described in this "commit":https://github.com/rails/rails/commit/302c912bf6bcd0fa200d964ec2dc4a44abe328a6. Hence the removal from the codebase, until such issues are fixed.
* Added a feature to dump/load internal state of +SchemaCache+ instance because we want to boot more quickly when we have many models.
# execute rake task.
RAILS_ENV=production bundle exec rake db:schema:cache:dump
=> generate db/schema_cache.dump
# add config.use_schema_cache_dump = true in config/production.rb. BTW, true is default.
# boot rails.
RAILS_ENV=production bundle exec rails server
=> use db/schema_cache.dump
# If you remove clear dumped cache, execute rake task.
RAILS_ENV=production bundle exec rake db:schema:cache:clear
=> remove db/schema_cache.dump
* Added support for partial indices to +PostgreSQL+ adapter.
* The +add_index+ method now supports a +where+ option that receives a string with the partial index criteria.
* Added the ActiveRecord::NullRelation class implementing the null object pattern for the Relation class.
* Implemented ActiveRecord::Relation#none method which returns a chainable relation with zero records (an instance of the +NullRelation+ class). Any subsequent condition chained to the returned relation will continue generating an empty relation and will not fire any query to the database.
* Added +create_join_table+ migration helper to create HABTM join tables.
create_join_table :products, :categories
# =>
# create_table :categories_products, :id => false do |td|
# td.integer :product_id, :null => false
# td.integer :category_id, :null => false
# end
* The primary key is always initialized in the +@attributes+ hash to nil (unless another value has been specified).
* In previous releases, the following would generate a single query with an OUTER JOIN comments, rather than two separate queries:
Post.includes(:comments).where("comments.name = 'foo'")
This behaviour relies on matching SQL string, which is an inherently flawed idea unless we write an SQL parser, which we do not wish to do. Therefore, it is now deprecated.
To avoid deprecation warnings and for future compatibility, you must explicitly state which tables you reference, when using SQL snippets:
Post.includes(:comments).where("comments.name = 'foo'").references(:comments)
Note that you do not need to explicitly specify references in the following cases, as they can be automatically inferred:
Post.where(comments: { name: 'foo' })
Post.where('comments.name' => 'foo')
Post.order('comments.name')
You also do not need to worry about this unless you are doing eager loading. Basically, don't worry unless you see a deprecation warning or (in future releases) an SQL error due to a missing JOIN.
* Support for the +schema_info+ table has been dropped. Please switch to +schema_migrations+.
* Connections *must* be closed at the end of a thread. If not, your connection pool can fill and an exception will be raised.
* Added the ActiveRecord::Model module which can be included in a class as an alternative to inheriting from ActiveRecord::Base:
class Post
include ActiveRecord::Model
end
* PostgreSQL hstore records can be created.
* PostgreSQL hstore types are automatically deserialized from the database.
h4(#activerecord_deprecations). Deprecations
* Deprecated most of the 'dynamic finder' methods. All dynamic methods except for +find_by_...+ and +find_by_...!+ are deprecated. Here's how you can rewrite the code:
find_all_by_... can be rewritten using where(...)
find_last_by_... can be rewritten using where(...).last
scoped_by_... can be rewritten using where(...)
find_or_initialize_by_... can be rewritten using where(...).first_or_initialize
find_or_create_by_... can be rewritten using where(...).first_or_create
find_or_create_by_...! can be rewritten using where(...).first_or_create!
The implementation of the deprecated dynamic finders has been moved to the +active_record_deprecated_finders+ gem.
* Deprecated the old-style hash based finder API. This means that methods which previously accepted "finder options" no longer do. For example this:
Post.find(:all, :conditions => { :comments_count => 10 }, :limit => 5)
should be rewritten in the new style which has existed since Rails 3:
Post.where(comments_count: 10).limit(5)
Note that as an interim step, it is possible to rewrite the above as:
Post.scoped(:where => { :comments_count => 10 }, :limit => 5)
This could save you a lot of work if there is a lot of old-style finder usage in your application.
Calling Post.scoped(options) is a shortcut for Post.scoped.merge(options). Relation#merge now accepts a hash of options, but they must be identical to the names of the equivalent finder method. These are mostly identical to the old-style finder option names, except in the following cases:
:conditions becomes :where
:include becomes :includes
:extend becomes :extending
The code to implement the deprecated features has been moved out to the +active_record_deprecated_finders+ gem. This gem is a dependency of Active Record in Rails 4.0. It will no longer be a dependency from Rails 4.1, but if your app relies on the deprecated features then you can add it to your own Gemfile. It will be maintained by the Rails core team until Rails 5.0 is released.
* Deprecate eager-evaluated scopes.
Don't use this:
scope :red, where(color: 'red')
default_scope where(color: 'red')
Use this:
scope :red, -> { where(color: 'red') }
default_scope { where(color: 'red') }
The former has numerous issues. It is a common newbie gotcha to do the following:
scope :recent, where(published_at: Time.now - 2.weeks)
Or a more subtle variant:
scope :recent, -> { where(published_at: Time.now - 2.weeks) }
scope :recent_red, recent.where(color: 'red')
Eager scopes are also very complex to implement within Active Record, and there are still bugs. For example, the following does not do what you expect:
scope :remove_conditions, except(:where)
where(...).remove_conditions # => still has conditions
* Added deprecation for the :dependent => :restrict association option.
* Up until now has_many and has_one, :dependent => :restrict option raised a DeleteRestrictionError at the time of destroying the object. Instead, it will add an error on the model.
* To fix this warning, make sure your code isn't relying on a DeleteRestrictionError and then add config.active_record.dependent_restrict_raises = false to your application config.
* New rails application would be generated with the config.active_record.dependent_restrict_raises = false in the application config.
h3. Active Model
* Changed AM::Serializers::JSON.include_root_in_json default value to false. Now, AM Serializers and AR objects have the same default behaviour.
class User < ActiveRecord::Base; end
class Person
include ActiveModel::Model
include ActiveModel::AttributeMethods
include ActiveModel::Serializers::JSON
attr_accessor :name, :age
def attributes
instance_values
end
end
user.as_json
=> {"id"=>1, "name"=>"Konata Izumi", "age"=>16, "awesome"=>true}
# root is not included
person.as_json
=> {"name"=>"Francesco", "age"=>22}
# root is not included
* Passing false hash values to +validates+ will no longer enable the corresponding validators.
* +ConfirmationValidator+ error messages will attach to :#{attribute}_confirmation instead of +attribute+.
* Added ActiveModel::Model, a mixin to make Ruby objects work with Action Pack out of the box.
* ActiveModel::Errors#to_json supports a new parameter :full_messages.
* Trims down the API by removing valid? and errors.full_messages.
h4(#activemodel_deprecations). Deprecations
h3. Active Resource
* Active Resource is removed from Rails 4.0 and is now a separate "gem":https://github.com/rails/activeresource.
h3. Active Support
* Time#change now works with time values with offsets other than UTC or the local time zone.
* Add Time#prev_quarter and Time#next_quarter short-hands for months_ago(3) and months_since(3).
* Remove obsolete and unused require_association method from dependencies.
* Add :instance_accessor option for config_accessor.
class User
include ActiveSupport::Configurable
config_accessor :allowed_access, instance_accessor: false
end
User.new.allowed_access = true # => NoMethodError
User.new.allowed_access # => NoMethodError
* ActionView::Helpers::NumberHelper methods have been moved to ActiveSupport::NumberHelper and are now available via Numeric#to_s.
* Numeric#to_s now accepts the formatting options :phone, :currency, :percentage, :delimited, :rounded, :human, and :human_size.
* Add Hash#transform_keys, Hash#transform_keys!, Hash#deep_transform_keys and Hash#deep_transform_keys!.
* Changed xml type datetime to dateTime (with upper case letter T).
* Add :instance_accessor option for class_attribute.
* +constantize+ now looks in the ancestor chain.
* Add Hash#deep_stringify_keys and Hash#deep_stringify_keys! to convert all keys from a +Hash+ instance into strings.
* Add Hash#deep_symbolize_keys and Hash#deep_symbolize_keys! to convert all keys from a +Hash+ instance into symbols.
* Object#try can't call private methods.
* AS::Callbacks#run_callbacks remove key argument.
* +deep_dup+ works more expectedly now and duplicates also values in +Hash+ instances and elements in +Array+ instances.
* Inflector no longer applies ice -> ouse to words like slice, police.
* Add ActiveSupport::Deprecations.behavior = :silence to completely ignore Rails runtime deprecations.
* Make Module#delegate stop using send - can no longer delegate to private methods.
* AS::Callbacks deprecate :rescuable option.
* Adds Integer#ordinal to get the ordinal suffix string of an integer.
* AS::Callbacks :per_key option is no longer supported.
* AS::Callbacks#define_callbacks add :skip_after_callbacks_if_terminated option.
* Add html_escape_once to ERB::Util, and delegate escape_once tag helper to it.
* Remove ActiveSupport::TestCase#pending method, use +skip+ instead.
* Deletes the compatibility method Module#method_names, use Module#methods from now on (which returns symbols).
* Deletes the compatibility method Module#instance_method_names, use Module#instance_methods from now on (which returns symbols).
* Unicode database updated to 6.1.0.
* Adds +encode_big_decimal_as_string+ option to force JSON serialization of BigDecimals as numeric instead of wrapping them in strings for safety.
h4(#activesupport_deprecations). Deprecations
* ActiveSupport::Callbacks: deprecate usage of filter object with #before and #after methods as around callback.
* BufferedLogger is deprecated. Use ActiveSupport::Logger or the +logger+ from Ruby stdlib.
* Deprecates the compatibility method Module#local_constant_names and use Module#local_constants instead (which returns symbols).
h3. Credits
See the "full list of contributors to Rails":http://contributors.rubyonrails.org/ for the many people who spent many hours making Rails, the stable and robust framework it is. Kudos to all of them.