2006-03-27 01:10:07 -05:00
|
|
|
require 'rubygems'
|
|
|
|
begin
|
|
|
|
require 'sendfile'
|
|
|
|
$mongrel_has_sendfile = true
|
|
|
|
STDERR.puts "** You have sendfile installed, will use that to serve files."
|
|
|
|
rescue Object
|
|
|
|
$mongrel_has_sendfile = false
|
|
|
|
end
|
2006-02-20 19:55:39 -05:00
|
|
|
|
|
|
|
module Mongrel
|
|
|
|
|
|
|
|
# You implement your application handler with this. It's very light giving
|
|
|
|
# just the minimum necessary for you to handle a request and shoot back
|
|
|
|
# a response. Look at the HttpRequest and HttpResponse objects for how
|
|
|
|
# to use them.
|
2006-03-25 16:15:30 -05:00
|
|
|
#
|
|
|
|
# This is used for very simple handlers that don't require much to operate.
|
|
|
|
# More extensive plugins or those you intend to distribute as GemPlugins
|
|
|
|
# should be implemented using the HttpHandlerPlugin mixin.
|
|
|
|
#
|
2006-02-20 19:55:39 -05:00
|
|
|
class HttpHandler
|
2006-03-25 16:15:30 -05:00
|
|
|
|
|
|
|
def process(request, response)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# This is used when your handler is implemented as a GemPlugin.
|
|
|
|
# The plugin always takes an options hash which you can modify
|
|
|
|
# and then access later. They are stored by default for
|
|
|
|
# the process method later.
|
|
|
|
module HttpHandlerPlugin
|
|
|
|
attr_reader :options
|
|
|
|
|
|
|
|
def initialize(options={})
|
|
|
|
@options = options
|
|
|
|
end
|
|
|
|
|
2006-02-20 19:55:39 -05:00
|
|
|
def process(request, response)
|
|
|
|
end
|
2006-03-25 16:15:30 -05:00
|
|
|
|
2006-02-20 19:55:39 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# The server normally returns a 404 response if an unknown URI is requested, but it
|
|
|
|
# also returns a lame empty message. This lets you do a 404 response
|
|
|
|
# with a custom message for special URIs.
|
|
|
|
class Error404Handler < HttpHandler
|
|
|
|
|
|
|
|
# Sets the message to return. This is constructed once for the handler
|
|
|
|
# so it's pretty efficient.
|
|
|
|
def initialize(msg)
|
|
|
|
@response = Const::ERROR_404_RESPONSE + msg
|
|
|
|
end
|
|
|
|
|
|
|
|
# Just kicks back the standard 404 response with your special message.
|
|
|
|
def process(request, response)
|
|
|
|
response.socket.write(@response)
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# Serves the contents of a directory. You give it the path to the root
|
|
|
|
# where the files are located, and it tries to find the files based on
|
|
|
|
# the PATH_INFO inside the directory. If the requested path is a
|
|
|
|
# directory then it returns a simple directory listing.
|
|
|
|
#
|
|
|
|
# It does a simple protection against going outside it's root path by
|
|
|
|
# converting all paths to an absolute expanded path, and then making sure
|
|
|
|
# that the final expanded path includes the root path. If it doesn't
|
|
|
|
# than it simply gives a 404.
|
|
|
|
class DirHandler < HttpHandler
|
|
|
|
MIME_TYPES = {
|
|
|
|
".css" => "text/css",
|
|
|
|
".gif" => "image/gif",
|
|
|
|
".htm" => "text/html",
|
|
|
|
".html" => "text/html",
|
|
|
|
".jpeg" => "image/jpeg",
|
|
|
|
".jpg" => "image/jpeg",
|
|
|
|
".js" => "text/javascript",
|
|
|
|
".png" => "image/png",
|
|
|
|
".swf" => "application/x-shockwave-flash",
|
|
|
|
".txt" => "text/plain"
|
|
|
|
}
|
|
|
|
|
2006-03-30 04:31:14 -05:00
|
|
|
ONLY_HEAD_GET="Only HEAD and GET allowed.".freeze
|
2006-02-20 19:55:39 -05:00
|
|
|
|
|
|
|
attr_reader :path
|
|
|
|
|
|
|
|
# You give it the path to the directory root and an (optional)
|
|
|
|
def initialize(path, listing_allowed=true, index_html="index.html")
|
|
|
|
@path = File.expand_path(path)
|
|
|
|
@listing_allowed=listing_allowed
|
|
|
|
@index_html = index_html
|
|
|
|
end
|
|
|
|
|
|
|
|
# Checks if the given path can be served and returns the full path (or nil if not).
|
|
|
|
def can_serve(path_info)
|
|
|
|
req = File.expand_path(File.join(@path,path_info), @path)
|
|
|
|
|
|
|
|
if req.index(@path) == 0 and File.exist? req
|
|
|
|
# it exists and it's in the right location
|
|
|
|
if File.directory? req
|
|
|
|
# the request is for a directory
|
|
|
|
index = File.join(req, @index_html)
|
|
|
|
if File.exist? index
|
|
|
|
# serve the index
|
|
|
|
return index
|
|
|
|
elsif @listing_allowed
|
|
|
|
# serve the directory
|
|
|
|
req
|
|
|
|
else
|
|
|
|
# do not serve anything
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
else
|
|
|
|
# it's a file and it's there
|
|
|
|
return req
|
|
|
|
end
|
|
|
|
else
|
|
|
|
# does not exist or isn't in the right spot
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# Returns a simplistic directory listing if they're enabled, otherwise a 403.
|
|
|
|
# Base is the base URI from the REQUEST_URI, dir is the directory to serve
|
|
|
|
# on the file system (comes from can_serve()), and response is the HttpResponse
|
|
|
|
# object to send the results on.
|
|
|
|
def send_dir_listing(base, dir, response)
|
|
|
|
# take off any trailing / so the links come out right
|
|
|
|
base.chop! if base[-1] == "/"[-1]
|
|
|
|
|
|
|
|
if @listing_allowed
|
|
|
|
response.start(200) do |head,out|
|
2006-03-30 04:31:14 -05:00
|
|
|
head[Const::CONTENT_TYPE] = "text/html"
|
2006-02-20 19:55:39 -05:00
|
|
|
out << "<html><head><title>Directory Listing</title></head><body>"
|
|
|
|
Dir.entries(dir).each do |child|
|
|
|
|
next if child == "."
|
|
|
|
|
|
|
|
if child == ".."
|
|
|
|
out << "<a href=\"#{base}/#{child}\">Up to parent..</a><br/>"
|
|
|
|
else
|
|
|
|
out << "<a href=\"#{base}/#{child}\">#{child}</a><br/>"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
out << "</body></html>"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
response.start(403) do |head,out|
|
|
|
|
out.write("Directory listings not allowed")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# Sends the contents of a file back to the user. Not terribly efficient since it's
|
|
|
|
# opening and closing the file for each read.
|
2006-03-27 21:06:50 -05:00
|
|
|
def send_file(req, response, header_only=false)
|
2006-02-20 19:55:39 -05:00
|
|
|
|
2006-03-27 01:10:07 -05:00
|
|
|
# first we setup the headers and status then we do a very fast send on the socket directly
|
|
|
|
response.status = 200
|
|
|
|
|
|
|
|
# set the mime type from our map based on the ending
|
|
|
|
dot_at = req.rindex(".")
|
|
|
|
if dot_at
|
|
|
|
ext = req[dot_at .. -1]
|
|
|
|
if MIME_TYPES[ext]
|
2006-03-30 04:31:14 -05:00
|
|
|
stat = File.stat(req)
|
2006-04-03 11:22:17 -04:00
|
|
|
response.header[Const::CONTENT_TYPE] = MIME_TYPES[ext] || "text"
|
2006-03-30 04:31:14 -05:00
|
|
|
# TODO: Confirm this works for rfc 1123
|
|
|
|
response.header[Const::LAST_MODIFIED] = HttpServer.httpdate(stat.mtime)
|
|
|
|
# TODO that this is a valid way to calculate an etag
|
|
|
|
response.header[Const::ETAG] = Const::ETAG_FORMAT % [stat.mtime.to_i, stat.size, stat.ino]
|
2006-02-20 19:55:39 -05:00
|
|
|
end
|
|
|
|
end
|
2006-03-27 01:10:07 -05:00
|
|
|
|
2006-03-27 21:06:50 -05:00
|
|
|
response.send_status(File.size(req))
|
2006-03-27 01:10:07 -05:00
|
|
|
response.send_header
|
|
|
|
|
2006-03-27 21:06:50 -05:00
|
|
|
if not header_only
|
|
|
|
begin
|
|
|
|
if $mongrel_has_sendfile
|
|
|
|
File.open(req, "rb") { |f| response.socket.sendfile(f) }
|
|
|
|
else
|
|
|
|
File.open(req, "rb") { |f| response.socket.write(f.read) }
|
|
|
|
end
|
2006-04-03 11:22:17 -04:00
|
|
|
rescue EOFError,Errno::ECONNRESET,Errno::EPIPE,Errno::EINVAL
|
2006-03-28 10:39:53 -05:00
|
|
|
# ignore these since it means the client closed off early
|
2006-03-27 21:06:50 -05:00
|
|
|
end
|
2006-03-28 10:39:53 -05:00
|
|
|
else
|
|
|
|
response.send_body # should send nothing
|
2006-03-27 01:10:07 -05:00
|
|
|
end
|
2006-02-20 19:55:39 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
# Process the request to either serve a file or a directory listing
|
|
|
|
# if allowed (based on the listing_allowed paramter to the constructor).
|
|
|
|
def process(request, response)
|
2006-03-30 04:31:14 -05:00
|
|
|
req_method = request.params[Const::REQUEST_METHOD] || Const::GET
|
|
|
|
req = can_serve request.params[Const::PATH_INFO]
|
2006-02-20 19:55:39 -05:00
|
|
|
if not req
|
|
|
|
# not found, return a 404
|
|
|
|
response.start(404) do |head,out|
|
|
|
|
out << "File not found"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
if File.directory? req
|
2006-03-30 04:31:14 -05:00
|
|
|
send_dir_listing(request.params[Const::REQUEST_URI],req, response)
|
|
|
|
elsif req_method == Const::HEAD
|
2006-03-27 21:06:50 -05:00
|
|
|
send_file(req, response, true)
|
2006-03-30 04:31:14 -05:00
|
|
|
elsif req_method == Const::GET
|
2006-03-27 21:06:50 -05:00
|
|
|
send_file(req, response, false)
|
|
|
|
else
|
2006-03-30 04:31:14 -05:00
|
|
|
response.start(403) {|head,out| out.write(ONLY_HEAD_GET) }
|
2006-02-20 19:55:39 -05:00
|
|
|
end
|
|
|
|
rescue => details
|
2006-03-28 10:39:53 -05:00
|
|
|
STDERR.puts "Error accessing file #{req}: #{details}"
|
2006-03-27 01:10:07 -05:00
|
|
|
STDERR.puts details.backtrace.join("\n")
|
2006-02-20 19:55:39 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# There is a small number of default mime types for extensions, but
|
|
|
|
# this lets you add any others you'll need when serving content.
|
|
|
|
def DirHandler::add_mime_type(extension, type)
|
|
|
|
MIME_TYPES[extension] = type
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|