/* * Copyright 2014 Attila Szegedi, Daniel Dekany, Jonathan Revusky * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package freemarker.template.utility; import java.io.IOException; import java.io.Writer; import java.util.Map; import freemarker.template.TemplateTransformModel; /** * Performs an XML escaping of a given template fragment. Specifically, * <tt><</tt> <tt>></tt> <tt>"</tt> <tt>'</tt> and <tt>&</tt> are all turned into entity references. * * <p>An instance of this transform is initially visible as shared * variable called <tt>xml_escape</tt>.</p> */ public class XmlEscape implements TemplateTransformModel { private static final char[] LT = "<".toCharArray(); private static final char[] GT = ">".toCharArray(); private static final char[] AMP = "&".toCharArray(); private static final char[] QUOT = """.toCharArray(); private static final char[] APOS = "'".toCharArray(); public Writer getWriter(final Writer out, Map args) { return new Writer() { @Override public void write(int c) throws IOException { switch(c) { case '<': out.write(LT, 0, 4); break; case '>': out.write(GT, 0, 4); break; case '&': out.write(AMP, 0, 5); break; case '"': out.write(QUOT, 0, 6); break; case '\'': out.write(APOS, 0, 6); break; default: out.write(c); } } @Override public void write(char cbuf[], int off, int len) throws IOException { int lastoff = off; int lastpos = off + len; for (int i = off; i < lastpos; i++) { switch (cbuf[i]) { case '<': out.write(cbuf, lastoff, i - lastoff); out.write(LT, 0, 4); lastoff = i + 1; break; case '>': out.write(cbuf, lastoff, i - lastoff); out.write(GT, 0, 4); lastoff = i + 1; break; case '&': out.write(cbuf, lastoff, i - lastoff); out.write(AMP, 0, 5); lastoff = i + 1; break; case '"': out.write(cbuf, lastoff, i - lastoff); out.write(QUOT, 0, 6); lastoff = i + 1; break; case '\'': out.write(cbuf, lastoff, i - lastoff); out.write(APOS, 0, 6); lastoff = i + 1; break; } } int remaining = lastpos - lastoff; if (remaining > 0) { out.write(cbuf, lastoff, remaining); } } @Override public void flush() throws IOException { out.flush(); } @Override public void close() { } }; } }