SPRING
2017.06.03 / 15:46

[½ÃÅ¥¾îÄÚµù½Ç½À] XSS ¹æ¾î¸¦ À§ÇÑ ½ÃÅ¥¾îÄÚµù ±â¹ý

±Ý±Ý±Ý
Ãßõ ¼ö 400


 domxss.jsp

 dom_xss.txt



1. Filter¸¦ ÀÌ¿ëÇÏ¿© XSS Ãë¾àÁ¡Á¦°Å 


 XSSFilter.java

 XSSRequestWrapper.java

 web.xml




XSSFilter.java ÀÛ¼º


public class XSSFilter implements Filter {


@Override

public void init(FilterConfig filterConfig) throws ServletException {

}


@Override

public void destroy() {

}


@Override

public void doFilter(ServletRequest request, ServletResponse response,

FilterChain chain) throws IOException, ServletException {

chain.doFilter(new XSSRequestWrapper((HttpServletRequest) request),

      response);

}

}



XSSRequestWrapper.java ÀÛ¼º


public class XSSRequestWrapper extends HttpServletRequestWrapper {


private static Pattern[] patterns = new Pattern[] {

Pattern.compile("<script>(.*?)</script>",                       

                                              Pattern.CASE_INSENSITIVE),

Pattern.compile("src[\r\n]*=[\r\n]*\\\'(.*?)\\\'",

                        Pattern.CASE_INSENSITIVE | Pattern.MULTILINE

                        | Pattern.DOTALL),

Pattern.compile("src[\r\n]*=[\r\n]*\\\"(.*?)\\\"",

                         Pattern.CASE_INSENSITIVE | Pattern.MULTILINE

                         | Pattern.DOTALL),

Pattern.compile("</script>", Pattern.CASE_INSENSITIVE),

Pattern.compile("<script(.*?)>", Pattern.CASE_INSENSITIVE

                        | Pattern.MULTILINE | Pattern.DOTALL),

Pattern.compile("eval\\((.*?)\\)", Pattern.CASE_INSENSITIVE

                        | Pattern.MULTILINE | Pattern.DOTALL),

Pattern.compile("expression\\((.*?)\\)", 

                        Pattern.CASE_INSENSITIVE

                        | Pattern.MULTILINE | Pattern.DOTALL),

Pattern.compile("javascript:", Pattern.CASE_INSENSITIVE),

Pattern.compile("vbscript:", Pattern.CASE_INSENSITIVE),

Pattern.compile("onload(.*?)=", Pattern.CASE_INSENSITIVE

                        | Pattern.MULTILINE | Pattern.DOTALL)

};


public XSSRequestWrapper(HttpServletRequest servletRequest) {

              super(servletRequest);

}


@Override

public String[] getParameterValues(String parameter) {

    String[] values = super.getParameterValues(parameter);

    if (values == null) {

        return null;

    }

    int count = values.length;

    String[] encodedValues = new String[count];

    for (int i = 0; i < count; i++) {

        encodedValues[i] = stripXSS(values[i]);

    }

    return encodedValues;

}


@Override

public String getParameter(String parameter) {

    String value = super.getParameter(parameter);

    return stripXSS(value);

}


@Override

public String getHeader(String name) {

    String value = super.getHeader(name);

    return stripXSS(value);

}


private String stripXSS(String value) {

   if (value != null) {


   // ESAPI library ÀÌ¿ëÇÏ¿© XSS ÇÊÅ͸¦ Àû¿ëÇÏ·Á¸é ¾Æ·¡ ÄÚµåÀÇ Ä¿¸ÇÆ®

   // ¸¦ Á¦°ÅÇÏ°í »ç¿ëÇÑ´Ù. °­·ÂÃßõ!!

   // value = ESAPI.encoder().canonicalize(value);


   // null ¹®ÀÚ¸¦ Á¦°ÅÇÑ´Ù.

   value = value.replaceAll("\0", "");


   // ÆÐÅÏÀ» Æ÷ÇÔÇϴ ÀԷ¿¡ ´ëÇØ <, > À» ÀÎÄÚµùÇÑ´Ù.

   for (Pattern scriptPattern : patterns) {

      if ( scriptPattern.matcher(value).find() ) {

         System.out.println("match.....");

         value=value.replaceAll("<", "&lt;").replaceAll(">", "&gt;");   }

      } 

    }

    return value;

  }

}




