kastner-rack 0.3.171

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (94) hide show
  1. data/AUTHORS +8 -0
  2. data/COPYING +18 -0
  3. data/KNOWN-ISSUES +18 -0
  4. data/README +273 -0
  5. data/Rakefile +185 -0
  6. data/bin/rackup +172 -0
  7. data/contrib/rack_logo.svg +111 -0
  8. data/example/lobster.ru +4 -0
  9. data/example/protectedlobster.rb +14 -0
  10. data/example/protectedlobster.ru +8 -0
  11. data/lib/rack.rb +85 -0
  12. data/lib/rack/adapter/camping.rb +22 -0
  13. data/lib/rack/auth/abstract/handler.rb +28 -0
  14. data/lib/rack/auth/abstract/request.rb +37 -0
  15. data/lib/rack/auth/basic.rb +58 -0
  16. data/lib/rack/auth/digest/md5.rb +124 -0
  17. data/lib/rack/auth/digest/nonce.rb +51 -0
  18. data/lib/rack/auth/digest/params.rb +55 -0
  19. data/lib/rack/auth/digest/request.rb +40 -0
  20. data/lib/rack/auth/openid.rb +437 -0
  21. data/lib/rack/builder.rb +67 -0
  22. data/lib/rack/cascade.rb +36 -0
  23. data/lib/rack/commonlogger.rb +61 -0
  24. data/lib/rack/conditionalget.rb +42 -0
  25. data/lib/rack/deflater.rb +63 -0
  26. data/lib/rack/directory.rb +149 -0
  27. data/lib/rack/file.rb +84 -0
  28. data/lib/rack/handler.rb +46 -0
  29. data/lib/rack/handler/cgi.rb +57 -0
  30. data/lib/rack/handler/evented_mongrel.rb +8 -0
  31. data/lib/rack/handler/fastcgi.rb +86 -0
  32. data/lib/rack/handler/lsws.rb +52 -0
  33. data/lib/rack/handler/mongrel.rb +78 -0
  34. data/lib/rack/handler/scgi.rb +57 -0
  35. data/lib/rack/handler/swiftiplied_mongrel.rb +8 -0
  36. data/lib/rack/handler/webrick.rb +61 -0
  37. data/lib/rack/head.rb +19 -0
  38. data/lib/rack/lint.rb +463 -0
  39. data/lib/rack/lobster.rb +65 -0
  40. data/lib/rack/methodoverride.rb +21 -0
  41. data/lib/rack/mime.rb +204 -0
  42. data/lib/rack/mock.rb +160 -0
  43. data/lib/rack/recursive.rb +57 -0
  44. data/lib/rack/reloader.rb +64 -0
  45. data/lib/rack/request.rb +217 -0
  46. data/lib/rack/response.rb +171 -0
  47. data/lib/rack/session/abstract/id.rb +140 -0
  48. data/lib/rack/session/cookie.rb +89 -0
  49. data/lib/rack/session/memcache.rb +97 -0
  50. data/lib/rack/session/pool.rb +73 -0
  51. data/lib/rack/showexceptions.rb +348 -0
  52. data/lib/rack/showstatus.rb +105 -0
  53. data/lib/rack/static.rb +38 -0
  54. data/lib/rack/urlmap.rb +48 -0
  55. data/lib/rack/utils.rb +318 -0
  56. data/rack.gemspec +31 -0
  57. data/test/cgi/lighttpd.conf +20 -0
  58. data/test/cgi/test +9 -0
  59. data/test/cgi/test.fcgi +8 -0
  60. data/test/cgi/test.ru +7 -0
  61. data/test/spec_rack_auth_basic.rb +69 -0
  62. data/test/spec_rack_auth_digest.rb +169 -0
  63. data/test/spec_rack_auth_openid.rb +137 -0
  64. data/test/spec_rack_builder.rb +84 -0
  65. data/test/spec_rack_camping.rb +51 -0
  66. data/test/spec_rack_cascade.rb +50 -0
  67. data/test/spec_rack_cgi.rb +89 -0
  68. data/test/spec_rack_commonlogger.rb +32 -0
  69. data/test/spec_rack_conditionalget.rb +41 -0
  70. data/test/spec_rack_deflater.rb +70 -0
  71. data/test/spec_rack_directory.rb +56 -0
  72. data/test/spec_rack_fastcgi.rb +89 -0
  73. data/test/spec_rack_file.rb +57 -0
  74. data/test/spec_rack_handler.rb +24 -0
  75. data/test/spec_rack_head.rb +30 -0
  76. data/test/spec_rack_lint.rb +371 -0
  77. data/test/spec_rack_lobster.rb +45 -0
  78. data/test/spec_rack_methodoverride.rb +31 -0
  79. data/test/spec_rack_mock.rb +152 -0
  80. data/test/spec_rack_mongrel.rb +170 -0
  81. data/test/spec_rack_recursive.rb +77 -0
  82. data/test/spec_rack_request.rb +426 -0
  83. data/test/spec_rack_response.rb +173 -0
  84. data/test/spec_rack_session_cookie.rb +78 -0
  85. data/test/spec_rack_session_memcache.rb +132 -0
  86. data/test/spec_rack_session_pool.rb +84 -0
  87. data/test/spec_rack_showexceptions.rb +21 -0
  88. data/test/spec_rack_showstatus.rb +72 -0
  89. data/test/spec_rack_static.rb +37 -0
  90. data/test/spec_rack_urlmap.rb +175 -0
  91. data/test/spec_rack_utils.rb +174 -0
  92. data/test/spec_rack_webrick.rb +123 -0
  93. data/test/testrequest.rb +45 -0
  94. metadata +177 -0
