Archive for the ‘rails’ Category
Ruby Redo’s: Sinatra
Posted on: February 7, 2013
- In: rails | ruby | ruby redos | sinatra
- Leave a Comment
Sinatra is described as “a DSL for quickly creating web applications” and since I have been playing with DSLs lately I thought I would try my hand at making something like it. You can see in Sinatra’s famous three line “Hello World” that they are calling get directly on the top-level object called main:
require 'sinatra' get '/hi' do "Hello World!" end
So first thing we know is that we need a Rack compatible class. Why do we know that? Because Rack deals with the webserver/HTTP stuff as long as we follow their interface spec which saves us a lot of fussing with servers. The spec is simple:
A Rack application is a Ruby object (not a class) that responds to call. It takes exactly one argument, the environment and returns an Array of exactly three values: The status, the headers, and the body.
So that actually tells us a fair bit about the class we need to write. As the saying goes, “There are only two hard things in Computer Science: cache invalidation and naming things”, I think I will name this thing HTTP. So here is the minimum to conform with the Rack spec:
class HTTP
def call env
[200, {}, ""]
end
end
If you are curious about what kind of stuff is going to end up in that env variable here is an example:
{
"SERVER_SOFTWARE"=>"thin 1.5.0 codename Knife",
"SERVER_NAME"=>"localhost",
"rack.input"=> StringIO.new(),
"rack.version"=>[1, 0],
"errors"=>STDERR,
"rack.multithread"=>false,
"rack.multiprocess"=>false,
"rack.run_once"=>false,
"REQUEST_METHOD"=>"GET",
"REQUEST_PATH"=>"/test",
"PATH_INFO" => "/test",
"REQUEST_URI"=>"/test",
"HTTP_VERSION"=>"HTTP/1.1",
"HTTP_USER_AGENT"=>"curl/7.27.0",
"HTTP_HOST"=>"localhost:8080",
"HTTP_ACCEPT"=>"*/*",
"GATEWTERFACE"=>"CGI/1.2",
"SERVER_PORT"=>"8080",
"QUERY_STRING"=>"",
"SERVER_PROTOCOL"=>"HTTP/1.1",
"rack.url_scheme"=>"http",
"SCRIPT_NAME"=>"",
"REMOTE_ADDR" => "127.0.0.1"
}
Rack actually has a bunch of helpful classes that we are going to use to get thing working, lets add them in:
require 'rack'
class HTTP
def call env
request = Rack::Request.new env
response = Rack::Response.new
response.write "Hello Whirled"
response.finish
end
end
at_exit do
Rack::Handler.default.run HTTP.new
end
You can see that we now have request (that wraps the environment hash that you can see us passing in) and a response object. Notice that response.finish is the last line of the call method now. This is because it returns the array of status, headers and body that Rack needs. The other thing to notice is that we are using Kernel#at_exit to run whatever Rack decides is the default webserver when the script exits. This actually gives us a working application that you can run like this:
mike@sleepycat:~/play/http$ ruby http.rb
>> Thin web server (v1.5.0 codename Knife)
>> Maximum connections set to 1024
>> Listening on 0.0.0.0:8080, CTRL+C to stop
Lets take another step and get that response.write out of there and put it in a Proc. Next, we use instance_eval to evaluate the Proc (which is being turned from a proc object back into a block with the & operator) in the context of the current object. Since the block is being evaluated in the context of the current object and not the context of the current method we need to make request and response instance variables and add some attr_accessor methods. You can see also that we are now storing the routes in a hash of hashes that is held in a class variable so that when we call new, our instance can still see the routes.
require 'rack'
class HTTP
attr_accessor :request, :response
@@routes = { get: {"/" => Proc.new {response.write "Hello Whirled"}} }
def call env
@request = Rack::Request.new env
@response = Rack::Response.new
the_proc = @@routes[:get]["/"]
instance_eval &the_proc
response.finish
end
end
at_exit do
Rack::Handler.default.run HTTP.new
end
Our next step is creating some way to add methods to that @@routes hash. For that we will create a class method called save_route. We’ll also add some other methods to the hash as well and make sure that if there is no block found for a url that we send back a 404:
require 'rack'
class HTTP
attr_accessor :response, :request
@@routes = { get: {}, put: {}, post: {}, delete: {}, patch: {}, head: {} }
def self.save_route *args
method, route, block = *args
@@routes[method.downcase.to_sym][route] = block
end
def call env
puts "Handling request for: #{env["PATH_INFO"]}"
@request = Rack::Request.new env
@response = Rack::Response.new
block = @@routes[request.request_method.downcase.to_sym][request.path_info]
# if no block is stored for that path; 404!
block ? instance_eval(&block) : throw_404
response.finish
end
private
def throw_404
@response.status = 404
@response.write "<html>Page Not Found</html>"
end
end
at_exit do
Rack::Handler.default.run HTTP.new
end
Now that we can add routes to our class, we need to turn our attention to adding methods to the main object. The methods we want to add are the usual HTTP methods, get, put and whatnot.
We want to define an method for each of them like this:
def get *args, &block HTTP.save_route(:get, args.first, block) end
But since that is kind of repetitive we are going to write some code that writes that code. Often you would use Module#define_method for this but the main object is weird and does not have that method. So I am going to use eval:
require 'rack'
class HTTP
#... all the HTTP code ...
end
def http_methods
%w(get put post delete head patch).each do |method_name|
eval <<-EOMETH
def #{method_name} *args, &block
HTTP.save_route(:#{method_name}, args.first, block)
end
EOMETH
end
end
at_exit do
# ... rack stuff ...
end
Then we need to open the eigenclass of main and attach the methods to that. This is so the methods are available as instance methods of the main object. We can do that with Ruby’s somewhat bizarre syntax:
require 'rack' class HTTP #... all the HTTP code ... end def http_methods #... all the HTTP code ... end class << self http_methods end at_exit do # ... rack stuff ... end
With that we now have a working toy version of Sinatra. We can use it like this:
require_relative 'http' get "/" do response.status = 200 response["HTTP-Referrer"] = "Mike" response.write "<html>Hello Whirled</html>" end
When you are working in Rails you often hear about Rack but its not usually something you interact directly with. Playing with this has really made me appreciate Sinatra for the intelligent, concise DSL that it is (which is easy to forget somehow). While playing with DSLs is fun, taking some time to get to know Rack has also really helped my understanding of Rails.
- In: rails | ruby
- Leave a Comment
It doesn’t take much time in the Ruby world to hear the term duck typing. It wasn’t until I was writing a method for a recent project that was to accept either one or many ids that I actually stopped and actually investigated it. I knew what most people know about it: “If it walks like a duck and quacks like a duck, then it’s a duck” and that in actual practice this means you call call respond_to? to see if a method exists.
But as I was working on the code I found I kept wanting to type object.kind_of?(Array) but since this seemed to clash with the idea of duck typing I thought it was time to investigate a little bit.
The reason the type of an object using #kind_of? is not that useful is that Ruby supports multiple inheritance via the inclusion of modules. Since the methods on a given object may have gotten there through either of those channels just checking one of them doesn’t make much sense. So you can see here that knowing that Tiger.new.kind_of?(Felidae) just returned true may not give you any idea whether or not it is safe to call #domesticated? on the object.
module Wild
def domesticated?
false
end
end
class Felidae
def claws?
true
end
end
class Tiger < Felidae
include Wild
end
It seems that there are two schools of thought on duck typing. First, “soft” duck typing. Paraphrasing an example from Dave Thomas’ pickaxe book shows the standard approach:
def some_method param
if param.respond_to? :<<
# must be something we can write to!
param << "a string to be added"
end
end
The difficulty here is that respond_to? famously fails when you are using method_missing to catch and respond to method calls. The only way to be sure that respond_to? is not deceiving you is to go ahead and call the method. It is this approach that people call “hard” duck typing:
def some_method param param << "a string to be added" end
As Dave explains:
“You don’t need to check the type of the arguments. If they support << (in the case of result) or title and artist (in the case of song), everything will just work. If they don’t, your method will throw an exception anyway (just as it would have done if you’d checked the types). But without the check, your method is suddenly a lot more flexible: you could pass it an array, a string, a file, or any other object that appends using <<, and it would just work.”
— Programming Ruby second edition pg 355 – Dave Thomas
The interesting thing to me here, is the rationale for just calling the method instead of doing something like this:
def some_method param
begin
param << "a string to be added"
rescue NoMethodError => e
raise ArgumentError, "Needs to support the << method"
end
end
Constantly wrapping method calls in begin/rescue/end is pretty heinous so I was pretty interested to find another approach from Jeremy Kemper, the top commiter to the Rails framework. With commit 609c1988d2e274b he added the #acts_like? method to Object:
class Object
# A duck-type assistant method. For example, Active Support extends Date
# to define an acts_like_date? method, and extends Time to define
# acts_like_time?. As a result, we can do "x.acts_like?(:time)" and
# "x.acts_like?(:date)" to do duck-type-safe comparisons, since classes that
# we want to act like Time simply need to define an acts_like_time? method.
def acts_like?(duck)
respond_to? :"acts_like_#{duck}?"
end
end
and then to each of the core classes in the standard library he added an acts_like_x? method. Here is what the one for Date looks like:
class Date
# Duck-types as a Date-like class. See Object#acts_like?.
def acts_like_date?
true
end
end
These were added to ActiveSupport back in 2009 but for some reason they don’t seem to be widely used in the codebase. This surprises me because this approach seems to avoid NoMethodErrors without cluttering your code with rescue statements. On top of that it has none of the problems inherent in respond_to?.
I hope that helps people out a little. I know it has helped me. Just one more reason to go and read the source.
Ruby redo’s: The Rails router
Posted on: December 12, 2012
- In: rails | ruby | ruby redos
- Leave a Comment
One of the things that Ruby is famous for is the ease with which you can build Domain Specific Languages (DSLs). The routing DSL in Rails is one of the more recognizable features of the framework and a good example of a Ruby DSL. If you’ve worked with Rails at all you have seen this in config/routes.rb:
MyApp::Application.routes.draw do match "/foo", to: "foo#bar" end
I’ve worked with Rails a fair bit and had a pretty good understanding of how to use the DSL but there is always more to be learned by implementing it (or something like it) yourself. So what are we implementing? We want something that behaves like the ActionDispatch Routeset:
[27] pry(main)> Rails.application.routes => #<ActionDispatch::Routing::RouteSet:0x0000000232c588> [28] pry(main)> Rails.application.routes.draw do [28] pry(main)* match "/foo", to: "bar#baz", via: :get [28] pry(main)* end => nil
So we can see that it has a draw method that accepts a block. That block contains a call to the method “match” and accepts a string and a hash of arguments. We called the match method in our code block above and passed that block to the draw methods. When match is evaluated it adds the given route to the set of routes for the application. Which means that if we dig into the routes we should find our path we specified (/foo). Sure enough:
[29] pry(main)> Rails.application.routes.router.routes.each{|r| puts r.path.spec};nil;
/assets
/foo(.:format)
/rails/info/properties(.:format)
As with most things in Ruby, its actually surprisingly little code to get such a thing working:
class RouteSet
def initialize(routes = {})
@routes = routes
end
def match(path, options)
@routes[path]= options
end
def draw(&block)
instance_eval &block
end
def to_s
puts @routes.inspect
end
end
We can play with it in Pry:
mike@sleepycat:~/projects/play$ pry -I.
[1] pry(main)> load 'routes.rb'
=> true
[2] pry(main)> routeset = RouteSet.new
{}
=> #<RouteSet:0x1561598>
[3] pry(main)> routeset.draw do
[3] pry(main)* match "/foo", to: "bar#baz", via: :get
[3] pry(main)* end
=> {:to=>"bar#baz", :via=>:get}
[4] pry(main)> routeset.to_s
{"/foo"=>{:to=>"bar#baz", :via=>:get}}
=> nil
The secret DSL sauce is all in the draw method. Notice the ampersand in front of the block parameter:
def draw(&block)
instance_eval &block
end
That ampersand operator wraps an incoming block in a Proc and then binds it to a local variable named block. The same operator is used to reverse that process, turning the contents of the block variable from a Proc back into a block. That block is then fed into instance_eval which evaluates the block in the context of the current object. The net effect is the same as if you had just written this:
def draw
match "/foo", to: "bar#baz", via: :get
end
This process of taking a block of code defined somewhere and evaluating it in some other context is the key to DSLs in Ruby. Understanding the ampersand operator and its conversion between blocks and Procs is really important since this is really common in Ruby code. While is may be common, its not cheap. All that “binding to variables” stuff can be slow so in those moments where you care about speed you will want to use this instead:
def draw
instance_eval &Proc.new
end
Playing with this stuff has really helped my understanding of both Ruby and Rails. I hope it helps you too.
Ruby redo’s: IRB
Posted on: December 2, 2012
- In: rails | ruby | ruby redos
- 2 Comments
I have come to the realization that the best thing you can do to advance your Rails knowledge is to get better at Ruby. Rails development tends to keep you in one particular corner of the language. To see the other parts you really need to step off Rails’ golden path and try writing something from scratch yourself.
To push myself into some interesting territory I have taken to creating toy implementations of some of the programs I use but have never really understood the internals of. Every time I do this I find I learn something new… sometimes something big, sometimes something small.
One of the small ones actually started with switching from IRB to Pry, and seeing the mind blowing simplicity of Josh Cheek’s example Read Evaluate Print Loop (REPL):
loop do puts eval gets end
When he showed that I nearly fell out of my chair. I’ve been using the Rails console for years now and had never really stopped to consider how it actually did what it does. Obviously IRB is a bit more more involved than a one-liner and a gemspec but this is one of those magic code examples that gets the concept across with searing clarity.
A learning opportunity is not far off either since its not long before you will get the urge to improve it after fat-fingering something and the having it exit. If you don’t know about Ruby’s Exception hierarchy, or like me needed a reminder because you have spent to much time with Rails, try adding some error handling catching the Exception class:
loop do
begin
puts eval gets
rescue Exception => e #Not a good idea!
puts e.message
end
end
Of course you will soon realize that you can’t end the program since it turns out that Kernel#abort just calls Kernel#exit and that actually exits the program by raising a SystemExit Error… which we just caught. Oops.
After discovering the joy of Kernel#exit! and doing some reading about the Exception hierarchy (Ahhaa StandardError!) I have to say that redoing even seven lines of IRB taught me a lot.
- In: linux | rails | ruby | ubuntu
- Leave a Comment
I just stumbled across this error in mid-yak-shave for one of my projects. While trying to get my app up and running to test something I get the usual native extensions whining:
Installing sqlite-ruby (2.2.3) with native extensions
Gem::Installer::ExtensionBuildError: ERROR: Failed to build gem native extension./home/mike/.rvm/rubies/ruby-1.9.2-p290/bin/ruby extconf.rb
checking for main() in -lsqlite… no
checking for sqlite.h… no
*** extconf.rb failed ***
Could not create Makefile due to some reason, probably lack of
necessary libraries and/or headers. Check the mkmf.log file for more
details. You may need configuration options.
After a you see this sort of thing a few times you pretty instinctively bang out the command:
sudo apt-get install libsqlite3-dev
I was pretty takenaback to get the same error after installing the dev libraries. It turns out that I was installing the gem “sqlite-ruby” when I needed “sqlite3” is what you need if you are using Ruby 1.9.2.
- In: rails | ruby
- Leave a Comment
Cruising through Stackoverflow I noticed some syntax that I had never seen before:
@blog.user.try(:username)
Interested, I looked it up. As far as I can tell, it seems that one of the clever folks behind Github posted it as a helpful snippet on his blog.
From there it seems to have made its way into the Rails ActiveSupport library and then into Ruby 1.9 itself. Basically its the same as send except it swallows errors and just returns nil.
An example from the docs:
@person ? @person.name : nil
could be replaced with:
@person.try(:name)
Coolness! I know this will be making its way into my own code shortly. Check out Scott Harvey’s excellent explanation of the try method and also take a look at the documentation here.
I am just finishing a simple app that doesn’t require a database. When running it on the server I was getting errors saying :
LoadError (no such file to load — sqlite3):
But it was already commented out of my Gemfile. It took some thinking before I realized that ActiveRecord is probably looking for that by default regardless of whether or not I am including it in my Gemfile. So the solution was to remove ActiveRecord. That used to be reasonably obvious in Rails 2.3 but Rails 3 made it a little less so. A standard config.application.rb file starts with the lines:
require File.expand_path(‘../boot’, __FILE__)
require ‘rails/all’
Rails 3 is broken up into a bunch of different parts and that require ‘rails/all’ pulls in, well all of them including ActiveRecord. An equivalent statement would be:
require “active_record/railtie”
require “action_controller/railtie”
require “action_mailer/railtie”
require “active_resource/railtie”
require “rails/test_unit/railtie”
Of course once you have exchanged ‘rails/all’ for that, you can simply comment out ActiveRecord and that’s that. If you are starting a new Rails app you can have Rails do that for you by using the flags -O or –skip-activerecord.
Now that AR is gone, my app starts without looking for a database and all is goodness and light.
Rails 3 on dreamhost
Posted on: March 21, 2011
- In: rails
- Leave a Comment
Since is seems strangely difficult to find a straight answer about getting Rails running on Dreamhost, here are the current versions of all the important programs:
_ \ _|_` | _` | | | -_)
.__/_|\__,_|\__, |\_,_|\___|
_| ____/
Welcome to prague.dreamhost.comAny malicious and/or unauthorized activity is strictly forbidden.
All activity may be logged by DreamHost Web Hosting.[prague]$ ruby -v
ruby 1.8.7 (2008-08-11 patchlevel 72) [x86_64-linux]
[prague]$ rails -v
Rails 3.0.3
[prague]$ gem -v
1.3.6
[prague]$ bundle -v
Bundler version 1.0.7
[prague]$ date
Sun Mar 20 19:04:34 PDT 2011
- In: git | rails | ruby
- Leave a Comment
I have a project that is up on Github and deployed to Heroku. As I was adding the user credentials file to my project it dawned on me: my application needs these credentials when I run it on Heroku but I don’t want them to be part of my project and therefore visible to everyone on Github.
Turning to the great minds at Stackoverflow, I was advised to use environmental variables.
Since my application is connecting to another site via SFTP I have a hash of all the usual stuff; username password and hostname. It turns out you can just reference environmental variables like this:
@sftp_credentials = {Rails.env.to_sym => {
:sftp_host => ENV['SFTP_HOST'],
:sftp_user => ENV['SFTP_USER'],
:sftp_password => ENV['SFTP_PASSWORD']
}}
And then make sure you have the environment vars set in your .bashrc so they are available during local development:
export SFTP_HOST=’somesite.com’
export SFTP_USER=’someuser’
export SFTP_PASSWORD=’somepassword’
And then set those variables on Heroku so they are available when your app runs there:
mike@sleepycat:~/projects/myapp$ heroku config:add SFTP_HOST=somesite.com SFTP_USER=someuser SFTP_PASSWORD=somepassword
Adding config vars:
SFTP_HOST => somesite.com
SFTP_PASSWORD => somepassword
SFTP_USER => someuser
Restarting app…done.
- In: mysql | rails | ubuntu
- Leave a Comment
I am working a bash script to set up a complicated application. As part of that setup I need to run a series of SQL commands to create the initial users and databases this app will require.
It turns out you can get mysql to execute arbitrary statements from the command line using the -e switch but that would be both ugly and tedious for a lot of statements. Sounds like a job for heredocs! Lets try it out:
mike@railsdev:~$ read -d ” test <<’EOT’
create database test;
select 1;
EOTmike@railsdev:~$ mysql -u root -ppassword -e “$test”
+—+
| 1 |
+—+
| 1 |
+—+
Perfect! I love it when things work the way you expect. On to useful stuff:
read -d ” MYSQL_SETUP <<'EOF'
create database myapp_test;
create database myapp_development;
create database myapp_production;
CREATE USER 'appuser'@'localhost' IDENTIFIED BY 'myapp_password';
grant all privileges on myapp_production.* to appuser@localhost;
grant all privileges on myapp_test.* to appuser@localhost;
grant all privileges on myapp_development.* to appuser@localhost;
EOFmysql -u root -ppassword -e "$MYSQL_SETUP"
And then use the same trick to write a nice new database.yml for the app to use those credentials:
read -d ” DB_CREDENTIALS <<'EOF'
production:
adapter: mysql
database: myapp_production
pool: 5
username: appuser
password: myapp_passworddevelopment:
adapter: mysql
database: myapp_development
pool: 5
username: appuser
password: myapp_passwordtest:
adapter: mysql
database: myapp_test
pool: 5
username: appuser
password: myapp_password
EOFread -d '' DB_CREDENTIALS < /var/www/myapp/config/database.yml