Skip to content

Commit

Permalink
Modified ActionDispatch::Static to allow passing multiple roots
Browse files Browse the repository at this point in the history
  • Loading branch information
drogus committed Sep 3, 2010
1 parent 5b6553e commit 401cd97
Show file tree
Hide file tree
Showing 6 changed files with 104 additions and 21 deletions.
60 changes: 50 additions & 10 deletions actionpack/lib/action_dispatch/middleware/static.rb
Original file line number Original file line Diff line number Diff line change
Expand Up @@ -2,27 +2,59 @@


module ActionDispatch module ActionDispatch
class Static class Static
class FileHandler
def initialize(at, root)
@at = at.chomp("/")
@file_server = ::Rack::File.new(root)
end

def file_exist?(path)
(path = full_readable_path(path)) && File.file?(path)
end

def directory_exist?(path)
(path = full_readable_path(path)) && File.directory?(path)
end

def call(env)
env["PATH_INFO"].gsub!(/^#{@at}/, "")
@file_server.call(env)
end

private
def includes_path?(path)
@at == "" || path =~ /^#{@at}/
end

def full_readable_path(path)
return unless includes_path?(path)
path = path.gsub(/^#{@at}/, "")
File.join(@file_server.root, ::Rack::Utils.unescape(path))
end
end

FILE_METHODS = %w(GET HEAD).freeze FILE_METHODS = %w(GET HEAD).freeze


def initialize(app, root) def initialize(app, roots)
@app = app @app = app
@file_server = ::Rack::File.new(root) roots = normalize_roots(roots)
@file_handlers = file_handlers(roots)
end end


def call(env) def call(env)
path = env['PATH_INFO'].chomp('/') path = env['PATH_INFO'].chomp('/')
method = env['REQUEST_METHOD'] method = env['REQUEST_METHOD']


if FILE_METHODS.include?(method) if FILE_METHODS.include?(method)
if file_exist?(path) if file_handler = file_exist?(path)
return @file_server.call(env) return file_handler.call(env)
else else
cached_path = directory_exist?(path) ? "#{path}/index" : path cached_path = directory_exist?(path) ? "#{path}/index" : path
cached_path += ::ActionController::Base.page_cache_extension cached_path += ::ActionController::Base.page_cache_extension


if file_exist?(cached_path) if file_handler = file_exist?(cached_path)
env['PATH_INFO'] = cached_path env['PATH_INFO'] = cached_path
return @file_server.call(env) return file_handler.call(env)
end end
end end
end end
Expand All @@ -32,13 +64,21 @@ def call(env)


private private
def file_exist?(path) def file_exist?(path)
full_path = File.join(@file_server.root, ::Rack::Utils.unescape(path)) @file_handlers.detect { |f| f.file_exist?(path) }
File.file?(full_path) && File.readable?(full_path)
end end


def directory_exist?(path) def directory_exist?(path)
full_path = File.join(@file_server.root, ::Rack::Utils.unescape(path)) @file_handlers.detect { |f| f.directory_exist?(path) }
File.directory?(full_path) && File.readable?(full_path) end

def normalize_roots(roots)
roots.is_a?(Hash) ? roots : { "/" => roots.chomp("/") }
end

def file_handlers(roots)
roots.map do |at, root|
FileHandler.new(at, root)
end
end end
end end
end end
61 changes: 50 additions & 11 deletions actionpack/test/dispatch/static_test.rb
Original file line number Original file line Diff line number Diff line change
@@ -1,35 +1,74 @@
require 'abstract_unit' require 'abstract_unit'


class StaticTest < ActiveSupport::TestCase module StaticTests
DummyApp = lambda { |env| def test_serves_dynamic_content
[200, {"Content-Type" => "text/plain"}, ["Hello, World!"]]
}
App = ActionDispatch::Static.new(DummyApp, "#{FIXTURE_LOAD_PATH}/public")

test "serves dynamic content" do
assert_equal "Hello, World!", get("/nofile") assert_equal "Hello, World!", get("/nofile")
end end


test "serves static index at root" do def test_serves_static_index_at_root
assert_equal "/index.html", get("/index.html") assert_equal "/index.html", get("/index.html")
assert_equal "/index.html", get("/index") assert_equal "/index.html", get("/index")
assert_equal "/index.html", get("/") assert_equal "/index.html", get("/")
end end


test "serves static file in directory" do def test_serves_static_file_in_directory
assert_equal "/foo/bar.html", get("/foo/bar.html") assert_equal "/foo/bar.html", get("/foo/bar.html")
assert_equal "/foo/bar.html", get("/foo/bar/") assert_equal "/foo/bar.html", get("/foo/bar/")
assert_equal "/foo/bar.html", get("/foo/bar") assert_equal "/foo/bar.html", get("/foo/bar")
end end


test "serves static index file in directory" do def test_serves_static_index_file_in_directory
assert_equal "/foo/index.html", get("/foo/index.html") assert_equal "/foo/index.html", get("/foo/index.html")
assert_equal "/foo/index.html", get("/foo/") assert_equal "/foo/index.html", get("/foo/")
assert_equal "/foo/index.html", get("/foo") assert_equal "/foo/index.html", get("/foo")
end end


private private
def get(path) def get(path)
Rack::MockRequest.new(App).request("GET", path).body Rack::MockRequest.new(@app).request("GET", path).body
end end
end end

class StaticTest < ActiveSupport::TestCase
DummyApp = lambda { |env|
[200, {"Content-Type" => "text/plain"}, ["Hello, World!"]]
}
App = ActionDispatch::Static.new(DummyApp, "#{FIXTURE_LOAD_PATH}/public")

def setup
@app = App
end

include StaticTests
end

class MultipleDirectorisStaticTest < ActiveSupport::TestCase
DummyApp = lambda { |env|
[200, {"Content-Type" => "text/plain"}, ["Hello, World!"]]
}
App = ActionDispatch::Static.new(DummyApp,
{ "/" => "#{FIXTURE_LOAD_PATH}/public",
"/blog" => "#{FIXTURE_LOAD_PATH}/blog_public",
"/foo" => "#{FIXTURE_LOAD_PATH}/non_existing_dir"
})

def setup
@app = App
end

include StaticTests

test "serves files from other mounted directories" do
assert_equal "/blog/index.html", get("/blog/index.html")
assert_equal "/blog/index.html", get("/blog/index")
assert_equal "/blog/index.html", get("/blog/")

assert_equal "/blog/blog.html", get("/blog/blog/")
assert_equal "/blog/blog.html", get("/blog/blog.html")
assert_equal "/blog/blog.html", get("/blog/blog")

assert_equal "/blog/subdir/index.html", get("/blog/subdir/index.html")
assert_equal "/blog/subdir/index.html", get("/blog/subdir/")
assert_equal "/blog/subdir/index.html", get("/blog/subdir")
end
end
1 change: 1 addition & 0 deletions actionpack/test/fixtures/blog_public/.gitignore
Original file line number Original file line Diff line number Diff line change
@@ -0,0 +1 @@
absolute/*
1 change: 1 addition & 0 deletions actionpack/test/fixtures/blog_public/blog.html
Original file line number Original file line Diff line number Diff line change
@@ -0,0 +1 @@
/blog/blog.html
1 change: 1 addition & 0 deletions actionpack/test/fixtures/blog_public/index.html
Original file line number Original file line Diff line number Diff line change
@@ -0,0 +1 @@
/blog/index.html
1 change: 1 addition & 0 deletions actionpack/test/fixtures/blog_public/subdir/index.html
Original file line number Original file line Diff line number Diff line change
@@ -0,0 +1 @@
/blog/subdir/index.html

0 comments on commit 401cd97

Please sign in to comment.