org.mapstruct.ap.internal.writer.ModelWriter Maven / Gradle / Ivy
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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 org.mapstruct.ap.internal.writer;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.net.URL;
import java.net.URLConnection;
import java.nio.charset.Charset;
import java.util.HashMap;
import java.util.Map;
import javax.tools.FileObject;
import org.mapstruct.ap.internal.writer.Writable.Context;
import freemarker.cache.StrongCacheStorage;
import freemarker.cache.TemplateLoader;
import freemarker.log.Logger;
import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapper;
/**
* Writes Java source files based on given mapper models, using a FreeMarker
* template.
*
* @author Gunnar Morling
*/
public class ModelWriter {
/**
* FreeMarker configuration. As per the documentation, thread-safe if not
* altered after original initialization
*/
private static final Configuration CONFIGURATION;
static {
try {
Logger.selectLoggerLibrary( Logger.LIBRARY_NONE );
}
catch ( ClassNotFoundException e ) {
throw new RuntimeException( e );
}
CONFIGURATION = new Configuration( Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS );
CONFIGURATION.setTemplateLoader( new SimpleClasspathLoader() );
CONFIGURATION.setObjectWrapper( new DefaultObjectWrapper( Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS ) );
CONFIGURATION.setSharedVariable(
"includeModel",
new ModelIncludeDirective( CONFIGURATION )
);
// do not refresh/gc the cached templates, as we never change them at runtime
CONFIGURATION.setCacheStorage( new StrongCacheStorage() );
CONFIGURATION.setTemplateUpdateDelay( Integer.MAX_VALUE );
CONFIGURATION.setLocalizedLookup( false );
}
public void writeModel(FileObject sourceFile, Writable model) {
try {
BufferedWriter writer = new BufferedWriter( new IndentationCorrectingWriter( sourceFile.openWriter() ) );
Map, Object> values = new HashMap, Object>();
values.put( Configuration.class, CONFIGURATION );
model.write( new DefaultModelElementWriterContext( values ), writer );
writer.flush();
writer.close();
}
catch ( RuntimeException e ) {
throw e;
}
catch ( Exception e ) {
throw new RuntimeException( e );
}
}
/**
* Simplified template loader that avoids reading modification timestamps and disables the jar-file caching.
*
* @author Andreas Gudian
*/
private static final class SimpleClasspathLoader implements TemplateLoader {
@Override
public Reader getReader(Object name, String encoding) throws IOException {
URL url = getClass().getClassLoader().getResource( String.valueOf( name ) );
if ( url == null ) {
throw new IllegalStateException( name + " not found on classpath" );
}
URLConnection connection = url.openConnection();
// don't use jar-file caching, as it caused occasionally closed input streams [at least under JDK 1.8.0_25]
connection.setUseCaches( false );
InputStream is = connection.getInputStream();
return new InputStreamReader( is, Charset.forName( "UTF-8" ) );
}
@Override
public long getLastModified(Object templateSource) {
return 0;
}
@Override
public Object findTemplateSource(String name) throws IOException {
return name;
}
@Override
public void closeTemplateSource(Object templateSource) throws IOException {
}
}
/**
* {@link Context} implementation which provides access to the current FreeMarker {@link Configuration}.
*
* @author Gunnar Morling
*/
static class DefaultModelElementWriterContext implements Context {
private final Map, Object> values;
DefaultModelElementWriterContext(Map, Object> values) {
this.values = new HashMap, Object>( values );
}
@Override
@SuppressWarnings("unchecked")
public T get(Class type) {
return (T) values.get( type );
}
}
}