1、JSP 安全编程实例浅析Java Server Page(JSP)作为建立动态网页的技术正在不断升温。JSP 和 ASP、PHP 、工作机制不太一样。一般说来,JSP 页面在执行时是编译式,而不是解释式的。首次调用JSP 文件其实是执行一个编译为 Servlet 的过程。当浏览器向服务器请求这一个 JSP 文件的时候,服务器将检查自上次编译后 JSP 文件是否有改变,如果没有改变,就直接执行Servlet,而不用再重新编译,这样,效率便得到了明显提高。今天我将和大家一起从脚本编程的角度看 JSP 的安全,那些诸如源码暴露类的安全隐患就不在这篇文章讨论范围之内了。写这篇文章的主要目的是给初学 J
2、SP 编程的朋友们提个醒,从一开始就要培养安全编程的意识,不要犯不该犯的错误,避免可以避免的损失。一、认证不严低级失误 在一论坛中, user_manager.jsp 是用户管理的页面,作者知道它的敏感性,加上了一把锁: if (session.getValue(“UserName“)=null)(session.getValue(“UserClass“)=null)(! session.getValue(“UserClass“).equals(“系统管理员“) response.sendRedirect(“err.jsp?id=14“); return; 如果要查看、修改某用户的信息,就要用
3、 modifyuser_manager.jsp 这个文件。管理员提交 http:/ 就是查看、修改 ID 为 51 的用户的资料(管理员默认的用户 ID 为 51) 。但是,如此重要的文件竟缺乏认证,普通用户(包括游客)也直接提交上述请求也可以对其一览无余(密码也是明文存储、显示的) 。modifyuser_manage.jsp 同样是门户大开,直到恶意用户把数据更新的操作执行完毕,重定向到 user_manager.jsp 的时候,他才会看见那个姗姗来迟的显示错误的页面。显然,只锁一扇门是远远不够的,编程的时候一定要不厌其烦地为每一个该加身份认证的地方加上身份认证。二、守好 JavaBean
4、 的入口 JSP 组件技术的核心是被称为 bean 的 java 组件。在程序中可把逻辑控制、数据库操作放在 javabeans 组件中,然后在 JSP 文件中调用它,这样可增加程序的清晰度及程序的可重用性。和传统的 ASP 或 PHP 页面相比,JSP 页面是非常简洁的,因为许多动态页面处理过程可以封装到 JavaBean 中。 要改变 JavaBean 属性,要用到“”标记。 下面的代码是假想的某电子购物系统的源码的一部分,这个文件是用来显示用户的购物框中的信息的,而 checkout.jsp 是用来结帐的。 You have added the item to your basket.
5、Your total is $ Proceed to checkout 注意到 property=“*“了吗?这表明用户在可见的 JSP 页面中输入的,或是直接通过 Query String 提交的全部变量的值,将存储到匹配的 bean 属性中。 一般,用户是这样提交请求的: http:/ /addToBasket.jsp?newItem=ITEM0105342 但是不守规矩的用户呢?他们可能会提交: http:/ /addToBasket.jsp?newItem=ITEM0105342 else return false; 四、时刻牢记 SQL 注入 一般的编程书籍在教初学者的时候都不注意让
6、他们从入门时就培养安全编程的习惯。著名的JSP 编程思想与实践就是这样向初学者示范编写带数据库的登录系统的(数据库为 MySQL): Statement stmt = conn.createStatement(); String checkUser = “select * from login where username = “ + userName + “ and userpassword = “ + userPassword + “; ResultSet rs = stmt.executeQuery(checkUser); if(rs.next() response.sendRedire
7、ct(“SuccessLogin.jsp“); else response.sendRedirect(“FailureLogin.jsp“); 这样使得尽信书的人长期使用这样先天“带洞”的登录代码。如果数据库里存在一个名叫“jack”的用户,那么在不知道密码的情况下至少有下面几种方法可以登录: 用户名:jack 密码: or a=a 用户名:jack 密码: or 1=1/* 用户名:jack or 1=1/* 密码:(任意) lybbs(凌云论坛)ver 2.9.Server 在 LogInOut.java 中是这样对登录提交的数据进行检查的: if(s.equals(“) s1.equal
8、s(“) throw new UserException(“用户名或密码不能空。“); if(s.indexOf(“) != -1 s.indexOf(“) != -1 s.indexOf(“,“) != -1 s.indexOf(“) != -1) throw new UserException(“用户名不能包括 “ , 等非法字符。“); if(s1.indexOf(“) != -1 s1.indexOf(“) != -1 s1.indexOf(“*“) != -1 s1.indexOf(“) != -1) throw new UserException(“密码不能包括 “ * 等非法字符
9、。“); if(s.startsWith(“ “) s1.startsWith(“ “) throw new UserException(“用户名或密码中不能用空格。“); 但是我不清楚为什么他只对密码而不对用户名过滤星号。另外,正斜杠似乎也应该被列到“黑名单”中。我还是认为用正则表达式只允许输入指定范围内的字符来得干脆。 这里要提醒一句:不要以为可以凭借某些数据库系统天生的“安全性”就可以有效地抵御所有的攻击。pinkeyes 的那篇PHP 注入实例就给那些依赖 PHP 的配置文件中的“magic_quotes_gpc = On”的人上了一课。五、String 对象带来的隐患 Java 平台
10、的确使安全编程更加方便了。Java 中无指针,这意味着 Java 程序不再像 C那样能对地址空间中的任意内存位置寻址了。在 JSP 文件被编译成 .class 文件时会被检查安全性问题,例如当访问超出数组大小的数组元素的尝试将被拒绝,这在很大程度上避免了缓冲区溢出攻击。但是,String 对象却会给我们带来一些安全上的隐患。如果密码是存储在 Java String 对象中的,则直到对它进行垃圾收集或进程终止之前,密码会一直驻留在内存中。即使进行了垃圾收集,它仍会存在于空闲内存堆中,直到重用该内存空间为止。密码 String 在内存中驻留得越久,遭到窃听的危险性就越大。更糟的是,如果实际内存减少
11、,则操作系统会将这个密码 String 换页调度到磁盘的交换空间,因此容易遭受磁盘块窃听攻击。为了将这种泄密的可能性降至最低(但不是消除) ,您应该将密码存储在 char 数组中,并在使用后对其置零(String 是不可变的,无法对其置零) 。六、线程安全初探 “JAVA 能做的,JSP 就能做” 。与 ASP、PHP 等脚本语言不一样,JSP 默认是以多线程方式执行的。以多线程方式执行可大大降低对系统的资源需求,提高系统的并发量及响应时间。线程在程序中是独立的、并发的执行路径,每个线程有它自己的堆栈、自己的程序计数器和自己的局部变量。虽然多线程应用程序中的大多数操作都可以并行进行,但也有某些
12、操作(如更新全局标志或处理共享文件)不能并行进行。如果没做好线程的同步,在大并发量访问时,不需要恶意用户的“热心参与” ,问题也会出现。最简单的解决方案就是在相关的 JSP 文件中加上: 指令,使它以单线程方式执行,这时,所有客户端的请求以串行方式执行。这样会严重降低系统的性能。我们可以仍让 JSP 文件以多线程方式执行,通过对函数上锁来对线程进行同步。一个函数加上 synchronized 关键字就获得了一个锁。看下面的示例: public class MyClass int a; public Init() /此方法可以多个线程同时调用 a = 0; public synchronized
13、 void Set() /两个线程不能同时调用此方法 if(a5) a= a-5; 但是这样仍然会对系统的性能有一定影响。一个更好的方案是采用局部变量代替实例变量。因为实例变量是在堆中分配的,被属于该实例的所有线程共享,不是线程安全的,而局部变量在堆栈中分配,因为每个线程都有它自己的堆栈空间,所以这样线程就是安全的了。比如凌云论坛中添加好友的代码: public void addFriend(int i, String s, String s1) throws DBConnectException try if else DBConnect dbconnect = new DBConnect(
14、“insert into friend (authorid,friendname) values (?,?)“); dbconnect.setInt(1, i); dbconnect.setString(2, s); dbconnect.executeUpdate(); dbconnect.close(); dbconnect = null; catch(Exception exception) throw new DBConnectException(exception.getMessage(); 下面是调用: friendName=ParameterUtils.getString(request,“friendname“); if(action.equals(“adduser“) forumFriend.addFriend(Integer.parseInt(cookieID),friendName,cookieName); errorInfo=forumFriend.getErrorInfo(); 如果采用的是实例变量,那么该实例变量属于该实例的所有线程共享,就有可能出现用户A 传递了某个参数后他的线程转为睡眠状态,而参数被用户 B 无意间修改,造成好友错配的现象。