lake2的专栏

千秋邈矣独留我,百战归来再读书

原创 PJBlog安全性分析收藏

新一篇: ASP.Net下的C/S后门——WebAdmin 2.Y应用详解 | 旧一篇: L-Blog的三个漏洞

Authorlake2http://lake2.0x54.org

Date2006-1-1

 

PJBlog2PuterJam开发的一款免费的ASP + Access的个人blog系统,这几天偶想弄个blog来玩玩,经过比较选中了功能、界面都相对较好的PJBlog2。经过试用,感觉这blog还不错,也发现几个安全方面的小问题,就把我的一点点见解发出来。我分析的版本是051211发布的PJBlog2 v2.4.1211版本。

 

 

一、 密码加密算法

PJBlog2没有采用常用的MD5算法对用户密码加密,而是使用的SHA1算法。SHA1算法跟MD5类似,也是单向散列函数,不过它对任意长度的数据进行处理输出160位的数值。

PJBlog2在创建新用户的时候会随机生成一个6位的字符串Salt,用户的明文密码加上这个Salt值再进行hash才得到加密后的密码。即:Password = SHA1(user_pwd & Salt)。这样子做的好处是即使两个一样的密码hash后的结果也完全不同。这个稍微“另类”的算法给破解密码带来了点难度。

呵呵,有难度不等于不能破解,网上没有现成的程序,需要自己动手写个才是。因为.NET提供了SHA1的类所以最初用VB.Net写的,由于狂耗资源的问题一直解决不了,只好换C了。用C实现SHA1的源代码我找了好久才在一个老外的站找到,还不错,呵呵,有兴趣的可以看下老外的SHA1类:http://www.codeproject.com/cpp/csha1.asp

