h1. XSLT(not!) with Elegant Dress
You'd swear XSLT is a joke, but it's not. It's
cruelty. XSLT is like this: it's good to rid the
city of this horrible vermin infestation-- Let's
nuke the city. It's an out-of-proportion
insanity. It's Turing-complete, and it's XML.
Hmm.
But what it is, is just a set of transformations
on fragments of your XML document. The beauty of
XSLT (they say), is that XSLT itself is XML, so
you can (in principle) use XSLT to transform
itself.
Hmm.
Oh I know, how about let's use Ruby?
Dress is an XML transformation library built on
top of Nokogiri. Nokogiri is awesome. Dress is
built on top of Nokogiri...
By syllogism, Dress is awesome.
h1. Transformation Elegant Dress
Honestly, there's not much to it. A @Dress@ is
just a sequence of transformations you perform on
a DOM tree. Unlike XSLT though, the
transformations are performed directly on the DOM
tree. That is to say, they are destructive.
This is a simple dress,
require 'dress'
doc = ""
dress = Dress {
match("brain") do
set("size","pea")
end
}
result = dress.on(doc)
puts result.to_s
#
We can have more matchers,
dress = Dress {
match("brain") do
set("size","pea")
end
match("my") do
each { |e| e.name = "homer"}
end
}
result = dress.on(doc)
puts result.to_s
#
Note that match always yield a
Nokogiri::Nodeset. But sometimes we'd like to
operate on just one node (or the first one we
find). For that there's the @at@ matcher.
dress = Dress {
at("my") do
me.name = "homer"
end
}
@me@ is always the object yielded by the matcher
string. For @match@ it would be a Nodeset, and for
@at@, it would be an Element.
We can define helper methods on a Dress. This is
one that implements a counter,
dress = Dress {
def count
@count ||= 0
@count += 1
@count
end
match("brain") do
each { |e| e["area"] = count.to_s }
end
}
puts dress.on(Nokogiri.make { my { brain; brain; brain; brain }}).to_s
#
We can combine dresses into lines,
d1 = Dress { ... }
d2 = Dress { ... }
(d1 | d2).on(doc)
We can link lines together,
d3 = Dress { ... }
d4 = Dress { ... }
((d1 | d2) | (d3 | d4)).on(doc)
A dress is a class inheriting from Dress.
class FancyDress < Dress
def helper1
end
def helper2
end
match(...) { ... }
...
end
class PrettyDress < Dress
...
end
Then we can combine these dresses,
(FancyDress | PrettyDress).on(document)
You can of course mix that with dynamic dresses,
(FancyDress | PrettyDress | Dress do ... end).on(doc)
h1. That Wasn't So Hard, Was It?
So Ruby saved us all from XSLT. Minus all the
XSLT-inspired buckets of tears, the world is a
better place. FTW.
h1. Copyright
Copyright (c) 2009 Howard Yeh. See LICENSE for details.