package org.jeecg.common.util.dynamic.db; import freemarker.cache.StringTemplateLoader; import freemarker.core.ParseException; import freemarker.template.Configuration; import freemarker.template.Template; import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang3.StringUtils; import org.jeecgframework.codegenerate.generate.util.SimpleFormat; import java.io.StringWriter; import java.util.Map; import java.util.regex.Pattern; /** * @author 赵俊夫 * @version V1.0 * @Title:FreemarkerHelper * @description:Freemarker引擎协助类 * @date Jul 5, 2013 2:58:29 PM */ @Slf4j public class FreemarkerParseFactory { private static final String ENCODE = "utf-8"; /** * 参数格式化工具类 */ private static final String MINI_DAO_FORMAT = "DaoFormat"; /** * 文件缓存 */ private static final Configuration _tplConfig = new Configuration(); /** * SQL 缓存 */ private static final Configuration _sqlConfig = new Configuration(); private static StringTemplateLoader stringTemplateLoader = new StringTemplateLoader(); // 使用内嵌的(?ms)打开单行和多行模式 private final static Pattern p = Pattern .compile("(?ms)/\\*.*?\\*/|^\\s*//.*?$"); static { _tplConfig.setClassForTemplateLoading( new FreemarkerParseFactory().getClass(), "/"); _tplConfig.setNumberFormat("0.#####################"); _sqlConfig.setTemplateLoader(stringTemplateLoader); _sqlConfig.setNumberFormat("0.#####################"); //classic_compatible设置,解决报空指针错误 _sqlConfig.setClassicCompatible(true); } /** * 判断模板是否存在 * * @throws Exception */ public static boolean isExistTemplate(String tplName) throws Exception { try { Template mytpl = _tplConfig.getTemplate(tplName, "UTF-8"); if (mytpl == null) { return false; } } catch (Exception e) { //update-begin--Author:scott Date:20180320 for:解决问题 - 错误提示sql文件不存在,实际问题是sql freemarker用法错误----- if (e instanceof ParseException) { log.error(e.getMessage(), e.fillInStackTrace()); throw new Exception(e); } log.debug("----isExistTemplate----" + e.toString()); //update-end--Author:scott Date:20180320 for:解决问题 - 错误提示sql文件不存在,实际问题是sql freemarker用法错误------ return false; } return true; } /** * 解析ftl模板 * * @param tplName 模板名 * @param paras 参数 * @return */ public static String parseTemplate(String tplName, Map paras) { try { log.debug(" minidao sql templdate : " + tplName); StringWriter swriter = new StringWriter(); Template mytpl = _tplConfig.getTemplate(tplName, ENCODE); if (paras.containsKey(MINI_DAO_FORMAT)) { throw new RuntimeException("DaoFormat 是 minidao 保留关键字,不允许使用 ,请更改参数定义!"); } paras.put(MINI_DAO_FORMAT, new SimpleFormat()); mytpl.process(paras, swriter); String sql = getSqlText(swriter.toString()); paras.remove(MINI_DAO_FORMAT); return sql; } catch (Exception e) { log.error(e.getMessage(), e.fillInStackTrace()); log.error("发送一次的模板key:{ " + tplName + " }"); //System.err.println(e.getMessage()); //System.err.println("模板名:{ "+ tplName +" }"); throw new RuntimeException("解析SQL模板异常"); } } /** * 解析ftl * * @param tplContent 模板内容 * @param paras 参数 * @return String 模板解析后内容 */ public static String parseTemplateContent(String tplContent, Map paras) { try { StringWriter swriter = new StringWriter(); if (stringTemplateLoader.findTemplateSource("sql_" + tplContent.hashCode()) == null) { stringTemplateLoader.putTemplate("sql_" + tplContent.hashCode(), tplContent); } Template mytpl = _sqlConfig.getTemplate("sql_" + tplContent.hashCode(), ENCODE); if (paras.containsKey(MINI_DAO_FORMAT)) { throw new RuntimeException("DaoFormat 是 minidao 保留关键字,不允许使用 ,请更改参数定义!"); } paras.put(MINI_DAO_FORMAT, new SimpleFormat()); mytpl.process(paras, swriter); String sql = getSqlText(swriter.toString()); paras.remove(MINI_DAO_FORMAT); return sql; } catch (Exception e) { log.error(e.getMessage(), e.fillInStackTrace()); log.error("发送一次的模板key:{ " + tplContent + " }"); //System.err.println(e.getMessage()); //System.err.println("模板内容:{ "+ tplContent +" }"); throw new RuntimeException("解析SQL模板异常"); } } /** * 除去无效字段,去掉注释 不然批量处理可能报错 去除无效的等于 */ private static String getSqlText(String sql) { // 将注释替换成"" sql = p.matcher(sql).replaceAll(""); sql = sql.replaceAll("\\n", " ").replaceAll("\\t", " ") .replaceAll("\\s{1,}", " ").trim(); // 去掉 最后是 where这样的问题 if (sql.endsWith("where") || sql.endsWith("where ")) { sql = sql.substring(0, sql.lastIndexOf("where")); } // 去掉where and 这样的问题 int index = 0; while ((index = StringUtils.indexOfIgnoreCase(sql, "where and", index)) != -1) { sql = sql.substring(0, index + 5) + sql.substring(index + 9, sql.length()); } // 去掉 , where 这样的问题 index = 0; while ((index = StringUtils.indexOfIgnoreCase(sql, ", where", index)) != -1) { sql = sql.substring(0, index) + sql.substring(index + 1, sql.length()); } // 去掉 最后是 ,这样的问题 if (sql.endsWith(",") || sql.endsWith(", ")) { sql = sql.substring(0, sql.lastIndexOf(",")); } return sql; } }