Consistent support for CompilationCustomizers as well as custom CompilerConfiguration
Issue: SPR-14585
This commit is contained in:
		
							parent
							
								
									a1b167a988
								
							
						
					
					
						commit
						6a0d9d3d97
					
				| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2015 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2016 the original author or authors.
 | 
			
		||||
 *
 | 
			
		||||
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
 * you may not use this file except in compliance with the License.
 | 
			
		||||
| 
						 | 
				
			
			@ -22,6 +22,8 @@ import java.util.Map;
 | 
			
		|||
import groovy.lang.Binding;
 | 
			
		||||
import groovy.lang.GroovyRuntimeException;
 | 
			
		||||
import groovy.lang.GroovyShell;
 | 
			
		||||
import org.codehaus.groovy.control.CompilerConfiguration;
 | 
			
		||||
import org.codehaus.groovy.control.customizers.CompilationCustomizer;
 | 
			
		||||
 | 
			
		||||
import org.springframework.beans.factory.BeanClassLoaderAware;
 | 
			
		||||
import org.springframework.scripting.ScriptCompilationException;
 | 
			
		||||
| 
						 | 
				
			
			@ -40,6 +42,8 @@ public class GroovyScriptEvaluator implements ScriptEvaluator, BeanClassLoaderAw
 | 
			
		|||
 | 
			
		||||
	private ClassLoader classLoader;
 | 
			
		||||
 | 
			
		||||
	private CompilerConfiguration compilerConfiguration = new CompilerConfiguration();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Construct a new GroovyScriptEvaluator.
 | 
			
		||||
