Recent notes
RSS feedGetting n..end in Rails
Nice one, henning. For anyone using Rails (ActiveSupport) a handy method called #from is also present.
[1, 2, 3, 4, 5].from(2) # => [3, 4, 5]
So our example would be
a.from(i)
It reads a lot better
update_all and serialized attributes
If you use update_all to change an attribute marked as serialized ( using ActiveRecord::Base.serialize ), you need to call to_yaml yourself:
User.update_all({ :preferences => { :first_name => 'John', :last_name => 'Doe' }.to_yaml })
add_to_base in Rails 3
use
model_instance.errors[:base] << "Msg"
instead of depracated
model_instance.errors.add_to_base("Msg")
for Rails 3
Depracated add_to_base
use
model_instance.errors[:base] << "Msg"
instead of depracated
model_instance.errors.add_to_base("Msg")
for Rails 3
Confusing log output
If you do a single Rails.cache.read('my_key'), your will usually see in your log something like
Cache read: my_key Cache read: my_key Cache write: my_key
Don’t worry about this.
What happens here is this: Before going to memcached, rails will first ask a local MemoryStore cache (which is responsible for the first and third line in the log), before falling through to memcached. This local cache is destroyed after each request.
Source is in activesupport/lib/active_support/cache/strategy/local_cache.rb.
In Rails3 use "unscoped" instead
The with_exclusive_scope examples no longer work in Rails3 because with_exclusive_scope is now a protected method which can and should not be used in a controller. Use the new unscoped method instead:
Article.unscoped
For mor details and examples have a look at: http://github.com/rails/rails/commit/bd1666ad1de88598ed6f04ceffb8488a77be4385.
When using enumerables
When using enumerables and storing them as strings in the database don’t forget to use .to_s or the select helper won’t automatically select your choice when viewing your data after save.
Exemple:
dates = 1900..Date.today.year f.select(:year, dates.collect {|d| [d.to_s,d.to_s]}, {:include_blank => "Select"}, {:class => "some_class"} )
Path Segments
url_options Returns a hash with path segments.
Example:
{:_path_segments=>{:action=>"show", :location=>"los-angeles-ca", :controller=>"city"}, :script_name=>"", :protocol=>"http://", :host=>"localhost:3000"}
Block_given?
Kernel#block_given? can be used to check if yield would be able to call a block.
Testing Named Scopes
Thanks for the example of testing named_scopes. Being new to Rails, I struggled to find examples that I could understand. Here is another “simple” test for a named_scope
Mine differs slightly from the one above in that I had to remove a set of {} in the :conditions in my test to avoid an “odd number list for Hash” error. I also replace the param-binding “?” with the number I expect to send in as an argument. My test would did know what args[0] was. I got an “undefined local variable” error.
The named scope in my model:
named_scope :up_to_and_including_year, lambda{ |*args| {
:conditions => [“to_char(grad_dt1,‘YYYY’) <= ?”, args[0]] }}
The test:
test "named_scope :up_to_and_including_year" do expected_options = { :conditions => ["to_char(grad_dt1,'YYYY') <= ?", '2010'] } assert_equal expected_options, Sso::CourseTaken.up_to_and_including_year('2010').proxy_options end
Inverse function
String#underscore is inverse for the camelize.
“active_record”.camelize.underscore # => “active_record”
assert_select negative with regex
(also, you can use instance vars)
assert_select "div#event_#{assigns[event].id}", { :count => 0, :html => /something/ }
label DOES translate
Maybe it used to not translate but I know it does as of 2.3.8. It is first lookup on the key:
helpers.label.<object_name>.<method_name>
If that doesn’t return anything it will use the human_attribute_name method on ActiveRecord::Base to translated which uses:
activerecord.attributes.<object>.<attribute>
I generally use both of these keys even when I don’t want to translate but just to have a single place where all my adjusted labels are stored.
If any form will need to use the same adjustment given the same object and attribute then I put in on the activerecord key. If the adjustment is form specific then I put it on the helpers key. Here is an example from real working code:
en: activerecord: attributes: "content/rich_text": name: Page name helpers: label: content_rich_text: testimonial_enabled: Enabled
Note that my AR object is a namespaced object (Content::RichText). In the activerecord key I need to change this to content/rich_text so if can find the correct key and put it in quotes to make it valid YAML. At the helper level on the other hand the namespace simply becomes an _.
:confirm, :popup, and :method override :onclick
upplying any combination of :confirm, :popup, and/or :method options to the link_to method results the :onclick option being overridden.
Example:
link_to "Delete", '#', :confirm=>"Are you sure?", :onclick=>"destroyJsFunction()" # expected output # => <a href="#" onclick="if(confirm('Are you sure?')) {destroyJsFunction()}; return false;">Delete</a> # actual output # => <a href="#" onclick="return confirm('Are you sure?');">Delete</a>
Note that the actual output doesn’t include any mention of the “destroyJsFunction()” passed to the link_to method.
Rails 3 will use unobtrusive JavaScript, and I haven’t tested how that will interact with the :onclick option.
Using strings as association names - beware of HashWithIndifferentAccess
If you merge a normal Hash into a HashWithIndifferentAccess, then the keys will convert to strings…
This will likely bite you if the merge is passed to AR find: as netmaniac said “Beware, that using strings as association names, when giving Hash to :include will render errors”.
Beware that params from your controller are HashWithIndifferentAccess like.
If your add_index is being ignored in your migration, see this
My add_index command was producing no change in my MySQL 5.0 database:
add_index :designations, [ :scope_type, :scope_id, :role_id, :user_id ], :unique => true
By just adding an index name, the problem was solved:
add_index :designations, [ :scope_type, :scope_id, :role_id, :user_id ], :unique => true, :name => 'my_index'
This happens when the autogenerated index name gets too long. For more info see:
To use in testing
If you want to use this in a test, add the following to test_helper.rb:
include ActionDispatch::TestProcess
(If using factory_girl, you can call it in your Factory, like so:
f.photo { fixture_file_upload 'test.png', 'image/png' }
Moved in Rails 3
In Rails 3, this has moved to ActionDispatch::TestProcess
(Which means, if you want to use it in a test, you need to add the following to test_helper.rb:)
include ActionDispatch::TestProcess
Works only inside the "it" block
Please note that stub_chain doesn’t work outside of the it...do...end block.
If you need to create more complicated chains using a function you need to use the old way.
When scripts don't end in .js
For example, Google Custom Search’s URL is http://www.google.com/jsapi
It’s an ugly hack, but works:
= javascript_include_tag('http://www.google.com/jsapi').sub('.js', '')
See in_place_editing as the new update
See http://github.com/cementhorses/in_place_editing for the new plugin to use as a substitute.
redirect_to :root
You can redirect to your main page using
redirect_to :root
Make sure to configure to root route first:
to set NULL => NO
use :null => false
change_column :my_table, :my_column, :integer, :default => 0, :null => false
Changing to MySql:BIGINT
I can change a column type from INT to BIGINT with this command:
change_column :my_table, :my_column, :bigint
Routing changes in Rails 3
See ActionDispatch::Routing for routing in Rails 3.
Doesn't output into STDOUT
Oddly enough it runs a rake task without any sort of output. To get around it you can simple substitute it with:
puts run('rake your_task')
Unless somebody has a better idea?
Looking for documentation of helper methods like: template, directory, file, etc?
Well, it’s not here. For Rails 3 they are part of Thor::Actions
If you need to pass a value
In the above example ‘value’ happens to be either true or false depending if the option was passed in or not. If you wish to capture an actual value you’ll want something like this:
def add_options!(opt) opt.on('-option=value') { |value| options[:option] = value } end
uniqueness
You can scope uniqueness as well
validates :user_name, :presence => true, :uniqueness => {:scope => :account_id} # the old way validates_uniqueness_of :user_name, :scope => :account_id