2005-01-02 17:31:13 -05:00
|
|
|
require "test/unit"
|
|
|
|
require "net/http"
|
|
|
|
require "tempfile"
|
|
|
|
require "webrick"
|
|
|
|
require "webrick/httpauth/basicauth"
|
2010-02-02 08:58:56 -05:00
|
|
|
require_relative "utils"
|
2005-01-02 17:31:13 -05:00
|
|
|
|
|
|
|
class TestWEBrickHTTPAuth < Test::Unit::TestCase
|
|
|
|
def test_basic_auth
|
2008-10-29 07:48:35 -04:00
|
|
|
TestWEBrick.start_httpserver{|server, addr, port, log|
|
2005-01-02 17:31:13 -05:00
|
|
|
realm = "WEBrick's realm"
|
|
|
|
path = "/basic_auth"
|
|
|
|
|
|
|
|
server.mount_proc(path){|req, res|
|
|
|
|
WEBrick::HTTPAuth.basic_auth(req, res, realm){|user, pass|
|
|
|
|
user == "webrick" && pass == "supersecretpassword"
|
2009-03-05 22:56:38 -05:00
|
|
|
}
|
2005-01-02 17:31:13 -05:00
|
|
|
res.body = "hoge"
|
|
|
|
}
|
|
|
|
http = Net::HTTP.new(addr, port)
|
|
|
|
g = Net::HTTP::Get.new(path)
|
|
|
|
g.basic_auth("webrick", "supersecretpassword")
|
2008-10-29 07:48:35 -04:00
|
|
|
http.request(g){|res| assert_equal("hoge", res.body, log.call)}
|
2005-01-02 17:31:13 -05:00
|
|
|
g.basic_auth("webrick", "not super")
|
2008-10-29 07:48:35 -04:00
|
|
|
http.request(g){|res| assert_not_equal("hoge", res.body, log.call)}
|
2005-01-02 17:31:13 -05:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_basic_auth2
|
2008-10-29 07:48:35 -04:00
|
|
|
TestWEBrick.start_httpserver{|server, addr, port, log|
|
2005-01-02 17:31:13 -05:00
|
|
|
realm = "WEBrick's realm"
|
|
|
|
path = "/basic_auth2"
|
|
|
|
|
|
|
|
tmpfile = Tempfile.new("test_webrick_auth")
|
|
|
|
tmpfile.close
|
|
|
|
tmp_pass = WEBrick::HTTPAuth::Htpasswd.new(tmpfile.path)
|
|
|
|
tmp_pass.set_passwd(realm, "webrick", "supersecretpassword")
|
|
|
|
tmp_pass.set_passwd(realm, "foo", "supersecretpassword")
|
|
|
|
tmp_pass.flush
|
|
|
|
|
|
|
|
htpasswd = WEBrick::HTTPAuth::Htpasswd.new(tmpfile.path)
|
|
|
|
users = []
|
|
|
|
htpasswd.each{|user, pass| users << user }
|
2008-10-29 07:48:35 -04:00
|
|
|
assert_equal(2, users.size, log.call)
|
|
|
|
assert(users.member?("webrick"), log.call)
|
|
|
|
assert(users.member?("foo"), log.call)
|
2005-01-02 17:31:13 -05:00
|
|
|
|
|
|
|
server.mount_proc(path){|req, res|
|
|
|
|
auth = WEBrick::HTTPAuth::BasicAuth.new(
|
|
|
|
:Realm => realm, :UserDB => htpasswd,
|
|
|
|
:Logger => server.logger
|
|
|
|
)
|
|
|
|
auth.authenticate(req, res)
|
|
|
|
res.body = "hoge"
|
|
|
|
}
|
|
|
|
http = Net::HTTP.new(addr, port)
|
|
|
|
g = Net::HTTP::Get.new(path)
|
|
|
|
g.basic_auth("webrick", "supersecretpassword")
|
2008-10-29 07:48:35 -04:00
|
|
|
http.request(g){|res| assert_equal("hoge", res.body, log.call)}
|
2005-01-02 17:31:13 -05:00
|
|
|
g.basic_auth("webrick", "not super")
|
2008-10-29 07:48:35 -04:00
|
|
|
http.request(g){|res| assert_not_equal("hoge", res.body, log.call)}
|
2013-03-23 22:06:32 -04:00
|
|
|
tmpfile.close(true)
|
2005-01-02 17:31:13 -05:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_basic_auth3
|
|
|
|
tmpfile = Tempfile.new("test_webrick_auth")
|
|
|
|
tmpfile.puts("webrick:{SHA}GJYFRpBbdchp595jlh3Bhfmgp8k=")
|
|
|
|
tmpfile.flush
|
2008-09-24 13:44:39 -04:00
|
|
|
assert_raise(NotImplementedError){
|
2005-01-02 17:31:13 -05:00
|
|
|
WEBrick::HTTPAuth::Htpasswd.new(tmpfile.path)
|
|
|
|
}
|
|
|
|
tmpfile.close(true)
|
|
|
|
|
|
|
|
tmpfile = Tempfile.new("test_webrick_auth")
|
|
|
|
tmpfile.puts("webrick:$apr1$IOVMD/..$rmnOSPXr0.wwrLPZHBQZy0")
|
|
|
|
tmpfile.flush
|
2008-09-24 13:44:39 -04:00
|
|
|
assert_raise(NotImplementedError){
|
2005-01-02 17:31:13 -05:00
|
|
|
WEBrick::HTTPAuth::Htpasswd.new(tmpfile.path)
|
|
|
|
}
|
|
|
|
tmpfile.close(true)
|
|
|
|
end
|
2011-07-05 01:06:05 -04:00
|
|
|
|
|
|
|
DIGESTRES_ = /
|
2013-04-02 02:28:13 -04:00
|
|
|
([a-zA-Z\-]+)
|
|
|
|
[ \t]*(?:\r\n[ \t]*)*
|
2011-07-05 01:06:05 -04:00
|
|
|
=
|
2013-04-02 02:28:13 -04:00
|
|
|
[ \t]*(?:\r\n[ \t]*)*
|
2011-07-05 01:06:05 -04:00
|
|
|
(?:
|
|
|
|
"((?:[^"]+|\\[\x00-\x7F])*)" |
|
|
|
|
([!\#$%&'*+\-.0-9A-Z^_`a-z|~]+)
|
|
|
|
)/x
|
|
|
|
|
|
|
|
def test_digest_auth
|
|
|
|
TestWEBrick.start_httpserver{|server, addr, port, log|
|
|
|
|
realm = "WEBrick's realm"
|
|
|
|
path = "/digest_auth"
|
|
|
|
|
|
|
|
tmpfile = Tempfile.new("test_webrick_auth")
|
|
|
|
tmpfile.close
|
|
|
|
tmp_pass = WEBrick::HTTPAuth::Htdigest.new(tmpfile.path)
|
|
|
|
tmp_pass.set_passwd(realm, "webrick", "supersecretpassword")
|
|
|
|
tmp_pass.set_passwd(realm, "foo", "supersecretpassword")
|
|
|
|
tmp_pass.flush
|
|
|
|
|
|
|
|
htdigest = WEBrick::HTTPAuth::Htdigest.new(tmpfile.path)
|
|
|
|
users = []
|
|
|
|
htdigest.each{|user, pass| users << user }
|
|
|
|
assert_equal(2, users.size, log.call)
|
|
|
|
assert(users.member?("webrick"), log.call)
|
|
|
|
assert(users.member?("foo"), log.call)
|
|
|
|
|
|
|
|
auth = WEBrick::HTTPAuth::DigestAuth.new(
|
|
|
|
:Realm => realm, :UserDB => htdigest,
|
|
|
|
:Algorithm => 'MD5',
|
|
|
|
:Logger => server.logger
|
|
|
|
)
|
|
|
|
server.mount_proc(path){|req, res|
|
|
|
|
auth.authenticate(req, res)
|
|
|
|
res.body = "hoge"
|
|
|
|
}
|
|
|
|
|
|
|
|
Net::HTTP.start(addr, port) do |http|
|
|
|
|
g = Net::HTTP::Get.new(path)
|
|
|
|
params = {}
|
|
|
|
http.request(g) do |res|
|
|
|
|
assert_equal('401', res.code, log.call)
|
|
|
|
res["www-authenticate"].scan(DIGESTRES_) do |key, quoted, token|
|
|
|
|
params[key.downcase] = token || quoted.delete('\\')
|
|
|
|
end
|
|
|
|
params['uri'] = "http://#{addr}:#{port}#{path}"
|
|
|
|
end
|
|
|
|
|
|
|
|
g['Authorization'] = credentials_for_request('webrick', "supersecretpassword", params)
|
|
|
|
http.request(g){|res| assert_equal("hoge", res.body, log.call)}
|
|
|
|
|
|
|
|
params['algorithm'].downcase! #4936
|
|
|
|
g['Authorization'] = credentials_for_request('webrick', "supersecretpassword", params)
|
|
|
|
http.request(g){|res| assert_equal("hoge", res.body, log.call)}
|
|
|
|
|
|
|
|
g['Authorization'] = credentials_for_request('webrick', "not super", params)
|
|
|
|
http.request(g){|res| assert_not_equal("hoge", res.body, log.call)}
|
|
|
|
end
|
2013-03-23 22:06:32 -04:00
|
|
|
tmpfile.close(true)
|
2011-07-05 01:06:05 -04:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
def credentials_for_request(user, password, params)
|
|
|
|
cnonce = "hoge"
|
|
|
|
nonce_count = 1
|
|
|
|
ha1 = "#{user}:#{params['realm']}:#{password}"
|
|
|
|
ha2 = "GET:#{params['uri']}"
|
|
|
|
request_digest =
|
|
|
|
"#{Digest::MD5.hexdigest(ha1)}:" \
|
|
|
|
"#{params['nonce']}:#{'%08x' % nonce_count}:#{cnonce}:#{params['qop']}:" \
|
|
|
|
"#{Digest::MD5.hexdigest(ha2)}"
|
2011-07-05 06:09:04 -04:00
|
|
|
"Digest username=\"#{user}\"" \
|
2011-07-05 01:06:05 -04:00
|
|
|
", realm=\"#{params['realm']}\"" \
|
|
|
|
", nonce=\"#{params['nonce']}\"" \
|
|
|
|
", uri=\"#{params['uri']}\"" \
|
|
|
|
", qop=#{params['qop']}" \
|
|
|
|
", nc=#{'%08x' % nonce_count}" \
|
|
|
|
", cnonce=\"#{cnonce}\"" \
|
|
|
|
", response=\"#{Digest::MD5.hexdigest(request_digest)}\"" \
|
|
|
|
", opaque=\"#{params['opaque']}\"" \
|
|
|
|
", algorithm=#{params['algorithm']}"
|
|
|
|
end
|
2005-01-02 17:31:13 -05:00
|
|
|
end
|