qoobaa-rack 1.0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- data/COPYING +18 -0
- data/KNOWN-ISSUES +18 -0
- data/RDOX +0 -0
- data/README +353 -0
- data/Rakefile +164 -0
- data/SPEC +164 -0
- data/bin/rackup +176 -0
- data/contrib/rack_logo.svg +111 -0
- data/example/lobster.ru +4 -0
- data/example/protectedlobster.rb +14 -0
- data/example/protectedlobster.ru +8 -0
- data/lib/rack/adapter/camping.rb +22 -0
- data/lib/rack/auth/abstract/handler.rb +37 -0
- data/lib/rack/auth/abstract/request.rb +37 -0
- data/lib/rack/auth/basic.rb +58 -0
- data/lib/rack/auth/digest/md5.rb +124 -0
- data/lib/rack/auth/digest/nonce.rb +51 -0
- data/lib/rack/auth/digest/params.rb +55 -0
- data/lib/rack/auth/digest/request.rb +40 -0
- data/lib/rack/auth/openid.rb +487 -0
- data/lib/rack/builder.rb +63 -0
- data/lib/rack/cascade.rb +41 -0
- data/lib/rack/chunked.rb +49 -0
- data/lib/rack/commonlogger.rb +52 -0
- data/lib/rack/conditionalget.rb +47 -0
- data/lib/rack/content_length.rb +29 -0
- data/lib/rack/content_type.rb +23 -0
- data/lib/rack/deflater.rb +96 -0
- data/lib/rack/directory.rb +153 -0
- data/lib/rack/file.rb +88 -0
- data/lib/rack/handler/cgi.rb +61 -0
- data/lib/rack/handler/evented_mongrel.rb +8 -0
- data/lib/rack/handler/fastcgi.rb +88 -0
- data/lib/rack/handler/lsws.rb +60 -0
- data/lib/rack/handler/mongrel.rb +87 -0
- data/lib/rack/handler/scgi.rb +62 -0
- data/lib/rack/handler/swiftiplied_mongrel.rb +8 -0
- data/lib/rack/handler/thin.rb +18 -0
- data/lib/rack/handler/webrick.rb +71 -0
- data/lib/rack/handler.rb +69 -0
- data/lib/rack/head.rb +19 -0
- data/lib/rack/lint.rb +546 -0
- data/lib/rack/lobster.rb +65 -0
- data/lib/rack/lock.rb +16 -0
- data/lib/rack/methodoverride.rb +27 -0
- data/lib/rack/mime.rb +204 -0
- data/lib/rack/mock.rb +187 -0
- data/lib/rack/recursive.rb +57 -0
- data/lib/rack/reloader.rb +107 -0
- data/lib/rack/request.rb +248 -0
- data/lib/rack/response.rb +183 -0
- data/lib/rack/rewindable_input.rb +100 -0
- data/lib/rack/session/abstract/id.rb +142 -0
- data/lib/rack/session/cookie.rb +91 -0
- data/lib/rack/session/memcache.rb +109 -0
- data/lib/rack/session/pool.rb +100 -0
- data/lib/rack/showexceptions.rb +349 -0
- data/lib/rack/showstatus.rb +106 -0
- data/lib/rack/static.rb +38 -0
- data/lib/rack/urlmap.rb +55 -0
- data/lib/rack/utils.rb +528 -0
- data/lib/rack.rb +90 -0
- data/rack.gemspec +60 -0
- data/test/cgi/lighttpd.conf +20 -0
- data/test/cgi/test +9 -0
- data/test/cgi/test.fcgi +8 -0
- data/test/cgi/test.ru +7 -0
- data/test/multipart/binary +0 -0
- data/test/multipart/empty +10 -0
- data/test/multipart/file1.txt +1 -0
- data/test/multipart/ie +6 -0
- data/test/multipart/nested +10 -0
- data/test/multipart/none +9 -0
- data/test/multipart/text +10 -0
- data/test/spec_rack_auth_basic.rb +73 -0
- data/test/spec_rack_auth_digest.rb +226 -0
- data/test/spec_rack_auth_openid.rb +84 -0
- data/test/spec_rack_builder.rb +84 -0
- data/test/spec_rack_camping.rb +51 -0
- data/test/spec_rack_cascade.rb +48 -0
- data/test/spec_rack_cgi.rb +89 -0
- data/test/spec_rack_chunked.rb +62 -0
- data/test/spec_rack_commonlogger.rb +61 -0
- data/test/spec_rack_conditionalget.rb +41 -0
- data/test/spec_rack_content_length.rb +43 -0
- data/test/spec_rack_content_type.rb +30 -0
- data/test/spec_rack_deflater.rb +127 -0
- data/test/spec_rack_directory.rb +61 -0
- data/test/spec_rack_fastcgi.rb +89 -0
- data/test/spec_rack_file.rb +75 -0
- data/test/spec_rack_handler.rb +43 -0
- data/test/spec_rack_head.rb +30 -0
- data/test/spec_rack_lint.rb +521 -0
- data/test/spec_rack_lobster.rb +45 -0
- data/test/spec_rack_lock.rb +38 -0
- data/test/spec_rack_methodoverride.rb +60 -0
- data/test/spec_rack_mock.rb +243 -0
- data/test/spec_rack_mongrel.rb +189 -0
- data/test/spec_rack_recursive.rb +77 -0
- data/test/spec_rack_request.rb +504 -0
- data/test/spec_rack_response.rb +218 -0
- data/test/spec_rack_rewindable_input.rb +118 -0
- data/test/spec_rack_session_cookie.rb +82 -0
- data/test/spec_rack_session_memcache.rb +250 -0
- data/test/spec_rack_session_pool.rb +172 -0
- data/test/spec_rack_showexceptions.rb +21 -0
- data/test/spec_rack_showstatus.rb +72 -0
- data/test/spec_rack_static.rb +37 -0
- data/test/spec_rack_thin.rb +91 -0
- data/test/spec_rack_urlmap.rb +185 -0
- data/test/spec_rack_utils.rb +467 -0
- data/test/spec_rack_webrick.rb +130 -0
- data/test/testrequest.rb +57 -0
- data/test/unregistered_handler/rack/handler/unregistered.rb +7 -0
- data/test/unregistered_handler/rack/handler/unregistered_long_one.rb +7 -0
- metadata +276 -0
@@ -0,0 +1,91 @@
|
|
1
|
+
require 'openssl'
|
2
|
+
require 'rack/request'
|
3
|
+
require 'rack/response'
|
4
|
+
|
5
|
+
module Rack
|
6
|
+
|
7
|
+
module Session
|
8
|
+
|
9
|
+
# Rack::Session::Cookie provides simple cookie based session management.
|
10
|
+
# The session is a Ruby Hash stored as base64 encoded marshalled data
|
11
|
+
# set to :key (default: rack.session).
|
12
|
+
# When the secret key is set, cookie data is checked for data integrity.
|
13
|
+
#
|
14
|
+
# Example:
|
15
|
+
#
|
16
|
+
# use Rack::Session::Cookie, :key => 'rack.session',
|
17
|
+
# :domain => 'foo.com',
|
18
|
+
# :path => '/',
|
19
|
+
# :expire_after => 2592000,
|
20
|
+
# :secret => 'change_me'
|
21
|
+
#
|
22
|
+
# All parameters are optional.
|
23
|
+
|
24
|
+
class Cookie
|
25
|
+
|
26
|
+
def initialize(app, options={})
|
27
|
+
@app = app
|
28
|
+
@key = options[:key] || "rack.session"
|
29
|
+
@secret = options[:secret]
|
30
|
+
@default_options = {:domain => nil,
|
31
|
+
:path => "/",
|
32
|
+
:expire_after => nil}.merge(options)
|
33
|
+
end
|
34
|
+
|
35
|
+
def call(env)
|
36
|
+
load_session(env)
|
37
|
+
status, headers, body = @app.call(env)
|
38
|
+
commit_session(env, status, headers, body)
|
39
|
+
end
|
40
|
+
|
41
|
+
private
|
42
|
+
|
43
|
+
def load_session(env)
|
44
|
+
request = Rack::Request.new(env)
|
45
|
+
session_data = request.cookies[@key]
|
46
|
+
|
47
|
+
if @secret && session_data
|
48
|
+
session_data, digest = session_data.split("--")
|
49
|
+
session_data = nil unless digest == generate_hmac(session_data)
|
50
|
+
end
|
51
|
+
|
52
|
+
begin
|
53
|
+
session_data = session_data.unpack("m*").first
|
54
|
+
session_data = Marshal.load(session_data)
|
55
|
+
env["rack.session"] = session_data
|
56
|
+
rescue
|
57
|
+
env["rack.session"] = Hash.new
|
58
|
+
end
|
59
|
+
|
60
|
+
env["rack.session.options"] = @default_options.dup
|
61
|
+
end
|
62
|
+
|
63
|
+
def commit_session(env, status, headers, body)
|
64
|
+
session_data = Marshal.dump(env["rack.session"])
|
65
|
+
session_data = [session_data].pack("m*")
|
66
|
+
|
67
|
+
if @secret
|
68
|
+
session_data = "#{session_data}--#{generate_hmac(session_data)}"
|
69
|
+
end
|
70
|
+
|
71
|
+
if session_data.size > (4096 - @key.size)
|
72
|
+
env["rack.errors"].puts("Warning! Rack::Session::Cookie data size exceeds 4K. Content dropped.")
|
73
|
+
[status, headers, body]
|
74
|
+
else
|
75
|
+
options = env["rack.session.options"]
|
76
|
+
cookie = Hash.new
|
77
|
+
cookie[:value] = session_data
|
78
|
+
cookie[:expires] = Time.now + options[:expire_after] unless options[:expire_after].nil?
|
79
|
+
response = Rack::Response.new(body, status, headers)
|
80
|
+
response.set_cookie(@key, cookie.merge(options))
|
81
|
+
response.to_a
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
def generate_hmac(data)
|
86
|
+
OpenSSL::HMAC.hexdigest(OpenSSL::Digest::SHA1.new, @secret, data)
|
87
|
+
end
|
88
|
+
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
@@ -0,0 +1,109 @@
|
|
1
|
+
# AUTHOR: blink <blinketje@gmail.com>; blink#ruby-lang@irc.freenode.net
|
2
|
+
|
3
|
+
require 'rack/session/abstract/id'
|
4
|
+
require 'memcache'
|
5
|
+
|
6
|
+
module Rack
|
7
|
+
module Session
|
8
|
+
# Rack::Session::Memcache provides simple cookie based session management.
|
9
|
+
# Session data is stored in memcached. The corresponding session key is
|
10
|
+
# maintained in the cookie.
|
11
|
+
# You may treat Session::Memcache as you would Session::Pool with the
|
12
|
+
# following caveats.
|
13
|
+
#
|
14
|
+
# * Setting :expire_after to 0 would note to the Memcache server to hang
|
15
|
+
# onto the session data until it would drop it according to it's own
|
16
|
+
# specifications. However, the cookie sent to the client would expire
|
17
|
+
# immediately.
|
18
|
+
#
|
19
|
+
# Note that memcache does drop data before it may be listed to expire. For
|
20
|
+
# a full description of behaviour, please see memcache's documentation.
|
21
|
+
|
22
|
+
class Memcache < Abstract::ID
|
23
|
+
attr_reader :mutex, :pool
|
24
|
+
DEFAULT_OPTIONS = Abstract::ID::DEFAULT_OPTIONS.merge \
|
25
|
+
:namespace => 'rack:session',
|
26
|
+
:memcache_server => 'localhost:11211'
|
27
|
+
|
28
|
+
def initialize(app, options={})
|
29
|
+
super
|
30
|
+
|
31
|
+
@mutex = Mutex.new
|
32
|
+
@pool = MemCache.
|
33
|
+
new @default_options[:memcache_server], @default_options
|
34
|
+
raise 'No memcache servers' unless @pool.servers.any?{|s|s.alive?}
|
35
|
+
end
|
36
|
+
|
37
|
+
def generate_sid
|
38
|
+
loop do
|
39
|
+
sid = super
|
40
|
+
break sid unless @pool.get(sid, true)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def get_session(env, sid)
|
45
|
+
session = @pool.get(sid) if sid
|
46
|
+
@mutex.lock if env['rack.multithread']
|
47
|
+
unless sid and session
|
48
|
+
env['rack.errors'].puts("Session '#{sid.inspect}' not found, initializing...") if $VERBOSE and not sid.nil?
|
49
|
+
session = {}
|
50
|
+
sid = generate_sid
|
51
|
+
ret = @pool.add sid, session
|
52
|
+
raise "Session collision on '#{sid.inspect}'" unless /^STORED/ =~ ret
|
53
|
+
end
|
54
|
+
session.instance_variable_set('@old', {}.merge(session))
|
55
|
+
return [sid, session]
|
56
|
+
rescue MemCache::MemCacheError, Errno::ECONNREFUSED # MemCache server cannot be contacted
|
57
|
+
warn "#{self} is unable to find server."
|
58
|
+
warn $!.inspect
|
59
|
+
return [ nil, {} ]
|
60
|
+
ensure
|
61
|
+
@mutex.unlock if env['rack.multithread']
|
62
|
+
end
|
63
|
+
|
64
|
+
def set_session(env, session_id, new_session, options)
|
65
|
+
expiry = options[:expire_after]
|
66
|
+
expiry = expiry.nil? ? 0 : expiry + 1
|
67
|
+
|
68
|
+
@mutex.lock if env['rack.multithread']
|
69
|
+
session = @pool.get(session_id) || {}
|
70
|
+
if options[:renew] or options[:drop]
|
71
|
+
@pool.delete session_id
|
72
|
+
return false if options[:drop]
|
73
|
+
session_id = generate_sid
|
74
|
+
@pool.add session_id, 0 # so we don't worry about cache miss on #set
|
75
|
+
end
|
76
|
+
old_session = new_session.instance_variable_get('@old') || {}
|
77
|
+
session = merge_sessions session_id, old_session, new_session, session
|
78
|
+
@pool.set session_id, session, expiry
|
79
|
+
return session_id
|
80
|
+
rescue MemCache::MemCacheError, Errno::ECONNREFUSED # MemCache server cannot be contacted
|
81
|
+
warn "#{self} is unable to find server."
|
82
|
+
warn $!.inspect
|
83
|
+
return false
|
84
|
+
ensure
|
85
|
+
@mutex.unlock if env['rack.multithread']
|
86
|
+
end
|
87
|
+
|
88
|
+
private
|
89
|
+
|
90
|
+
def merge_sessions sid, old, new, cur=nil
|
91
|
+
cur ||= {}
|
92
|
+
unless Hash === old and Hash === new
|
93
|
+
warn 'Bad old or new sessions provided.'
|
94
|
+
return cur
|
95
|
+
end
|
96
|
+
|
97
|
+
delete = old.keys - new.keys
|
98
|
+
warn "//@#{sid}: delete #{delete*','}" if $VERBOSE and not delete.empty?
|
99
|
+
delete.each{|k| cur.delete k }
|
100
|
+
|
101
|
+
update = new.keys.select{|k| new[k] != old[k] }
|
102
|
+
warn "//@#{sid}: update #{update*','}" if $VERBOSE and not update.empty?
|
103
|
+
update.each{|k| cur[k] = new[k] }
|
104
|
+
|
105
|
+
cur
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
end
|
@@ -0,0 +1,100 @@
|
|
1
|
+
# AUTHOR: blink <blinketje@gmail.com>; blink#ruby-lang@irc.freenode.net
|
2
|
+
# THANKS:
|
3
|
+
# apeiros, for session id generation, expiry setup, and threadiness
|
4
|
+
# sergio, threadiness and bugreps
|
5
|
+
|
6
|
+
require 'rack/session/abstract/id'
|
7
|
+
require 'thread'
|
8
|
+
|
9
|
+
module Rack
|
10
|
+
module Session
|
11
|
+
# Rack::Session::Pool provides simple cookie based session management.
|
12
|
+
# Session data is stored in a hash held by @pool.
|
13
|
+
# In the context of a multithreaded environment, sessions being
|
14
|
+
# committed to the pool is done in a merging manner.
|
15
|
+
#
|
16
|
+
# The :drop option is available in rack.session.options if you with to
|
17
|
+
# explicitly remove the session from the session cache.
|
18
|
+
#
|
19
|
+
# Example:
|
20
|
+
# myapp = MyRackApp.new
|
21
|
+
# sessioned = Rack::Session::Pool.new(myapp,
|
22
|
+
# :domain => 'foo.com',
|
23
|
+
# :expire_after => 2592000
|
24
|
+
# )
|
25
|
+
# Rack::Handler::WEBrick.run sessioned
|
26
|
+
|
27
|
+
class Pool < Abstract::ID
|
28
|
+
attr_reader :mutex, :pool
|
29
|
+
DEFAULT_OPTIONS = Abstract::ID::DEFAULT_OPTIONS.merge :drop => false
|
30
|
+
|
31
|
+
def initialize(app, options={})
|
32
|
+
super
|
33
|
+
@pool = Hash.new
|
34
|
+
@mutex = Mutex.new
|
35
|
+
end
|
36
|
+
|
37
|
+
def generate_sid
|
38
|
+
loop do
|
39
|
+
sid = super
|
40
|
+
break sid unless @pool.key? sid
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def get_session(env, sid)
|
45
|
+
session = @pool[sid] if sid
|
46
|
+
@mutex.lock if env['rack.multithread']
|
47
|
+
unless sid and session
|
48
|
+
env['rack.errors'].puts("Session '#{sid.inspect}' not found, initializing...") if $VERBOSE and not sid.nil?
|
49
|
+
session = {}
|
50
|
+
sid = generate_sid
|
51
|
+
@pool.store sid, session
|
52
|
+
end
|
53
|
+
session.instance_variable_set('@old', {}.merge(session))
|
54
|
+
return [sid, session]
|
55
|
+
ensure
|
56
|
+
@mutex.unlock if env['rack.multithread']
|
57
|
+
end
|
58
|
+
|
59
|
+
def set_session(env, session_id, new_session, options)
|
60
|
+
@mutex.lock if env['rack.multithread']
|
61
|
+
session = @pool[session_id]
|
62
|
+
if options[:renew] or options[:drop]
|
63
|
+
@pool.delete session_id
|
64
|
+
return false if options[:drop]
|
65
|
+
session_id = generate_sid
|
66
|
+
@pool.store session_id, 0
|
67
|
+
end
|
68
|
+
old_session = new_session.instance_variable_get('@old') || {}
|
69
|
+
session = merge_sessions session_id, old_session, new_session, session
|
70
|
+
@pool.store session_id, session
|
71
|
+
return session_id
|
72
|
+
rescue
|
73
|
+
warn "#{new_session.inspect} has been lost."
|
74
|
+
warn $!.inspect
|
75
|
+
ensure
|
76
|
+
@mutex.unlock if env['rack.multithread']
|
77
|
+
end
|
78
|
+
|
79
|
+
private
|
80
|
+
|
81
|
+
def merge_sessions sid, old, new, cur=nil
|
82
|
+
cur ||= {}
|
83
|
+
unless Hash === old and Hash === new
|
84
|
+
warn 'Bad old or new sessions provided.'
|
85
|
+
return cur
|
86
|
+
end
|
87
|
+
|
88
|
+
delete = old.keys - new.keys
|
89
|
+
warn "//@#{sid}: dropping #{delete*','}" if $DEBUG and not delete.empty?
|
90
|
+
delete.each{|k| cur.delete k }
|
91
|
+
|
92
|
+
update = new.keys.select{|k| new[k] != old[k] }
|
93
|
+
warn "//@#{sid}: updating #{update*','}" if $DEBUG and not update.empty?
|
94
|
+
update.each{|k| cur[k] = new[k] }
|
95
|
+
|
96
|
+
cur
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
@@ -0,0 +1,349 @@
|
|
1
|
+
require 'ostruct'
|
2
|
+
require 'erb'
|
3
|
+
require 'rack/request'
|
4
|
+
require 'rack/utils'
|
5
|
+
|
6
|
+
module Rack
|
7
|
+
# Rack::ShowExceptions catches all exceptions raised from the app it
|
8
|
+
# wraps. It shows a useful backtrace with the sourcefile and
|
9
|
+
# clickable context, the whole Rack environment and the request
|
10
|
+
# data.
|
11
|
+
#
|
12
|
+
# Be careful when you use this on public-facing sites as it could
|
13
|
+
# reveal information helpful to attackers.
|
14
|
+
|
15
|
+
class ShowExceptions
|
16
|
+
CONTEXT = 7
|
17
|
+
|
18
|
+
def initialize(app)
|
19
|
+
@app = app
|
20
|
+
@template = ERB.new(TEMPLATE)
|
21
|
+
end
|
22
|
+
|
23
|
+
def call(env)
|
24
|
+
@app.call(env)
|
25
|
+
rescue StandardError, LoadError, SyntaxError => e
|
26
|
+
backtrace = pretty(env, e)
|
27
|
+
[500,
|
28
|
+
{"Content-Type" => "text/html",
|
29
|
+
"Content-Length" => backtrace.join.size.to_s},
|
30
|
+
backtrace]
|
31
|
+
end
|
32
|
+
|
33
|
+
def pretty(env, exception)
|
34
|
+
req = Rack::Request.new(env)
|
35
|
+
path = (req.script_name + req.path_info).squeeze("/")
|
36
|
+
|
37
|
+
frames = exception.backtrace.map { |line|
|
38
|
+
frame = OpenStruct.new
|
39
|
+
if line =~ /(.*?):(\d+)(:in `(.*)')?/
|
40
|
+
frame.filename = $1
|
41
|
+
frame.lineno = $2.to_i
|
42
|
+
frame.function = $4
|
43
|
+
|
44
|
+
begin
|
45
|
+
lineno = frame.lineno-1
|
46
|
+
lines = ::File.readlines(frame.filename)
|
47
|
+
frame.pre_context_lineno = [lineno-CONTEXT, 0].max
|
48
|
+
frame.pre_context = lines[frame.pre_context_lineno...lineno]
|
49
|
+
frame.context_line = lines[lineno].chomp
|
50
|
+
frame.post_context_lineno = [lineno+CONTEXT, lines.size].min
|
51
|
+
frame.post_context = lines[lineno+1..frame.post_context_lineno]
|
52
|
+
rescue
|
53
|
+
end
|
54
|
+
|
55
|
+
frame
|
56
|
+
else
|
57
|
+
nil
|
58
|
+
end
|
59
|
+
}.compact
|
60
|
+
|
61
|
+
env["rack.errors"].puts "#{exception.class}: #{exception.message}"
|
62
|
+
env["rack.errors"].puts exception.backtrace.map { |l| "\t" + l }
|
63
|
+
env["rack.errors"].flush
|
64
|
+
|
65
|
+
[@template.result(binding)]
|
66
|
+
end
|
67
|
+
|
68
|
+
def h(obj) # :nodoc:
|
69
|
+
case obj
|
70
|
+
when String
|
71
|
+
Utils.escape_html(obj)
|
72
|
+
else
|
73
|
+
Utils.escape_html(obj.inspect)
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
# :stopdoc:
|
78
|
+
|
79
|
+
# adapted from Django <djangoproject.com>
|
80
|
+
# Copyright (c) 2005, the Lawrence Journal-World
|
81
|
+
# Used under the modified BSD license:
|
82
|
+
# http://www.xfree86.org/3.3.6/COPYRIGHT2.html#5
|
83
|
+
TEMPLATE = <<'HTML'
|
84
|
+
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
|
85
|
+
<html lang="en">
|
86
|
+
<head>
|
87
|
+
<meta http-equiv="content-type" content="text/html; charset=utf-8" />
|
88
|
+
<meta name="robots" content="NONE,NOARCHIVE" />
|
89
|
+
<title><%=h exception.class %> at <%=h path %></title>
|
90
|
+
<style type="text/css">
|
91
|
+
html * { padding:0; margin:0; }
|
92
|
+
body * { padding:10px 20px; }
|
93
|
+
body * * { padding:0; }
|
94
|
+
body { font:small sans-serif; }
|
95
|
+
body>div { border-bottom:1px solid #ddd; }
|
96
|
+
h1 { font-weight:normal; }
|
97
|
+
h2 { margin-bottom:.8em; }
|
98
|
+
h2 span { font-size:80%; color:#666; font-weight:normal; }
|
99
|
+
h3 { margin:1em 0 .5em 0; }
|
100
|
+
h4 { margin:0 0 .5em 0; font-weight: normal; }
|
101
|
+
table {
|
102
|
+
border:1px solid #ccc; border-collapse: collapse; background:white; }
|
103
|
+
tbody td, tbody th { vertical-align:top; padding:2px 3px; }
|
104
|
+
thead th {
|
105
|
+
padding:1px 6px 1px 3px; background:#fefefe; text-align:left;
|
106
|
+
font-weight:normal; font-size:11px; border:1px solid #ddd; }
|
107
|
+
tbody th { text-align:right; color:#666; padding-right:.5em; }
|
108
|
+
table.vars { margin:5px 0 2px 40px; }
|
109
|
+
table.vars td, table.req td { font-family:monospace; }
|
110
|
+
table td.code { width:100%;}
|
111
|
+
table td.code div { overflow:hidden; }
|
112
|
+
table.source th { color:#666; }
|
113
|
+
table.source td {
|
114
|
+
font-family:monospace; white-space:pre; border-bottom:1px solid #eee; }
|
115
|
+
ul.traceback { list-style-type:none; }
|
116
|
+
ul.traceback li.frame { margin-bottom:1em; }
|
117
|
+
div.context { margin: 10px 0; }
|
118
|
+
div.context ol {
|
119
|
+
padding-left:30px; margin:0 10px; list-style-position: inside; }
|
120
|
+
div.context ol li {
|
121
|
+
font-family:monospace; white-space:pre; color:#666; cursor:pointer; }
|
122
|
+
div.context ol.context-line li { color:black; background-color:#ccc; }
|
123
|
+
div.context ol.context-line li span { float: right; }
|
124
|
+
div.commands { margin-left: 40px; }
|
125
|
+
div.commands a { color:black; text-decoration:none; }
|
126
|
+
#summary { background: #ffc; }
|
127
|
+
#summary h2 { font-weight: normal; color: #666; }
|
128
|
+
#summary ul#quicklinks { list-style-type: none; margin-bottom: 2em; }
|
129
|
+
#summary ul#quicklinks li { float: left; padding: 0 1em; }
|
130
|
+
#summary ul#quicklinks>li+li { border-left: 1px #666 solid; }
|
131
|
+
#explanation { background:#eee; }
|
132
|
+
#template, #template-not-exist { background:#f6f6f6; }
|
133
|
+
#template-not-exist ul { margin: 0 0 0 20px; }
|
134
|
+
#traceback { background:#eee; }
|
135
|
+
#requestinfo { background:#f6f6f6; padding-left:120px; }
|
136
|
+
#summary table { border:none; background:transparent; }
|
137
|
+
#requestinfo h2, #requestinfo h3 { position:relative; margin-left:-100px; }
|
138
|
+
#requestinfo h3 { margin-bottom:-1em; }
|
139
|
+
.error { background: #ffc; }
|
140
|
+
.specific { color:#cc3300; font-weight:bold; }
|
141
|
+
</style>
|
142
|
+
<script type="text/javascript">
|
143
|
+
//<!--
|
144
|
+
function getElementsByClassName(oElm, strTagName, strClassName){
|
145
|
+
// Written by Jonathan Snook, http://www.snook.ca/jon;
|
146
|
+
// Add-ons by Robert Nyman, http://www.robertnyman.com
|
147
|
+
var arrElements = (strTagName == "*" && document.all)? document.all :
|
148
|
+
oElm.getElementsByTagName(strTagName);
|
149
|
+
var arrReturnElements = new Array();
|
150
|
+
strClassName = strClassName.replace(/\-/g, "\\-");
|
151
|
+
var oRegExp = new RegExp("(^|\\s)" + strClassName + "(\\s|$$)");
|
152
|
+
var oElement;
|
153
|
+
for(var i=0; i<arrElements.length; i++){
|
154
|
+
oElement = arrElements[i];
|
155
|
+
if(oRegExp.test(oElement.className)){
|
156
|
+
arrReturnElements.push(oElement);
|
157
|
+
}
|
158
|
+
}
|
159
|
+
return (arrReturnElements)
|
160
|
+
}
|
161
|
+
function hideAll(elems) {
|
162
|
+
for (var e = 0; e < elems.length; e++) {
|
163
|
+
elems[e].style.display = 'none';
|
164
|
+
}
|
165
|
+
}
|
166
|
+
window.onload = function() {
|
167
|
+
hideAll(getElementsByClassName(document, 'table', 'vars'));
|
168
|
+
hideAll(getElementsByClassName(document, 'ol', 'pre-context'));
|
169
|
+
hideAll(getElementsByClassName(document, 'ol', 'post-context'));
|
170
|
+
}
|
171
|
+
function toggle() {
|
172
|
+
for (var i = 0; i < arguments.length; i++) {
|
173
|
+
var e = document.getElementById(arguments[i]);
|
174
|
+
if (e) {
|
175
|
+
e.style.display = e.style.display == 'none' ? 'block' : 'none';
|
176
|
+
}
|
177
|
+
}
|
178
|
+
return false;
|
179
|
+
}
|
180
|
+
function varToggle(link, id) {
|
181
|
+
toggle('v' + id);
|
182
|
+
var s = link.getElementsByTagName('span')[0];
|
183
|
+
var uarr = String.fromCharCode(0x25b6);
|
184
|
+
var darr = String.fromCharCode(0x25bc);
|
185
|
+
s.innerHTML = s.innerHTML == uarr ? darr : uarr;
|
186
|
+
return false;
|
187
|
+
}
|
188
|
+
//-->
|
189
|
+
</script>
|
190
|
+
</head>
|
191
|
+
<body>
|
192
|
+
|
193
|
+
<div id="summary">
|
194
|
+
<h1><%=h exception.class %> at <%=h path %></h1>
|
195
|
+
<h2><%=h exception.message %></h2>
|
196
|
+
<table><tr>
|
197
|
+
<th>Ruby</th>
|
198
|
+
<td><code><%=h frames.first.filename %></code>: in <code><%=h frames.first.function %></code>, line <%=h frames.first.lineno %></td>
|
199
|
+
</tr><tr>
|
200
|
+
<th>Web</th>
|
201
|
+
<td><code><%=h req.request_method %> <%=h(req.host + path)%></code></td>
|
202
|
+
</tr></table>
|
203
|
+
|
204
|
+
<h3>Jump to:</h3>
|
205
|
+
<ul id="quicklinks">
|
206
|
+
<li><a href="#get-info">GET</a></li>
|
207
|
+
<li><a href="#post-info">POST</a></li>
|
208
|
+
<li><a href="#cookie-info">Cookies</a></li>
|
209
|
+
<li><a href="#env-info">ENV</a></li>
|
210
|
+
</ul>
|
211
|
+
</div>
|
212
|
+
|
213
|
+
<div id="traceback">
|
214
|
+
<h2>Traceback <span>(innermost first)</span></h2>
|
215
|
+
<ul class="traceback">
|
216
|
+
<% frames.each { |frame| %>
|
217
|
+
<li class="frame">
|
218
|
+
<code><%=h frame.filename %></code>: in <code><%=h frame.function %></code>
|
219
|
+
|
220
|
+
<% if frame.context_line %>
|
221
|
+
<div class="context" id="c<%=h frame.object_id %>">
|
222
|
+
<% if frame.pre_context %>
|
223
|
+
<ol start="<%=h frame.pre_context_lineno+1 %>" class="pre-context" id="pre<%=h frame.object_id %>">
|
224
|
+
<% frame.pre_context.each { |line| %>
|
225
|
+
<li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h line %></li>
|
226
|
+
<% } %>
|
227
|
+
</ol>
|
228
|
+
<% end %>
|
229
|
+
|
230
|
+
<ol start="<%=h frame.lineno %>" class="context-line">
|
231
|
+
<li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h frame.context_line %><span>...</span></li></ol>
|
232
|
+
|
233
|
+
<% if frame.post_context %>
|
234
|
+
<ol start='<%=h frame.lineno+1 %>' class="post-context" id="post<%=h frame.object_id %>">
|
235
|
+
<% frame.post_context.each { |line| %>
|
236
|
+
<li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h line %></li>
|
237
|
+
<% } %>
|
238
|
+
</ol>
|
239
|
+
<% end %>
|
240
|
+
</div>
|
241
|
+
<% end %>
|
242
|
+
</li>
|
243
|
+
<% } %>
|
244
|
+
</ul>
|
245
|
+
</div>
|
246
|
+
|
247
|
+
<div id="requestinfo">
|
248
|
+
<h2>Request information</h2>
|
249
|
+
|
250
|
+
<h3 id="get-info">GET</h3>
|
251
|
+
<% unless req.GET.empty? %>
|
252
|
+
<table class="req">
|
253
|
+
<thead>
|
254
|
+
<tr>
|
255
|
+
<th>Variable</th>
|
256
|
+
<th>Value</th>
|
257
|
+
</tr>
|
258
|
+
</thead>
|
259
|
+
<tbody>
|
260
|
+
<% req.GET.sort_by { |k, v| k.to_s }.each { |key, val| %>
|
261
|
+
<tr>
|
262
|
+
<td><%=h key %></td>
|
263
|
+
<td class="code"><div><%=h val.inspect %></div></td>
|
264
|
+
</tr>
|
265
|
+
<% } %>
|
266
|
+
</tbody>
|
267
|
+
</table>
|
268
|
+
<% else %>
|
269
|
+
<p>No GET data.</p>
|
270
|
+
<% end %>
|
271
|
+
|
272
|
+
<h3 id="post-info">POST</h3>
|
273
|
+
<% unless req.POST.empty? %>
|
274
|
+
<table class="req">
|
275
|
+
<thead>
|
276
|
+
<tr>
|
277
|
+
<th>Variable</th>
|
278
|
+
<th>Value</th>
|
279
|
+
</tr>
|
280
|
+
</thead>
|
281
|
+
<tbody>
|
282
|
+
<% req.POST.sort_by { |k, v| k.to_s }.each { |key, val| %>
|
283
|
+
<tr>
|
284
|
+
<td><%=h key %></td>
|
285
|
+
<td class="code"><div><%=h val.inspect %></div></td>
|
286
|
+
</tr>
|
287
|
+
<% } %>
|
288
|
+
</tbody>
|
289
|
+
</table>
|
290
|
+
<% else %>
|
291
|
+
<p>No POST data.</p>
|
292
|
+
<% end %>
|
293
|
+
|
294
|
+
|
295
|
+
<h3 id="cookie-info">COOKIES</h3>
|
296
|
+
<% unless req.cookies.empty? %>
|
297
|
+
<table class="req">
|
298
|
+
<thead>
|
299
|
+
<tr>
|
300
|
+
<th>Variable</th>
|
301
|
+
<th>Value</th>
|
302
|
+
</tr>
|
303
|
+
</thead>
|
304
|
+
<tbody>
|
305
|
+
<% req.cookies.each { |key, val| %>
|
306
|
+
<tr>
|
307
|
+
<td><%=h key %></td>
|
308
|
+
<td class="code"><div><%=h val.inspect %></div></td>
|
309
|
+
</tr>
|
310
|
+
<% } %>
|
311
|
+
</tbody>
|
312
|
+
</table>
|
313
|
+
<% else %>
|
314
|
+
<p>No cookie data.</p>
|
315
|
+
<% end %>
|
316
|
+
|
317
|
+
<h3 id="env-info">Rack ENV</h3>
|
318
|
+
<table class="req">
|
319
|
+
<thead>
|
320
|
+
<tr>
|
321
|
+
<th>Variable</th>
|
322
|
+
<th>Value</th>
|
323
|
+
</tr>
|
324
|
+
</thead>
|
325
|
+
<tbody>
|
326
|
+
<% env.sort_by { |k, v| k.to_s }.each { |key, val| %>
|
327
|
+
<tr>
|
328
|
+
<td><%=h key %></td>
|
329
|
+
<td class="code"><div><%=h val %></div></td>
|
330
|
+
</tr>
|
331
|
+
<% } %>
|
332
|
+
</tbody>
|
333
|
+
</table>
|
334
|
+
|
335
|
+
</div>
|
336
|
+
|
337
|
+
<div id="explanation">
|
338
|
+
<p>
|
339
|
+
You're seeing this error because you use <code>Rack::ShowExceptions</code>.
|
340
|
+
</p>
|
341
|
+
</div>
|
342
|
+
|
343
|
+
</body>
|
344
|
+
</html>
|
345
|
+
HTML
|
346
|
+
|
347
|
+
# :startdoc:
|
348
|
+
end
|
349
|
+
end
|