- rest-graph: A simple wrapper to access Facebook's Graph API.
- cerialize: A drop-in replacement for Rails' "serialize" to use Marshal (or others) instead of slow YAML.
method_missing
Futzing around with Ruby, other languages, and further occasional idiosyncrasies.
Saturday, May 15, 2010
Open-sourced Projects
We've been busy these last few months, here's a couple of libraries we open sourced:
Thursday, October 15, 2009
Rails i18n truly default locales
The now built-in i18n functionality in Rails is a great and simple way of internationalizing a Rails app. However, one annoying out-of-the-box feature is that if one of your locales is missing a translation string, it will default it to a message like "translation missing: es, whassup", so you have to make sure that every string on every locale is set, which can be a pain.
A gentler approach would be to fallback to a "default" locale, and look up the string there. Not to be confused with the I18n.default_locale which is the locale to use when no locale at all is specified by the request.
Luckily the designers left us a backdoor to deal with this in the form of the exception_handler. To setup the fallback functionality have this code run somewhere (an initializer perhaps):
A gentler approach would be to fallback to a "default" locale, and look up the string there. Not to be confused with the I18n.default_locale which is the locale to use when no locale at all is specified by the request.
Luckily the designers left us a backdoor to deal with this in the form of the exception_handler. To setup the fallback functionality have this code run somewhere (an initializer perhaps):
require 'i18n'
module I18n # Reopen library module
def self.fallback_default_exception_handler(exception, locale, key, options)
# Recursive case should fall through
if exception.kind_of?(MissingTranslationData) && locale != self.default_locale
begin
return translate(key, options.merge(:locale => self.default_locale, :raise => true))
# Always raise so we can catch it
rescue MissingTranslationData
# Fall through and handle as below
end
end
send :default_exception_handler, exception, locale, key, options
end
end
I18n.exception_handler = :fallback_default_exception_handler
Sunday, September 27, 2009
Argument defaults for partials in Rails
It has bugged me, ever since I started using Rails, that when using partials with arguments, it is not easy to define default values for arguments. Matthew Moore sets out an example and has a good solution.
His solution is to check the parameters thusly:
of an undocumented hack (and so subject to change), and also a bit (actually documented, thanks Benjamin) long to remember and type, we could do better by hiding this behind a helper. However, local_assigns is a local variable itself, so how to access it from inside the helper?
Ah, but there is a back door! eval allows code to be executed in another context (and hence access to its local variables) provided we have a block to extract a binding from. So, we can make our helper interface be used like this:
It turns out however that eval is a rather slow call (the string must be parsed and compiled), so a few benchmarks show that it is better to preemptively lump the two evals together:
His solution is to check the parameters thusly:
title = nil if local_assigns[:title].nil?This can be improved by checking instead whether the value key is there, to handle the case in which you actually want to set the value to nil:
title = :defaultvalue unless local_assigns.has_key? :titleSince this use of local_assigns is a bit
Ah, but there is a back door! eval allows code to be executed in another context (and hence access to its local variables) provided we have a block to extract a binding from. So, we can make our helper interface be used like this:
title = opt(:title) { :defaultvalue }
Then we implement our helper thusly:module OptHelper
def opt(name, &block)
if eval("local_assigns.has_key? :#{name}", block.binding)
eval name.to_s, block.binding
else
yield
end
end
end
The first eval checks to see whether the variable was assigned, second eval is so that we can return that value. The yield executes the passed-in block to get the default value. This implementation also has the advantage that the default value can be an expensive operation that only gets executed if it's needed.It turns out however that eval is a rather slow call (the string must be parsed and compiled), so a few benchmarks show that it is better to preemptively lump the two evals together:
module OptHelper
def opt(name, &block)
was_assigned, value = eval(
"[ local_assigns.has_key?(:#{name}), local_assigns[:#{name}] ]",
block.binding)
if was_assigned
value
else
yield
end
end
end
Friday, September 25, 2009
begin
Why? because some of my colleagues believe code they find on blogs more than the some of the stuff I write, so the solution: publish my own blog. But seriously, also to give some back for the myriad times I've gotten unstuck thanks to some helpful blogger.
Some logistics: I'll expect to be having source code, so some easy way to publish code is needed. I found this: SyntaxHighlighter. Does it work?
The documentation is a bit unclear, but to get it working using the author's hosted css and js files, you add this to the bottom of the page:
I added this to the bottom of the Blogger template, and then simply surrounded the code with
The .swf (Adobe/Macromedia Shockwave) is to enable the neat clipboard functionality.
Some logistics: I'll expect to be having source code, so some easy way to publish code is needed. I found this: SyntaxHighlighter. Does it work?
class DoesItWork def yay! end end
The documentation is a bit unclear, but to get it working using the author's hosted css and js files, you add this to the bottom of the page:
<link href='http://alexgorbatchev.com/pub/sh/2.0.320/styles/shCore.css' rel='stylesheet' type='text/css'> <link href='http://alexgorbatchev.com/pub/sh/2.0.320/styles/shThemeDefault.css' rel='stylesheet' type='text/css'> <script language='javascript' src='http://alexgorbatchev.com/pub/sh/2.0.320/scripts/shCore.js'> <script language='javascript' src='http://alexgorbatchev.com/pub/sh/2.0.320/scripts/shBrushRuby.js'> <script language='javascript' src='http://alexgorbatchev.com/pub/sh/2.0.320/scripts/shBrushJScript.js'> <script language='javascript' src='http://alexgorbatchev.com/pub/sh/2.0.320/scripts/shBrushXml.js'> <script language='javascript'> SyntaxHighlighter.config.bloggerMode = true; SyntaxHighlighter.defaults.gutter = false; SyntaxHighlighter.config.clipboardSwf = 'http://alexgorbatchev.com/pub/sh/2.0.320/scripts/clipboard.swf'; SyntaxHighlighter.all(); </script>
I added this to the bottom of the Blogger template, and then simply surrounded the code with
<pre name="code" class="brush:ruby"> class DoesItWork def yay! end end </pre>
The .swf (Adobe/Macromedia Shockwave) is to enable the neat clipboard functionality.
Subscribe to:
Posts (Atom)