Akhir tahun 2015 Ruby on Rails 5 Beta 1 dirilis. Tidak lama kemudian dirilis Rails versi Beta 1.1 dan awal Februari 2016 dirilis Rails 5 Beta 2. Perubahan versi major dari Rails diikuti banyak perubahan.

Pada tulisan akan dibahas beberapa fitur baru Rails 5 dan mungkin dilanjutkan tulisan lain mengenai Rails 5.

Action Cable

Fitur Action Cable memungkinkan Rails 5 memberikan support terhadap WebSocket, hal ini mempermudah dalam membuat aplikasi real time. Awalnya sempat menjadi kontroversi karena adanya dependensi terhadap Celluloid, Redis, serta EventMachine.

Namun pada rilis Rails 5 Beta 2 dependensi tersebut dihilangkan. Sebagai gantinya Rails menggunakan adapter alternatif ke Redis sebagai pubsub dan menggunakan non-EventMachine adapter Redis. Selain itu, pada rilis ini Rails resmi tidak lagi support PostgreSQL dibawah versi 9.1.

Rails juga memperkenalkan ActionController::Renderer dimana memungkinkan untuk melakukan render template diluar controller. Sangat berguna ketika ingin reuse template dari server-side sebagai respon WebSocket.

Demo Action Cable oleh DHH

Rails API

Fitur selanjutnya dari Rails 5 adalah Rails API. Fitur ini memungkinkan untuk generate aplikasi API-only dimana aplikasi yang dihasilkan generator Rails mengasumsikan menggunakan JSON sebagai respon serta menghilangkan bagian - bagian yang tidak digunakan ketika membuat aplikasi pure hanya sebagai backend.

rails new my-awesome-api-app --api

Rails Command

Bagi pemula sepertinya banyak yang mengalami kebingungan tentang rake terkait perbedaan dengan command rails. Dengan adanya perubahan pada Rails Command, pada Rails 5 semua command yang menggunakan rake akan diganti menjadi rails. Sehingga perintah seperti rake db:migrate akan diganti menjadi rails db:migrate.

Attributes API

Fitur Attributes API digunakan untuk mendefinisikan type pada Model dan memungkinkan untuk melakukan override attribute yang ada jika diperlukan. Fitur ini juga memungkinkan untuk mendefinisikan attribute tanpa memiliki kolom database.

Contoh 1

# db/schema.rb
create_table :store_listings, force: true do |t|
 t.decimal :price_in_cents
end

# app/models/store_listing.rb
class StoreListing < ActiveRecord::Base
end

store_listing = StoreListing.new(price_in_cents: '9.1')

# sebelum Rails 5
store_listing.price_in_cents # => BigDecimal.new(9.1)

class StoreListing < ActiveRecord::Base
 attribute :price_in_cents, :integer
end

# Rails 5
store_listing.price_in_cents # => 9

Contoh 2

# Attribute tanpa kolom di database

class MyModel < ActiveRecord::Base
 attribute :my_string, :string
 attribute :my_int_array, :integer, array: true
 attribute :my_float_range, :float, range: true
end

model = MyModel.new(
 my_string: "string",
 my_int_array: ["11", "12", "13"],
 my_float_range: "[2,4.5]",
)

model.attributes
# =>
#  {
#    my_string: "string",
#    my_int_array: [11, 12, 13],
#    my_float_range: 2.0..4.5
#  }

ApplicationRecord

ApplicationRecord mirip ApplicationController. Superclass ini bertujuan untuk berbagi fungsionalitas yang sama di semua model sehingga tidak diperlukan lagi monkey patch terhadap ActiveRecord::Base

module MyCustomValidationModule
  def do_something
    puts "Yadaaa! Yadaa!"
  end
end

# sebelum Rails 5
ActiveRecord::Base.include(MyCustomValidationModule)

# Rails 5
class ApplicationRecord < ActiveRecord::Base
  include MyCustomValidationModule

  self.abstract_class = true
end

ActiveRecord::Relation#or

ActiveRecord::Relation#or memungkinkan melakukan query #or.

Post.where(id: 1).or(Post.where(id: 2))
# => SELECT * FROM posts WHERE (id = 1) OR (id = 2)

# Sayang masih belum support untuk syntax seperti ini.
Post.where(id: 1).or(id: 2)
# NoMethodError: undefined method `limit_value' for {:id=>2}:Hash

