qoobaa-rack 1.0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (116) hide show
  1. data/COPYING +18 -0
  2. data/KNOWN-ISSUES +18 -0
  3. data/RDOX +0 -0
  4. data/README +353 -0
  5. data/Rakefile +164 -0
  6. data/SPEC +164 -0
  7. data/bin/rackup +176 -0
  8. data/contrib/rack_logo.svg +111 -0
  9. data/example/lobster.ru +4 -0
  10. data/example/protectedlobster.rb +14 -0
  11. data/example/protectedlobster.ru +8 -0
  12. data/lib/rack/adapter/camping.rb +22 -0
  13. data/lib/rack/auth/abstract/handler.rb +37 -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 +487 -0
  21. data/lib/rack/builder.rb +63 -0
  22. data/lib/rack/cascade.rb +41 -0
  23. data/lib/rack/chunked.rb +49 -0
  24. data/lib/rack/commonlogger.rb +52 -0
  25. data/lib/rack/conditionalget.rb +47 -0
  26. data/lib/rack/content_length.rb +29 -0
  27. data/lib/rack/content_type.rb +23 -0
  28. data/lib/rack/deflater.rb +96 -0
  29. data/lib/rack/directory.rb +153 -0
  30. data/lib/rack/file.rb +88 -0
  31. data/lib/rack/handler/cgi.rb +61 -0
  32. data/lib/rack/handler/evented_mongrel.rb +8 -0
  33. data/lib/rack/handler/fastcgi.rb +88 -0
  34. data/lib/rack/handler/lsws.rb +60 -0
  35. data/lib/rack/handler/mongrel.rb +87 -0
  36. data/lib/rack/handler/scgi.rb +62 -0
  37. data/lib/rack/handler/swiftiplied_mongrel.rb +8 -0
  38. data/lib/rack/handler/thin.rb +18 -0
  39. data/lib/rack/handler/webrick.rb +71 -0
  40. data/lib/rack/handler.rb +69 -0
  41. data/lib/rack/head.rb +19 -0
  42. data/lib/rack/lint.rb +546 -0
  43. data/lib/rack/lobster.rb +65 -0
  44. data/lib/rack/lock.rb +16 -0
  45. data/lib/rack/methodoverride.rb +27 -0
  46. data/lib/rack/mime.rb +204 -0
  47. data/lib/rack/mock.rb +187 -0
  48. data/lib/rack/recursive.rb +57 -0
  49. data/lib/rack/reloader.rb +107 -0
  50. data/lib/rack/request.rb +248 -0
  51. data/lib/rack/response.rb +183 -0
  52. data/lib/rack/rewindable_input.rb +100 -0
  53. data/lib/rack/session/abstract/id.rb +142 -0
  54. data/lib/rack/session/cookie.rb +91 -0
  55. data/lib/rack/session/memcache.rb +109 -0
  56. data/lib/rack/session/pool.rb +100 -0
  57. data/lib/rack/showexceptions.rb +349 -0
  58. data/lib/rack/showstatus.rb +106 -0
  59. data/lib/rack/static.rb +38 -0
  60. data/lib/rack/urlmap.rb +55 -0
  61. data/lib/rack/utils.rb +528 -0
  62. data/lib/rack.rb +90 -0
  63. data/rack.gemspec +60 -0
  64. data/test/cgi/lighttpd.conf +20 -0
  65. data/test/cgi/test +9 -0
  66. data/test/cgi/test.fcgi +8 -0
  67. data/test/cgi/test.ru +7 -0
  68. data/test/multipart/binary +0 -0
  69. data/test/multipart/empty +10 -0
  70. data/test/multipart/file1.txt +1 -0
  71. data/test/multipart/ie +6 -0
  72. data/test/multipart/nested +10 -0
  73. data/test/multipart/none +9 -0
  74. data/test/multipart/text +10 -0
  75. data/test/spec_rack_auth_basic.rb +73 -0
  76. data/test/spec_rack_auth_digest.rb +226 -0
  77. data/test/spec_rack_auth_openid.rb +84 -0
  78. data/test/spec_rack_builder.rb +84 -0
  79. data/test/spec_rack_camping.rb +51 -0
  80. data/test/spec_rack_cascade.rb +48 -0
  81. data/test/spec_rack_cgi.rb +89 -0
  82. data/test/spec_rack_chunked.rb +62 -0
  83. data/test/spec_rack_commonlogger.rb +61 -0
  84. data/test/spec_rack_conditionalget.rb +41 -0
  85. data/test/spec_rack_content_length.rb +43 -0
  86. data/test/spec_rack_content_type.rb +30 -0
  87. data/test/spec_rack_deflater.rb +127 -0
  88. data/test/spec_rack_directory.rb +61 -0
  89. data/test/spec_rack_fastcgi.rb +89 -0
  90. data/test/spec_rack_file.rb +75 -0
  91. data/test/spec_rack_handler.rb +43 -0
  92. data/test/spec_rack_head.rb +30 -0
  93. data/test/spec_rack_lint.rb +521 -0
  94. data/test/spec_rack_lobster.rb +45 -0
  95. data/test/spec_rack_lock.rb +38 -0
  96. data/test/spec_rack_methodoverride.rb +60 -0
  97. data/test/spec_rack_mock.rb +243 -0
  98. data/test/spec_rack_mongrel.rb +189 -0
  99. data/test/spec_rack_recursive.rb +77 -0
  100. data/test/spec_rack_request.rb +504 -0
  101. data/test/spec_rack_response.rb +218 -0
  102. data/test/spec_rack_rewindable_input.rb +118 -0
  103. data/test/spec_rack_session_cookie.rb +82 -0
  104. data/test/spec_rack_session_memcache.rb +250 -0
  105. data/test/spec_rack_session_pool.rb +172 -0
  106. data/test/spec_rack_showexceptions.rb +21 -0
  107. data/test/spec_rack_showstatus.rb +72 -0
  108. data/test/spec_rack_static.rb +37 -0
  109. data/test/spec_rack_thin.rb +91 -0
  110. data/test/spec_rack_urlmap.rb +185 -0
  111. data/test/spec_rack_utils.rb +467 -0
  112. data/test/spec_rack_webrick.rb +130 -0
  113. data/test/testrequest.rb +57 -0
  114. data/test/unregistered_handler/rack/handler/unregistered.rb +7 -0
  115. data/test/unregistered_handler/rack/handler/unregistered_long_one.rb +7 -0
  116. 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