Yet Another ObjectMother pattern implementation for rails testing
We meet some problems with factory-girl:
-
We require quite complex logic for creation of test models
-
We require use factories with running rails application for integration tests
-
So why don’t make factory just another class in rails lib, and get extensibility of factories/mothers and rails lazy loading and dependency management?
-
Create folder app/famili under project root
-
Add app/famili to autoload path
# application.rb config.autoload_paths.push 'app/famili'
To define factory/mother for models User and Article just add following files to your app/famili directory:
#app/famili/user_famili.rb class UserFamili < Famili::Mother fist_name { 'nicola' } last_name { 'nicola' } email { "#{last_name}@mail.lv" } def before_save(user) #... end def after_create(user) #... end end #app/famili/article_famili.rb class ArticleFamili < Famili::Mother #creating association user { UserFamili.create } title { "article by #{user.last_name}" } end
And you can use it anywhere in tests or controllers:
UserFamili.create(:fist_name=>'Override') # create model UserFamili.build(:fist_name=>'Override') # build model (do not save) UserFamili.build_hash(:fist_name=>'Override') # get attributes hash ArticleFamili.create #create article with user
You can inherite mothers just like plain ruby classes, Just think each declaration field_name {…} as method definition
class UserFamili < Famili::Mother name { "nicola" } end class PersonFamili < UserFamili email { "#{name}@emial.com" } end
Mother have some usable methods, which can be used
class UserFamili < Famili::Mother last_name { 'nicola' } login { "#{last_name}_#{unique}" } number { sequence_number } end
You can add named set of attributes to override or extend default values
class UserFamili < Famili::Mother last_name { 'nicola' } login { "#{last_name}_#{unique}" } number { sequence_number } trait :unidentified do last_name { 'unknown' } end end UserFamili.unidentified.create(:first_name => 'john') # john unknown
Its also possible to use named scopes (like ActiveRecord)
class UserFamili < Famili::Mother last_name { 'nicola' } scope :prefixed do |prefix| scoped(last_name: "#{prefix}#{attributes[:last_name]}") end scope :suffixed do |suffix| scoped(last_name: "#{attributes[:last_name]}#{suffix}") end scope :mr_junior do prefixed('Mr ').suffixed(', Jr') end end
shared = UserFamili.scoped(:first_name => 'jeffry') shared.create(:last_name => 'stone') # jeffry stone shared.create(:last_name => 'snow') # jeffry snow
You can rewrite ArticleFamili declared above with using declarative association syntax:
class ArticleFamili < Famili::Mother has :user do last_name { 'Smith' } end end
When you need to create number of similar objects, you can use {build,create}_brothers methods:
brothers = UserFamili.create_brothers(2, :first_name => 'john') # both john nicola, but with different login and number
If you have complex initialization logic, which you want apply just after object was initialized, you can do it in initialization block:
UserFamili.create do |user| user.login = "updated_#{user.login}" end
You even can use famili for not active record models. You don’t need change everything if you just need to build object. But if you need custom persistence you may write custom save method:
class Person attr_accessor :persisted, :name end class PersonFamili < Famili::Mother name { 'John Smith' } def save(model) model.persisted = true end end
Now, you can create your model as usual:
PersonFamili.create(name: 'Barry Redwell') # persisted == true
If you need some additional objects for persistence (a.e. you are using some persistence service), then you have option to use famili instance instead of singleton:
class XmlSerializer def serialize(model) end end class PersonFamili < Famili::Mother name { 'John Smith' } def initialize(serializer) @serializer = serializer end def save(person) serializer.serialize(person) end end
and usage sample:
PersonFamili.new(XmlSerializer.new).create
If you want to create object with parameterized initialize method you can write own instantiate method
class Range attr :from, :to def initialize(from, to) @from, @to = from, to end end def RangeFamili < Famili::Mother from { 100 } to { 200 } def instantiate(attributes) Range.new(attributes[:from], attributes[:to]) end end
Note, if you access attribute once it will never be evaluated again, so it will not try to set from and to attributes for object and it safe to make attributes read-only. Other limitation - you can’t use other factoried properties until object will be created (f.e. you can’t write “to { from + 100 }”).
class HashFamili < Mother::Mother last_name { 'Smith' } first_name { 'John' } full_name { "#{last_name}, #{first_name}" } def instantiate(attributes) {} end def born(child) child.resolve_attributes(instantiate(child)) { |h, key, value| h[key.to_sym] = value } end end HashFamili.build(last_name: 'Rock')[:full_name] # => 'Rock, John'
As you can see in this example you can call ‘resolve_attributes` with new instance and callback to be called for each resolved attribute.
-
sequence_number - incremented with each instance
-
unique - just unique string
-
we a planing add more
Put following lineinto your Gemfile
gem 'famili'
-
1.3.0 - support custom instantion methods
-
1.2.0 - added support for custom objects instantiation throgh Famili::Mother#instantiate method
-
1.1.2 - fixed bug with dirty object state for factoried object
-
1.1.1 - fixed bug with famili associations (has)
-
1.1.0 - support custom persistence, support building & creating of objects using famili instance
-
1.0.0 - changed API (old scope renamed to trait, scope conception refined)
-
0.1.9 - optimize creation of relations
-
0.1.8 - support brother index in brothers init block, fix bug with using same mother instance for all childs, support declarative association syntax (has :user)
-
0.1.7 - fix build_hash result to not return updated_at and created_at (it caused errors in models created by new migrations in Rails 3.2)
-
0.1.6 - support creation of models with properties which have only set accessors
-
0.1.5 - fix method_missing in define_method
-
0.1.4 - famili now creates child in scope of model. So, self is a reference to model.
-
0.1.3 - supported anonym scopes with scoped. Supported collection creation methods: build_brothers, create_brothers. Supported initialization block for build, create, build_brothers, create_brothers.
-
0.1.2 - migrated to Rails 3 and Ruby 1.9 (no backward compatibility). Supported scopes & access to model methods. Fixed multiple access to calculated properties.
-
0.0.6 - rename Mother#hash to Mother#build_hash (to avoid conflicts with Object#hash in Ruby 1.9). Old name keeped as alias when using with oldest versions of Ruby for backward compatibility.
-
0.0.5 - add raise NoMethodError if property declared without block (becose it is error-prone), fix Famili::Mother.class#name method
-
0.0.3 - fix Mother.create call model.save!; Mother.hash return symbolized hash
-
0.0.2 - add inheritance, and mother methods [unique,sequence_number]
-
0.0.1 - created
-
generators
(The MIT License)
Copyright © 2010 niquola
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the ‘Software’), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED ‘AS IS’, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.