2012-11-06 04:49:57 +04:00
|
|
|
# coding: US-ASCII
|
2015-12-16 08:07:31 +03:00
|
|
|
# frozen_string_literal: false
|
2010-02-02 16:58:56 +03:00
|
|
|
require_relative "utils"
|
2010-01-18 10:50:08 +03:00
|
|
|
require "webrick"
|
2005-01-10 09:29:58 +03:00
|
|
|
require "test/unit"
|
|
|
|
|
|
|
|
class TestWEBrickCGI < Test::Unit::TestCase
|
2008-08-17 21:33:13 +04:00
|
|
|
CRLF = "\r\n"
|
|
|
|
|
2016-01-05 09:09:17 +03:00
|
|
|
def teardown
|
|
|
|
WEBrick::Utils::TimeoutHandler.terminate
|
|
|
|
super
|
|
|
|
end
|
|
|
|
|
2014-11-09 17:01:20 +03:00
|
|
|
def start_cgi_server(log_tester=TestWEBrick::DefaultLogTester, &block)
|
2005-01-10 09:29:58 +03:00
|
|
|
config = {
|
2008-05-18 17:33:24 +04:00
|
|
|
:CGIInterpreter => TestWEBrick::RubyBin,
|
2005-01-10 09:29:58 +03:00
|
|
|
:DocumentRoot => File.dirname(__FILE__),
|
2005-05-27 21:16:06 +04:00
|
|
|
:DirectoryIndex => ["webrick.cgi"],
|
2011-01-12 03:37:12 +03:00
|
|
|
:RequestCallback => Proc.new{|req, res|
|
2006-02-22 10:27:20 +03:00
|
|
|
def req.meta_vars
|
|
|
|
meta = super
|
|
|
|
meta["RUBYLIB"] = $:.join(File::PATH_SEPARATOR)
|
2011-11-16 07:06:05 +04:00
|
|
|
meta[RbConfig::CONFIG['LIBPATHENV']] = ENV[RbConfig::CONFIG['LIBPATHENV']] if RbConfig::CONFIG['LIBPATHENV']
|
2006-02-22 10:27:20 +03:00
|
|
|
return meta
|
|
|
|
end
|
|
|
|
},
|
2005-01-10 09:29:58 +03:00
|
|
|
}
|
2013-08-07 11:39:23 +04:00
|
|
|
if RUBY_PLATFORM =~ /mswin|mingw|cygwin|bccwin32/
|
2005-09-20 12:35:52 +04:00
|
|
|
config[:CGIPathEnv] = ENV['PATH'] # runtime dll may not be in system dir.
|
|
|
|
end
|
2014-11-09 17:01:20 +03:00
|
|
|
TestWEBrick.start_httpserver(config, log_tester){|server, addr, port, log|
|
2008-10-29 14:48:35 +03:00
|
|
|
block.call(server, addr, port, log)
|
2008-08-17 21:33:13 +04:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_cgi
|
2008-10-29 14:48:35 +03:00
|
|
|
start_cgi_server{|server, addr, port, log|
|
2005-01-10 09:29:58 +03:00
|
|
|
http = Net::HTTP.new(addr, port)
|
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi")
|
2008-10-29 14:48:35 +03:00
|
|
|
http.request(req){|res| assert_equal("/webrick.cgi", res.body, log.call)}
|
2005-01-10 09:29:58 +03:00
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi/path/info")
|
2008-10-29 14:48:35 +03:00
|
|
|
http.request(req){|res| assert_equal("/path/info", res.body, log.call)}
|
2005-01-10 09:29:58 +03:00
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi/%3F%3F%3F?foo=bar")
|
2008-10-29 14:48:35 +03:00
|
|
|
http.request(req){|res| assert_equal("/???", res.body, log.call)}
|
2015-12-06 12:28:02 +03:00
|
|
|
unless RUBY_PLATFORM =~ /mswin|mingw|cygwin|bccwin32/
|
|
|
|
# Path info of res.body is passed via ENV.
|
|
|
|
# ENV[] returns different value on Windows depending on locale.
|
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi/%A4%DB%A4%B2/%A4%DB%A4%B2")
|
2012-10-12 17:30:11 +04:00
|
|
|
http.request(req){|res|
|
|
|
|
assert_equal("/\xA4\xDB\xA4\xB2/\xA4\xDB\xA4\xB2", res.body, log.call)}
|
|
|
|
end
|
2005-01-10 09:29:58 +03:00
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi?a=1;a=2;b=x")
|
2008-10-29 14:48:35 +03:00
|
|
|
http.request(req){|res| assert_equal("a=1, a=2, b=x", res.body, log.call)}
|
2005-01-10 09:29:58 +03:00
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi?a=1&a=2&b=x")
|
2008-10-29 14:48:35 +03:00
|
|
|
http.request(req){|res| assert_equal("a=1, a=2, b=x", res.body, log.call)}
|
2005-01-10 09:29:58 +03:00
|
|
|
|
|
|
|
req = Net::HTTP::Post.new("/webrick.cgi?a=x;a=y;b=1")
|
|
|
|
req["Content-Type"] = "application/x-www-form-urlencoded"
|
|
|
|
http.request(req, "a=1;a=2;b=x"){|res|
|
2008-10-29 14:48:35 +03:00
|
|
|
assert_equal("a=1, a=2, b=x", res.body, log.call)}
|
2005-01-10 09:29:58 +03:00
|
|
|
req = Net::HTTP::Post.new("/webrick.cgi?a=x&a=y&b=1")
|
|
|
|
req["Content-Type"] = "application/x-www-form-urlencoded"
|
|
|
|
http.request(req, "a=1&a=2&b=x"){|res|
|
2008-10-29 14:48:35 +03:00
|
|
|
assert_equal("a=1, a=2, b=x", res.body, log.call)}
|
2005-05-27 21:16:06 +04:00
|
|
|
req = Net::HTTP::Get.new("/")
|
|
|
|
http.request(req){|res|
|
2006-11-02 03:21:28 +03:00
|
|
|
ary = res.body.lines.to_a
|
2008-10-29 14:48:35 +03:00
|
|
|
assert_match(%r{/$}, ary[0], log.call)
|
|
|
|
assert_match(%r{/webrick.cgi$}, ary[1], log.call)
|
2005-05-27 21:16:06 +04:00
|
|
|
}
|
2005-10-30 23:40:05 +03:00
|
|
|
|
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi")
|
|
|
|
req["Cookie"] = "CUSTOMER=WILE_E_COYOTE; PART_NUMBER=ROCKET_LAUNCHER_0001"
|
|
|
|
http.request(req){|res|
|
|
|
|
assert_equal(
|
|
|
|
"CUSTOMER=WILE_E_COYOTE\nPART_NUMBER=ROCKET_LAUNCHER_0001\n",
|
2008-10-29 14:48:35 +03:00
|
|
|
res.body, log.call)
|
2005-10-30 23:40:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi")
|
|
|
|
cookie = %{$Version="1"; }
|
|
|
|
cookie << %{Customer="WILE_E_COYOTE"; $Path="/acme"; }
|
|
|
|
cookie << %{Part_Number="Rocket_Launcher_0001"; $Path="/acme"; }
|
|
|
|
cookie << %{Shipping="FedEx"; $Path="/acme"}
|
|
|
|
req["Cookie"] = cookie
|
|
|
|
http.request(req){|res|
|
|
|
|
assert_equal("Customer=WILE_E_COYOTE, Shipping=FedEx",
|
2008-10-29 14:48:35 +03:00
|
|
|
res["Set-Cookie"], log.call)
|
2005-10-30 23:40:05 +03:00
|
|
|
assert_equal("Customer=WILE_E_COYOTE\n" +
|
|
|
|
"Part_Number=Rocket_Launcher_0001\n" +
|
2008-10-29 14:48:35 +03:00
|
|
|
"Shipping=FedEx\n", res.body, log.call)
|
2005-10-30 23:40:05 +03:00
|
|
|
}
|
2005-01-10 09:29:58 +03:00
|
|
|
}
|
|
|
|
end
|
2008-08-17 21:33:13 +04:00
|
|
|
|
|
|
|
def test_bad_request
|
2014-11-09 17:01:20 +03:00
|
|
|
log_tester = lambda {|log, access_log|
|
|
|
|
assert_match(/BadRequest/, log.join)
|
|
|
|
}
|
|
|
|
start_cgi_server(log_tester) {|server, addr, port, log|
|
2008-08-17 21:33:13 +04:00
|
|
|
sock = TCPSocket.new(addr, port)
|
|
|
|
begin
|
|
|
|
sock << "POST /webrick.cgi HTTP/1.0" << CRLF
|
|
|
|
sock << "Content-Type: application/x-www-form-urlencoded" << CRLF
|
|
|
|
sock << "Content-Length: 1024" << CRLF
|
|
|
|
sock << CRLF
|
|
|
|
sock << "a=1&a=2&b=x"
|
|
|
|
sock.close_write
|
2008-10-29 14:48:35 +03:00
|
|
|
assert_match(%r{\AHTTP/\d.\d 400 Bad Request}, sock.read, log.call)
|
2008-08-17 21:33:13 +04:00
|
|
|
ensure
|
|
|
|
sock.close
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
2010-01-10 12:33:47 +03:00
|
|
|
|
2016-07-22 19:43:12 +03:00
|
|
|
def test_cgi_env
|
|
|
|
start_cgi_server do |server, addr, port, log|
|
|
|
|
http = Net::HTTP.new(addr, port)
|
|
|
|
req = Net::HTTP::Get.new("/webrick.cgi/dumpenv")
|
|
|
|
req['proxy'] = 'http://example.com/'
|
|
|
|
req['hello'] = 'world'
|
|
|
|
http.request(req) do |res|
|
|
|
|
env = Marshal.load(res.body)
|
|
|
|
assert_equal 'world', env['HTTP_HELLO']
|
|
|
|
assert_not_operator env, :include?, 'HTTP_PROXY'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-01-10 12:33:47 +03:00
|
|
|
CtrlSeq = [0x7f, *(1..31)].pack("C*").gsub(/\s+/, '')
|
|
|
|
CtrlPat = /#{Regexp.quote(CtrlSeq)}/o
|
|
|
|
DumpPat = /#{Regexp.quote(CtrlSeq.dump[1...-1])}/o
|
|
|
|
|
|
|
|
def test_bad_uri
|
2014-11-09 17:01:20 +03:00
|
|
|
log_tester = lambda {|log, access_log|
|
|
|
|
assert_equal(1, log.length)
|
|
|
|
assert_match(/ERROR bad URI/, log[0])
|
|
|
|
}
|
|
|
|
start_cgi_server(log_tester) {|server, addr, port, log|
|
2010-01-10 12:33:47 +03:00
|
|
|
res = TCPSocket.open(addr, port) {|sock|
|
|
|
|
sock << "GET /#{CtrlSeq}#{CRLF}#{CRLF}"
|
|
|
|
sock.close_write
|
|
|
|
sock.read
|
|
|
|
}
|
|
|
|
assert_match(%r{\AHTTP/\d.\d 400 Bad Request}, res)
|
|
|
|
s = log.call.each_line.grep(/ERROR bad URI/)[0]
|
|
|
|
assert_match(DumpPat, s)
|
|
|
|
assert_not_match(CtrlPat, s)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_bad_header
|
2014-11-09 17:01:20 +03:00
|
|
|
log_tester = lambda {|log, access_log|
|
|
|
|
assert_equal(1, log.length)
|
|
|
|
assert_match(/ERROR bad header/, log[0])
|
|
|
|
}
|
|
|
|
start_cgi_server(log_tester) {|server, addr, port, log|
|
2010-01-10 12:33:47 +03:00
|
|
|
res = TCPSocket.open(addr, port) {|sock|
|
|
|
|
sock << "GET / HTTP/1.0#{CRLF}#{CtrlSeq}#{CRLF}#{CRLF}"
|
|
|
|
sock.close_write
|
|
|
|
sock.read
|
|
|
|
}
|
|
|
|
assert_match(%r{\AHTTP/\d.\d 400 Bad Request}, res)
|
|
|
|
s = log.call.each_line.grep(/ERROR bad header/)[0]
|
|
|
|
assert_match(DumpPat, s)
|
|
|
|
assert_not_match(CtrlPat, s)
|
|
|
|
}
|
|
|
|
end
|
2005-01-10 09:29:58 +03:00
|
|
|
end
|