Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* added support for threaded processing * switched to bundler gem building * added documentation
- Loading branch information
Florian Aßmann
committed
Sep 28, 2011
1 parent
500b9d1
commit 581a618
Showing
15 changed files
with
286 additions
and
186 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,21 +1,4 @@ | ||
## MAC OS | ||
.DS_Store | ||
|
||
## TEXTMATE | ||
*.tmproj | ||
tmtags | ||
|
||
## EMACS | ||
*~ | ||
\#* | ||
.\#* | ||
|
||
## VIM | ||
*.swp | ||
|
||
## PROJECT::GENERAL | ||
coverage | ||
rdoc | ||
pkg | ||
|
||
## PROJECT::SPECIFIC | ||
*.gem | ||
.bundle | ||
Gemfile.lock | ||
pkg/* |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
source "http://rubygems.org" | ||
gemspec | ||
|
||
gem "rack" | ||
gem "nokogiri" | ||
# gem "patron" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
PATH | ||
remote: . | ||
specs: | ||
rack-esi (0.2.0) | ||
nokogiri | ||
rack | ||
|
||
GEM | ||
remote: http://rubygems.org/ | ||
specs: | ||
nokogiri (1.5.0) | ||
rack (1.3.3) | ||
riot (0.12.5) | ||
rr | ||
rr (1.0.4) | ||
|
||
PLATFORMS | ||
ruby | ||
|
||
DEPENDENCIES | ||
nokogiri | ||
rack | ||
rack-esi! | ||
riot |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,54 +1,71 @@ | ||
# rack-esi | ||
|
||
Nokogiri based ESI middleware implementation for Rack with (limited) support | ||
for include, remove and comment. | ||
Rack-ESI is a Nokogiri based ESI middleware implementation for Rack with support for include tags, all other ESI namespaced nodes are just removed. | ||
|
||
To make this gem work you must define the (xmlns:esi)[http://www.edge-delivery.org/esi/1.0] namespace in your text/html response. | ||
|
||
Note: This gem should only be used in development. For production use setup varnish or any other ESI enabled server. | ||
|
||
## Features | ||
|
||
* path blacklisting (:skip => nil, expects Regexp) | ||
* type whitelisting (:only => /^text\/(?:x|ht)ml/) | ||
* recursion limit (:depth => 5) | ||
* include limits (:includes => 32) | ||
* support for <include> alt and noerror attributes | ||
* threaded (in case we have slow IOs) | ||
* PATH_INFO blacklisting (:skip => nil, should respond to ===) | ||
* support for esi|include[alt] and esi|include[noerror] fallbacks | ||
|
||
## Dependencies | ||
|
||
* Nokogiri | ||
* Rack | ||
|
||
## Setup | ||
|
||
### w/o Gemfile | ||
|
||
_It's for development purpose..._ | ||
$ gem install rack-esi | ||
|
||
## Installation | ||
### w/ Gemfile | ||
|
||
gem install rack-esi | ||
gem 'rack-esi' | ||
|
||
## Rails Setup (environment.rb) | ||
### rackup | ||
|
||
config.gem 'rack-esi' | ||
require 'rack-esi' | ||
config.middleware.insert_before config.middleware.first, Rack::ESI | ||
use Rack::ESI, options || {} | ||
run Application.new | ||
|
||
### Rails: environment.rb | ||
|
||
config.gem 'rack-esi' # for setups w/o Gemfile | ||
config.middleware.use Rack::ESI, options || {} | ||
|
||
## Options | ||
|
||
* poolsize: 4 | ||
Number of worker threads. A value of 1 disables threading model. | ||
* skip: nil | ||
This should be an object which responds to #===(PATH_INFO). | ||
* parser: Nokogiri::XML::Document | ||
You can change this to Nokogiri::HTML::Document, but you should change the serializer, too (see below). | ||
* serializer: :to_xhtml | ||
The serializer value specifies the method name which is send to the object created by the parser#parse. | ||
|
||
## TODO | ||
|
||
* write documentation | ||
* write more tests | ||
* support more ESI elements | ||
* switch to Nokogiri::XML::SAX::Document? | ||
|
||
## Dependencies | ||
|
||
* Nokogiri | ||
* Rack | ||
|
||
## Note on Patches/Pull Requests | ||
|
||
* Fork the project. | ||
* Make your feature addition or bug fix. | ||
* Add tests for it. This is important so I don't break it in a | ||
future version unintentionally. | ||
* Add tests for it. | ||
* Commit, do not mess with rakefile, version, or history. | ||
(if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull) | ||
* Send me a pull request. Bonus points for topic branches. | ||
* Send me a pull request. | ||
|
||
## Thanks | ||
|
||
tenderlove and Qerub | ||
|
||
## Copyright | ||
|
||
Copyright (c) 2009 Florian Assmann. See LICENSE for details. | ||
Copyright (c) 2009 Florian Aßmann. See LICENSE for details. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,53 +1 @@ | ||
require 'rubygems' | ||
require 'rake' | ||
|
||
begin | ||
require 'jeweler' | ||
Jeweler::Tasks.new do |gem| | ||
gem.name = "rack-esi" | ||
gem.summary = %Q{ESI middleware implementation for Rack.} | ||
gem.description = %Q{Nokogiri based ESI middleware implementation for Rack with (limited) support for include, remove and comment.} | ||
gem.email = "florian.assmann@email.de" | ||
gem.homepage = "http://github.com/boof/rack-esi" | ||
gem.authors = ["Florian Aßmann"] | ||
gem.add_development_dependency "riot", ">= 0" | ||
gem.add_development_dependency "yard", ">= 0" | ||
gem.add_dependency 'nokogiri', '>= 0' | ||
end | ||
Jeweler::GemcutterTasks.new | ||
rescue LoadError | ||
puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler" | ||
end | ||
|
||
require 'rake/testtask' | ||
Rake::TestTask.new(:test) do |test| | ||
test.libs << 'lib' << 'test' | ||
test.pattern = 'test/**/*_test.rb' | ||
test.verbose = true | ||
end | ||
|
||
begin | ||
require 'rcov/rcovtask' | ||
Rcov::RcovTask.new do |test| | ||
test.libs << 'test' | ||
test.pattern = 'test/**/*_test.rb' | ||
test.verbose = true | ||
end | ||
rescue LoadError | ||
task :rcov do | ||
abort "RCov is not available. In order to run rcov, you must: sudo gem install spicycode-rcov" | ||
end | ||
end | ||
|
||
task :test => :check_dependencies | ||
|
||
task :default => :test | ||
|
||
begin | ||
require 'yard' | ||
YARD::Rake::YardocTask.new | ||
rescue LoadError | ||
task :yardoc do | ||
abort "YARD is not available. In order to run yardoc, you must: sudo gem install yard" | ||
end | ||
end | ||
require "bundler/gem_tasks" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,97 +1,47 @@ | ||
require 'rack' | ||
require 'nokogiri' | ||
require 'bundler' | ||
Bundler.require | ||
|
||
class Rack::ESI | ||
NS = { 'esi' => 'http://www.edge-delivery.org/esi/1.0' } | ||
METHODS = { 'include' => :esi_include, 'remove' => nil, 'comment' => nil } | ||
CSS = METHODS.keys.map { |cmd| "esi|#{ cmd }" } * ',' | ||
require File.expand_path('../rack-esi/processor', __FILE__) | ||
|
||
class Error < RuntimeError | ||
def initialize(status, headers, response) | ||
@status, @headers, @response = status, headers, response | ||
end | ||
def finish | ||
return [@status, @headers, backtrace] | ||
end | ||
end | ||
class Rack::ESI | ||
|
||
def initialize(app, options = {}) | ||
@app = app | ||
|
||
@paths = options[:skip] | ||
@types = options[:only] || /^text\/(?:x|ht)ml/ | ||
@max_includes = options[:includes] || 32 | ||
@max_recursion = options[:depth] || 5 | ||
@parser = options.fetch :parser, Nokogiri::XML::Document | ||
@serializer = options.fetch :serializer, :to_xhtml | ||
@skip = options[:skip] | ||
@poolsize = options.fetch :poolsize, 4 | ||
@processor = @poolsize == 1 ? Processor::Linear : Processor::Threaded | ||
|
||
super app, options | ||
end | ||
|
||
def call env, counter = { :recursion => 0, :includes => 0 } | ||
return @app.call(env) if skip_path? env['PATH_INFO'] | ||
def queue(&block) | ||
unless @queue | ||
@queue, @group = Queue.new, ThreadGroup.new | ||
@poolsize.times { @group.add Worker.new(@queue) } | ||
|
||
status, headers, input = @app.call env.dup | ||
return status, headers, input if skip_type? headers['Content-Type'] | ||
|
||
output = [] | ||
input.each { |body| output << compile_body(body, env, counter) } | ||
|
||
Rack::Response.new(output, status, headers).finish | ||
end | ||
|
||
private | ||
|
||
def with_compiled_path(env, path) | ||
# TODO: should compile variables. | ||
env.merge 'PATH_INFO' => path, 'REQUEST_URI' => path | ||
at_exit { Finisher.wait @queue } | ||
end | ||
|
||
def fetch(path, env, counter) | ||
call with_compiled_path(env, path), counter if path | ||
rescue => e | ||
return [500, {}, e.backtrace] | ||
end | ||
@queue.push block | ||
end | ||
|
||
# Should I use XML::SAX::Parser? | ||
def compile_body(body, env, counter) | ||
document = Nokogiri.XML body | ||
def build_processor(env) | ||
@processor.new self, env | ||
end | ||
|
||
document.css(CSS, NS).each do |node| | ||
method = METHODS[node.name] and send method, node, env, counter | ||
node.unlink | ||
end | ||
attr_reader :parser, :serializer | ||
|
||
document.to_xhtml | ||
end | ||
def call(env) | ||
return app.call(env) if @skip === env['PATH_INFO'] | ||
|
||
def skip_path?(path) | ||
@paths =~ path if @paths | ||
end | ||
def skip_type?(type) | ||
@types !~ type | ||
end | ||
status, headers, body = app.call env.dup | ||
|
||
def max?(counter) | ||
not counter[:includes] < @max_includes && | ||
counter[:recursion] < @max_recursion | ||
if status == 200 and headers['Content-Type'] =~ /text\/html/ | ||
body = build_processor(env).process body | ||
end | ||
|
||
def esi_include(node, env, counter) | ||
return if max? counter | ||
|
||
counter[:includes] += 1 | ||
counter[:recursion] += 1 | ||
|
||
status, headers, response = fetch node['src'], env, counter | ||
status, headers, response = fetch node['alt'], env, counter if status != 200 | ||
|
||
if status == 200 | ||
data = '' | ||
response.each { |inc| data << inc } | ||
node.before data | ||
elsif node['onerror'] != 'continue' | ||
raise Error.new(status, headers, response) | ||
end | ||
|
||
ensure | ||
counter[:recursion] -= 1 | ||
end | ||
return status, headers, body | ||
end | ||
|
||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
class Rack::ESI | ||
class Processor < Struct.new(:esi, :env) | ||
|
||
class Linear < self | ||
def process_document(d) | ||
d.xpath('//e:*', 'e' => NAMESPACE).each { |n| process_node n } | ||
end | ||
end | ||
autoload :Threaded, File.expand_path('../threaded', __FILE__) | ||
|
||
NAMESPACE = 'http://www.edge-delivery.org/esi/1.0' | ||
Error = Class.new RuntimeError | ||
|
||
def read(enumerable, buffer = '') | ||
enumerable.each { |str| buffer << str } | ||
buffer | ||
end | ||
|
||
def include(path) | ||
# RADAR patron here? | ||
esi.call env.merge('PATH_INFO' => path, 'REQUEST_URI' => path) | ||
rescue => e | ||
return 500, {}, [] | ||
end | ||
def process_node(node) | ||
case node.name | ||
when 'include' | ||
status, headers, body = include node['src'] | ||
|
||
unless status == 200 or node['alt'].nil? | ||
status, headers, body = include node['alt'] | ||
end | ||
|
||
if status == 200 | ||
node.replace read(body) | ||
elsif node['onerror'] != 'continue' | ||
raise Error | ||
end | ||
else | ||
node.remove | ||
end | ||
end | ||
def process_document(document) | ||
raise NotImplementedError | ||
end | ||
def process(body) | ||
document = esi.parser.parse read(body) | ||
process_document document | ||
document.send esi.serializer | ||
end | ||
|
||
end | ||
end |
Oops, something went wrong.