简介
本文的第二部分主要是介绍如何使用asp.net实现我们自己的表格认证的方法。在第一部分中,我们讨论了表格认证的基本概念和原理。在阅读本文的第二部分之前,读者需要了解表格认证的一些基本概念,或者已经阅读过第一部分。
自定义表格认证的建立
使用的页面:default.aspx、login.aspx、web.config、users.xml、hashpassword.aspx
在这个自定义表格认证的例子中,我们将自始至终地使用一个xml文档存储用户名和口令。建立该自定义表格认证所需要的一些准备工作:
- 在互联网服务器的根目录下建立名字为customforms的目录。
- 使该文件夹成为互联网服务管理器中的一个应用。
- 创建名字为unsecure的子目录。
- 创建名字为hashpassword.aspx的文件,并将它移到unsecure目录。
web.config概览
web.config文件中包含了web应用程序的所有可配置的设置选项。我加亮显示了需要认真研究的代码:
web.config代码
<configuration>
<system.web>
<customerrors mode="off"/>
<authentication mode="forms">
<forms name="authcookie" path="/" loginurl="login.aspx" protection="all" timeout="10">
</forms>
</authentication>
<authorization>
<deny users="?" />
</authorization>
</system.web>
这部分加亮显示
<location path="unsecure">
<system.web>
<authorization>
<allow users="*"/>
</authorization>
</system.web>
</location>
这部分加亮显示
</configuration>
web.config详解
这个例子增加了一个名字为location的配置小节,它允许我们覆盖web.config文件中system.web配置小节的设置。在本例中,我们希望允许匿名或没有通过认证的用户访问unsecure目录,常见的例子是整个web应用都是安全的,只有注册页是个例外。通过允许匿名用户访问一个目录,我们可以将能够被任何人浏览的文件存储到该目录中。如果有必要,我们可以创建多个location小节。
users.xml概览
在这个文件中,我们存储了所有认证需要的数据,例如用户名和口令。口令采用了sha1算法进行加密,稍后我们会对sha1算法进行解释。
users.xml代码
<?xml version="1.0"?>
<users>
<jeff>a94a8fe5ccb19ba61c4c0873d391e987982fbbd3</jeff>
<mike>a94a8fe5ccb19ba61c4c0873d391e987982fbbd3</mike>
</users>
users.xml详解
在该文件中,有一个被称作users的小节,其中包含有每个用户的个人节点,在节点的开始、结束标志之间,有一个经过哈希处理的口令。很明显的是,该文件中也可以包含更多的信息,例如姓、名以及电话号码等。
login.aspx概览
该文件包含对一个用户进行认证所需要的全部逻辑。在本例中,我们将使用一个xml文件对用户进行认证,当然了我们也可以将本页的逻辑用于使用一个数据库对用户进行认证。
login.aspx代码
<%@page language="vb" %>
<%@import namespace="system.web.security" %>
<%@import namespace="system.xml" %>
<script language="vb" runat="server">
sub processlogin(objsender as object, objargs as eventargs)
dim strcurrentpath as string = request.physicalpath
dim strxmldocpath as string = left(strcurrentpath, instrrev(strcurrentpath, "/")) & "users.xml"
dim struser as string = txtuser.text
dim strpassword as string = txtpassword.text
dim strencpassword as string = gethashedpass(strpassword)
dim blnisauthenticated as boolean
dim objxmldoc as new xmldocument()
try
objxmldoc.load(strxmldocpath)
catch objerror as exception
errormessage.innerhtml = "<b> the xml document could not be loaded.</b>.<br>" & _
objerror.message & "<br />" & objerror.source
exit sub
end try
dim usernodes as xmlnodelist
usernodes = objxmldoc.getelementsbytagname(struser)
是否有用户名与输入的用户名相同的元素
if not usernodes is nothing then
dim blnuserexists as boolean = true
dim strusercheck as string
try
strusercheck = usernodes(0).firstchild().value
catch objerror as exception
errormessage.innerhtml = "<b>invalid username</b> please re-enter..."
blnuserexists = false
end try
if blnuserexists = true then
if strencpassword = usernodes(0).firstchild().value then
blnisauthenticated = true
else
errormessage.innerhtml = "<b>invalid password</b> please re-enter..."
end if
end if
end if
if blnisauthenticated then
formsauthentication.redirectfromloginpage(struser, chkpersistlogin.checked)
end if
end sub
function gethashedpass(byval apassword as string) as string
return formsauthentication.hashpasswordforstoringinconfigfile(apassword,"sha1")
end function
</script>
<html>
<head>
<title>custom forms authentication login form</title>
</head>
<body bgcolor="#ffffff" text="#000000">
<form runat="server">
<table width="400" border="0" cellspacing="0" cellpadding="0">
<tr>
<td width="80">username : </td>
<td width="10"> </td>
<td><asp:textbox id="txtuser" runat="server"/></td>
</tr>
<tr>
<td>password : </td>
<td width="10"> </td>
<td><asp:textbox id="txtpassword" textmode="password" runat="server"/></td>
</tr>
<tr>
<tr>
<td></td>
<td width="10"> </td>
<td><asp:checkbox id="chkpersistlogin" runat="server" />remember my credentials
<br>
</td>
</tr>
<tr>
<td> </td>
<td width="10"> </td>
<td><asp:button id="cmdlogin" onclick="processlogin" text="login" runat="server" /></td>
</tr>
</table>
<br>
<br>
<div id="errormessage" runat="server" />
</form>
</body>
</html>
login.aspx详解
在本例中,我添加了对system.web.security和system.xml的引用,因为我们将会用到这二个名字空间中的类。我们还编写了一个名字为processlogin的过程,它的作用是检查表格数据(用户名和口令)与xml文件中所包含的口令和用户名是否相同。
首先,我们为文本编辑框创建了一些局部变量。由于需要得到users.xml文件的全路径,因此我们使用了
request.physicalpath方法,然后对脚本文件名进行整理。另外我们还创建了一个变量,保存经过哈希处理后的口令。
其次,我们将xmldoc.load方法调用放在try...catch语句中。try...catch语句是asp.net中新增添的,它是处理错误和异常的一种很好的方式。在下面的代码,我们将用户节点表作为一个变量数组,使用getelementsbytagname方法将xml文件中的用户节点赋给它。然后检查该用户是否存在,如果存在,则检查用户输入的口令是否与存储在xml文件中的相同。如果用户存在,而且口令也相同,我们就将blnisauthenticated的值勤设置为true。在过程的末尾,如果blnisauthenticated的值为true,我们就调用redirectfromloginpage方法。当然了,我们也可以使用setauthcookie方法来完成同样的功能,但不会把用户引导到另一个网页。
在login.aspx文件的接口或html部分,我们有开发了2个服务器端正文框,1个服务器端复选框,1个按钮,在该按钮的onclick事件中,调用了processlogin。我们还有一个在服务器端运行的div,它能够向用户显示出错信息。
default.aspx概览
该文件中的代码与本篇文章第一部分的default.aspx文件相同。
default.aspx的代码
<%@page language="vb" %>
<%@import namespace="system.web.security" %>
<script language="vb" runat="server">
sub signout(objsender as object, objargs as eventargs)
删除用户认证的cookie并退出
formsauthentication.signout()
将用户引导到提交的网页
response.redirect(request.urlreferrer.tostring())
end sub
sub page_load()
验证认证
if user.identity.isauthenticated then
显示认证信息
displaycredentials.innerhtml = "current user : <b>" & user.identity.name & _
"</b><br><br>authentication used : <b>" & _
user.identity.authenticationtype & "</b>"
else
显示错误信息
displaycredentials.innerhtml = "sorry, you have not been authenticated."
cmdsignout.disabled = true
end if
end sub
</script>
<html>
<head>
<title>forms authentication</title>
</head>
<body bgcolor="#ffffff" text="#000000">
<span class="header">forms based authentication using custom method</span>
<br>
<br>
<div id="displaycredentials" runat="server" />
<br>
<br>
<form runat="server">
<input id="cmdsignout" type="submit" value="sign out" runat="server" onserverclick="signout" /><p />
</form>
</body>
</html>
default.aspx详解
该网页与本文第一部分中的default.aspx的功能完全相同,它只是简单地显示用户名和使用的认证方法。
hashpassword.aspx概览
这个网页允许一个未经认证的用户创建加密的口令,它可以用于在web.config的credentials小节、xml文件或数据库中存储口令。
<ccid_nobr><b>hashpassword.aspx代码</b></ccid_nobr>
<%@page language="vb" %>
<%@import namespace="system.web.security" %>
<script language="vb" runat="server">
sub gethashedpass(objsender as object, objargs as eventargs)
dim strencpass as string
strencpass = formsauthentication.hashpasswordforstoringinconfigfile(txtpassword.value,"sha1")
hashedpass.innerhtml = "hashed password for web.config, xml file or database<br><b>" & _
strencpass & "</b>"
end sub
</script>
<html>
<head>
<title>create hashed password</title>
</head>
<body bgcolor="#ffffff" text="#000000">
<b>create hashed password</b>
<form runat="server">
<table width="100%" border="0" cellspacing="0" cellpadding="0">
<tr>
<td>password to encrypt:
<input id="txtpassword" type="password" runat="server" name="text"/>
<input type="submit" value="hash pass" runat="server" onserverclick="gethashedpass"/>
</td>
</tr>
<tr>
<tr>
<td> </td>
</tr>
<tr>
<td>
<div id="hashedpass" runat="server"/>
</td>
</tr>
</table>
</form>
</body>
</html>
hashpassword.aspx详解
为了使用forms authentication名字空间,我们需要再次使用system.web.security名字空间。这里,我们使用一个过程接收文本框的文本,并使用sha1哈希算法对它进行哈希处理。完成这一功能的方法的名字是hashpasswordforstoringinconfigfile(这很可能是我看到的最长的方法名字了),该方法接收二个参数,一个是需要进行哈希处理的字符串,另一个是要使用的算法,在该方法中,我们可以使用sha1或md5算法。
结束语
就象在上面的稿子中看到的那样,在web应用程序的开发中,表格认证是一个功能强大的工具。如果能够掌握好它的使用,会给我们的开发工作带来很大的方便。
注册会员,创建你的web开发资料库,