Web.xml ÆÄÀÏ¿¡ ÇÊÅÍ°¡ Àû¿ëµÇµµ·Ï ¼öÁ¤ÇÑ´Ù.

** ¸ÖƼÆÄÆ® Æû µ¥ÀÌÅÍ¿¡ ´ëÇØ XSS ÇÊÅ͸¦ Àû¿ëÇϱâ À§Çؼ­´Â XSSFilter¸¦ Àû¿ëÇϱâÀü¿¡ ¸ÕÀú MultipartFilter¸¦ Àû¿ëÇØ¾ß ÇÑ´Ù.

<filter>
    <filter-name>multipartFilter</filter-name>
    <filter-class>
               org.springframework.web.multipart.support.MultipartFilter
    </filter-class>
</filter>
  
  <filter-mapping>
    <filter-name>multipartFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
  
  <filter>
    <filter-name>xssFilter</filter-name>
    <filter-class>kr.co.openeg.lab.common.filter.XSSFilter</filter-class>
  </filter>
  
  <filter-mapping>
    <filter-name>xssFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping> 


dispatcher-servlet.xml ÆÄÀÏ¿¡¼­ ¸ÖƼÆÄÆ®¸¦ ó¸®ÇÏ´Â ¸®Á¹¹öÀÇ beanÀ̸§À» filterMultipartResolver·Î º¯°æÇÑ´Ù. 

    <bean id="filterMultipartResolver"
    class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
    </bean>




2. Ãâ·Â°ª¿¡ ´ëÇÑ ÇÊÅ͸µ ¶Ç´Â HTML ÀÎÄÚµùÀ» 
  
    ÀÌ¿ëÇÑ XSS Ãë¾àÁ¡Á¦°Å


CASE1:  JSP ÆÄÀÏ¿¡¼­ JSTL <c:out>À» ÀÌ¿ëÇÏ¿© Ãâ·Â°ªÀ» ÅؽºÆ®·Î ó¸®ÇÑ´Ù.
<p><c:out value="${board.content}"/></p>


CASE2: JSTLÀÇ fn:escapeXml() ÇÔ¼ö¸¦ ÀÌ¿ëÇÏ¿© HTML ÀÎÄÚµù ó¸®ÇÏ¿© Ãâ·ÂÇÑ´Ù.
<p>${fn:escapeXml(board.content)} </p> 


CASE3: Servlet¿¡¼­ ³×À̹öÀÇ lucy-xss Filter¸¦ ÀÌ¿ëÇÏ¿© ÇÊÅ͸µ ÇÏ¿© Ãâ·ÂÇÑ´Ù.
String data=request.getParameter("data");
XssFilter filter = XssFilter.getInstance("lucy-xss-superset.xml");
out.println(filter.doFilter(data));


lucy-xss ÇÊÅÍ Àû¿ë¹æ¹ýÀº http://openeg.co.kr/484 À» ÂüÁ¶ÇÑ´Ù.



Filter¸¦ Àû¿ëÇϱâ Àü¿¡ ÀԷ°ª¿¡ ´ëÇÑ Á¤±ÔÈ­ ¶Ç´Â ±Ô¹üÈ­ ÀÛ¾÷ÀÌ ¿ä±¸µÈ´Ù. ¿¹¸¦ µé¾î ÀԷ°ªÀÌ URL ÀÎÄÚµùµÇ¾î Àü´ÞµÇ´Â °æ¿ì µðÄÚµùÀ» ¼öÇàÇÏ¿© ÇÊÅ͸¦ Àû¿ëÇÏ¿© ÇÑ´Ù. ¶Ç´Â ´õºí ÀÎÄÚµù¿¡ ´ëÇؼ­µµ ÀûÀýÇÏ°Ô ´ëÀÀÇÏ¿© ÇÑ´Ù.

@RequestMapping(value="/test/xss_test.do", method = RequestMethod.POST)

@ResponseBody

public String testXss(HttpServletRequest request) {

StringBuffer buffer=new StringBuffer();

String data=request.getParameter("data");

System.out.println("Before Decde:  "+data);

Pattern p=Pattern.compile("(25)*3[cC]");

try {

     data=URLDecoder.decode(data, "UTF-8");

     System.out.println("After Decde:  "+data);

     if ( p.matcher(data).find() ) {

      return "XSS °ø°ÝÀÌ °¨ÁöµÊ";

     }

} catch (UnsupportedEncodingException e) {

          e.printStackTrace();

}

XssFilter filter = XssFilter.getInstance("lucy-xss-superset.xml");

buffer.append(filter.doFilter(data));

        return buffer.toString();

}


