All Downloads are FREE. Search and download functionalities are using the official Maven repository.

retrofit.processor.TemplateVars Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2014 Google, Inc.
 *
 * 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 retrofit.processor;

import com.google.common.collect.ImmutableList;

import org.apache.velocity.VelocityContext;
import org.apache.velocity.runtime.log.NullLogChute;
import org.apache.velocity.runtime.RuntimeConstants;
import org.apache.velocity.runtime.RuntimeInstance;
import org.apache.velocity.runtime.parser.ParseException;
import org.apache.velocity.runtime.parser.node.SimpleNode;

import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import org.apache.velocity.runtime.resource.ResourceCacheImpl;

/**
 * A template and a set of variables to be substituted into that template. A concrete subclass of
 * this class defines a set of fields that are template variables, and an implementation of the
 * {@link #parsedTemplate()} method which is the template to substitute them into. Once the values
 * of the fields have been assigned, the {@link #toText()} method returns the result of substituting
 * them into the template.
 *
 * 

The subclass must be a direct subclass of this class. Fields cannot be static unless they are * also final. They cannot be private, though they can be package-private if the class is in the * same package as this class. They cannot be primitive or null, so that there is a clear indication * when a field has not been set. * * @author Éamonn McManus */ abstract class TemplateVars { abstract SimpleNode parsedTemplate(); private static final RuntimeInstance velocityRuntimeInstance = new RuntimeInstance(); static { // Ensure that $undefinedvar will produce an exception rather than outputting $undefinedvar. velocityRuntimeInstance.setProperty(RuntimeConstants.RUNTIME_REFERENCES_STRICT, "true"); velocityRuntimeInstance.setProperty(RuntimeConstants.RUNTIME_LOG_LOGSYSTEM_CLASS, new NullLogChute()); velocityRuntimeInstance.setProperty(RuntimeConstants.RESOURCE_MANAGER_CACHE_CLASS, ResourceCacheImpl.class.getName()); // Setting ResourceCacheImpl is should not be necessary since that is the default value, but // ensures that Maven shading sees that Apache Commons classes referenced from ResourceCacheImpl // are indeed referenced and cannot be removed during minimization. // Disable any logging that Velocity might otherwise see fit to do. velocityRuntimeInstance.setProperty(RuntimeConstants.RUNTIME_LOG_LOGSYSTEM, new NullLogChute()); // Velocity likes its "managers", LogManager and ResourceManager, which it loads through the // context class loader. If that loader can see another copy of Velocity then that will lead // to hard-to-diagnose exceptions during initialization. Thread currentThread = Thread.currentThread(); ClassLoader oldContextLoader = currentThread.getContextClassLoader(); try { currentThread.setContextClassLoader(TemplateVars.class.getClassLoader()); velocityRuntimeInstance.init(); } finally { currentThread.setContextClassLoader(oldContextLoader); } } private final ImmutableList fields; TemplateVars() { if (getClass().getSuperclass() != TemplateVars.class) { throw new IllegalArgumentException("Class must extend TemplateVars directly"); } ImmutableList.Builder fields = ImmutableList.builder(); Field[] declaredFields = getClass().getDeclaredFields(); for (Field field : declaredFields) { if (field.isSynthetic() || isStaticFinal(field)) { continue; } if (Modifier.isPrivate(field.getModifiers())) { throw new IllegalArgumentException("Field cannot be private: " + field); } if (Modifier.isStatic(field.getModifiers())) { throw new IllegalArgumentException("Field cannot be static unless also final: " + field); } if (field.getType().isPrimitive()) { throw new IllegalArgumentException("Field cannot be primitive: " + field); } fields.add(field); } this.fields = fields.build(); } /** * Returns the result of substituting the variables defined by the fields of this class * (a concrete subclass of TemplateVars) into the template returned by {@link #parsedTemplate()}. */ String toText() { VelocityContext velocityContext = toVelocityContext(); StringWriter writer = new StringWriter(); SimpleNode parsedTemplate = parsedTemplate(); boolean rendered = velocityRuntimeInstance.render( velocityContext, writer, parsedTemplate.getTemplateName(), parsedTemplate); if (!rendered) { // I don't know when this happens. Usually you get an exception during rendering. throw new IllegalArgumentException("Template rendering failed"); } return writer.toString(); } private VelocityContext toVelocityContext() { VelocityContext velocityContext = new VelocityContext(); for (Field field : fields) { Object value = fieldValue(field, this); if (value == null) { throw new IllegalArgumentException("Field cannot be null (was it set?): " + field); } Object old = velocityContext.put(field.getName(), value); if (old != null) { throw new IllegalArgumentException("Two fields called " + field.getName() + "?!"); } } return velocityContext; } static SimpleNode parsedTemplateForResource(String templateStr, String resourceName) { try { return velocityRuntimeInstance.parse(templateStr, resourceName); } catch (ParseException e) { throw new AssertionError(e); } } static SimpleNode parsedTemplateForResource(String resourceName) { InputStream in = RetrofitTemplateVars.class.getResourceAsStream(resourceName); if (in == null) { throw new IllegalArgumentException("Could not find resource: " + resourceName); } try { Reader reader = new InputStreamReader(in, "UTF-8"); return velocityRuntimeInstance.parse(reader, resourceName); } catch (UnsupportedEncodingException e) { throw new AssertionError(e); } catch (ParseException e) { throw new AssertionError(e); } } private static Object fieldValue(Field field, Object container) { try { return field.get(container); } catch (IllegalAccessException e) { throw new RuntimeException(e); } } private static boolean isStaticFinal(Field field) { int modifiers = field.getModifiers(); return Modifier.isStatic(modifiers) && Modifier.isFinal(modifiers); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy