The guys over at NeverBlock have released a database adapter for Rails application that will severely increase the performance of ActiveRecord. Its also really easy to integrate into your application. Heres how:
Add a line to environment.rb for mongrel or thin servers:
require 'never_block/servers/thin'
or
require 'never_block/servers/mongrel'
Change the adapter in database.yml:
adapter: neverblock_postgresql
or
adapter: neverblock_mysql
You can also specify the number of connections (default of 4):
More information, along with benchmarks, can be found here.
I’m sure you all know how to use the :conditions attribute when using ActiveRecord:
User.find(:all, :conditions=>['active = ?', true])
And you may even use associations this way:
User.find(:all, :include=>[:photos], :conditions=>['photos.removed = ? and users.active = ?', false, true])
But did you know that you can do this easier through hashes?
User.find(:all, :conditions=>{:active=>true})
User.find(:all, :include=>[:photos], :conditions=>{'photos.removed'=>false, 'users.active'=>true})
Nothing special there, but I thought it was pretty cool. One thing you have to remember when using associations, is to include that model.
Sometimes you need to create another environment for your rails application aside from development, test, production. In this example we will create a “stage” environment. Here is how you do it.
First create the entry in your config/database.yml file:
# Stage database configuration
stage:
adapter: sqlite3
database: db/stage.sqlite3
timeout: 5000
Next create a file called stage.rb and place it into config/environments. I usually just copy my development.rb file and then change the values as needed:
Finally, In your config/environment.rb file, change the ENV[’RAILS_ENV’] to:
ENV['RAILS_ENV'] ||= 'stage'
Now when you boot up your server or console, just specify the “stage” environment.
I’m sure most of you already would know this, or use restful authentication that handles it for you. However, if you have some custom setup where you are loading a user object, and then storing it in session, slap to you!
Basically what I am talking about is doing this in your login method:
session[:current_user] = @user
Instead you should do:
session[:current_user] = @user.id
And then in your application controller, setup a before filter like so:
def set_current_user
@current_user = User.find(session[:current_user])
end
One main reason not to do that would be if you had to update some user information. If you had it stored in session, then the user would have to log out and log back in for the changes to take effect. This is of course a basic rough draft, but you get the idea.
Even though rails makes every effort to help with security in your apps, you should still be proactive about it. Don’t just assume that your data will be safe no matter how you code. Here is a prime example.
You have a login form and you process the request like this:
user = User.find(:first, :conditions=>["login = '#{params[:login]}'"])
You just essentially told every hacker to kill your data by doing something like “‘; delete from users;–”, or even worse a database drop. The appropriate way would be like this:
user = User.find(:first, :conditions=>['login = ?', params[:login]])
Other things you want to make sure you do is to sanitize your views as well:
Assume the worse and check all your user input to make sure they can’t do anything you don’t want them to and you will have a happy APP!
If you ever find yourself using CVS as your repository, and are trying to deploy your projects via capistrano, you may run into a connection refused issue.
The easiest way I found a solution around this was to modify the capistrano file: “/capistrano-2.1.0/lib/capistrano/recipes/deploy/scm/cvs.rb”. This of course is in your ruby gems folder.I could have added just the change to my project, but I am lazy, and need this for multiple CVS project deploying from this machine.
The change I made was on line 145, which I added the CVS_RSH=ssh line:
"export CVS_RSH=ssh && mkdir -p #{ dest } && cd #{ dest }"
Here is a quick, clean and easy way to get use from strftime on your date/time fields. Create a file called date_format.rb in the config/initializers directory and add the following code:
ActiveSupport::CoreExtensions::Time::Conversions::DATE_FORMATS|>.merge!(
:datetime_military => '%Y-%m-%d %H:%M',
:datetime => '%Y-%m-%d %I:%M%P',
:time => '%I:%M%P',
:time_military => '%H:%M%P',
:datetime_short => '%m/%d %I:%M'
)
Then in your views, you can do the following:
@object.time_field.to_s(:datetime) # Or any other format you created
For a while, I kept getting exceptions from my app in the form of “TypeError (can’t dump File):”. I finally found out that this was caused when I was using active_record_store with something like file_column, attachment_fu, or paperclip. Basically whenever you’re storing a file in session, that was too large for the session, you would experience this issue. Here is how to get around it:
Say you have a model like so (This is using file_column):
class Model < ActiveRecord::Base
file_column :filename
end
Then in your controller you would want to add a line before you redirect off to clear that session:
class Controller < ApplicationController
def create
@model = Model.find(params[:model])
@model.save!
params[:model][:filename] = nil rescue nil # Reset value here
redirect_to models_path(@model)
end
end
As you can see, I am resetting the filename value on the model. Now it shouldn’t complain that it can’t dump the file. Happy RAILSING!
If you come across something that stumps you with ruby or rails, or there is something that you don’t quite understand, let me know. I would love to help out with trying to solve your problem.
Also, if you have something you would like me to explain or create a “how-to” on, let me know about that too. Leave a comment here, or write me an email.
You can contact me at randy@freezzo.com
While writing some tests the other day, I came across a little bit of a stump. I have an action that required the use of a multi-dimensional param such as:
This is exactly what I was doing, but you get the picture. I could have easily changed it to a single array, but that not the point. The solution in this example, would be to nest your hash in the test such as:
def test_should_do_something
post :create, :some_object=>{
:name=>'Bob'
}, :user=>{ :name=>'Something' }
end