CASE4: Spring MVC ¸¦ »ç¿ëÇÏ´Â °æ¿ì web.xml ¿¡¼­ Àüü ¾ÖÇø®ÄÉÀ̼ǿ¡ HTML ÀÎÄÚµùÀ» Àû¿ëÇÑ´Ù.

<context-param> 

            <param-name>defaultHtmlEscape</param-name> 

            <param-value>true</param-value> 

</context-param> 



CASE5: Spring MVC ¸¦ »ç¿ëÇÏ´Â °æ¿ì page ·¹º§¿¡¼­ HTML ÀÎÄÚµùÀ» Àû¿ëÇÑ´Ù.

<spring:htmlEscape defaultHtmlEscape="true" />



CASE6: Spring MVC ¸¦ »ç¿ëÇÏ´Â °æ¿ì form tagÀÇ ¼Ó¼º°ªÀ» »ç¿ëÇؼ­ HTML ÀÎÄÚµùÀ» Àû¿ëÇÑ´Ù.

<form:input path="name" htmlEscape="true" />




DOM XSS Å×½ºÆ®


 domxss.jsp




3. Interceptor¸¦ ÀÌ¿ëÇÏ¿© XSS Ãë¾àÁ¡Á¦°Å



XSSInterceptor.java ¿¡¼­ ÇÊÅ͸µ Á¤Ã¥À» ÀÛ¼ºÇÑ´Ù.

package kims.interceptor;

import java.util.Enumeration;

import java.util.Iterator;

import java.util.Map;

import java.util.Set;

import java.util.regex.Pattern;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import org.springframework.web.multipart.MultipartHttpServletRequest;

import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;

public class XSSInterceptor extends HandlerInterceptorAdapter {

 

 // ¿©±â¼­´Â ÇÊÅ͸µ Á¤Ã¥À» ºí·¢¸®½ºÆ®·Î ¼³Á¤. ÆÐÅÏÀÌ Æ÷ÇÔµÈ ¿äûÀ» ¸ðµÎ ¹ö¸².

 // È­ÀÌÆ® ¸®½ºÆ® Á¤Ã¥À» ¼³Á¤ÇÏ¿© Çã¿ëÇÏ°íÀÚ ÇÏ´Â °ª¸¸ Åë°ú ½ÃÅ°µµ·Ï Çϴ°ÍÀÌ 

 // ´õ ¾ÈÀüÇÑ ¹æ¹ýÀÓ.

 // ¸¸¾à ºí·¢¸®½ºÆ® Á¤Ã¥À» ¾²´Â °æ¿ì ÀԷ°ª¿¡ ´ëÇÑ ÇÊÅ͸µ ÀÛ¾÷Àº ¹«ÁöÇÏ°Ô ¾î·Æ´Ù.

 // ±×·³¿¡µµ ºÒ±¸ÇÏ°í ºí·¢¸®½ºÆ® Á¤Ã¥À» ¾´´Ù¸é Ãâ·Â°ª¿¡ ´ëÇÑ ÇÊÅ͸µ Á¤Ã¥À» Àû¿ë

 // ÇÏ´Â°Í ÀÌ ±×³ª¸¶ ³ªÀº ¹æ¹ýÀÌ´Ù.

 

 // ÀÌ ¿¹Á¦¿¡¼­´Â °£´ÜÇÏ°Ô Interceptor¸¦ ÀÌ¿ëÇÏ¿© ¾î¶»°Ô ÇÊÅ͸µÀÌ Àû¿ëµÇ´ÂÁö¸¸ 

 // º¸¿©ÁØ´Ù.


