volocity html特殊字符,顶改造Velocity模板引擎让$[!]{}输出默认进行html转义,并增加$#{}语法支持不转义输出...-程序员宅基地

技术标签: volocity html特殊字符  

一直以来在项目中使用Apache Velocity模板引擎作为视图层输出,为了解决XSS漏洞,需要对输出到页面的内容进行HTML转义,我一般采用2种方式实现:

使用过滤器 Filter,在其中进行 HttpServletRequestWrapper的 getParameter( )等方法重载,在底层进行HTML转义,然后页面直接输出;

这种方式很轻松很直接,业务代码不需要修改就完成了所有的转义工作;但是也带来了问题:修改了用户的原始输入数据,如果需要用到用户的原始输入数据,又得反转义回去,很麻烦。

在页面上使用 velocity-tools 的 $esc.html( ) 进行手工转义:$esc.html($task.content);

这种方式工作量比较大,需要在转义的变量上增加 $esc.html(),很容易遗漏。

那有没有别的方法来解决呢?

后来我发现 Velocity 为我们提供了EscapeHtmlReference  event handler 用于在引用变量值输出之前进行HTML转义处理,但是这个HTML转义功能需要通过eventhandler.escape.html.match=//配置指定变量前缀来进行,特别对于有的变量需要转义,有的变量不需要转义的情况下非常的不方便;在用了一段时间后,感觉很麻烦,污染变量命名,不爽;

正巧这段时间也使用了artTemplate.js的前端模板引擎,发现这个模板引擎提供了2种变量输出方式: 和 ,其中 是默认的常用输出方式(会对变量值中的HTML字符进行转义输出), 则是原样输出(不进行任何转义);感觉这样的方式非常的好,即满足了大部分的转义输出,也满足了小部分的非转义输出,而且不用对业务代码进行修改,而是由模板引擎提供不同的输出方法。

那能否修改 Velocity 的语法,也支持一种非转义输出呢?这样不就完美解决这个问题了吗?

$[ ! ][ { ] param [ } ] 是 Velocity 的默认引用定义和输出语法,为了自定义的语法简单易用,想采用 $#{ ... } 语法格式,这样和原来的语法只有一个字的差别:! -> #

// 默认转义输出

$task.content

${task.content}

$!task.content

$!{task.content}

// 原样不转义输出

$#task.content

$#{task.content}

于是我开始研究 Velocity 的语法解析代码,Velocity 采用 AST 语法树进行模板的语法解析的,所有的语法定义在 Parser.jjt 文件中,然后使用JAVACC 编译Parser.jjt文件生成语法解析代码(ASTAddNode,ASTEQNode, ASTReference, ASTSetDirective等等),$[ ! ][ { ] ... [ } ]语法Velocity定义为 Reference,因此 ASTReference.java 就是用来处理 Reference的。

在 Parser.jjt 文件的 730 行定义了这样的语法配置:

这段语法是用来支持 $! 的,如果我把最后的"!" 变成 ("!"|"#") 不就支持 $# 了嘛,呵呵,修改之(当然其它的语法定义也必须都看一遍):