@@ -0,0 +1,140 @@
1
+ # AUTHOR: blink <blinketje@gmail.com>; blink#ruby-lang@irc.freenode.net
2
+ # bugrep: Andreas Zehnder
3
+
4
+ require 'rack/utils'
5
+ require 'time'
6
+
7
+ module Rack
8
+ module Session
9
+ module Abstract
10
+ # ID sets up a basic framework for implementing an id based sessioning
11
+ # service. Cookies sent to the client for maintaining sessions will only
12
+ # contain an id reference. Only #get_session and #set_session should
13
+ # need to be overwritten.
14
+ #
15
+ # All parameters are optional.
16
+ # * :key determines the name of the cookie, by default it is
17
+ # 'rack.session'
18
+ # * :domain and :path set the related cookie values, by default
19
+ # domain is nil, and the path is '/'.
20
+ # * :expire_after is the number of seconds in which the session
21
+ # cookie will expire. By default it is set not to provide any
22
+ # expiry time.
23
+ class ID
24
+ attr_reader :key
25
+ DEFAULT_OPTIONS = {
26
+ :key => 'rack.session',
27
+ :path => '/',
28
+ :domain => nil,
29
+ :expire_after => nil
30
+ }
31
+
32
+ def initialize(app, options={})
33
+ @default_options = self.class::DEFAULT_OPTIONS.merge(options)
34
+ @key = @default_options[:key]
35
+ @default_context = context app
36
+ end
37
+
38
+ def call(env)
39
+ @default_context.call(env)
40
+ end
41
+
42
+ def context(app)
43
+ Rack::Utils::Context.new self, app do |env|
44
+ load_session env
45
+ response = app.call(env)
46
+ commit_session env, response
47
+ response
48
+ end
49
+ end
50
+
51
+ private
52
+
53
+ # Extracts the session id from provided cookies and passes it and the
54
+ # environment to #get_session. It then sets the resulting session into
55
+ # 'rack.session', and places options and session metadata into
56
+ # 'rack.session.options'.
57
+ def load_session(env)
58
+ sid = (env['HTTP_COOKIE']||'')[/#{@key}=([^,;]+)/,1]
59
+ sid, session = get_session(env, sid)
60
+ unless session.is_a?(Hash)
61
+ puts 'Session: '+sid.inspect+"\n"+session.inspect if $DEBUG
62
+ raise TypeError, 'Session not a Hash'
63
+ end
64
+
65
+ options = @default_options.
66
+ merge({ :id => sid, :by => self, :at => Time.now })
67
+
68
+ env['rack.session'] = session
69
+ env['rack.session.options'] = options
70
+
71
+ return true
72
+ end
73
+
74
+ # Acquires the session from the environment and the session id from
75
+ # the session options and passes them to #set_session. It then
76
+ # proceeds to set a cookie up in the response with the session's id.
77
+ def commit_session(env, response)
78
+ unless response.is_a?(Array)
79
+ puts 'Response: '+response.inspect if $DEBUG
80
+ raise ArgumentError, 'Response is not an array.'
81
+ end
82
+
83
+ options = env['rack.session.options']
84
+ unless options.is_a?(Hash)
85
+ puts 'Options: '+options.inspect if $DEBUG
86
+ raise TypeError, 'Options not a Hash'
87
+ end
88
+
89
+ sid, time, z = options.values_at(:id, :at, :by)
90
+ unless self == z
91
+ warn "#{self} not managing this session."
92
+ return
93
+ end
94
+
95
+ unless env['rack.session'].is_a?(Hash)
96
+ warn 'Session: '+sid.inspect+"\n"+session.inspect if $DEBUG
97
+ raise TypeError, 'Session not a Hash'
98
+ end
99
+
100
+ unless set_session(env, sid)
101
+ warn "Session not saved." if $DEBUG
102
+ warn "#{env['rack.session'].inspect} has been lost."if $DEBUG
103
+ return false
104
+ end
105
+
106
+ cookie = Utils.escape(@key)+'='+Utils.escape(sid)
107
+ cookie<< "; domain=#{options[:domain]}" if options[:domain]
108
+ cookie<< "; path=#{options[:path]}" if options[:path]
109
+ if options[:expire_after]
110
+ expiry = time + options[:expire_after]
111
+ cookie<< "; expires=#{expiry.httpdate}"
112
+ end
113
+
114
+ case a = (h = response[1])['Set-Cookie']
115
+ when Array then a << cookie
116
+ when String then h['Set-Cookie'] = [a, cookie]
117
+ when nil then h['Set-Cookie'] = cookie
118
+ end
119
+
120
+ return true
121
+ end
122
+
123
+ # Should return [session_id, session]. All thread safety and session
124
+ # retrival proceedures should occur here.
125
+ # If nil is provided as the session id, generation of a new valid id
126
+ # should occur within.
127
+ def get_session(env, sid)
128
+ raise '#get_session needs to be implemented.'
129
+ end
130
+
131
+ # All thread safety and session storage proceedures should occur here.
132
+ # Should return true or false dependant on whether or not the session
133
+ # was saved or not.
134
+ def set_session(env, sid)
135
+ raise '#set_session needs to be implemented.'
136
+ end
137
+ end
138
+ end
139
+ end
140
+ end
@@ -0,0 +1,89 @@
1
+ require 'openssl'
2
+
3
+ module Rack
4
+
5
+ module Session
6
+
7
+ # Rack::Session::Cookie provides simple cookie based session management.
8
+ # The session is a Ruby Hash stored as base64 encoded marshalled data
9
+ # set to :key (default: rack.session).
10
+ # When the secret key is set, cookie data is checked for data integrity.
11
+ #
12
+ # Example:
13
+ #
14
+ # use Rack::Session::Cookie, :key => 'rack.session',
15
+ # :domain => 'foo.com',
16
+ # :path => '/',
17
+ # :expire_after => 2592000,
18
+ # :secret => 'change_me'
19
+ #
20
+ # All parameters are optional.
21
+
22
+ class Cookie
23
+
24
+ def initialize(app, options={})
25
+ @app = app
26
+ @key = options[:key] || "rack.session"
27
+ @secret = options[:secret]
28
+ @default_options = {:domain => nil,
29
+ :path => "/",
30
+ :expire_after => nil}.merge(options)
31
+ end
32
+
33
+ def call(env)
34
+ load_session(env)
35
+ status, headers, body = @app.call(env)
36
+ commit_session(env, status, headers, body)
37
+ end
38
+
39
+ private
40
+
41
+ def load_session(env)
42
+ request = Rack::Request.new(env)
43
+ session_data = request.cookies[@key]
44
+
45
+ if @secret && session_data
46
+ session_data, digest = session_data.split("--")
47
+ session_data = nil unless digest == generate_hmac(session_data)
48
+ end
49
+
50
+ begin
51
+ session_data = session_data.unpack("m*").first
52
+ session_data = Marshal.load(session_data)
53
+ env["rack.session"] = session_data
54
+ rescue
55
+ env["rack.session"] = Hash.new
56
+ end
57
+
58
+ env["rack.session.options"] = @default_options.dup
59
+ end
60
+
61
+ def commit_session(env, status, headers, body)
62
+ session_data = Marshal.dump(env["rack.session"])
63
+ session_data = [session_data].pack("m*")
64
+
65
+ if @secret
66
+ session_data = "#{session_data}--#{generate_hmac(session_data)}"
67
+ end
68
+
69
+ if session_data.size > (4096 - @key.size)
70
+ env["rack.errors"].puts("Warning! Rack::Session::Cookie data size exceeds 4K. Content dropped.")
71
+ [status, headers, body]
72
+ else
73
+ options = env["rack.session.options"]
74
+ cookie = Hash.new
75
+ cookie[:value] = session_data
76
+ cookie[:expires] = Time.now + options[:expire_after] unless options[:expire_after].nil?
77
+ response = Rack::Response.new(body, status, headers)
78
+ response.set_cookie(@key, cookie.merge(options))
79
+ response.to_a
80
+ end
81
+ end
82
+
83
+ def generate_hmac(data)
84
+ OpenSSL::HMAC.hexdigest(OpenSSL::Digest::SHA1.new, @secret, data)
85
+ end
86
+
87
+ end
88
+ end
89
+ end
@@ -0,0 +1,97 @@
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
+
29
+ def initialize(app, options={})
30
+ super
31
+ @pool = MemCache.new @default_options[:memcache_server], @default_options
32
+ unless @pool.servers.any?{|s|s.alive?}
33
+ raise "#{self} unable to find server during initialization."
34
+ end
35
+ @mutex = Mutex.new
36
+ end
37
+
38
+ private
39
+
40
+ def get_session(env, sid)
41
+ session = sid && @pool.get(sid)
42
+ unless session and session.is_a?(Hash)
43
+ session = {}
44
+ lc = 0
45
+ @mutex.synchronize do
46
+ begin
47
+ raise RuntimeError, 'Unique id finding looping excessively' if (lc+=1) > 1000
48
+ sid = "%08x" % rand(0xffffffff)
49
+ ret = @pool.add(sid, session)
50
+ end until /^STORED/ =~ ret
51
+ end
52
+ end
53
+ class << session
54
+ @deleted = []
55
+ def delete key
56
+ (@deleted||=[]) << key
57
+ super
58
+ end
59
+ end
60
+ [sid, session]
61
+ rescue MemCache::MemCacheError, Errno::ECONNREFUSED # MemCache server cannot be contacted
62
+ warn "#{self} is unable to find server."
63
+ warn $!.inspect
64
+ return [ nil, {} ]
65
+ end
66
+
67
+ def set_session(env, sid)
68
+ session = env['rack.session']
69
+ options = env['rack.session.options']
70
+ expiry = options[:expire_after] || 0
71
+ o, s = @mutex.synchronize do
72
+ old_session = @pool.get(sid)
73
+ unless old_session.is_a?(Hash)
74
+ warn 'Session not properly initialized.' if $DEBUG
75
+ old_session = {}
76
+ @pool.add sid, old_session, expiry
77
+ end
78
+ session.instance_eval do
79
+ @deleted.each{|k| old_session.delete(k) } if defined? @deleted
80
+ end
81
+ @pool.set sid, old_session.merge(session), expiry
82
+ [old_session, session]
83
+ end
84
+ s.each do |k,v|
85
+ next unless o.has_key?(k) and v != o[k]
86
+ warn "session value assignment collision at #{k.inspect}:"+
87
+ "\n\t#{o[k].inspect}\n\t#{v.inspect}"
88
+ end if $DEBUG and env['rack.multithread']
89
+ return true
90
+ rescue MemCache::MemCacheError, Errno::ECONNREFUSED # MemCache server cannot be contacted
91
+ warn "#{self} is unable to find server."
92
+ warn $!.inspect
93
+ return false
94
+ end
95
+ end
96
+ end
97
+ end
@@ -0,0 +1,73 @@
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
+ # Example:
17
+ # myapp = MyRackApp.new
18
+ # sessioned = Rack::Session::Pool.new(myapp,
19
+ # :key => 'rack.session',
20
+ # :domain => 'foo.com',
21
+ # :path => '/',
22
+ # :expire_after => 2592000
23
+ # )
24
+ # Rack::Handler::WEBrick.run sessioned
25
+
26
+ class Pool < Abstract::ID
27
+ attr_reader :mutex, :pool
28
+ DEFAULT_OPTIONS = Abstract::ID::DEFAULT_OPTIONS.dup
29
+
30
+ def initialize(app, options={})
31
+ super
32
+ @pool = Hash.new
33
+ @mutex = Mutex.new
34
+ end
35
+
36
+ private
37
+
38
+ def get_session(env, sid)
39
+ session = @mutex.synchronize do
40
+ unless sess = @pool[sid] and ((expires = sess[:expire_at]).nil? or expires > Time.now)
41
+ @pool.delete_if{|k,v| expiry = v[:expire_at] and expiry < Time.now }
42
+ begin
43
+ sid = "%08x" % rand(0xffffffff)
44
+ end while @pool.has_key?(sid)
45
+ end
46
+ @pool[sid] ||= {}
47
+ end
48
+ [sid, session]
49
+ end
50
+
51
+ def set_session(env, sid)
52
+ options = env['rack.session.options']
53
+ expiry = options[:expire_after] && options[:at]+options[:expire_after]
54
+ @mutex.synchronize do
55
+ old_session = @pool[sid]
56
+ old_session[:expire_at] = expiry if expiry
57
+ session = old_session.merge(env['rack.session'])
58
+ @pool[sid] = session
59
+ session.each do |k,v|
60
+ next unless old_session.has_key?(k) and v != old_session[k]
61
+ warn "session value assignment collision at #{k}: #{old_session[k]} <- #{v}"
62
+ end if $DEBUG and env['rack.multithread']
63
+ end
64
+ return true
65
+ rescue
66
+ warn "#{self} is unable to find server."
67
+ warn "#{env['rack.session'].inspect} has been lost."
68
+ warn $!.inspect
69
+ return false
70
+ end
71
+ end
72
+ end
73
+ end
@@ -0,0 +1,348 @@
1
+ require 'ostruct'
2
+ require 'erb'
3
+ require 'rack/request'
4
+
5
+ module Rack
6
+ # Rack::ShowExceptions catches all exceptions raised from the app it
7
+ # wraps. It shows a useful backtrace with the sourcefile and
8
+ # clickable context, the whole Rack environment and the request
9
+ # data.
10
+ #
11
+ # Be careful when you use this on public-facing sites as it could
12
+ # reveal information helpful to attackers.
13
+
14
+ class ShowExceptions
15
+ CONTEXT = 7
16
+
17
+ def initialize(app)
18
+ @app = app
19
+ @template = ERB.new(TEMPLATE)
20
+ end
21
+
22
+ def call(env)
23
+ @app.call(env)
24
+ rescue StandardError, LoadError, SyntaxError => e
25
+ backtrace = pretty(env, e)
26
+ [500,
27
+ {"Content-Type" => "text/html",
28
+ "Content-Length" => backtrace.join.size.to_s},
29
+ backtrace]
30
+ end
31
+
32
+ def pretty(env, exception)
33
+ req = Rack::Request.new(env)
34
+ path = (req.script_name + req.path_info).squeeze("/")
35
+
36
+ frames = exception.backtrace.map { |line|
37
+ frame = OpenStruct.new
38
+ if line =~ /(.*?):(\d+)(:in `(.*)')?/
39
+ frame.filename = $1
40
+ frame.lineno = $2.to_i
41
+ frame.function = $4
42
+
43
+ begin
44
+ lineno = frame.lineno-1
45
+ lines = ::File.readlines(frame.filename)
46
+ frame.pre_context_lineno = [lineno-CONTEXT, 0].max
47
+ frame.pre_context = lines[frame.pre_context_lineno...lineno]
48
+ frame.context_line = lines[lineno].chomp
49
+ frame.post_context_lineno = [lineno+CONTEXT, lines.size].min
50
+ frame.post_context = lines[lineno+1..frame.post_context_lineno]
51
+ rescue
52
+ end
53
+
54
+ frame
55
+ else
56
+ nil
57
+ end
58
+ }.compact
59
+
60
+ env["rack.errors"].puts "#{exception.class}: #{exception.message}"
61
+ env["rack.errors"].puts exception.backtrace.map { |l| "\t" + l }
62
+ env["rack.errors"].flush
63
+
64
+ [@template.result(binding)]
65
+ end
66
+
67
+ def h(obj) # :nodoc:
68
+ case obj
69
+ when String
70
+ Utils.escape_html(obj)
71
+ else
72
+ Utils.escape_html(obj.inspect)
73
+ end
74
+ end
75
+
76
+ # :stopdoc:
77
+
78
+ # adapted from Django <djangoproject.com>
79
+ # Copyright (c) 2005, the Lawrence Journal-World
80
+ # Used under the modified BSD license:
81
+ # http://www.xfree86.org/3.3.6/COPYRIGHT2.html#5
82
+ TEMPLATE = <<'HTML'
83
+ <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
84
+ <html lang="en">
85
+ <head>
86
+ <meta http-equiv="content-type" content="text/html; charset=utf-8" />
87
+ <meta name="robots" content="NONE,NOARCHIVE" />
88
+ <title><%=h exception.class %> at <%=h path %></title>
89
+ <style type="text/css">
90
+ html * { padding:0; margin:0; }
91
+ body * { padding:10px 20px; }
92
+ body * * { padding:0; }
93
+ body { font:small sans-serif; }
94
+ body>div { border-bottom:1px solid #ddd; }
95
+ h1 { font-weight:normal; }
96
+ h2 { margin-bottom:.8em; }
97
+ h2 span { font-size:80%; color:#666; font-weight:normal; }
98
+ h3 { margin:1em 0 .5em 0; }
99
+ h4 { margin:0 0 .5em 0; font-weight: normal; }
100
+ table {
101
+ border:1px solid #ccc; border-collapse: collapse; background:white; }
102
+ tbody td, tbody th { vertical-align:top; padding:2px 3px; }
103
+ thead th {
104
+ padding:1px 6px 1px 3px; background:#fefefe; text-align:left;
105
+ font-weight:normal; font-size:11px; border:1px solid #ddd; }
106
+ tbody th { text-align:right; color:#666; padding-right:.5em; }
107
+ table.vars { margin:5px 0 2px 40px; }
108
+ table.vars td, table.req td { font-family:monospace; }
109
+ table td.code { width:100%;}
110
+ table td.code div { overflow:hidden; }
111
+ table.source th { color:#666; }
112
+ table.source td {
113
+ font-family:monospace; white-space:pre; border-bottom:1px solid #eee; }
114
+ ul.traceback { list-style-type:none; }
115
+ ul.traceback li.frame { margin-bottom:1em; }
116
+ div.context { margin: 10px 0; }
117
+ div.context ol {
118
+ padding-left:30px; margin:0 10px; list-style-position: inside; }
119
+ div.context ol li {
120
+ font-family:monospace; white-space:pre; color:#666; cursor:pointer; }
121
+ div.context ol.context-line li { color:black; background-color:#ccc; }
122
+ div.context ol.context-line li span { float: right; }
123
+ div.commands { margin-left: 40px; }
124
+ div.commands a { color:black; text-decoration:none; }
125
+ #summary { background: #ffc; }
126
+ #summary h2 { font-weight: normal; color: #666; }
127
+ #summary ul#quicklinks { list-style-type: none; margin-bottom: 2em; }
128
+ #summary ul#quicklinks li { float: left; padding: 0 1em; }
129
+ #summary ul#quicklinks>li+li { border-left: 1px #666 solid; }
130
+ #explanation { background:#eee; }
131
+ #template, #template-not-exist { background:#f6f6f6; }
132
+ #template-not-exist ul { margin: 0 0 0 20px; }
133
+ #traceback { background:#eee; }
134
+ #requestinfo { background:#f6f6f6; padding-left:120px; }
135
+ #summary table { border:none; background:transparent; }
136
+ #requestinfo h2, #requestinfo h3 { position:relative; margin-left:-100px; }
137
+ #requestinfo h3 { margin-bottom:-1em; }
138
+ .error { background: #ffc; }
139
+ .specific { color:#cc3300; font-weight:bold; }
140
+ </style>
141
+ <script type="text/javascript">
142
+ //<!--
143
+ function getElementsByClassName(oElm, strTagName, strClassName){
144
+ // Written by Jonathan Snook, http://www.snook.ca/jon;
145
+ // Add-ons by Robert Nyman, http://www.robertnyman.com
146
+ var arrElements = (strTagName == "*" && document.all)? document.all :
147
+ oElm.getElementsByTagName(strTagName);
148
+ var arrReturnElements = new Array();
149
+ strClassName = strClassName.replace(/\-/g, "\\-");
150
+ var oRegExp = new RegExp("(^|\\s)" + strClassName + "(\\s|$$)");
151
+ var oElement;
152
+ for(var i=0; i<arrElements.length; i++){
153
+ oElement = arrElements[i];
154
+ if(oRegExp.test(oElement.className)){
155
+ arrReturnElements.push(oElement);
156
+ }
157
+ }
158
+ return (arrReturnElements)
159
+ }
160
+ function hideAll(elems) {
161
+ for (var e = 0; e < elems.length; e++) {
162
+ elems[e].style.display = 'none';
163
+ }
164
+ }
165
+ window.onload = function() {
166
+ hideAll(getElementsByClassName(document, 'table', 'vars'));
167
+ hideAll(getElementsByClassName(document, 'ol', 'pre-context'));
168
+ hideAll(getElementsByClassName(document, 'ol', 'post-context'));
169
+ }
170
+ function toggle() {
171
+ for (var i = 0; i < arguments.length; i++) {
172
+ var e = document.getElementById(arguments[i]);
173
+ if (e) {
174
+ e.style.display = e.style.display == 'none' ? 'block' : 'none';
175
+ }
176
+ }
177
+ return false;
178
+ }
179
+ function varToggle(link, id) {
180
+ toggle('v' + id);
181
+ var s = link.getElementsByTagName('span')[0];
182
+ var uarr = String.fromCharCode(0x25b6);
183
+ var darr = String.fromCharCode(0x25bc);
184
+ s.innerHTML = s.innerHTML == uarr ? darr : uarr;
185
+ return false;
186
+ }
187
+ //-->
188
+ </script>
189
+ </head>
190
+ <body>
191
+
192
+ <div id="summary">
193
+ <h1><%=h exception.class %> at <%=h path %></h1>
194
+ <h2><%=h exception.message %></h2>
195
+ <table><tr>
196
+ <th>Ruby</th>
197
+ <td><code><%=h frames.first.filename %></code>: in <code><%=h frames.first.function %></code>, line <%=h frames.first.lineno %></td>
198
+ </tr><tr>
199
+ <th>Web</th>
200
+ <td><code><%=h req.request_method %> <%=h(req.host + path)%></code></td>
201
+ </tr></table>
202
+
203
+ <h3>Jump to:</h3>
204
+ <ul id="quicklinks">
205
+ <li><a href="#get-info">GET</a></li>
206
+ <li><a href="#post-info">POST</a></li>
207
+ <li><a href="#cookie-info">Cookies</a></li>
208
+ <li><a href="#env-info">ENV</a></li>
209
+ </ul>
210
+ </div>
211
+
212
+ <div id="traceback">
213
+ <h2>Traceback <span>(innermost first)</span></h2>
214
+ <ul class="traceback">
215
+ <% frames.each { |frame| %>
216
+ <li class="frame">
217
+ <code><%=h frame.filename %></code>: in <code><%=h frame.function %></code>
218
+
219
+ <% if frame.context_line %>
220
+ <div class="context" id="c<%=h frame.object_id %>">
221
+ <% if frame.pre_context %>
222
+ <ol start="<%=h frame.pre_context_lineno+1 %>" class="pre-context" id="pre<%=h frame.object_id %>">
223
+ <% frame.pre_context.each { |line| %>
224
+ <li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h line %></li>
225
+ <% } %>
226
+ </ol>
227
+ <% end %>
228
+
229
+ <ol start="<%=h frame.lineno %>" class="context-line">
230
+ <li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h frame.context_line %><span>...</span></li></ol>
231
+
232
+ <% if frame.post_context %>
233
+ <ol start='<%=h frame.lineno+1 %>' class="post-context" id="post<%=h frame.object_id %>">
234
+ <% frame.post_context.each { |line| %>
235
+ <li onclick="toggle('pre<%=h frame.object_id %>', 'post<%=h frame.object_id %>')"><%=h line %></li>
236
+ <% } %>
237
+ </ol>
238
+ <% end %>
239
+ </div>
240
+ <% end %>
241
+ </li>
242
+ <% } %>
243
+ </ul>
244
+ </div>
245
+
246
+ <div id="requestinfo">
247
+ <h2>Request information</h2>
248
+
249
+ <h3 id="get-info">GET</h3>
250
+ <% unless req.GET.empty? %>
251
+ <table class="req">
252
+ <thead>
253
+ <tr>
254
+ <th>Variable</th>
255
+ <th>Value</th>
256
+ </tr>
257
+ </thead>
258
+ <tbody>
259
+ <% req.GET.sort_by { |k, v| k.to_s }.each { |key, val| %>
260
+ <tr>
261
+ <td><%=h key %></td>
262
+ <td class="code"><div><%=h val.inspect %></div></td>
263
+ </tr>
264
+ <% } %>
265
+ </tbody>
266
+ </table>
267
+ <% else %>
268
+ <p>No GET data.</p>
269
+ <% end %>
270
+
271
+ <h3 id="post-info">POST</h3>
272
+ <% unless req.POST.empty? %>
273
+ <table class="req">
274
+ <thead>
275
+ <tr>
276
+ <th>Variable</th>
277
+ <th>Value</th>
278
+ </tr>
279
+ </thead>
280
+ <tbody>
281
+ <% req.POST.sort_by { |k, v| k.to_s }.each { |key, val| %>
282
+ <tr>
283
+ <td><%=h key %></td>
284
+ <td class="code"><div><%=h val.inspect %></div></td>
285
+ </tr>
286
+ <% } %>
287
+ </tbody>
288
+ </table>
289
+ <% else %>
290
+ <p>No POST data.</p>
291
+ <% end %>
292
+
293
+
294
+ <h3 id="cookie-info">COOKIES</h3>
295
+ <% unless req.cookies.empty? %>
296
+ <table class="req">
297
+ <thead>
298
+ <tr>
299
+ <th>Variable</th>
300
+ <th>Value</th>
301
+ </tr>
302
+ </thead>
303
+ <tbody>
304
+ <% req.cookies.each { |key, val| %>
305
+ <tr>
306
+ <td><%=h key %></td>
307
+ <td class="code"><div><%=h val.inspect %></div></td>
308
+ </tr>
309
+ <% } %>
310
+ </tbody>
311
+ </table>
312
+ <% else %>
313
+ <p>No cookie data.</p>
314
+ <% end %>
315
+
316
+ <h3 id="env-info">Rack ENV</h3>
317
+ <table class="req">
318
+ <thead>
319
+ <tr>
320
+ <th>Variable</th>
321
+ <th>Value</th>
322
+ </tr>
323
+ </thead>
324
+ <tbody>
325
+ <% env.sort_by { |k, v| k.to_s }.each { |key, val| %>
326
+ <tr>
327
+ <td><%=h key %></td>
328
+ <td class="code"><div><%=h val %></div></td>
329
+ </tr>
330
+ <% } %>
331
+ </tbody>
332
+ </table>
333
+
334
+ </div>
335
+
336
+ <div id="explanation">
337
+ <p>
338
+ You're seeing this error because you use <code>Rack::ShowException</code>.
339
+ </p>
340
+ </div>
341
+
342
+ </body>
343
+ </html>
344
+ HTML
345
+
346
+ # :startdoc:
347
+ end
348
+ end