 XSS_Filter_Rules.txt

String[] pattern= {

       "<script>(.*?)</script>",

       "src[\r\n]*=[\r\n]*\\\'(.*?)\\\'",

       "src[\r\n]*=[\r\n]*\\\"(.*?)\\\"",

       "</script>",

       "<script(.*?)>",

       "eval\\((.*?)\\)",

       "expression\\((.*?)\\)",

       "javascript:", 

       "vbscript:",

       "onload(.*?)="  

  };

 




@Override

public boolean preHandle(HttpServletRequest request,

   HttpServletResponse response, Object handler) throws Exception {


  if ( request instanceof MultipartHttpServletRequest) {

      // ParameterMap  À» »ç¿ëÇϴ°ÍÀÌ ´õ ±ÇÀå»çÇ×ÀÓ.

      Enumeration<String> names=request.getParameterNames();

      while ( names.hasMoreElements()) {

         String a=names.nextElement();

            String value=request.getParameter(a);

            System.out.println("parameter: "+a);

               if ( ! isValidInput(value)) {

             response.sendRedirect("xss_detect.do");

             return false;

            }

      }

  }

     return true; 

 }




private boolean isValidInput(String value) {      

          if (value != null) {             

              value = value.replaceAll("", "");

              for ( String ptn : pattern ) {

                    Pattern scriptPattern = Pattern.compile(ptn, 

                                                  Pattern.CASE_INSENSITIVE);

                    if ( scriptPattern.matcher(value).find() ) return false;

              }

              return true;

          } else {

           return false;

          }             

  }      

}





web.xml ¿¡ ContextLoadListener¸¦ µî·ÏÇØÁØ´Ù.


 web.xml.txt


  <context-param> 

     <param-name>contextConfigLocation</param-name>

     <param-value>WEB-INF/dispatcher-servlet.xml</param-value>

  </context-param>

  <listener> 

    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

  </listener>




dispatcher-servlet.xml ¿¡¼­ interceptor¸¦ µî·ÏÇØÁØ´Ù.


 dispatcher-servlet.xml.txt


 <mvc:interceptors>

         <mvc:interceptor>

                <mvc:mapping path="/board/write.do"/>

                <bean class="kims.interceptor.XSSInterceptor"></bean>

         </mvc:interceptor>

 </mvc:interceptors>



<Âü°í> Interceptor¿¡¼­ XSSFilter Àû¿ë¿¹Á¦.

           Interceptor¿¡¼­ request°ªÀ» wrappingÇؼ­ ó¸®ÇÏ´Â Maven ÇÁ·ÎÁ§Æ® ¿¹Á¦.

           ÀÌŬ¸³½º¿¡¼­ import ÇÒ¶§ ±âÁ¸ Maven ÇÁ·ÎÁ§Æ® import ÇÏ¿© Å×½ºÆ® ÇÑ´Ù.

 stripes-xss-master.zip



<Âü°í> XSSÇÊÅ͸µ ¶óÀ̺귯¸® 

 html_filter.zip




================================================================================================
<Âü°í> À§ÀÇ ¿¹Á¦´Â ºí·¢¸®½ºÆ® Á¤Ã¥À» Àû¿ëÇÑ °ÍÀÌ´Ù. 
          È­ÀÌÆ®¸®½ºÆ® Á¤Ã¥Àº ´ÙÀ½°ú °°ÀÌ Àû¿ëÇÒ ¼ö ÀÖ´Ù.

-------------------------------------------------------------------------------
package com.josephoconnell.html;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import junit.framework.Assert;
import junit.framework.TestCase;

/**
 
 * HTML filtering utility for protecting against XSS (Cross Site Scripting).
 *
 * This code is licensed under a Creative Commons Attribution-ShareAlike 2.5 License
 * http://creativecommons.org/licenses/by-sa/2.5/
 
 * This code is a Java port of the original work in PHP by Cal Hendersen.
 * http://code.iamcal.com/php/lib_filter/
 *
 * The trickiest part of the translation was handling the differences in regex handling
 * between PHP and Java.  These resources were helpful in the process:
 
 * http://java.sun.com/j2se/1.4.2/docs/api/java/util/regex/Pattern.html
 * http://us2.php.net/manual/en/reference.pcre.pattern.modifiers.php
 * http://www.regular-expressions.info/modifiers.html
 
 * A note on naming conventions: instance variables are prefixed with a "v"; global
 * constants are in all caps.
 
 * Sample use:
 * String input = ...
 * String clean = new HTMLInputFilter().filter( input );
 
 * If you find bugs or have suggestions on improvement (especially regarding 
 * perfomance), please contact me at the email below.  The latest version of this
 * source can be found at
 
 * http://josephoconnell.com/java/xss-html-filter/
 *
 @author Joseph O'Connell <joe.oconnell at gmail dot com>
 @version 1.0 
 */