ActiveRecord::Relation#in_batches

Method ActiveRecord::Relation#in_batches memungkinkan melakukan proses terhadap batch atau sekumpulan record sekaligus.

# dalam block
User.where("age > 25").in_batches do |relation|
  relation.delete_all
  sleep(10) # Throttle the delete queries
end

# tanpa block
User.in_batches.delete_all
User.in_batches.update_all(mantap: true)
User.in_batches.each_record(&:count_sign_in!)

Options dari ActiveRecord::Relation#in_batches

  • of - Set ukuran batch. Default 1000.
  • load - Set apakah relation harus di-_load_. Default false.
  • start - Set berapa nilai primary key untuk memulai batches. Angka yang dimasukkan merupakan inklusif.
  • finish - Set berapa nilai primary key untuk bacthes berakhir. Angka yang dimasukkan merupakan inklusif.

Dengan adanya fitur dan options diatas, sangat berguna ketika harus melakukan update record per kelompok (batches). Juga sangat berguna ketika ada beberapa worker yang memproses queue yang sama dimana worker 1 memproses record dengan ID 1 sampai 1000, worker 2 dengan ID 1001 sampai 2000, dan seterusnya.

User.in_batches(of: 2000, start: 1000).update_all(awesome: true)

User.in_batches.each do |relation|
  relation.update_all('age = age + 5')
  relation.where('age > 25').update_all(awesome: true)
  relation.where('age <= 25').delete_all
end

ActiveRecord::Base#has_secure_token

Fitur Rails ini memungkinkan untuk melakukan generate token unik pada model. Sangat berguna ketika membangun API dimana sering membutuhkan unique token yang digunakan sebagai authentication token.

Token unik yang di-_generate_ sepanjang 24 karakter dengan menggunakan SecureRandom::base58. Kemungkinan terjadi token kembar masih ada.

# Schema: User(token:string, auth_token:string)
class User < ActiveRecord::Base
 has_secure_token
 has_secure_token :auth_token
end

user = User.new
user.save
user.token # => "ZM27zsMN2ViQKta1bGfLmVs9"
user.auth_token # => "99TMHrHJFvFDwodq8w7Ev2y3"
user.regenerate_token # => true
user.regenerate_auth_token # => true

Method find(ids)

Sebelum Rails 5, ketika melakukan find(ids) atau where(ids: array_of_id), record yang dihasilkan tidak berurutan sesuai ID yang diberikan. Pada Rails 5 secara default akan diurutkan berdasarkan id yang dimasukkan.

ids = [3, 4, 2, 5]

# sebelum Rails 5
posts = Post.find(ids)
ordered_posts = ids.collect do |id|
  posts.detect {|post| post.id == id }
end

# Rails 5
posts = Post.find(ids)

ActiveModel::Errors#details

class User < ActiveRecord::Base
  validates :email, presence: true
end

user = User.new
user.valid?
user.errors.details
# => {email: [{error: :blank}]}

Multiple Konteks pada valid? dan invalid?

class User
  include ActiveModel::Validations

  attr_reader :email, :name
  validates_presence_of :email, on: :create
  validates_presence_of :name, on: :update
end

user = User.new
user.valid?([:create, :update])    # => false
user.errors.messages               # => {:email=>["can't be blank"], :name=>["can't be blank"]}

Callback Baru: after_{create,update,delete}_commit

# Sebelum Rails 5

after_commit :add_to_index_later, on: :create
after_commit :update_in_index_later, on: :update
after_commit :remove_from_index_later, on: :destroy

# Rails 5
after_create_commit  :add_to_index_later
after_update_commit  :update_in_index_later
after_destroy_commit :remove_from_index_later

Perubahan ActiveRecord::Relation#update

Pada Rails 5 bisa melakukan banyak record tanpa mengirimkan id dari record yang akan di-_update_.

# Sebelum Rails 5
# ArgumentError: wrong number of arguments (1 for 2)
Post.where(published: true).update(body: "Group of Software Engineer")

# Rails 5
# OK
Post.where(published: true).update(body: "Group of Software Engineer")

Terakhir, Rails 5 hanya support versi Ruby 2.2.2 atau lebih baru, jadi sebelum update Rails pastikan update Ruby terlebih dahulu.

Referensi