发现在 ASTReference.java 文件的 getRoot() 方法中进行了 $!和$!{ 的处理:

if (t.image.startsWith("$!")) {

referenceType = QUIET_REFERENCE;

/*

*  only if we aren't escaped do we want to null the output

*/

if (!escaped) nullString = "";

if (t.image.startsWith("$!{")) {

/*

*  ex : $!{provider.Title}

*/

return t.next.image;

} else {

/*

*  ex : $!provider.Title

*/

return t.image.substring(2);

}

}

在有了上面的了解后,修改Velocity的Reference 语法规则就简单了,迅速修改 Parser.jjt 文件,然后使用 JAVACC(我使用javacc-eclipse插件) 编译Parser.jjt,生成了和Velocity src 下源码一样的AST代码结构,由于我修改的仅仅是 Reference 的语法,因此生成的代码文件中我只保留了 ParserTokenManager.java 文件(用于替换src的该文件,这个文件代码很多,又没有发现可以上传附件的地方,因此代码就不贴出了),其它的文件使用原始 src 下的文件(注意:原始src下的AST文件大部分被后期手工修改完善,因此不能完全使用新编译生成的AST代码文件)。

同时对原始的 ASTReference.java 的getRoot() 方法中的代码进行扩充,支持 $# 的处理:

if (t.image.startsWith("$!") || t.image.startsWith("$#")) {

referenceType = QUIET_REFERENCE;

/*

*  only if we aren't escaped do we want to null the output

*/

if (!escaped) nullString = "";

if (t.image.startsWith("$!{") || t.image.startsWith("$#{"))

{

/*

*  ex : $!{provider.Title} OR $#{provider.Title}

*/

return t.next.image;

}

else

{

/*

*  ex : $!provider.Title OR $#provider.Title

*/

return t.image.substring(2);

}

}

然后测试,一切OK,这样 Velocity 就多了一个语法:$#[ { ] ... [ } ]支持。

哦,还没完呢,呵呵,此时 $#{} 仅仅和 $!{} 一样,下面开始对这2个命令进行处理:$!{} -- 转义输出, $#{} -- 不转义输出:

为了不修改原始核心代码,方便扩展,这里采用扩展 Velocity 提供的 ReferenceInsertionEventHandler 接口实现进行(类似EscapeHtmlReference),直接上代码和配置:

import org.apache.velocity.app.event.ReferenceInsertionEventHandler;

import org.apache.velocity.runtime.RuntimeServices;

import org.apache.velocity.util.RuntimeServicesAware;

/**

* HTML转义输出

*/

public class VelocityEscapeHtmlOutput

implements ReferenceInsertionEventHandler, RuntimeServicesAware

{

private RuntimeServices rs = null;

public Object referenceInsert(String reference, Object value)

{

// 呵呵,这里... 凡是以 $#开头的reference,其值直接返回(^_^)

if(reference.startsWith("$#")) {

return value;

}

// 其它默认转义

return escapeHtml(value);

}

public void setRuntimeServices(RuntimeServices rs)

{

this.rs = rs;

}

protected RuntimeServices getRuntimeServices()

{

return this.rs;

}

/**

* 转义HTML字符串

* @param str

* @return

*/

private static Object escapeHtml(Object value)

{

if(value == null)

{

return null;

}

if(!(value instanceof String))

{

return value;

}

String str = value.toString();

StringBuilder sb = new StringBuilder(str.length() + 30);

for(int i = 0, len = str.length(); i 

{

char c = str.charAt(i);

// 去除不可见字符

if((int)c 

{

continue;

}

switch(c)

{

case '

sb.append("<");

break;

case '>':

sb.append(">");

break;

case '&':

sb.append("&");

break;

case '"':

sb.append(""");

break;

case '\'':

sb.append("'");

break;

case '/':

sb.append("/");

break;

default:

sb.append(c);

break;

}

}

str = null;

return sb.toString();

}

}

Velocity 提供的EventHandler接口需要在 velocity.properties中进行配置才能生效:

eventhandler.referenceinsertion.class = com.xxx.VelocityEscapeHtmlOutput

到此为止,就完成了对 Velocity 的 Reference 语法改造了!尽情享用吧!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_29383401/article/details/117846701

智能推荐

python简易爬虫v1.0-程序员宅基地

文章浏览阅读1.8k次,点赞4次,收藏6次。python简易爬虫v1.0作者:William Ma (the_CoderWM)进阶python的首秀,大部分童鞋肯定是做个简单的爬虫吧,众所周知,爬虫需要各种各样的第三方库,例如scrapy, bs4, requests, urllib3等等。此处,我们先从最简单的爬虫开始。首先,我们需要安装两个第三方库:requests和bs4。在cmd中输入以下代码:pip install requestspip install bs4等安装成功后,就可以进入pycharm来写爬虫了。爬

安装flask后vim出现:error detected while processing /home/zww/.vim/ftplugin/python/pyflakes.vim:line 28_freetorn.vim-程序员宅基地

文章浏览阅读2.6k次。解决方法:解决方法可以去github重新下载一个pyflakes.vim。执行如下命令git clone --recursive git://github.com/kevinw/pyflakes-vim.git然后进入git克降目录,./pyflakes-vim/ftplugin,通过如下命令将python目录下的所有文件复制到~/.vim/ftplugin目录下即可。cp -R ...._freetorn.vim

HIT CSAPP大作业:程序人生—Hello‘s P2P-程序员宅基地

文章浏览阅读210次,点赞7次,收藏3次。本文简述了hello.c源程序的预处理、编译、汇编、链接和运行的主要过程,以及hello程序的进程管理、存储管理与I/O管理,通过hello.c这一程序周期的描述,对程序的编译、加载、运行有了初步的了解。_hit csapp

18个顶级人工智能平台-程序员宅基地

文章浏览阅读1w次,点赞2次,收藏27次。来源:机器人小妹  很多时候企业拥有重复,乏味且困难的工作流程,这些流程往往会减慢生产速度并增加运营成本。为了降低生产成本,企业别无选择,只能自动化某些功能以降低生产成本。  通过数字化..._人工智能平台

electron热加载_electron-reloader-程序员宅基地

文章浏览阅读2.2k次。热加载能够在每次保存修改的代码后自动刷新 electron 应用界面,而不必每次去手动操作重新运行,这极大的提升了开发效率。安装 electron 热加载插件热加载虽然很方便,但是不是每个 electron 项目必须的,所以想要舒服的开发 electron 就只能给 electron 项目单独的安装热加载插件[electron-reloader]:// 在项目的根目录下安装 electron-reloader,国内建议使用 cnpm 代替 npmnpm install electron-relo._electron-reloader

android 11.0 去掉recovery模式UI页面的选项_android recovery 删除 部分菜单-程序员宅基地

文章浏览阅读942次。在11.0 进行定制化开发,会根据需要去掉recovery模式的一些选项 就是在device.cpp去掉一些选项就可以了。_android recovery 删除 部分菜单

随便推点

echart省会流向图(物流运输、地图)_java+echart地图+物流跟踪-程序员宅基地

文章浏览阅读2.2k次,点赞2次,收藏6次。继续上次的echart博客,由于省会流向图是从echart画廊中直接取来的。所以直接上代码<!DOCTYPE html><html><head> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no" /&_java+echart地图+物流跟踪

Ceph源码解析:读写流程_ceph 发送数据到其他副本的源码-程序员宅基地

文章浏览阅读1.4k次。一、OSD模块简介1.1 消息封装:在OSD上发送和接收信息。cluster_messenger -与其它OSDs和monitors沟通client_messenger -与客户端沟通1.2 消息调度:Dispatcher类,主要负责消息分类1.3 工作队列:1.3.1 OpWQ: 处理ops(从客户端)和sub ops(从其他的OSD)。运行在op_tp线程池。1...._ceph 发送数据到其他副本的源码

进程调度(一)——FIFO算法_进程调度fifo算法代码-程序员宅基地

文章浏览阅读7.9k次,点赞3次,收藏22次。一 定义这是最早出现的置换算法。该算法总是淘汰最先进入内存的页面,即选择在内存中驻留时间最久的页面予以淘汰。该算法实现简单,只需把一个进程已调入内存的页面,按先后次序链接成一个队列,并设置一个指针,称为替换指针,使它总是指向最老的页面。但该算法与进程实际运行的规律不相适应,因为在进程中,有些页面经常被访问,比如,含有全局变量、常用函数、例程等的页面,FIFO 算法并不能保证这些页面不被淘汰。这里,我_进程调度fifo算法代码

mysql rownum写法_mysql应用之类似oracle rownum写法-程序员宅基地

文章浏览阅读133次。rownum是oracle才有的写法,rownum在oracle中可以用于取第一条数据,或者批量写数据时限定批量写的数量等mysql取第一条数据写法SELECT * FROM t order by id LIMIT 1;oracle取第一条数据写法SELECT * FROM t where rownum =1 order by id;ok,上面是mysql和oracle取第一条数据的写法对比,不过..._mysql 替换@rownum的写法

eclipse安装教程_ecjelm-程序员宅基地

文章浏览阅读790次,点赞3次,收藏4次。官网下载下载链接:http://www.eclipse.org/downloads/点击Download下载完成后双击运行我选择第2个,看自己需要(我选择企业级应用,如果只是单纯学习java选第一个就行)进入下一步后选择jre和安装路径修改jvm/jre的时候也可以选择本地的(点后面的文件夹进去),但是我们没有11版本的,所以还是用他的吧选择接受安装中安装过程中如果有其他界面弹出就点accept就行..._ecjelm

Linux常用网络命令_ifconfig 删除vlan-程序员宅基地

文章浏览阅读245次。原文链接:https://linux.cn/article-7801-1.htmlifconfigping &lt;IP地址&gt;:发送ICMP echo消息到某个主机traceroute &lt;IP地址&gt;:用于跟踪IP包的路由路由:netstat -r: 打印路由表route add :添加静态路由路径routed:控制动态路由的BSD守护程序。运行RIP路由协议gat..._ifconfig 删除vlan