public class HTMLInputFilter
{  
  /** 
   * flag determining whether to try to make tags when presented with "unbalanced"
   * angle brackets (e.g. "<b text </b>" becomes "<b> text </b>").  If set to false,
   * unbalanced angle brackets will be html escaped.
   */
  protected static final boolean ALWAYS_MAKE_TAGS = true;
  
  /**
   * flag determing whether comments are allowed in input String.
   */
  protected static final boolean STRIP_COMMENTS = true;
  
  /** regex flag union representing /si modifiers in php **/
  protected static final int REGEX_FLAGS_SI = Pattern.CASE_INSENSITIVE | Pattern.DOTALL;
  
  /** set of allowed html elements, along with allowed attributes for each element **/
  protected Map<String,List<String>> vAllowed;
  
  /** counts of open tags for each (allowable) html element **/
  protected Map<String,Integer> vTagCounts;
  
  /** html elements which must always be self-closing (e.g. "<img />") **/
  protected String[] vSelfClosingTags;
  
  /** html elements which must always have separate opening and closing tags (e.g. "<b></b>") **/
  protected String[] vNeedClosingTags;
  
  /** attributes which should be checked for valid protocols **/
  protected String[] vProtocolAtts;
  
  /** allowed protocols **/
  protected String[] vAllowedProtocols;
  
  /** tags which should be removed if they contain no content (e.g. "<b></b>" or "<b />") **/
  protected String[] vRemoveBlanks;
  
  /** entities allowed within html markup **/
  protected String[] vAllowedEntities;
  
  protected boolean vDebug;
  
  public HTMLInputFilter()
  {
    this(false);
  }
  