| 
						 | 
				
			
			@ -56,6 +60,35 @@ public class GroovyScriptEvaluator implements ScriptEvaluator, BeanClassLoaderAw
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Set a custom compiler configuration for this evaluator.
 | 
			
		||||
	 * @since 4.3.3
 | 
			
		||||
	 * @see #setCompilationCustomizers
 | 
			
		||||
	 */
 | 
			
		||||
	public void setCompilerConfiguration(CompilerConfiguration compilerConfiguration) {
 | 
			
		||||
		this.compilerConfiguration =
 | 
			
		||||
				(compilerConfiguration != null ? compilerConfiguration : new CompilerConfiguration());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Return this evaluator's compiler configuration (never {@code null}).
 | 
			
		||||
	 * @since 4.3.3
 | 
			
		||||
	 * @see #setCompilerConfiguration
 | 
			
		||||
	 */
 | 
			
		||||
	public CompilerConfiguration getCompilerConfiguration() {
 | 
			
		||||
		return this.compilerConfiguration;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Set one or more customizers to be applied to this evaluator's compiler configuration.
 | 
			
		||||
	 * <p>Note that this modifies the shared compiler configuration held by this evaluator.
 | 
			
		||||
	 * @since 4.3.3
 | 
			
		||||
	 * @see #setCompilerConfiguration
 | 
			
		||||
	 */
 | 
			
		||||
	public void setCompilationCustomizers(CompilationCustomizer... compilationCustomizers) {
 | 
			
		||||
		this.compilerConfiguration.addCompilationCustomizers(compilationCustomizers);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public void setBeanClassLoader(ClassLoader classLoader) {
 | 
			
		||||
		this.classLoader = classLoader;
 | 
			
		||||
| 
						 | 
				
			
			@ -69,7 +102,8 @@ public class GroovyScriptEvaluator implements ScriptEvaluator, BeanClassLoaderAw
 | 
			
		|||
 | 
			
		||||
	@Override
 | 
			
		||||
	public Object evaluate(ScriptSource script, Map<String, Object> arguments) {
 | 
			
		||||
		GroovyShell groovyShell = new GroovyShell(this.classLoader, new Binding(arguments));
 | 
			
		||||
		GroovyShell groovyShell = new GroovyShell(
 | 
			
		||||
				this.classLoader, new Binding(arguments), this.compilerConfiguration);
 | 
			
		||||
		try {
 | 
			
		||||
			String filename = (script instanceof ResourceScriptSource ?
 | 
			
		||||
					((ResourceScriptSource) script).getResource().getFilename() : null);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -36,6 +36,7 @@ import org.springframework.scripting.ScriptFactory;
 | 
			
		|||
import org.springframework.scripting.ScriptSource;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
import org.springframework.util.ClassUtils;
 | 
			
		||||
import org.springframework.util.ObjectUtils;
 | 
			
		||||
import org.springframework.util.ReflectionUtils;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
| 
						 | 
				
			
			@ -104,23 +105,39 @@ public class GroovyScriptFactory implements ScriptFactory, BeanFactoryAware, Bea
 | 
			
		|||
		this.groovyObjectCustomizer = groovyObjectCustomizer;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Create a new GroovyScriptFactory for the given script source,
 | 
			
		||||
	 * specifying a strategy interface that can create a custom MetaClass
 | 
			
		||||
	 * to supply missing methods and otherwise change the behavior of the object.
 | 
			
		||||
	 * @param scriptSourceLocator a locator that points to the source of the script.
 | 
			
		||||
	 * Interpreted by the post-processor that actually creates the script.
 | 
			
		||||
	 * @param compilerConfiguration a custom compiler configuration to be applied
 | 
			
		||||
	 * to the GroovyClassLoader (may be {@code null})
 | 
			
		||||
	 * @since 4.3.3
 | 
			
		||||
	 * @see GroovyClassLoader#GroovyClassLoader(ClassLoader, CompilerConfiguration)
 | 
			
		||||
	 */
 | 
			
		||||
	public GroovyScriptFactory(String scriptSourceLocator, CompilerConfiguration compilerConfiguration) {
 | 
			
		||||
		this(scriptSourceLocator);
 | 
			
		||||
		this.compilerConfiguration = compilerConfiguration;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Create a new GroovyScriptFactory for the given script source,
 | 
			
		||||
	 * specifying a strategy interface that can customize Groovy's compilation
 | 
			
		||||
	 * process within the underlying GroovyClassLoader.
 | 
			
		||||
	 * @param scriptSourceLocator a locator that points to the source of the script.
 | 
			
		||||
	 * Interpreted by the post-processor that actually creates the script.
 | 
			
		||||
	 * @param compilationCustomizer a customizer to be applied to the GroovyClassLoader
 | 
			
		||||
	 * compiler configuration (may be {@code null})
 | 
			
		||||
	 * @param compilationCustomizers one or more customizers to be applied to the
 | 
			
		||||
	 * GroovyClassLoader compiler configuration
 | 
			
		||||
	 * @since 4.3.3
 | 
			
		||||
	 * @see CompilerConfiguration#addCompilationCustomizers
 | 
			
		||||
	 * @see org.codehaus.groovy.control.customizers.ImportCustomizer
 | 
			
		||||
	 */
 | 
			
		||||
	public GroovyScriptFactory(String scriptSourceLocator, CompilationCustomizer compilationCustomizer) {
 | 
			
		||||
	public GroovyScriptFactory(String scriptSourceLocator, CompilationCustomizer... compilationCustomizers) {
 | 
			
		||||
		this(scriptSourceLocator);
 | 
			
		||||
		if (compilationCustomizer != null) {
 | 
			
		||||
		if (!ObjectUtils.isEmpty(compilationCustomizers)) {
 | 
			
		||||
			this.compilerConfiguration = new CompilerConfiguration();
 | 
			
		||||
			this.compilerConfiguration.addCompilationCustomizers(compilationCustomizer);
 | 
			
		||||
			this.compilerConfiguration.addCompilationCustomizers(compilationCustomizers);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -19,6 +19,7 @@ package org.springframework.scripting.groovy;
 | 
			
		|||
import java.util.HashMap;
 | 
			
		||||
import java.util.Map;
 | 
			
		||||
 | 
			
		||||
import org.codehaus.groovy.control.customizers.ImportCustomizer;
 | 
			
		||||
import org.junit.Test;
 | 
			
		||||
 | 
			
		||||
import org.springframework.core.io.ClassPathResource;
 | 
			
		||||
| 
						 | 
				
			
			@ -58,6 +59,26 @@ public class GroovyScriptEvaluatorTests {
 | 
			
		|||
		assertEquals(6, result);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void testGroovyScriptWithCompilerConfiguration() {
 | 
			
		||||
		GroovyScriptEvaluator evaluator = new GroovyScriptEvaluator();
 | 
			
		||||
		MyBytecodeProcessor processor = new MyBytecodeProcessor();
 | 
			
		||||
		evaluator.getCompilerConfiguration().setBytecodePostprocessor(processor);
 | 
			
		||||
		Object result = evaluator.evaluate(new StaticScriptSource("return 3 * 2"));
 | 
			
		||||
		assertEquals(6, result);
 | 
			
		||||
		assertTrue(processor.processed.contains("Script1"));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void testGroovyScriptWithImportCustomizer() {
 | 
			
		||||
		GroovyScriptEvaluator evaluator = new GroovyScriptEvaluator();
 | 
			
		||||
		ImportCustomizer importCustomizer = new ImportCustomizer();
 | 
			
		||||
		importCustomizer.addStarImports("org.springframework.util");
 | 
			
		||||
		evaluator.setCompilationCustomizers(importCustomizer);
 | 
			
		||||
		Object result = evaluator.evaluate(new StaticScriptSource("return ResourceUtils.CLASSPATH_URL_PREFIX"));
 | 
			
		||||
		assertEquals("classpath:", result);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void testGroovyScriptFromStringUsingJsr223() {
 | 
			
		||||
		StandardScriptEvaluator evaluator = new StandardScriptEvaluator();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -470,6 +470,8 @@ public class GroovyScriptFactoryTests {
 | 
			
		|||
		ApplicationContext ctx = new ClassPathXmlApplicationContext("groovy-with-xsd.xml", getClass());
 | 
			
		||||
		Map<?, Messenger> beans = ctx.getBeansOfType(Messenger.class);
 | 
			
		||||
		assertEquals(4, beans.size());
 | 
			
		||||
		assertTrue(ctx.getBean(MyBytecodeProcessor.class).processed.contains(
 | 
			
		||||
				"org.springframework.scripting.groovy.GroovyMessenger2"));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,37 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2016 the original author or authors.
 | 
			
		||||
 *
 | 
			
		||||
 * 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.springframework.scripting.groovy;
 | 
			
		||||
 | 
			
		||||
import java.util.HashSet;
 | 
			
		||||
import java.util.Set;
 | 
			
		||||
 | 
			
		||||
import org.codehaus.groovy.control.BytecodeProcessor;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * @author Juergen Hoeller
 | 
			
		||||
 */
 | 
			
		||||
public class MyBytecodeProcessor implements BytecodeProcessor {
 | 
			
		||||
 | 
			
		||||
	public final Set<String> processed = new HashSet<String>();
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public byte[] processBytecode(String name, byte[] original) {
 | 
			
		||||
		this.processed.add(name);
 | 
			
		||||
		return original;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -39,8 +39,8 @@ class GroovyCalculator implements Calculator {
 | 
			
		|||
	<lang:groovy id="groovyObjectCustomizer" customizer-ref="importCustomizer">
 | 
			
		||||
		<lang:inline-script><![CDATA[
 | 
			
		||||
public class TestCustomizer implements GroovyObjectCustomizer {
 | 
			
		||||
	public void customize(GroovyObject o) {
 | 
			
		||||
		println "customizing ${o}.."
 | 
			
		||||
	public void customize(GroovyObject go) {
 | 
			
		||||
		println "customizing ${go}..."
 | 
			
		||||
	}
 | 
			
		||||
}]]>
 | 
			
		||||
		</lang:inline-script>
 | 
			
		||||
| 
						 | 
				
			
			@ -53,8 +53,15 @@ public class TestCustomizer implements GroovyObjectCustomizer {
 | 
			
		|||
		<lang:property name="message" value="Hello World!"/>
 | 
			
		||||
	</lang:groovy>
 | 
			
		||||
 | 
			
		||||
	<lang:groovy script-source="classpath:org/springframework/scripting/groovy/Messenger.groovy">
 | 
			
		||||
	<lang:groovy script-source="classpath:org/springframework/scripting/groovy/Messenger.groovy"
 | 
			
		||||
			customizer-ref="compilerConfiguration">
 | 
			
		||||
		<lang:property name="message" value="Hello World!"/>
 | 
			
		||||
	</lang:groovy>
 | 
			
		||||
 | 
			
		||||
	<bean id="compilerConfiguration" class="org.codehaus.groovy.control.CompilerConfiguration">
 | 
			
		||||
		<property name="bytecodePostprocessor" ref="bytecodeProcessor"/>
 | 
			
		||||
	</bean>
 | 
			
		||||
 | 
			
		||||
	<bean id="bytecodeProcessor" class="org.springframework.scripting.groovy.MyBytecodeProcessor"/>
 | 
			
		||||
 | 
			
		||||
</beans>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue