NAME
attributes.rb
INSTALL
gem install attributes
URIS
http://codeforpeople.com/lib/ruby
http://rubyforge.org/projects/codeforpeople/
http://codeforpeople.rubyforge.org/svn/
SYNOPSIS
attributes.rb provides a set of attr_* like method with several user
friendly additions. attributes.rb is similar to the traits.rb package but
sacrifices a few features for simplicity of implementation.
the implementation of attributes.rb borrows many of the best ideas from the
metakoans.rb ruby quiz
http://www.rubyquiz.com/quiz67.html
in particular the solutions of Christian Neukirchen and Florian Gross along
with concepts from the original traits lib
key features provided by attributes are
- ability to specify default values for attrs and definition time. values
can be literal objects or blocks, which are evaluated in the context of
self to initialize the variable
- classes remember which attributes they've defined and this information is
available to client code
- a whole suite of methods is defined by calls to #attributes including
getter, setter, query (var?) and banger (var! - which forces
re-initialization from the default value)
- ability to define multiple attributes at once using key => value pairs
- fast lookup of whether or not a class has defined a certain attribute
- attributes can be defined on objects on a per singleton basis as well
- getters acts as setters if an argument is given to them
- block caching, calling an attribute with a block sets the instance
variable to that block
all this in < 100 lines of code
HISTORY
5.0.1
- removed wrong dep on pervasives from gemspec.rb
5.0.0
- added support for block caching. for example
- simple block caching:
class Filter
attribute :process
end
filter = Filter.new
filter.process{|line| line.upcase}
lines.each do |line|
p filter.process.call(line)
end
- using block caching to delay block evaluation/class-factory:
module MigrationDSL
attribute :migration
def migration_class
model = self
Class.new(::ActiveRecord::Migration) do
singleton_class =
class << self
self
end
singleton_class.module_eval{ attribute :model => model }
singleton_class.module_eval &model.migration
end
end
end
class Table < ActiveRecord::Base
extend MigrationDSL
end
class Jobs < Table
migration do
def up
create_table model.table_name, :primary_key => model.primary_key do |t|
t.column 'vinyl_shoes', :text
end
end
def down
create_table model.table_name
end
end
end
...
JobsMigration = Jobs.migration_class
4.1.0
- 4.0.0 introduced a bug where a query (foo?) would not initialize a var -
4.1.0 fixes that
4.0.0
- removed dependancy on, and bundle of, pervasives
- faster. as fast as normal method definition.
- faster lookup for MyClass.attributes.include?('foobar')
3.7.0
small patch to use 'instance_variable_defined?' instead of defined?
keyword
3.5.0
migrated to a pervasives based impl to attributes should work on any
object - even blankslate objects
3.3.0
moved to an instance variable-less model using an module level closure for
the attributes list
SAMPLES
<========< samples/a.rb >========>
~ > cat samples/a.rb
#
# basic usage is like attr, but note that attribute defines a suite of methods
#
require 'attributes'
class C
attribute 'a'
end
c = C.new
c.a = 42
p c.a #=> 42
p 'forty-two' if c.a? #=> 'forty-two'
#
# attributes works on object too
#
o = Object.new
o.attribute 'answer' => 42
p o.answer #=> 42
~ > ruby samples/a.rb
42
"forty-two"
42
<========< samples/b.rb >========>
~ > cat samples/b.rb
#
# default values may be given either directly or as a block which will be
# evaluated in the context of self. in both cases (value or block) the
# default is set only once and only if needed - it's a lazy evaluation. the
# 'banger' method can be used to re-initialize a variable at any point whether
# or not it's already been initialized.
#
require 'attributes'
class C
attribute :a => 42
attribute(:b){ Float a }
end
c = C.new
p c.a #=> 42
p c.b #=> 42.0
c.a = 43
p c.a #=> 43
c.a!
p c.a #=> 42
~ > ruby samples/b.rb
42
42.0
43
42
<========< samples/c.rb >========>
~ > cat samples/c.rb
#
# multiple values may by given, plain names and key/val pairs may be mixed.
#
require 'attributes'
class C
attributes 'x', 'y' => 0b101000, 'z' => 0b10
end
c = C.new
c.x = c.y + c.z
p c.x #=> 42
~ > ruby samples/c.rb
42
<========< samples/d.rb >========>
~ > cat samples/d.rb
#
# a nice feature is that all attributes are enumerated in the class. this,
# combined with the fact that the getter method is defined so as to delegate
# to the setter when an argument is given, means bulk initialization and/or
# attribute traversal is very easy.
#
require 'attributes'
class C
attributes %w( x y z )
def attributes
self.class.attributes
end
def initialize
attributes.each_with_index{|a,i| send a, i}
end
def to_hash
attributes.inject({}){|h,a| h.update a => send(a)}
end
def inspect
to_hash.inspect
end
end
c = C.new
p c.attributes
p c
c.x 'forty-two'
p c.x
~ > ruby samples/d.rb
["x", "y", "z"]
{"x"=>0, "y"=>1, "z"=>2}
"forty-two"
<========< samples/e.rb >========>
~ > cat samples/e.rb
#
# my favourite element of attributes is that getters can also be setters.
# this allows incredibly clean looking code like
#
require 'attributes'
class Config
attributes %w( host port)
def initialize(&block) instance_eval &block end
end
conf = Config.new{
host 'codeforpeople.org'
port 80
}
p conf
~ > ruby samples/e.rb
#<Config:0x1fb58 @port=80, @host="codeforpeople.org">
<========< samples/f.rb >========>
~ > cat samples/f.rb
#
# of course attributes works as well at class/module level as at instance
# level
#
require 'attributes'
module Logging
Level_names = {
0 => 'INFO',
# ...
42 => 'DEBUG',
}
class << self
attribute 'level' => 42
attribute('level_name'){ Level_names[level] }
end
end
p Logging.level
p Logging.level_name
~ > ruby samples/f.rb
42
"DEBUG"