  public HTMLInputFilterboolean debug )
  {
    vDebug = debug;
    
    vAllowed = new HashMap<String,List<String>>();
    vTagCounts = new HashMap<String,Integer>();
    
    ArrayList<String> a_atts = new ArrayList<String>();
    a_atts.add"href" );
    a_atts.add"target" );
    vAllowed.put"a", a_atts );
    
    ArrayList<String> img_atts = new ArrayList<String>();
    img_atts.add"src" );
    img_atts.add"width" );
    img_atts.add"height" );
    img_atts.add"alt" );
    vAllowed.put"img", img_atts );
    
    ArrayList<String> no_atts = new ArrayList<String>();
    vAllowed.put"b", no_atts );
    vAllowed.put"strong", no_atts );
    vAllowed.put"i", no_atts );
    vAllowed.put"em", no_atts );
    
    vSelfClosingTags = new String[] { "img" };
    vNeedClosingTags = new String[] { "a""b""strong""i""em" };
    vAllowedProtocols = new String[] { "http""mailto" }// no ftp.
    vProtocolAtts = new String[] { "src""href" };
    vRemoveBlanks = new String[] { "a""b""strong""i""em" };
    vAllowedEntities = new String[] { "amp""gt""lt""quot" };
  }
  
  protected void reset()
  {
    vTagCounts = new HashMap<String,Integer>();
  }
  
  protected void debugString msg )
  {
    if (vDebug)
      System.out.printlnmsg );
  }
  
  //---------------------------------------------------------------
  // my versions of some PHP library functions
  
  public static String chrint decimal )
  {
    return String.valueOf( (chardecimal );
  }
  
  public static String htmlSpecialCharsString s )
  {
    s = s.replaceAll"&""&amp;" );
    s = s.replaceAll"\"""&quot;" );
    s = s.replaceAll"<""&lt;" );
    s = s.replaceAll">""&gt;" );
    return s;
  }
  
  //---------------------------------------------------------------
  
  /**
   * given a user submitted input String, filter out any invalid or restricted
   * html.
   
   @param input text (i.e. submitted by a user) than may contain html
   @return "clean" version of input, with only valid, whitelisted html elements allowed
   */
  public synchronized String filterString input )
  {
    reset();
    String s = input;
    
    debug"************************************************" );
    debug"              INPUT: " + input );
    
    s = escapeComments(s);
    debug"     escapeComments: " + s );
    
    s = balanceHTML(s);
    debug"        balanceHTML: " + s );
    
    s = checkTags(s);
    debug"          checkTags: " + s );
    
    s = processRemoveBlanks(s);
    debug"processRemoveBlanks: " + s );
    
    s = validateEntities(s);
    debug"    validateEntites: " + s );
    
    debug"************************************************\n\n" );
    return s;
  }
  
  protected String escapeCommentsString s )
  {
    Pattern p = Pattern.compile"<!--(.*?)-->", Pattern.DOTALL );
    Matcher m = p.matcher);
    StringBuffer buf = new StringBuffer();
    if (m.find()) {
      String match = m.group)//(.*?)
      m.appendReplacementbuf, "<!--" + htmlSpecialCharsmatch "-->" );
    }
    m.appendTailbuf );
    
    return buf.toString();
  }
  
  protected String balanceHTMLString s )
  {
    if (ALWAYS_MAKE_TAGS
    {
      //
      // try and form html
      //
      s = regexReplace("^>""", s);
      s = regexReplace("<([^>]*?)(?=<|$)""<$1>", s);
      s = regexReplace("(^|>)([^<]*?)(?=>)""$1<$2", s);
      
    
    else
    {
      //
      // escape stray brackets
      //
      s = regexReplace("<([^>]*?)(?=<|$)""&lt;$1", s);
      s = regexReplace("(^|>)([^<]*?)(?=>)""$1$2&gt;<", s);
      
      //
      // the last regexp causes '<>' entities to appear
      // (we need to do a lookahead assertion so that the last bracket can
      // be used in the next pass of the regexp)
      //
      s = s.replaceAll("<>""");
    }
    
    return s;
  }
  
  protected String checkTagsString s )
  {    
    Pattern p = Pattern.compile"<(.*?)>", Pattern.DOTALL );
    Matcher m = p.matcher);
    
    StringBuffer buf = new StringBuffer();
    while (m.find()) {
      String replaceStr = m.group);
      replaceStr = processTagreplaceStr );
      m.appendReplacement(buf, replaceStr);
    }
    m.appendTail(buf);
    
    s = buf.toString();
    
    // these get tallied in processTag
    // (remember to reset before subsequent calls to filter method)
    forString key : vTagCounts.keySet())
    {
      for(int ii=0; ii<vTagCounts.get(key); ii++) {
        s += "</" + key + ">";
      }
    }
    
    return s;
  }
  
  protected String processRemoveBlanksString s )
  {
    forString tag : vRemoveBlanks )
    {
      s = regexReplace"<" + tag + "(\\s[^>]*)?></" + tag + ">""", s );
      s = regexReplace"<" + tag + "(\\s[^>]*)?/>""", s );
    }
    
    return s;
  }
  
  protected String regexReplaceString regex_pattern, String replacement, String s )
  {
    Pattern p = Pattern.compileregex_pattern );
    Matcher m = p.matcher);
    return m.replaceAllreplacement );
  }
  
  protected String processTagString s )
  {    
    // ending tags
    Pattern p = Pattern.compile"^/([a-z0-9]+)", REGEX_FLAGS_SI );
    Matcher m = p.matcher);
    if (m.find()) {
      String name = m.group(1).toLowerCase();
      if (vAllowed.containsKeyname )) {
        if (!inArray(name, vSelfClosingTags)) {
          if (vTagCounts.containsKeyname )) {
            vTagCounts.put(name, vTagCounts.get(name)-1);
            return "</" + name + ">";
          }
        }
      }
    }
    
    // starting tags
    p = Pattern.compile("^([a-z0-9]+)(.*?)(/?)$", REGEX_FLAGS_SI);
    m = p.matcher);
    if (m.find()) {
      String name = m.group(1).toLowerCase();
      String body = m.group(2);
      String ending = m.group(3);
      
      //debug( "in a starting tag, name='" + name + "'; body='" + body + "'; ending='" + ending + "'" );
      if (vAllowed.containsKeyname )) {
        String params = "";
        
        Pattern p2 = Pattern.compile("([a-z0-9]+)=([\"'])(.*?)\\2", REGEX_FLAGS_SI);
        Pattern p3 = Pattern.compile("([a-z0-9]+)(=)([^\"\\s']+)", REGEX_FLAGS_SI);
        Matcher m2 = p2.matcherbody );
        Matcher m3 = p3.matcherbody );
        List<String> paramNames = new ArrayList<String>();
        List<String> paramValues = new ArrayList<String>();
        while (m2.find()) {
          paramNames.add(m2.group(1))//([a-z0-9]+)
          paramValues.add(m2.group(3))//(.*?)
        }
        while (m3.find()) {
          paramNames.add(m3.group(1))//([a-z0-9]+)
          paramValues.add(m3.group(3))//([^\"\\s']+)
        }
        
        String paramName, paramValue;
        forint ii=0; ii<paramNames.size(); ii++ ) {
          paramName = paramNames.get(ii).toLowerCase();
          paramValue = paramValues.get(ii);
          
          //debug( "paramName='" + paramName + "'" );
          //debug( "paramValue='" + paramValue + "'" );
          //debug( "allowed? " + vAllowed.get( name ).contains( paramName ) );
          
          if (vAllowed.getname ).containsparamName )) {
            if (inArrayparamName, vProtocolAtts )) {
              paramValue = processParamProtocolparamValue );
            }
            params += " " + paramName + "=\"" + paramValue + "\"";
          }
        }
        
        if (inArrayname, vSelfClosingTags )) {
          ending = " /";
        }
        
        if (inArrayname, vNeedClosingTags )) {
          ending = "";
        }
        
        if (ending == null || ending.length() 1) {
          if (vTagCounts.containsKeyname )) {
            vTagCounts.putname, vTagCounts.get(name)+);
          else {
            vTagCounts.putname, );
          }
        else {
          ending = " /";
        }
        return "<" + name + params + ending + ">";
      else {
        return "";
      }
    }
    
    // comments
    p = Pattern.compile"^!--(.*)--$", REGEX_FLAGS_SI );
    m = p.matcher);
    if (m.find()) {
      String comment = m.group();
      if (STRIP_COMMENTS) {
        return "";
      else {
        return "<" + comment + ">"
      }
    }
    
    return "";
  }
  
  protected String processParamProtocolString s )
  {
    s = decodeEntities);
    Pattern p = Pattern.compile"^([^:]+):", REGEX_FLAGS_SI );
    Matcher m = p.matcher);
    if (m.find()) {
      String protocol = m.group(1);
      if (!inArrayprotocol, vAllowedProtocols )) {
        // bad protocol, turn into local anchor link instead
        s = "#" + s.substringprotocol.length()+1, s.length() );
        if (s.startsWith("#//")) s = "#" + s.substring3, s.length() );
      }
    }
    
    return s;
  }
  
  protected String decodeEntitiesString s )
  {
    StringBuffer buf = new StringBuffer();
    
    Pattern p = Pattern.compile"&#(\\d+);?" );
    Matcher m = p.matcher);
    while (m.find()) {
      String match = m.group);
      int decimal = Integer.decodematch ).intValue();
      m.appendReplacementbuf, chrdecimal ) );
    }
    m.appendTailbuf );
    s = buf.toString();
    
    buf = new StringBuffer();
    p = Pattern.compile"&#x([0-9a-f]+);?");
    m = p.matcher);
    while (m.find()) {
      String match = m.group);
      int decimal = Integer.decodematch ).intValue();
      m.appendReplacementbuf, chrdecimal ) );
    }
    m.appendTailbuf );
    s = buf.toString();
    
    buf = new StringBuffer();
    p = Pattern.compile"%([0-9a-f]{2});?");
    m = p.matcher);
    while (m.find()) {
      String match = m.group);
      int decimal = Integer.decodematch ).intValue();
      m.appendReplacementbuf, chrdecimal ) );
    }
    m.appendTailbuf );
    s = buf.toString();
    
    s = validateEntities);
    return s;
  }
  
  protected String validateEntitiesString s )
  {
    // validate entities throughout the string
    Pattern p = Pattern.compile"&([^&;]*)(?=(;|&|$))" );
    Matcher m = p.matcher);
    if (m.find()) {
      String one = m.group)//([^&;]*) 
      String two = m.group)//(?=(;|&|$))
      s = checkEntityone, two );
    }
    
    // validate quotes outside of tags
    p = Pattern.compile"(>|^)([^<]+?)(<|$)", Pattern.DOTALL );
    m = p.matcher);
    StringBuffer buf = new StringBuffer();
    if (m.find()) {
      String one = m.group)//(>|^) 
      String two = m.group)//([^<]+?) 
      String three = m.group)//(<|$) 
      m.appendReplacementbuf, one + two.replaceAll"\"""&quot;" + three);
    }
    m.appendTailbuf );
    
    return s;
  }
  
  protected String checkEntityString preamble, String term )
  {
    if (!term.equals(";")) {
      return "&amp;" + preamble;
    }
    
    if isValidEntitypreamble ) ) {
      return "&" + preamble;
    }
    
    return "&amp;" + preamble;
  }
  
  protected boolean isValidEntityString entity )
  {
    return inArrayentity, vAllowedEntities );
  }
  
  private boolean inArrayString s, String[] array )
  {
    for (String item : array)
      if (item != null && item.equals(s))
        return true;
    
    return false;
  }
  
  // ============================================ START-UNIT-TEST =========================================
  public static class Test extends TestCase
  {  
    protected HTMLInputFilter vFilter;
    
    protected void setUp() 
    
      vFilter = new HTMLInputFiltertrue );
    }
    
    protected void tearDown()
    {
      vFilter = null;
    }
    
    private void tString input, String result )
    {
      Assert.assertEqualsresult, vFilter.filter(input) );
    }
    
    public void test_basics()
    {
      t"""" );
      t"hello""hello" );
    }
    
    public void test_balancing_tags()
    {
      t"<b>hello""<b>hello</b>" );
      t"<b>hello""<b>hello</b>" );
      t"hello<b>""hello" );
      t"hello</b>""hello" );
      t"hello<b/>""hello" );
      t"<b><b><b>hello""<b><b><b>hello</b></b></b>" );
      t"</b><b>""" );
    }
    
    public void test_end_slashes()
    {
      t("<img>","<img />");
      t("<img/>","<img />");
      t("<b/></b>","");
    }
    
    public void test_balancing_angle_brackets()
    {
      if (ALWAYS_MAKE_TAGS) {
        t("<img src=\"foo\"","<img src=\"foo\" />");
        t("i>","");
        t("<img src=\"foo\"/","<img src=\"foo\" />");
        t(">","");
        t("foo<b","foo");
        t("b>foo","<b>foo</b>");
        t("><b","");
        t("b><","");
        t("><b>","");
      else {
        t("<img src=\"foo\"","&lt;img src=\"foo\"");
        t("b>","b&gt;");
        t("<img src=\"foo\"/","&lt;img src=\"foo\"/");
        t(">","&gt;");
        t("foo<b","foo&lt;b");
        t("b>foo","b&gt;foo");
        t("><b","&gt;&lt;b");
        t("b><","b&gt;&lt;");
        t("><b>","&gt;");
      }
    }
    
    public void test_attributes()
    {
      t("<img src=foo>","<img src=\"foo\" />")
      t("<img asrc=foo>","<img />");
      t("<img src=test test>","<img src=\"test\" />")
    }
    
    public void test_disallow_script_tags()
    {
      t("<script>","");
      if (ALWAYS_MAKE_TAGS) { t("<script","");  else t("<script","&lt;script")}
      t("<script/>","");
      t("</script>","");
      t("<script woo=yay>","");
      t("<script woo=\"yay\">","");
      t("<script woo=\"yay>","");
      t("<script woo=\"yay<b>","");
      t("<script<script>>","");
      t("<<script>script<script>>","script");
      t("<<script><script>>","");
      t("<<script>script>>","");
      t("<<script<script>>","");
    }
    
    public void test_protocols()
    {
      t("<a href=\"http://foo\">bar</a>""<a href=\"http://foo\">bar</a>");
      // we don't allow ftp. t("<a href=\"ftp://foo\">bar</a>", "<a href=\"ftp://foo\">bar</a>");
      t("<a href=\"mailto:foo\">bar</a>""<a href=\"mailto:foo\">bar</a>");
      t("<a href=\"javascript:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"java script:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"java\tscript:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"java\nscript:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"java" + chr(1"script:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"jscript:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"vbscript:foo\">bar</a>""<a href=\"#foo\">bar</a>");
      t("<a href=\"view-source:foo\">bar</a>""<a href=\"#foo\">bar</a>");
    }
    
    public void test_self_closing_tags()
    {
      t("<img src=\"a\">","<img src=\"a\" />");
      t("<img src=\"a\">foo</img>""<img src=\"a\" />foo");
      t("</img>""");
    }
    
    public void test_comments()
    {
      if (STRIP_COMMENTS) {
        t("<!-- a<b --->""");
      else {
        t("<!-- a<b --->""<!-- a&lt;b --->");
      }
    }

  }
  // ============================================ END-UNIT-TEST ===========================================
}






3. AOP¸¦ ÀÌ¿ëÇÏ¿© XSS Ãë¾àÁ¡ Á¦°Å 
4. Validator¸¦ ÀÌ¿ëÇÏ¿© XSS Ãë¾àÁ¡ Á¦°Å



 








Ãâó: http://openeg.co.kr/383 [Open Expert Group]