程序可以在这里( http://www.0x54.org/lake2/program/PJBlogCracker.exe )下载,是命令行下的,功能比较弱,单线程速度超级慢,可能还有bug,诸多问题以后再改咯。

 

二、 登陆认证

PJBlog2的认证方式是用的CookiesIP。当用户登陆成功,系统随机产生一个Hashkey写入Cookies并记录到数据库中,然后通过Cookies里的HashkeyUsername还有IP来判断用户。Cookies好办,可以用跨站、下数据库等方式拿,可以这IP就难办了,看来进行Cookies欺骗的可能性很小啊。呵呵,那就不看这个咯。

 

三、 几个地方过滤不严

第一个就是统计访问的referer过滤不严。看:

 

       Guest_Refer=Trim(Request.ServerVariables("HTTP_REFERER"))

         Conn.ExeCute("INSERT INTO blog_Counter(coun_IP,coun_OS,coun_Browser,coun_Referer) VALUES ('"&Guest_IP&"','"&Guest_Browser(1)&"','"&Guest_Browser(0)&"','"&CheckStr(Guest_Refer)&"')")

 

呵呵,只是把referer过滤后用CheckStr检查之,看CheckStr代码:

 

'*************************************

'过滤特殊字符

'*************************************

Function CheckStr(byVal ChkStr)

         Dim Str:Str=ChkStr

         Str=Trim(Str)

         If IsNull(Str) Then

                   CheckStr = ""

                   Exit Function

         End If

    Str = Replace(Str, "&", "&")

    Str = Replace(Str,"'","'")

    Str = Replace(Str,"""",""")

         Dim re

         Set re=new RegExp

         re.IgnoreCase =True

         re.Global=True

         re.Pattern="(w)(here)"

    Str = re.replace(Str,"$1here")

         re.Pattern="(s)(elect)"

    Str = re.replace(Str,"$1elect")

         re.Pattern="(i)(nsert)"

    Str = re.replace(Str,"$1nsert")

         re.Pattern="(c)(reate)"

    Str = re.replace(Str,"$1reate")

         re.Pattern="(d)(rop)"

    Str = re.replace(Str,"$1rop")

         re.Pattern="(a)(lter)"

    Str = re.replace(Str,"$1lter")

         re.Pattern="(d)(elete)"

    Str = re.replace(Str,"$1elete")

         re.Pattern="(u)(pdate)"

    Str = re.replace(Str,"$1pdate")

         re.Pattern="(\s)(or)"

    Str = re.replace(Str,"$1or")

         Set re=Nothing

         CheckStr=Str

End Function

 

过滤了单引号、双引号、连接符等,不过最重要的“<”和“>”却没有过滤。呵呵,跨站脚本攻击又有用武之地了。注意只显示前面40个字符在页面,要好好构造哦。

第二就是游客评论输入用户名也是用的CheckStr过滤,用户名数据库有限制,24个字符,这里构造CSS就更有难度了,不过可以有别的用处,具体的嘛,呵呵,后文详述。

有些blog主人禁止了游客评论,所以就只好注册后再评论了,但是注册后评论的名字文本框被设成了注册名而且只读,怎么办?呵呵,没关系,可以外部提交数据的。

再一个就是blog的留言板插件,还是用户名没过滤好,这个更难利用,只有20个字符哦。

 

四、 数据库的问题

PJBlog2的默认数据库名为pblog.asp,虽然数据库里面有个貌似防下载的blog_Notdownload表,呵呵,访问数据库试试,可以下载的哦。

既然可以下载,当然可以插入asp代码运行咯。有些地方(比如游客评论内容)插入asp代码会被扯开,不知道原因,郁闷。

经测试评论的姓名里是可以插ASP代码的。前面说了这里过滤不严,这样插入数据库的asp不会被显示出来的,看到的只是个没名字的家伙在唧唧歪歪,呵呵。这个可要好好构造哦,24个字符而且"被过滤了,想了一下,正好找到一个最短的,恰好24个字符:<%eval request(chr(9))%>

留言版的内容也可以插入asp代码,不过管理员看留言时看得到的哦。

呵呵,当然大多数站长应该都是改了数据库名称的,但是测试时还是发现少数人不改……

 

五、 上传文件

出于安全的考虑,PJBlog2限制了上传文件类型,包括aspasaaspxcercdxhtr。其实现在很多虚拟主机不仅支持asp,还会支持aspxphpperl的,而且也可以上传shtml等格式,所以如果既然要限制最好是把所有服务器执行文件的类型一起限制才是。

上传文件这里还有点问题,看attachment.asp里的代码:

 

Dim F_File,F_Type

Set F_File=FileUP.File("File")

F_Name=randomStr(1)&Year(now)&Month(now)&Day(now)&Hour(now)&Minute(now)&Second(now)&"."&F_File.FileExt

F_Type=FixName(F_File.FileExt)

IF F_File.FileSize > Int(UP_FileSize) Then

         Response.Write("<div style=""padding:6px""><a href='attachment.asp'>文件大小超出,请返回重新上传</a></div>")

ElseIF IsvalidFile(UCase(F_Type)) = False Then

         Response.Write("<div style=""padding:6px""><a href='attachment.asp'>文件格式非法,请返回重新上传</a></div>" )

Else

         F_File.SaveAs Server.MapPath("attachments/"&D_Name&"/"&F_Name)

         response.write "<script>addUploadItem('"&F_Type&"','attachments/"&D_Name&"/"&F_Name&"',"&Request.QueryString("MSave")&")</script>"

         Response.Write("<div style=""padding:6px""><a href='attachment.asp'>文件上传成功,请返回继续上传</a></div>")

End IF

 

保存文件的后缀是F_File.FileExt,而检查的是经过FixName()函数处理的后缀,那就看看fixname函数定义,在function.asp中:

 

'*************************************

'过滤文件名字

'*************************************

Function FixName(UpFileExt)

         If IsEmpty(UpFileExt) Then Exit Function

         FixName = Ucase(UpFileExt)

         FixName = Replace(FixName,Chr(0),"")

         FixName = Replace(FixName,".","")

         FixName = Replace(FixName,"ASP","")

         FixName = Replace(FixName,"ASA","")

         FixName = Replace(FixName,"ASPX","")

         FixName = Replace(FixName,"CER","")

         FixName = Replace(FixName,"CDX","")

         FixName = Replace(FixName,"HTR","")

End Function

 

呵呵,它把危险后缀过滤了的,如果我的文件后缀是asp(0x00)gif,那么检查的时候过滤chr(0)asp,后缀就成了gif,通过,保存的时候就是asp(0x00)gif。从理论上来说是对的啊,我搞了半天也不行,郁闷。有知道的大侠告诉我啊。

不过我们可以利用这个上传aspx格式。呵呵,我们上传aspx文件,fixname函数过滤asp,于是后缀就成了x。只要把x设为可以上传的格式,就可以传aspx文件了。

 

六、 附件管理问题

管理员登陆后有个附件管理功能,看他参数类似于http://localhost/blog/ConContent.asp?Fmenu=SQLFile&Smenu=Attachments&AttPath=attachments/month_0512,可以通过指定AttPath浏览web目录的哦。

但是系统限制了站外提交数据,所以不能直接改url,那就“曲线救国”吧。找到友情链接添加功能,url就填上我们构造的url,保存,然后点查看,就绕过站外的限制了。注意构造的AttPath的第一个字符不能是“.”和“/”哦,程序有检查的。我们这样构造就可以跳到blog根目录:http://localhost/blog/ConContent.asp?Fmenu=SQLFile&Smenu=Attachments&AttPath=attachments/..

 

 

呵呵,最后还是希望大家支持PJBlog2,真的是很好用的啊。希望作者再接再厉,把这套blog系统做得更好,造福众多网民J

PS:作者真是勤快啊,补丁已经出来了,呵呵

发表于 @ 2006年01月08日 17:42:00|评论(loading...)|编辑

新一篇: ASP.Net下的C/S后门——WebAdmin 2.Y应用详解 | 旧一篇: L-Blog的三个漏洞

评论

#NetSnow 发表于2006-01-09 14:35:00  IP: 219.148.133.*
不是吧?才几天啊?从L-BLOG到PJBLOG2..
接下来是不是要到LBS-BLOG了?我晕!!
搞的我现在对BLOG危机从丛的~
#h4k_b4n 发表于2006-01-09 13:44:00  IP: 61.144.115.*
Lake2疯了啊!!!!!!!!!光是砍死Blog啊`~`~`~Blog得罪Lake2了啊
#lake2 发表于2006-01-09 15:49:00  IP: 125.71.3.*
狂汗,我,我,我真的只是想弄个自己的blog来玩玩而已嘛,我不知道弄个blog还要回答这么多问题~~~
呵呵,今年我的blog还比较热闹,嘿嘿~~~大家都快乐哈~~
#NetSnow 发表于2006-01-10 16:33:00  IP: 210.82.175.*
呵呵..人多好呀..人多了可热闹了.多好玩呀..
#h4k_b4n 发表于2006-01-09 19:55:00  IP: 61.144.115.*
Lake可以试试Z-Blog或者LBS,Z-BLOG我觉得安全还是有点隐患`~但是总说不出到底是什么回事
#Cloudream 发表于2006-05-28 09:46:00  IP: 222.36.4.*
分析下Z-Blog吧^^
#kuhanzhu 发表于2006-07-05 18:51:00  IP: 125.112.31.*
感谢,一年前分析我的程序,我就比pjblog出名了.
#kuhanzhu 发表于2006-07-05 19:03:00  IP: 125.112.31.*
感觉,用过滤后缀的方法很不安全,pjblog出现过,oblog也出现过这个漏洞。
pjblog到2006年初还没搞明白防下载表是很不应该的,这步走在所有blog程序后面(几乎所有ASP程序了)。
另外,问:request.servervariables("http_referer")可以构造跨站,那么Request.ServerVariables("HTTP_USER_AGENT")应该也可以,是吧?
#kuhanzhu 发表于2006-07-05 19:09:00  IP: 125.112.31.*
说下SHA1的缺点,别说;运算速度快与慢,160位,浪费数据库空间~
LBLOG的作者好象也中意SHA1了(是z-blog还是?忘记了)
其实用md5就可以了。
大家都知道md5是不可逆的。既然不可逆就必须得穷举法破解。
一个程序好坏,主要看编程者的好坏而非语言本身(说这可没贬低pjblog的作者哦~偶是指对加密方法的利用上说的)。
md5,可以这样运用:md5(admin&psw)
这样,就算密码是123,我用用户名kuhanzhu,最终存入数据库数据也是kuhanzhu123,如果用工具破解,11位,怕很难是吧。所以,用md5足够了。如果不放心,还可以如此这般:md5(admin&psw&"%$#")
加上两三个特殊字符(俗称salt的东西),如此加密有什么不放心?
发表评论  


当前用户设置只有注册用户才能发表评论。如果你没有登录,请点击登录
Csdn Blog version 3.1a
Copyright © lake2