You are viewing a plain text version of this content. The canonical link for it is here.
Posted to notifications@freemarker.apache.org by dd...@apache.org on 2017/05/15 21:24:01 UTC

[45/51] [abbrv] [partial] incubator-freemarker git commit: Restructured project so that freemarker-test-utils depends on freemarker-core (and hence can provide common classes for testing templates, and can use utility classes defined in the core). As a c

http://git-wip-us.apache.org/repos/asf/incubator-freemarker/blob/28a276c8/freemarker-core-test/src/test/java/org/apache/freemarker/core/ObjectBuilderSettingsTest.java
----------------------------------------------------------------------
diff --git a/freemarker-core-test/src/test/java/org/apache/freemarker/core/ObjectBuilderSettingsTest.java b/freemarker-core-test/src/test/java/org/apache/freemarker/core/ObjectBuilderSettingsTest.java
new file mode 100644
index 0000000..49534fa
--- /dev/null
+++ b/freemarker-core-test/src/test/java/org/apache/freemarker/core/ObjectBuilderSettingsTest.java
@@ -0,0 +1,1499 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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.apache.freemarker.core;
+
+import static org.apache.freemarker.test.hamcerst.Matchers.*;
+import static org.hamcrest.Matchers.*;
+import static org.junit.Assert.*;
+
+import java.io.IOException;
+import java.io.Serializable;
+import java.io.Writer;
+import java.math.BigDecimal;
+import java.math.BigInteger;
+import java.nio.charset.Charset;
+import java.nio.charset.StandardCharsets;
+import java.nio.charset.UnsupportedCharsetException;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Properties;
+import java.util.TimeZone;
+
+import org.apache.freemarker.core.arithmetic.ArithmeticEngine;
+import org.apache.freemarker.core.arithmetic.impl.BigDecimalArithmeticEngine;
+import org.apache.freemarker.core.model.ObjectWrapper;
+import org.apache.freemarker.core.model.impl.DefaultObjectWrapper;
+import org.apache.freemarker.core.templateresolver.CacheStorage;
+import org.apache.freemarker.core.templateresolver.TemplateLoader;
+import org.apache.freemarker.core.templateresolver.TemplateLoaderSession;
+import org.apache.freemarker.core.templateresolver.TemplateLoadingResult;
+import org.apache.freemarker.core.templateresolver.TemplateLoadingSource;
+import org.apache.freemarker.core.templateresolver.impl.MruCacheStorage;
+import org.apache.freemarker.core.userpkg.PublicWithMixedConstructors;
+import org.junit.Test;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+
+@SuppressWarnings("boxing")
+public class ObjectBuilderSettingsTest {
+
+    @Test
+    public void newInstanceTest() throws Exception {
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(4f, res.f, 0);
+            assertFalse(res.b);
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(4f, res.f, 0);
+            assertFalse(res.b);
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(1.5, -20, 8589934592, true)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(1.5f, res.f, 0);
+            assertEquals(Integer.valueOf(-20), res.i);
+            assertEquals(8589934592l, res.l);
+            assertTrue(res.b);
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(1, true)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(2f, res.f, 0);
+            assertEquals(Integer.valueOf(1), res.i);
+            assertEquals(2l, res.l);
+            assertTrue(res.b);
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(11, 22)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(3f, res.f, 0);
+            assertEquals(Integer.valueOf(11), res.i);
+            assertEquals(22l, res.l);
+            assertFalse(res.b);
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(p1 = 1, p2 = 2, p3 = true, p4 = 's')",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(4f, res.f, 0);
+            assertFalse(res.b);
+            assertEquals(1d, res.getP1(), 0);
+            assertEquals(2, res.getP2());
+            assertTrue(res.isP3());
+            assertEquals("s", res.getP4());
+        }
+
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1("
+                    + "null, 2, p1 = 1, p2 = 2, p3 = false, p4 = null)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertNull(res.i);
+            assertEquals(2, res.l, 0);
+            assertEquals(3f, res.f, 0);
+            assertFalse(res.b);
+            assertEquals(1d, res.getP1(), 0);
+            assertEquals(2, res.getP2());
+            assertFalse(res.isP3());
+            assertNull(res.getP4());
+        }
+        
+        {
+            // Deliberately odd spacings
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "\t\torg.apache.freemarker .core  . \n"
+                    + "\tObjectBuilderSettingsTest$TestBean1(\n\r\tp1=1\n,p2=2,p3=true,p4='s'  )",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(4f, res.f, 0);
+            assertFalse(res.b);
+            assertEquals(1d, res.getP1(), 0);
+            assertEquals(2, res.getP2());
+            assertTrue(res.isP3());
+            assertEquals("s", res.getP4());
+        }
+        
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(1, true, p2 = 2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(2f, res.f, 0);
+            assertEquals(Integer.valueOf(1), res.i);
+            assertEquals(2l, res.l);
+            assertTrue(res.b);
+            assertEquals(0d, res.getP1(), 0);
+            assertEquals(2, res.getP2());
+            assertFalse(res.isP3());
+        }
+    }
+    
+    @Test
+    public void builderTest() throws Exception {
+        {
+            // `()`-les syntax:
+            TestBean2 res = (TestBean2) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertTrue(res.built);
+            assertEquals(0, res.x);
+        }
+        
+        {
+            TestBean2 res = (TestBean2) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertTrue(res.built);
+            assertEquals(0, res.x);
+        }
+        
+        {
+            TestBean2 res = (TestBean2) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(x = 1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertTrue(res.built);
+            assertEquals(1, res.x);
+        }
+        
+        {
+            TestBean2 res = (TestBean2) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertTrue(res.built);
+            assertEquals(1, res.x);
+        }
+    }
+
+    @Test
+    public void staticInstanceTest() throws Exception {
+        // ()-les syntax:
+        {
+            TestBean5 res = (TestBean5) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(0, res.i);
+            assertEquals(0, res.x);
+            assertSame(TestBean5.INSTANCE, res); //!
+        }
+        
+        {
+            TestBean5 res = (TestBean5) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(0, res.i);
+            assertEquals(0, res.x);
+            assertSame(TestBean5.INSTANCE, res); //!
+        }
+        
+        {
+            TestBean5 res = (TestBean5) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5(x = 1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(0, res.i);
+            assertEquals(1, res.x);
+            assertNotSame(TestBean5.INSTANCE, res);
+        }
+
+        {
+            TestBean5 res = (TestBean5) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5(1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(1, res.i);
+            assertEquals(0, res.x);
+            assertNotSame(TestBean5.INSTANCE, res);
+        }
+    }
+
+    @Test
+    public void stringLiteralsTest() throws Exception {
+        {
+            TestBean4 res = (TestBean4) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean4(\"\", '', s3 = r\"\", s4 = r'')",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("", res.getS1());
+            assertEquals("", res.getS2());
+            assertEquals("", res.getS3());
+            assertEquals("", res.getS4());
+        }
+        
+        {
+            TestBean4 res = (TestBean4) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean4(\"a\", 'b', s3 = r\"c\", s4 = r'd')",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("a", res.getS1());
+            assertEquals("b", res.getS2());
+            assertEquals("c", res.getS3());
+            assertEquals("d", res.getS4());
+        }
+        
+        {
+            TestBean4 res = (TestBean4) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean4(\"a'A\", 'b\"B', s3 = r\"c'C\", s4 = r'd\"D')",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("a'A", res.getS1());
+            assertEquals("b\"B", res.getS2());
+            assertEquals("c'C", res.getS3());
+            assertEquals("d\"D", res.getS4());
+        }
+        
+        {
+            TestBean4 res = (TestBean4) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean4("
+                    + "\"a\\nA\\\"a\\\\A\", 'a\\nA\\'a\\\\A', s3 = r\"a\\n\\A\", s4 = r'a\\n\\A')",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("a\nA\"a\\A", res.getS1());
+            assertEquals("a\nA'a\\A", res.getS2());
+            assertEquals("a\\n\\A", res.getS3());
+            assertEquals("a\\n\\A", res.getS4());
+        }
+    }
+
+    @Test
+    public void nestedBuilderTest() throws Exception {
+        {
+            TestBean6 res = (TestBean6) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean6("
+                    + "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(11, 22, p4 = 'foo'),"
+                    + "1,"
+                    + "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(11),"
+                    + "y=2,"
+                    + "b3=org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(x = 22)"
+                    + ")",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(Integer.valueOf(11), res.b1.i);
+            assertEquals(22, res.b1.l);
+            assertEquals("foo", res.b1.p4);
+            assertEquals(1, res.x);
+            assertEquals(11, res.b2.x);
+            assertEquals(2, res.y);
+            assertEquals(22, res.b3.x);
+            assertNull(res.b4);
+        }
+        
+        {
+            TestBean6 res = (TestBean6) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean6("
+                    + "null,"
+                    + "-1,"
+                    + "null,"
+                    + "b4=org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean6("
+                    + "   org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(11, 22, p4 = 'foo'),"
+                    + "   1,"
+                    + "   org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(11),"
+                    + "   y=2,"
+                    + "   b3=org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean2(x = 22)"
+                    + "),"
+                    + "y=2"
+                    + ")",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertNull(res.b1);
+            assertEquals(-1, res.x);
+            assertNull(res.b2);
+            assertEquals(2, res.y);
+            assertEquals(Integer.valueOf(11), res.b4.b1.i);
+            assertEquals(22, res.b4.b1.l);
+            assertEquals("foo", res.b4.b1.p4);
+            assertEquals(1, res.b4.x);
+            assertEquals(11, res.b4.b2.x);
+            assertEquals(2, res.b4.y);
+            assertEquals(22, res.b4.b3.x);
+            assertNull(res.b4.b4);
+        }
+    }
+
+    @Test
+    public void defaultObjectWrapperTest() throws Exception {
+        DefaultObjectWrapper ow = (DefaultObjectWrapper) _ObjectBuilderSettingEvaluator.eval(
+                "DefaultObjectWrapper(3.0.0)",
+                ObjectWrapper.class, false, _SettingEvaluationEnvironment.getCurrent());
+        assertEquals(Configuration.VERSION_3_0_0, ow.getIncompatibleImprovements());
+        assertFalse(ow.isExposeFields());
+    }
+
+    @Test
+    public void defaultObjectWrapperTest2() throws Exception {
+        DefaultObjectWrapper ow = (DefaultObjectWrapper) _ObjectBuilderSettingEvaluator.eval(
+                "DefaultObjectWrapper(3.0.0, exposeFields=true)",
+                ObjectWrapper.class, false, _SettingEvaluationEnvironment.getCurrent());
+        assertEquals(Configuration.VERSION_3_0_0, ow.getIncompatibleImprovements());
+        assertTrue(ow.isExposeFields());
+    }
+
+    @Test
+    public void configurationPropertiesTest() throws Exception {
+        final Configuration.Builder cfgB = new Configuration.Builder(Configuration.getVersion());
+        
+        {
+            Properties props = new Properties();
+            props.setProperty(MutableProcessingConfiguration.OBJECT_WRAPPER_KEY,
+                    "org.apache.freemarker.core.model.impl.DefaultObjectWrapper(3.0.0)");
+            props.setProperty(MutableProcessingConfiguration.ARITHMETIC_ENGINE_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyArithmeticEngine");
+            props.setProperty(MutableProcessingConfiguration.TEMPLATE_EXCEPTION_HANDLER_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyTemplateExceptionHandler");
+            props.setProperty(Configuration.ExtendableBuilder.CACHE_STORAGE_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyCacheStorage()");
+            props.setProperty(MutableProcessingConfiguration.NEW_BUILTIN_CLASS_RESOLVER_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyNewBuiltinClassResolver()");
+            props.setProperty(Configuration.ExtendableBuilder.SOURCE_ENCODING_KEY, "utf-8");
+            props.setProperty(Configuration.ExtendableBuilder.TEMPLATE_LOADER_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyTemplateLoader()");
+            cfgB.setSettings(props);
+            assertEquals(DefaultObjectWrapper.class, cfgB.getObjectWrapper().getClass());
+            assertEquals(
+                    Configuration.VERSION_3_0_0, ((DefaultObjectWrapper) cfgB.getObjectWrapper()).getIncompatibleImprovements());
+            assertEquals(DummyArithmeticEngine.class, cfgB.getArithmeticEngine().getClass());
+            assertEquals(DummyTemplateExceptionHandler.class, cfgB.getTemplateExceptionHandler().getClass());
+            assertEquals(DummyCacheStorage.class, cfgB.getCacheStorage().getClass());
+            assertEquals(DummyNewBuiltinClassResolver.class, cfgB.getNewBuiltinClassResolver().getClass());
+            assertEquals(DummyTemplateLoader.class, cfgB.getTemplateLoader().getClass());
+            assertEquals(StandardCharsets.UTF_8, cfgB.getSourceEncoding());
+        }
+        
+        {
+            Properties props = new Properties();
+            props.setProperty(MutableProcessingConfiguration.OBJECT_WRAPPER_KEY, "defAult");
+            props.setProperty(MutableProcessingConfiguration.ARITHMETIC_ENGINE_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyArithmeticEngine(x = 1)");
+            props.setProperty(MutableProcessingConfiguration.TEMPLATE_EXCEPTION_HANDLER_KEY,
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$DummyTemplateExceptionHandler(x = 1)");
+            props.setProperty(Configuration.ExtendableBuilder.CACHE_STORAGE_KEY,
+                    "soft: 500, strong: 100");
+            props.setProperty(MutableProcessingConfiguration.NEW_BUILTIN_CLASS_RESOLVER_KEY,
+                    "allows_nothing");
+            cfgB.setSettings(props);
+            assertEquals(DefaultObjectWrapper.class, cfgB.getObjectWrapper().getClass());
+            assertEquals(1, ((DummyArithmeticEngine) cfgB.getArithmeticEngine()).getX());
+            assertEquals(1, ((DummyTemplateExceptionHandler) cfgB.getTemplateExceptionHandler()).getX());
+            assertEquals(Configuration.VERSION_3_0_0,
+                    ((DefaultObjectWrapper) cfgB.getObjectWrapper()).getIncompatibleImprovements());
+            assertEquals(500, ((MruCacheStorage) cfgB.getCacheStorage()).getSoftSizeLimit());
+            assertEquals(TemplateClassResolver.ALLOWS_NOTHING_RESOLVER, cfgB.getNewBuiltinClassResolver());
+            assertEquals(StandardCharsets.UTF_8, cfgB.getSourceEncoding());
+        }
+
+        {
+            Properties props = new Properties();
+            props.setProperty(MutableProcessingConfiguration.OBJECT_WRAPPER_KEY, "Default");
+            props.setProperty(MutableProcessingConfiguration.ARITHMETIC_ENGINE_KEY, "bigdecimal");
+            props.setProperty(MutableProcessingConfiguration.TEMPLATE_EXCEPTION_HANDLER_KEY, "rethrow");
+            cfgB.setSettings(props);
+            assertEquals(DefaultObjectWrapper.class, cfgB.getObjectWrapper().getClass());
+            assertSame(BigDecimalArithmeticEngine.INSTANCE, cfgB.getArithmeticEngine());
+            assertSame(TemplateExceptionHandler.RETHROW_HANDLER, cfgB.getTemplateExceptionHandler());
+            assertEquals(Configuration.VERSION_3_0_0,
+                    ((DefaultObjectWrapper) cfgB.getObjectWrapper()).getIncompatibleImprovements());
+        }
+        
+        {
+            Properties props = new Properties();
+            props.setProperty(MutableProcessingConfiguration.OBJECT_WRAPPER_KEY, "DefaultObjectWrapper(3.0.0)");
+            cfgB.setSettings(props);
+            assertEquals(DefaultObjectWrapper.class, cfgB.getObjectWrapper().getClass());
+            assertEquals(
+                    Configuration.VERSION_3_0_0,
+                    ((DefaultObjectWrapper) cfgB.getObjectWrapper()).getIncompatibleImprovements());
+        }
+    }
+    
+    @Test
+    public void timeZoneTest() throws _ObjectBuilderSettingEvaluationException, ClassNotFoundException,
+    InstantiationException, IllegalAccessException {
+        for (String timeZoneId : new String[] { "GMT+01", "GMT", "UTC" }) {
+            TestBean8 result = (TestBean8) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(timeZone=TimeZone('"
+                    + timeZoneId + "'))",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            assertEquals(TimeZone.getTimeZone(timeZoneId), result.getTimeZone());
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(timeZone=TimeZone('foobar'))",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getCause().getMessage(),
+                    allOf(containsStringIgnoringCase("unrecognized"), containsString("foobar")));
+        }
+    }
+
+    @Test
+    public void charsetTest() throws _ObjectBuilderSettingEvaluationException, ClassNotFoundException,
+            InstantiationException, IllegalAccessException {
+        for (String timeZoneId : new String[] { "uTf-8", "GMT", "UTC" }) {
+            TestBean8 result = (TestBean8) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(charset=Charset('iso-8859-1'))",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            assertEquals(StandardCharsets.ISO_8859_1, result.getCharset());
+        }
+
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(charset=Charset('noSuchCS'))",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getCause(), instanceOf(UnsupportedCharsetException.class));
+        }
+    }
+
+    @Test
+    public void configureBeanTest() throws Exception {
+        final TestBean7 bean = new TestBean7();
+        final String src = "a/b(s='foo', x=1, b=true), bar";
+        int nextPos = _ObjectBuilderSettingEvaluator.configureBean(src, src.indexOf('(') + 1, bean,
+                _SettingEvaluationEnvironment.getCurrent());
+        assertEquals("foo", bean.getS());
+        assertEquals(1, bean.getX());
+        assertTrue(bean.isB());
+        assertEquals(", bar", src.substring(nextPos));
+    }
+    
+    @Test
+    public void parsingErrors() throws Exception {
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(1,,2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("\",\""));
+        }
+
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(x=1,2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsStringIgnoringCase("must precede named"));
+        }
+
+
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(x=1;2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("\";\""));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(1,2))",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("\")\""));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "foo.Bar('s${x}s'))",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("${...}"));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "foo.Bar('s#{x}s'))",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("#{...}"));
+        }
+    }
+
+    @Test
+    public void semanticErrors() throws Exception {
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$XTestBean1(1,2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsStringIgnoringCase("Failed to get class"));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(true, 2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsStringIgnoringCase("constructor"));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(x = 1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsStringIgnoringCase("no writeable JavaBeans property called \"x\""));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1(p1 = 1, p1 = 2)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("twice"));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "java.util.HashMap()",
+                    ObjectWrapper.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("is not a(n) " + ObjectWrapper.class.getName()));
+        }
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "null",
+                    ObjectWrapper.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("can't be null"));
+        }
+    }
+    
+    @Test
+    public void testLiteralAsObjectBuilder() throws Exception {
+        assertNull(_ObjectBuilderSettingEvaluator.eval(
+                "null",
+                ObjectWrapper.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals("foo", _ObjectBuilderSettingEvaluator.eval(
+                "'foo'",
+                CharSequence.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Boolean.TRUE, _ObjectBuilderSettingEvaluator.eval(
+                "  true  ",
+                Object.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Double.valueOf("1.23"), _ObjectBuilderSettingEvaluator.eval(
+                "1.23 ",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(new Version(1, 2, 3), _ObjectBuilderSettingEvaluator.eval(
+                " 1.2.3",
+                Object.class, true, _SettingEvaluationEnvironment.getCurrent()));
+    }
+
+    @Test
+    public void testNumberLiteralJavaTypes() throws Exception {
+        assertEquals(Double.valueOf("1.0"), _ObjectBuilderSettingEvaluator.eval(
+                "1.0",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+
+        assertEquals(new BigInteger("-9223372036854775809"), _ObjectBuilderSettingEvaluator.eval(
+                "-9223372036854775809",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(new BigInteger("9223372036854775808"), _ObjectBuilderSettingEvaluator.eval(
+                "9223372036854775808",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        
+        assertEquals(Long.valueOf(-9223372036854775808L), _ObjectBuilderSettingEvaluator.eval(
+                "-9223372036854775808",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Long.valueOf(9223372036854775807L), _ObjectBuilderSettingEvaluator.eval(
+                "9223372036854775807",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        
+        assertEquals(Integer.valueOf(-2147483648), _ObjectBuilderSettingEvaluator.eval(
+                "-2147483648",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Integer.valueOf(2147483647), _ObjectBuilderSettingEvaluator.eval(
+                "2147483647",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        
+        assertEquals(Integer.valueOf(-1), _ObjectBuilderSettingEvaluator.eval(
+                "-1",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Integer.valueOf(1), _ObjectBuilderSettingEvaluator.eval(
+                "1",
+                Number.class, true, _SettingEvaluationEnvironment.getCurrent()));
+    }
+    
+    @Test
+    public void testListLiterals() throws Exception {
+        {
+            ArrayList<Object> expected = new ArrayList();
+            expected.add("s");
+            expected.add(null);
+            expected.add(true);
+            expected.add(new TestBean9(1));
+            expected.add(ImmutableList.of(11, 22, 33));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    "['s', null, true, org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1), [11, 22, 33]]",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    "  [  's'  ,  null ,  true , org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1) ,"
+                    + "  [ 11 , 22 , 33 ]  ]  ",
+                    Collection.class, false, _SettingEvaluationEnvironment.getCurrent()));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    "['s',null,true,org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1),[11,22,33]]",
+                    List.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        }
+        
+        assertEquals(Collections.emptyList(), _ObjectBuilderSettingEvaluator.eval(
+                "[]",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Collections.emptyList(), _ObjectBuilderSettingEvaluator.eval(
+                "[  ]",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+
+        assertEquals(Collections.singletonList(123), _ObjectBuilderSettingEvaluator.eval(
+                "[123]",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Collections.singletonList(123), _ObjectBuilderSettingEvaluator.eval(
+                "[ 123 ]",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        
+        assertEquals(new TestBean9(1, ImmutableList.of("a", "b")), _ObjectBuilderSettingEvaluator.eval(
+                "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1, ['a', 'b'])",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "[1,]",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \"]\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "[,1]",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \",\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "1]",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \"]\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "[1",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("end of"));
+        }
+    }
+
+    @Test
+    public void testMapLiterals() throws Exception {
+        {
+            HashMap<String, Object> expected = new HashMap();
+            expected.put("k1", "s");
+            expected.put("k2", null);
+            expected.put("k3", true);
+            expected.put("k4", new TestBean9(1));
+            expected.put("k5", ImmutableList.of(11, 22, 33));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    "{'k1': 's', 'k2': null, 'k3': true, "
+                    + "'k4': org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1), 'k5': [11, 22, 33]}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    " {  'k1'  :  's'  ,  'k2' :  null  , 'k3' : true , "
+                    + "'k4' : org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9 ( 1 ) , 'k5' : [ 11 , 22 , 33 ] } ",
+                    Map.class, false, _SettingEvaluationEnvironment.getCurrent()));
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    " {'k1':'s','k2':null,'k3':true,"
+                    + "'k4':org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1),'k5':[11,22,33]}",
+                    LinkedHashMap.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        }
+        
+        {
+            HashMap<Object, String> expected = new HashMap();
+            expected.put(true, "T");
+            expected.put(1, "O");
+            expected.put(new TestBean9(1), "B");
+            expected.put(ImmutableList.of(11, 22, 33), "L");
+            assertEquals(expected, _ObjectBuilderSettingEvaluator.eval(
+                    "{ true: 'T', 1: 'O', org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1): 'B', "
+                    + "[11, 22, 33]: 'L' }",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        }
+        
+        assertEquals(Collections.emptyMap(), _ObjectBuilderSettingEvaluator.eval(
+                "{}",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Collections.emptyMap(), _ObjectBuilderSettingEvaluator.eval(
+                "{  }",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+
+        assertEquals(Collections.singletonMap("k1", 123), _ObjectBuilderSettingEvaluator.eval(
+                "{'k1':123}",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        assertEquals(Collections.singletonMap("k1", 123), _ObjectBuilderSettingEvaluator.eval(
+                "{ 'k1' : 123 }",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        
+        assertEquals(new TestBean9(1, ImmutableMap.of(11, "a", 22, "b")), _ObjectBuilderSettingEvaluator.eval(
+                "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean9(1, { 11: 'a', 22: 'b' })",
+                Object.class, false, _SettingEvaluationEnvironment.getCurrent()));
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "{1:2,}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \"}\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "{,1:2}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \",\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "1:2}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \":\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "1}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("found character \"}\""));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "{1",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("end of"));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "{1:",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("end of"));
+        }
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "{null:1}",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(), containsString("null as key"));
+        }
+    }
+
+    @Test
+    public void testMethodParameterNumberTypes() throws Exception {
+        {
+            TestBean8 result = (TestBean8) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(anyObject=1)",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            assertEquals(result.getAnyObject(), 1);
+        }
+        {
+            TestBean8 result = (TestBean8) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(anyObject=2147483649)",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            assertEquals(result.getAnyObject(), 2147483649L);
+        }
+        {
+            TestBean8 result = (TestBean8) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean8(anyObject=1.0)",
+                    TestBean8.class, false, new _SettingEvaluationEnvironment());
+            // Like in FTL, non-integer numbers are BigDecimal-s, that are later coerced to the actual parameter type.
+            // However, here the type is Object, so it remains BigDecimal.
+            assertEquals(new BigDecimal("1.0"), result.getAnyObject());
+        }
+    }
+    
+    @Test
+    public void testNonMethodParameterNumberTypes() throws Exception {
+        assertEqualsEvaled(Integer.valueOf(1), "1");
+        assertEqualsEvaled(Double.valueOf(1), "1.0");
+        assertEqualsEvaled(Long.valueOf(2147483649l), "2147483649");
+
+        assertEqualsEvaled(Double.valueOf(1), "1d");
+        assertEqualsEvaled(Double.valueOf(1), "1D");
+        assertEqualsEvaled(Float.valueOf(1), "1f");
+        assertEqualsEvaled(Float.valueOf(1), "1F");
+        assertEqualsEvaled(Long.valueOf(1), "1l");
+        assertEqualsEvaled(Long.valueOf(1), "1L");
+        assertEqualsEvaled(BigDecimal.valueOf(1), "1bd");
+        assertEqualsEvaled(BigDecimal.valueOf(1), "1Bd");
+        assertEqualsEvaled(BigDecimal.valueOf(1), "1BD");
+        assertEqualsEvaled(BigInteger.valueOf(1), "1bi");
+        assertEqualsEvaled(BigInteger.valueOf(1), "1bI");
+        
+        assertEqualsEvaled(Float.valueOf(1.5f), "1.5f");
+        assertEqualsEvaled(Double.valueOf(1.5), "1.5d");
+        assertEqualsEvaled(BigDecimal.valueOf(1.5), "1.5bd");
+        
+        assertEqualsEvaled(
+                ImmutableList.of(-1, -0.5, new BigDecimal("-0.1")),
+                "[ -1, -0.5, -0.1bd ]");
+        assertEqualsEvaled(
+                ImmutableMap.of(-1, -11, -0.5, -0.55, new BigDecimal("-0.1"), new BigDecimal("-0.11")),
+                "{ -1: -11, -0.5: -0.55, -0.1bd: -0.11bd }");
+    }
+    
+    @Test
+    public void testStaticFields() throws Exception {
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1("
+                    + "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST, true)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(TestStaticFields.CONST, (int) res.i);
+        }
+        {
+            TestBean1 res = (TestBean1) _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1("
+                    + "p2 = org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(TestStaticFields.CONST, res.getP2());
+        }
+        assertEqualsEvaled(123, "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST");
+        
+        // With shorthand class name:
+        assertEqualsEvaled(ParsingConfiguration.AUTO_DETECT_TAG_SYNTAX, "Configuration.AUTO_DETECT_TAG_SYNTAX");
+        
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean1("
+                    + "p2 = org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST())",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(),
+                    containsString("org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST"));
+        }
+        try {
+            assertEqualsEvaled(123, "org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST()");
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(),
+                    containsString("org.apache.freemarker.core.ObjectBuilderSettingsTest$TestStaticFields.CONST"));
+        }
+        try {
+            assertEqualsEvaled(123, "java.lang.String(org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5.INSTANCE)");
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertThat(e.getMessage(),
+                    containsString("org.apache.freemarker.core.ObjectBuilderSettingsTest$TestBean5()"));
+        }
+    }
+    
+    private void assertEqualsEvaled(Object expectedValue, String s)
+            throws _ObjectBuilderSettingEvaluationException, ClassNotFoundException, InstantiationException,
+            IllegalAccessException {
+        Object actualValue = _ObjectBuilderSettingEvaluator.eval(
+                s, Object.class, true, _SettingEvaluationEnvironment.getCurrent());
+        assertEquals(expectedValue, actualValue);
+    }
+    
+    @Test
+    public void visibilityTest() throws Exception {
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PackageVisibleAll()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertEquals(IllegalAccessException.class, e.getCause().getClass());
+        }
+
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PackageVisibleWithPublicConstructor()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertEquals(IllegalAccessException.class, e.getCause().getClass());
+        }
+
+        try {
+            _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PublicWithPackageVisibleConstructor()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            fail();
+        } catch (_ObjectBuilderSettingEvaluationException e) {
+            assertEquals(IllegalAccessException.class, e.getCause().getClass());
+        }
+        
+        {
+            Object o = _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PublicAll()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals(org.apache.freemarker.core.userpkg.PublicAll.class, o.getClass());
+        }
+        
+        {
+            Object o = _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PublicWithMixedConstructors(1)",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("Integer", ((PublicWithMixedConstructors) o).getS());
+        }
+        
+        
+        {
+            Object o = _ObjectBuilderSettingEvaluator.eval(
+                    "org.apache.freemarker.core.userpkg.PackageVisibleAllWithBuilder()",
+                    Object.class, false, _SettingEvaluationEnvironment.getCurrent());
+            assertEquals("org.apache.freemarker.core.userpkg.PackageVisibleAllWithBuilder", o.getClass().getName());
+        }
+    }
+
+    public static class TestBean1 {
+        float f;
+        Integer i;
+        long l;
+        boolean b;
+        
+        double p1;
+        int p2;
+        boolean p3;
+        String p4;
+        
+        public TestBean1(float f, Integer i, long l, boolean b) {
+            this.f = f;
+            this.i = i;
+            this.l = l;
+            this.b = b;
+        }
+        
+        public TestBean1(Integer i, boolean b) {
+            f = 2;
+            this.i = i;
+            l = 2;
+            this.b = b;
+        }
+    
+        public TestBean1(Integer i, long l) {
+            f = 3;
+            this.i = i;
+            this.l = l;
+            b = false;
+        }
+        
+        public TestBean1() {
+            f = 4;
+        }
+    
+        public double getP1() {
+            return p1;
+        }
+    
+        public void setP1(double p1) {
+            this.p1 = p1;
+        }
+    
+        public int getP2() {
+            return p2;
+        }
+    
+        public void setP2(int p2) {
+            this.p2 = p2;
+        }
+    
+        public boolean isP3() {
+            return p3;
+        }
+    
+        public void setP3(boolean p3) {
+            this.p3 = p3;
+        }
+
+        public String getP4() {
+            return p4;
+        }
+
+        public void setP4(String p4) {
+            this.p4 = p4;
+        }
+        
+    }
+    
+    public static class TestBean2 {
+        final boolean built;
+        final int x;
+
+        public TestBean2() {
+            built = false;
+            x = 0;
+        }
+        
+        public TestBean2(int x) {
+            built = false;
+            this.x = x;
+        }
+
+        public TestBean2(TestBean2Builder builder) {
+            built = true;
+            x = builder.x;
+        }
+        
+    }
+
+    public static class TestBean2Builder {
+        int x;
+        
+        public TestBean2Builder() { }
+
+        public TestBean2Builder(int x) {
+            this.x = x;
+        }
+        
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+        
+        public TestBean2 build() {
+            return new TestBean2(this);
+        }
+        
+    }
+
+    public static class TestBean3 {
+        
+        private int x;
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+        
+    }
+    
+    public static class TestBean4 {
+        private final String s1, s2;
+        private String s3, s4;
+        
+        public TestBean4(String s1, String s2) {
+            this.s1 = s1;
+            this.s2 = s2;
+        }
+        
+        public String getS1() {
+            return s1;
+        }
+
+        public String getS2() {
+            return s2;
+        }
+
+        public String getS3() {
+            return s3;
+        }
+        
+        public void setS3(String s3) {
+            this.s3 = s3;
+        }
+        
+        public String getS4() {
+            return s4;
+        }
+        
+        public void setS4(String s4) {
+            this.s4 = s4;
+        }
+        
+    }
+    
+    public static class TestBean5 {
+        
+        public final static TestBean5 INSTANCE = new TestBean5();
+        
+        private final int i;
+        private int x;
+        
+        public TestBean5() {
+            i = 0;
+        }
+
+        public TestBean5(int i) {
+            this.i = i;
+        }
+
+        public int getI() {
+            return i;
+        }
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+        
+    }
+
+    public static class TestBean6 {
+        private final TestBean1 b1;
+        private int x;
+        private final TestBean2 b2;
+        private int y;
+        private TestBean2 b3;
+        private TestBean6 b4;
+        
+        public TestBean6(TestBean1 b1, int x, TestBean2 b2) {
+            this.b1 = b1;
+            this.x = x;
+            this.b2 = b2;
+        }
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+
+        public int getY() {
+            return y;
+        }
+
+        public void setY(int y) {
+            this.y = y;
+        }
+
+        public TestBean2 getB3() {
+            return b3;
+        }
+
+        public void setB3(TestBean2 b3) {
+            this.b3 = b3;
+        }
+
+        public TestBean1 getB1() {
+            return b1;
+        }
+
+        public TestBean2 getB2() {
+            return b2;
+        }
+
+        public TestBean6 getB4() {
+            return b4;
+        }
+
+        public void setB4(TestBean6 b4) {
+            this.b4 = b4;
+        }
+        
+    }
+    
+    public class TestBean7 {
+
+        private String s;
+        private int x;
+        private boolean b;
+
+        public String getS() {
+            return s;
+        }
+
+        public void setS(String s) {
+            this.s = s;
+        }
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+
+        public boolean isB() {
+            return b;
+        }
+
+        public void setB(boolean b) {
+            this.b = b;
+        }
+
+        @Override
+        public String toString() {
+            return "TestBean [s=" + s + ", x=" + x + ", b=" + b + "]";
+        }
+
+    }
+    
+    public static class TestBean8 {
+        private TimeZone timeZone;
+        private Charset charset;
+        private Object anyObject;
+        private List<?> list;
+        
+        public TimeZone getTimeZone() {
+            return timeZone;
+        }
+        
+        public void setTimeZone(TimeZone timeZone) {
+            this.timeZone = timeZone;
+        }
+
+        public Charset getCharset() {
+            return charset;
+        }
+
+        public void setCharset(Charset charset) {
+            this.charset = charset;
+        }
+
+        public Object getAnyObject() {
+            return anyObject;
+        }
+        
+        public void setAnyObject(Object anyObject) {
+            this.anyObject = anyObject;
+        }
+
+        public List<?> getList() {
+            return list;
+        }
+        
+        public void setList(List<?> list) {
+            this.list = list;
+        }
+        
+    }
+    
+    public static class TestBean9 {
+        
+        private final int n;
+        private final List<?> list;
+        private final Map<?, ?> map;
+
+        public TestBean9(int n) {
+            this(n, null, null);
+        }
+
+        public TestBean9(int n, List<?> list) {
+            this(n, list, null);
+        }
+
+        public TestBean9(int n, Map<?, ?> map) {
+            this(n, null, map);
+        }
+        
+        public TestBean9(int n, List<?> list, Map<?, ?> map) {
+            this.n = n;
+            this.list = list;
+            this.map = map;
+        }
+
+        @Override
+        public int hashCode() {
+            final int prime = 31;
+            int result = 1;
+            result = prime * result + ((list == null) ? 0 : list.hashCode());
+            result = prime * result + ((map == null) ? 0 : map.hashCode());
+            result = prime * result + n;
+            return result;
+        }
+
+        @Override
+        public boolean equals(Object obj) {
+            if (this == obj) return true;
+            if (obj == null) return false;
+            if (getClass() != obj.getClass()) return false;
+            TestBean9 other = (TestBean9) obj;
+            if (list == null) {
+                if (other.list != null) return false;
+            } else if (!list.equals(other.list)) return false;
+            if (map == null) {
+                if (other.map != null) return false;
+            } else if (!map.equals(other.map)) return false;
+            return n == other.n;
+        }
+        
+    }
+    
+    public static class TestStaticFields {
+        public static final int CONST = 123;
+    }
+    
+    public static class DummyArithmeticEngine extends ArithmeticEngine {
+        
+        private int x;
+
+        @Override
+        public int compareNumbers(Number first, Number second) throws TemplateException {
+            return 0;
+        }
+
+        @Override
+        public Number add(Number first, Number second) throws TemplateException {
+            return null;
+        }
+
+        @Override
+        public Number subtract(Number first, Number second) throws TemplateException {
+            return null;
+        }
+
+        @Override
+        public Number multiply(Number first, Number second) throws TemplateException {
+            return null;
+        }
+
+        @Override
+        public Number divide(Number first, Number second) throws TemplateException {
+            return null;
+        }
+
+        @Override
+        public Number modulus(Number first, Number second) throws TemplateException {
+            return null;
+        }
+
+        @Override
+        public Number toNumber(String s) {
+            return null;
+        }
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+        
+    }
+    
+    public static class DummyTemplateExceptionHandler implements TemplateExceptionHandler {
+        
+        private int x;
+
+        @Override
+        public void handleTemplateException(TemplateException te, Environment env, Writer out) throws TemplateException {
+        }
+
+        public int getX() {
+            return x;
+        }
+
+        public void setX(int x) {
+            this.x = x;
+        }
+        
+    }
+    
+    public static class DummyCacheStorage implements CacheStorage {
+        
+        @Override
+        public Object get(Object key) {
+            return null;
+        }
+
+        @Override
+        public void put(Object key, Object value) {
+        }
+
+        @Override
+        public void remove(Object key) {
+        }
+
+        @Override
+        public void clear() {
+        }
+        
+    }
+    
+    public static class DummyNewBuiltinClassResolver implements TemplateClassResolver {
+
+        @Override
+        public Class resolve(String className, Environment env, Template template) throws TemplateException {
+            return null;
+        }
+        
+    }
+    
+    public static class DummyTemplateLoader implements TemplateLoader {
+
+        @Override
+        public TemplateLoaderSession createSession() {
+            return null;
+        }
+
+        @Override
+        public TemplateLoadingResult load(String name, TemplateLoadingSource ifSourceDiffersFrom,
+                Serializable ifVersionDiffersFrom, TemplateLoaderSession session) throws IOException {
+            return TemplateLoadingResult.NOT_FOUND;
+        }
+
+        @Override
+        public void resetState() {
+            // Do nothing
+        }
+        
+    }
+    
+}

http://git-wip-us.apache.org/repos/asf/incubator-freemarker/blob/28a276c8/freemarker-core-test/src/test/java/org/apache/freemarker/core/OptInTemplateClassResolverTest.java
----------------------------------------------------------------------
diff --git a/freemarker-core-test/src/test/java/org/apache/freemarker/core/OptInTemplateClassResolverTest.java b/freemarker-core-test/src/test/java/org/apache/freemarker/core/OptInTemplateClassResolverTest.java
new file mode 100644
index 0000000..06e863c
--- /dev/null
+++ b/freemarker-core-test/src/test/java/org/apache/freemarker/core/OptInTemplateClassResolverTest.java
@@ -0,0 +1,230 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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.apache.freemarker.core;
+
+import java.util.ArrayList;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+
+import org.apache.freemarker.core.util.OptInTemplateClassResolver;
+import org.apache.freemarker.test.TestConfigurationBuilder;
+
+import junit.framework.AssertionFailedError;
+import junit.framework.TestCase;
+
+public class OptInTemplateClassResolverTest extends TestCase {
+
+    public OptInTemplateClassResolverTest(String name) {
+        super(name);
+    }
+    
+    private static final Set ALLOWED_CLASSES = new HashSet();
+    static {
+        ALLOWED_CLASSES.add("java.lang.String");
+        ALLOWED_CLASSES.add("java.lang.Integer");
+    }
+    
+    private static final List TRUSTED_TEMPLATES = new ArrayList();
+    static {
+        TRUSTED_TEMPLATES.add("lib/*");
+        TRUSTED_TEMPLATES.add("/include/*");
+        TRUSTED_TEMPLATES.add("trusted.ftl");
+    }
+    
+    private OptInTemplateClassResolver resolver = new OptInTemplateClassResolver(
+            ALLOWED_CLASSES, TRUSTED_TEMPLATES);
+    
+
+    public void testOptIn() throws TemplateException {
+        Template dummyTemp = Template.createPlainTextTemplate("foo.ftl", "",
+                new TestConfigurationBuilder().build());
+
+        assertEquals(String.class, resolver.resolve("java.lang.String", null, dummyTemp));
+        assertEquals(Integer.class, resolver.resolve("java.lang.Integer", null, dummyTemp));
+        try {
+            resolver.resolve("java.lang.Long", null, dummyTemp);
+            fail();
+        } catch (TemplateException e) {
+            // Expected
+        }
+    }
+
+    public void testTrusted() throws TemplateException {
+        Configuration cfg = new TestConfigurationBuilder().build();
+
+        assertEquals(Long.class, resolver.resolve("java.lang.Long", null,
+                Template.createPlainTextTemplate("lib/foo.ftl", "", cfg)));
+        assertEquals(String.class, resolver.resolve("java.lang.String", null,
+                Template.createPlainTextTemplate("lib/foo.ftl", "", cfg)));
+        assertEquals(Long.class, resolver.resolve("java.lang.Long", null,
+                Template.createPlainTextTemplate("/lib/foo.ftl", "", cfg)));
+        assertEquals(Long.class, resolver.resolve("java.lang.Long", null,
+                Template.createPlainTextTemplate("include/foo.ftl", "", cfg)));
+        assertEquals(Long.class, resolver.resolve("java.lang.Long", null,
+                Template.createPlainTextTemplate("trusted.ftl", "", cfg)));
+        assertEquals(Long.class, resolver.resolve("java.lang.Long", null,
+                Template.createPlainTextTemplate("/trusted.ftl", "", cfg)));
+    }
+
+    public void testCraftedTrusted() throws TemplateException {
+        testTrusted_checkFails("lib/../foo.ftl");
+        testTrusted_checkFails("lib\\..\\foo.ftl");
+        testTrusted_checkFails("lib\\../foo.ftl");
+        testTrusted_checkFails("lib/..\\foo.ftl");
+        testTrusted_checkFails("lib/..");
+        testTrusted_checkFails("lib%2f%2E%2e%5cfoo.ftl");
+        testTrusted_checkFails("/lib%5C%.%2e%2Efoo.ftl");
+        
+        try {
+            testTrusted_checkFails("lib/./foo.ftl");
+            fail();
+        } catch (AssertionFailedError e) {
+            // Expected
+        }
+        try {
+            testTrusted_checkFails("lib/foo..ftl");
+            fail();
+        } catch (AssertionFailedError e) {
+            // Expected
+        }
+        try {
+            testTrusted_checkFails("lib/%2e/foo.ftl");
+            fail();
+        } catch (AssertionFailedError e) {
+            // Expected
+        }
+    }
+    
+    public void testTrusted_checkFails(String templateName) {
+        try {
+            resolver.resolve("java.lang.Long", null,
+                    Template.createPlainTextTemplate(templateName, "",
+                            new TestConfigurationBuilder().build()));
+            fail();
+        } catch (TemplateException e) {
+            // Expected
+        }
+    }
+    
+    public void testSettingParser() throws Exception {
+        {
+            Configuration cfg = new TestConfigurationBuilder()
+                    .setting(
+                            "new_builtin_class_resolver",
+                            "trusted_templates: foo.ftl, \"lib/*\"")
+                    .build();
+
+            TemplateClassResolver res = cfg.getNewBuiltinClassResolver();
+            assertEquals(String.class, res.resolve("java.lang.String", null,
+                    Template.createPlainTextTemplate("foo.ftl", "", cfg)));
+            assertEquals(String.class, res.resolve("java.lang.String", null,
+                    Template.createPlainTextTemplate("lib/bar.ftl", "", cfg)));
+            try {
+                res.resolve("java.lang.String", null,
+                        Template.createPlainTextTemplate("bar.ftl", "", cfg));
+                fail();
+            } catch (TemplateException e) {
+                // Expected
+            }
+        }
+
+        {
+            Configuration cfg = new TestConfigurationBuilder()
+                    .setting(
+                            "new_builtin_class_resolver",
+                            "allowed_classes: java.lang.String, java.lang.Integer")
+                    .build();
+
+            TemplateClassResolver res = cfg.getNewBuiltinClassResolver();
+            assertEquals(String.class, res.resolve("java.lang.String", null,
+                    Template.createPlainTextTemplate("foo.ftl", "", cfg)));
+            assertEquals(Integer.class, res.resolve("java.lang.Integer", null,
+                    Template.createPlainTextTemplate("foo.ftl", "", cfg)));
+            try {
+                res.resolve("java.lang.Long", null,
+                        Template.createPlainTextTemplate("foo.ftl", "", cfg));
+                fail();
+            } catch (TemplateException e) {
+                // good
+            }
+        }
+
+        {
+            Configuration cfg = new TestConfigurationBuilder()
+                    .setting(
+                            "new_builtin_class_resolver",
+                            "trusted_templates: foo.ftl, 'lib/*', allowed_classes: 'java.lang.String',"
+                            + " java.lang.Integer")
+                    .build();
+            TemplateClassResolver res = cfg.getNewBuiltinClassResolver();
+            assertEquals(String.class, res.resolve("java.lang.String", null,
+                    Template.createPlainTextTemplate("x.ftl", "", cfg)));
+            assertEquals(Integer.class, res.resolve("java.lang.Integer", null,
+                    Template.createPlainTextTemplate("x.ftl", "", cfg)));
+            try {
+                res.resolve("java.lang.Long", null,
+                        Template.createPlainTextTemplate("x.ftl", "", cfg));
+                fail();
+            } catch (TemplateException e) {
+                // Expected
+            }
+            assertEquals(Long.class, res.resolve("java.lang.Long", null,
+                    Template.createPlainTextTemplate("foo.ftl", "", cfg)));
+            assertEquals(Long.class, res.resolve("java.lang.Long", null,
+                    Template.createPlainTextTemplate("lib/bar.ftl", "", cfg)));
+            try {
+                res.resolve("java.lang.Long", null,
+                        Template.createPlainTextTemplate("x.ftl", "", cfg));
+                fail();
+            } catch (TemplateException e) {
+                // Expected
+            }
+        }
+        
+        try {
+            new TestConfigurationBuilder().setSetting("new_builtin_class_resolver", "wrong: foo");
+            fail();
+        } catch (ConfigurationException e) {
+            // Expected
+        }
+
+        {
+            Configuration cfg = new TestConfigurationBuilder()
+                    .setting(
+                            "new_builtin_class_resolver",
+                            "\"allowed_classes\"  :  java.lang.String  ,  'trusted_templates' :\"lib:*\"")
+                    .build();
+            TemplateClassResolver res = cfg.getNewBuiltinClassResolver();
+            assertEquals(String.class, res.resolve("java.lang.String", null,
+                    Template.createPlainTextTemplate("x.ftl", "", cfg)));
+            try {
+                res.resolve("java.lang.Long", null,
+                        Template.createPlainTextTemplate("x.ftl", "", cfg));
+                fail();
+            } catch (TemplateException e) {
+                // Expected
+            }
+            assertEquals(Long.class, res.resolve("java.lang.Long", null,
+                    Template.createPlainTextTemplate("lib:bar.ftl", "", cfg)));
+        }
+    }
+    
+}