Introduction
Know how you can’t just use ‘form_for’ in rails with a newly initialized ActiveResource object? Ever get annoyed at having to add default attributes to your newly initialized ActiveResource objects? Fortify gives you the power to DRY’ly define default attributes for your ActiveResource derived classes.
Motivation
Let’s suppose you’re going to create a form for creating Books. However, your Book is an ActiveResource::Base derived class:
# app/models/book.rb
class Book < ActiveResource::Base
end
# app/controllers/books_controller.rb
class BooksController < ActiveResource::Base
def new
@book = Book.new
end
end
# app/view/books/new.haml
- form_for @book do |f|
.field
= f.label :title
= f.text_field :title
.field
= f.label :author
= f.text_field :author
.field
= f.label :genre
= f.text_field :genre
This, of course fails. Why? When you create a new Book object, unlike ActiveRecord (which has the luxury of querying the database to determine the object’s attributes), ActiveResource starts off as a blank slate. In other words:
console> b = Book.new
console> b.attributes.inspect
==> {}
To get this to work, you would have to give your new Book object some default attributes:
# app/controllers/books_controller.rb
class BooksController < ActiveResource::Base
def new
@book = Book.new :title => nil, :author => nil, :genre => nil
end
end
As you can imagine, this approach might not be very DRY. This is where fortify steps in.
Usage
First, install fortify by adding the following line to your config/environment.rb and then running “sudo rake gems:install”:
config.gem :fortify, :lib => 'fortify', :source => 'http://gemcutter.org'
Next, go back to your book model and add some fortification to it. You can do this in one of three ways. Let’s first assume that we just want to make sure the Book model has “author”, “title”, and “genre” attributes.
# app/models/book.rb
class Book < ActiveResource::Base
self.site = ''
fortify :author, :title, :genre
end
Now, your controller / views as you originally specified them will work, since now:
console> b = Book.new
console> b.attributes.inspect
==> {:author => nil, :title => nil, :genre => nil}
What if, however, we wanted default values for each of those attributes?
# app/models/book.rb
class Book < ActiveResource::Base
self.site = ''
fortify :author => 'Anonymous', :genre => 'Unknown', :title => 'Untitled'
end
In this case, a Book.new would give you:
console> b = Book.new
console> b.attributes.inspect
==> {:author => 'Anonymous', :title => 'Untitled', :genre => 'Unknown'}
Lastly, what if we only wanted a default value for the :author attribute? Try this:
# app/models/book.rb
class Book < ActiveResource::Base
self.site = ''
fortify do |attrs|
attrs. = 'Anonymous'
attrs.title
attrs.genre
end
end
Console would report something like this:
console> b = Book.new
console> b.attributes.inspect
==> {:author => 'Anonymous', :title => nil, :genre => nil}