Refactor withFallback() implementations to share more logic among subtypes

Also add some tests and fix some correctness issues.

If we fall back to a value that has ignoresFallbacks()=true, then we need
to "catch" its ignoresFallbacks.

Don't create new RootConfig objects if the underlying has not changed.

Get rid of ConfigImpl.merge() utility, no longer useful.
This commit is contained in:
Havoc Pennington 2011-11-18 16:12:46 -05:00
parent 7df8e342f6
commit 8b6f4c6156
10 changed files with 268 additions and 150 deletions

View File

@ -106,50 +106,57 @@ abstract class AbstractConfigObject extends AbstractConfigValue implements
boolean ignoresFallbacks);
@Override
public AbstractConfigObject withFallback(ConfigMergeable mergeable) {
ConfigValue other = mergeable.toValue();
protected AbstractConfigObject newCopy(boolean ignoresFallbacks) {
return newCopy(resolveStatus(), ignoresFallbacks);
}
if (ignoresFallbacks()) {
return this;
} else if (other instanceof Unmergeable) {
List<AbstractConfigValue> stack = new ArrayList<AbstractConfigValue>();
stack.add(this);
stack.addAll(((Unmergeable) other).unmergedValues());
return new ConfigDelayedMergeObject(mergeOrigins(stack), stack);
} else if (other instanceof AbstractConfigObject) {
AbstractConfigObject fallback = (AbstractConfigObject) other;
if (fallback.isEmpty()) {
return this; // nothing to do
} else {
boolean allResolved = true;
Map<String, AbstractConfigValue> merged = new HashMap<String, AbstractConfigValue>();
Set<String> allKeys = new HashSet<String>();
allKeys.addAll(this.keySet());
allKeys.addAll(fallback.keySet());
for (String key : allKeys) {
AbstractConfigValue first = this.peek(key);
AbstractConfigValue second = fallback.peek(key);
AbstractConfigValue kept;
if (first == null)
kept = second;
else if (second == null)
kept = first;
else
kept = first.withFallback(second);
merged.put(key, kept);
if (kept.resolveStatus() == ResolveStatus.UNRESOLVED)
allResolved = false;
}
return new SimpleConfigObject(mergeOrigins(this, fallback),
merged, ResolveStatus.fromBoolean(allResolved),
ignoresFallbacks());
}
@Override
protected final AbstractConfigObject mergedWithTheUnmergeable(Unmergeable fallback) {
if (ignoresFallbacks())
throw new ConfigException.BugOrBroken("should not be reached");
List<AbstractConfigValue> stack = new ArrayList<AbstractConfigValue>();
if (this instanceof Unmergeable) {
stack.addAll(((Unmergeable) this).unmergedValues());
} else {
// falling back to a non-object doesn't merge anything, and also
// prohibits merging any objects that we fall back to later.
// so we have to switch to ignoresFallbacks mode.
return newCopy(resolveStatus(), true /* ignoresFallbacks */);
stack.add(this);
}
stack.addAll(fallback.unmergedValues());
return new ConfigDelayedMergeObject(mergeOrigins(stack), stack,
((AbstractConfigValue) fallback).ignoresFallbacks());
}
@Override
protected AbstractConfigObject mergedWithObject(AbstractConfigObject fallback) {
if (ignoresFallbacks())
throw new ConfigException.BugOrBroken("should not be reached");
boolean allResolved = true;
Map<String, AbstractConfigValue> merged = new HashMap<String, AbstractConfigValue>();
Set<String> allKeys = new HashSet<String>();
allKeys.addAll(this.keySet());
allKeys.addAll(fallback.keySet());
for (String key : allKeys) {
AbstractConfigValue first = this.peek(key);
AbstractConfigValue second = fallback.peek(key);
AbstractConfigValue kept;
if (first == null)
kept = second;
else if (second == null)
kept = first;
else
kept = first.withFallback(second);
merged.put(key, kept);
if (kept.resolveStatus() == ResolveStatus.UNRESOLVED)
allResolved = false;
}
return new SimpleConfigObject(mergeOrigins(this, fallback), merged,
ResolveStatus.fromBoolean(allResolved), fallback.ignoresFallbacks());
}
@Override
public AbstractConfigObject withFallback(ConfigMergeable mergeable) {
return (AbstractConfigObject) super.withFallback(mergeable);
}
static ConfigOrigin mergeOrigins(
@ -169,7 +176,9 @@ abstract class AbstractConfigObject extends AbstractConfigValue implements
if (desc.startsWith(prefix))
desc = desc.substring(prefix.length());
if (v instanceof ConfigObject && ((ConfigObject) v).isEmpty()) {
if (v instanceof AbstractConfigObject
&& ((AbstractConfigObject) v).resolveStatus() == ResolveStatus.RESOLVED
&& ((ConfigObject) v).isEmpty()) {
// don't include empty files or the .empty()
// config in the description, since they are
// likely to be "implementation details"

View File

@ -3,6 +3,7 @@
*/
package com.typesafe.config.impl;
import com.typesafe.config.ConfigException;
import com.typesafe.config.ConfigMergeable;
import com.typesafe.config.ConfigOrigin;
import com.typesafe.config.ConfigResolveOptions;
@ -75,6 +76,10 @@ abstract class AbstractConfigValue implements ConfigValue {
return this;
}
protected AbstractConfigValue newCopy(boolean ignoresFallbacks) {
return this;
}
// this is virtualized rather than a field because only some subclasses
// really need to store the boolean, and they may be able to pack it
// with another boolean to save space.
@ -82,9 +87,50 @@ abstract class AbstractConfigValue implements ConfigValue {
return true;
}
private ConfigException badMergeException() {
if (ignoresFallbacks())
throw new ConfigException.BugOrBroken(
"method should not have been called with ignoresFallbacks=true"
+ getClass().getSimpleName());
else
throw new ConfigException.BugOrBroken("should override this in "
+ getClass().getSimpleName());
}
protected AbstractConfigValue mergedWithTheUnmergeable(Unmergeable fallback) {
throw badMergeException();
}
protected AbstractConfigValue mergedWithObject(AbstractConfigObject fallback) {
throw badMergeException();
}
@Override
public AbstractConfigValue withFallback(ConfigMergeable other) {
return this;
public AbstractConfigValue withFallback(ConfigMergeable mergeable) {
if (ignoresFallbacks()) {
return this;
} else {
ConfigValue other = mergeable.toValue();
if (other instanceof Unmergeable) {
return mergedWithTheUnmergeable((Unmergeable) other);
} else if (other instanceof AbstractConfigObject) {
AbstractConfigObject fallback = (AbstractConfigObject) other;
if (fallback.resolveStatus() == ResolveStatus.RESOLVED && fallback.isEmpty()) {
if (fallback.ignoresFallbacks())
return newCopy(true /* ignoresFallbacks */);
else
return this;
} else {
return mergedWithObject((AbstractConfigObject) other);
}
} else {
// falling back to a non-object doesn't merge anything, and also
// prohibits merging any objects that we fall back to later.
// so we have to switch to ignoresFallbacks mode.
return newCopy(true /* ignoresFallbacks */);
}
}
}
protected boolean canEqual(Object other) {

View File

@ -8,10 +8,8 @@ import java.util.Collection;
import java.util.List;
import com.typesafe.config.ConfigException;
import com.typesafe.config.ConfigMergeable;
import com.typesafe.config.ConfigOrigin;
import com.typesafe.config.ConfigResolveOptions;
import com.typesafe.config.ConfigValue;
import com.typesafe.config.ConfigValueType;
/**
@ -29,8 +27,8 @@ final class ConfigDelayedMerge extends AbstractConfigValue implements
final private List<AbstractConfigValue> stack;
final private boolean ignoresFallbacks;
private ConfigDelayedMerge(ConfigOrigin origin,
List<AbstractConfigValue> stack, boolean ignoresFallbacks) {
ConfigDelayedMerge(ConfigOrigin origin, List<AbstractConfigValue> stack,
boolean ignoresFallbacks) {
super(origin);
this.stack = stack;
this.ignoresFallbacks = ignoresFallbacks;
@ -38,6 +36,11 @@ final class ConfigDelayedMerge extends AbstractConfigValue implements
throw new ConfigException.BugOrBroken(
"creating empty delayed merge value");
for (AbstractConfigValue v : stack) {
if (v instanceof ConfigDelayedMerge || v instanceof ConfigDelayedMergeObject)
throw new ConfigException.BugOrBroken(
"placed nested DelayedMerge in a ConfigDelayedMerge, should have consolidated stack");
}
}
ConfigDelayedMerge(ConfigOrigin origin, List<AbstractConfigValue> stack) {
@ -67,18 +70,19 @@ final class ConfigDelayedMerge extends AbstractConfigValue implements
List<AbstractConfigValue> stack, SubstitutionResolver resolver,
int depth, ConfigResolveOptions options) {
// to resolve substitutions, we need to recursively resolve
// the stack of stuff to merge, and then merge the stack.
List<AbstractConfigValue> toMerge = new ArrayList<AbstractConfigValue>();
// the stack of stuff to merge, and merge the stack so
// we won't be a delayed merge anymore.
AbstractConfigValue merged = null;
for (AbstractConfigValue v : stack) {
AbstractConfigValue resolved = resolver.resolve(v, depth, options);
toMerge.add(resolved);
if (merged == null)
merged = resolved;
else
merged = merged.withFallback(resolved);
}
// we shouldn't have a delayed merge object with an empty stack, so
// it should be safe to ignore the toMerge.isEmpty case.
return ConfigImpl.merge(AbstractConfigValue.class, toMerge.get(0),
toMerge.subList(1, toMerge.size()));
return merged;
}
@Override
@ -101,29 +105,31 @@ final class ConfigDelayedMerge extends AbstractConfigValue implements
}
@Override
public AbstractConfigValue withFallback(ConfigMergeable mergeable) {
ConfigValue other = mergeable.toValue();
protected final ConfigDelayedMerge mergedWithTheUnmergeable(Unmergeable fallback) {
if (ignoresFallbacks)
throw new ConfigException.BugOrBroken("should not be reached");
if (ignoresFallbacks) {
return this;
} else if (other instanceof AbstractConfigObject
|| other instanceof Unmergeable) {
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.addAll(stack);
if (other instanceof Unmergeable)
newStack.addAll(((Unmergeable) other).unmergedValues());
else
newStack.add((AbstractConfigValue) other);
return new ConfigDelayedMerge(
AbstractConfigObject.mergeOrigins(newStack), newStack,
ignoresFallbacks);
} else {
// if the other is not an object, there won't be anything
// to merge with, so we are it even if we are an object.
return new ConfigDelayedMerge(origin(), stack, true /* ignoresFallbacks */);
}
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.addAll(stack);
newStack.addAll(fallback.unmergedValues());
return new ConfigDelayedMerge(AbstractConfigObject.mergeOrigins(newStack), newStack,
((AbstractConfigValue) fallback).ignoresFallbacks());
}
@Override
protected final ConfigDelayedMerge mergedWithObject(AbstractConfigObject fallback) {
if (ignoresFallbacks)
throw new ConfigException.BugOrBroken("should not be reached");
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.addAll(stack);
newStack.add(fallback);
return new ConfigDelayedMerge(AbstractConfigObject.mergeOrigins(newStack), newStack,
fallback.ignoresFallbacks());
}
@Override

View File

@ -28,8 +28,8 @@ class ConfigDelayedMergeObject extends AbstractConfigObject implements
this(origin, stack, false /* ignoresFallbacks */);
}
private ConfigDelayedMergeObject(ConfigOrigin origin,
List<AbstractConfigValue> stack, boolean ignoresFallbacks) {
ConfigDelayedMergeObject(ConfigOrigin origin, List<AbstractConfigValue> stack,
boolean ignoresFallbacks) {
super(origin);
this.stack = stack;
this.ignoresFallbacks = ignoresFallbacks;
@ -40,10 +40,16 @@ class ConfigDelayedMergeObject extends AbstractConfigObject implements
if (!(stack.get(0) instanceof AbstractConfigObject))
throw new ConfigException.BugOrBroken(
"created a delayed merge object not guaranteed to be an object");
for (AbstractConfigValue v : stack) {
if (v instanceof ConfigDelayedMerge || v instanceof ConfigDelayedMergeObject)
throw new ConfigException.BugOrBroken(
"placed nested DelayedMerge in a ConfigDelayedMergeObject, should have consolidated stack");
}
}
@Override
public ConfigDelayedMergeObject newCopy(ResolveStatus status,
protected ConfigDelayedMergeObject newCopy(ResolveStatus status,
boolean ignoresFallbacks) {
if (status != resolveStatus())
throw new ConfigException.BugOrBroken(
@ -86,30 +92,22 @@ class ConfigDelayedMergeObject extends AbstractConfigObject implements
}
@Override
public ConfigDelayedMergeObject withFallback(ConfigMergeable mergeable) {
ConfigValue other = mergeable.toValue();
protected ConfigDelayedMergeObject mergedWithObject(AbstractConfigObject fallback) {
if (ignoresFallbacks)
throw new ConfigException.BugOrBroken("should not be reached");
if (ignoresFallbacks) {
return this;
} else if (other instanceof AbstractConfigObject
|| other instanceof Unmergeable) {
// since we are an object, and the fallback could be,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.addAll(stack);
if (other instanceof Unmergeable)
newStack.addAll(((Unmergeable) other).unmergedValues());
else
newStack.add((AbstractConfigValue) other);
return new ConfigDelayedMergeObject(
AbstractConfigObject.mergeOrigins(newStack), newStack,
ignoresFallbacks);
} else {
// if the other is not an object, there won't be anything
// to merge with but we do need to ignore any fallbacks
// that occur after it.
return newCopy(resolveStatus(), true /* ignoresFallbacks */);
}
// since we are an object, and the fallback is, we'll need to
// merge the fallback once we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.addAll(stack);
newStack.add(fallback);
return new ConfigDelayedMergeObject(AbstractConfigObject.mergeOrigins(newStack), newStack,
fallback.ignoresFallbacks());
}
@Override
public ConfigDelayedMergeObject withFallback(ConfigMergeable mergeable) {
return (ConfigDelayedMergeObject) super.withFallback(mergeable);
}
@Override

View File

@ -5,7 +5,6 @@ package com.typesafe.config.impl;
import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
@ -16,7 +15,6 @@ import com.typesafe.config.Config;
import com.typesafe.config.ConfigException;
import com.typesafe.config.ConfigIncludeContext;
import com.typesafe.config.ConfigIncluder;
import com.typesafe.config.ConfigMergeable;
import com.typesafe.config.ConfigObject;
import com.typesafe.config.ConfigOrigin;
import com.typesafe.config.ConfigParseOptions;
@ -28,21 +26,6 @@ import com.typesafe.config.ConfigValue;
/** This is public but is only supposed to be used by the "config" package */
public class ConfigImpl {
static <T extends ConfigMergeable> T merge(Class<T> klass, T first,
ConfigMergeable... others) {
List<ConfigMergeable> stack = Arrays.asList(others);
return merge(klass, first, stack);
}
static <T extends ConfigMergeable> T merge(Class<T> klass, T first,
List<? extends ConfigMergeable> stack) {
ConfigMergeable merged = first;
for (ConfigMergeable fallback : stack) {
merged = merged.withFallback(fallback);
}
return klass.cast(merged);
}
private interface NameSource {
ConfigParseable nameToParseable(String name);
}

View File

@ -9,7 +9,6 @@ import java.util.Collections;
import java.util.List;
import com.typesafe.config.ConfigException;
import com.typesafe.config.ConfigMergeable;
import com.typesafe.config.ConfigOrigin;
import com.typesafe.config.ConfigResolveOptions;
import com.typesafe.config.ConfigValue;
@ -57,28 +56,41 @@ final class ConfigSubstitution extends AbstractConfigValue implements
}
@Override
public AbstractConfigValue withFallback(ConfigMergeable mergeable) {
ConfigValue other = mergeable.toValue();
protected ConfigSubstitution newCopy(boolean ignoresFallbacks) {
return new ConfigSubstitution(origin(), pieces, prefixLength, ignoresFallbacks);
}
if (ignoresFallbacks) {
return this;
} else if (other instanceof AbstractConfigObject
|| other instanceof Unmergeable) {
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.add(this);
if (other instanceof Unmergeable)
newStack.addAll(((Unmergeable) other).unmergedValues());
else
newStack.add((AbstractConfigValue) other);
return new ConfigDelayedMerge(
AbstractConfigObject.mergeOrigins(newStack), newStack);
} else {
// if the other is not an object, there won't be anything
// to merge with, so we are it even if we are an object.
return new ConfigSubstitution(origin(), pieces, prefixLength, true /* ignoresFallbacks */);
}
@Override
protected boolean ignoresFallbacks() {
return ignoresFallbacks;
}
@Override
protected AbstractConfigValue mergedWithTheUnmergeable(Unmergeable fallback) {
if (ignoresFallbacks)
throw new ConfigException.BugOrBroken("should not be reached");
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.add(this);
newStack.addAll(fallback.unmergedValues());
return new ConfigDelayedMerge(AbstractConfigObject.mergeOrigins(newStack), newStack,
((AbstractConfigValue) fallback).ignoresFallbacks());
}
@Override
protected AbstractConfigValue mergedWithObject(AbstractConfigObject fallback) {
if (ignoresFallbacks)
throw new ConfigException.BugOrBroken("should not be reached");
// if we turn out to be an object, and the fallback also does,
// then a merge may be required; delay until we resolve.
List<AbstractConfigValue> newStack = new ArrayList<AbstractConfigValue>();
newStack.add(this);
newStack.add(fallback);
return new ConfigDelayedMerge(AbstractConfigObject.mergeOrigins(newStack), newStack,
fallback.ignoresFallbacks());
}
@Override

View File

@ -35,14 +35,14 @@ final class RootConfig extends SimpleConfig implements ConfigRoot {
// if the object is already resolved then we should end up returning
// "this" here, since asRoot() should return this if the path
// is unchanged.
SimpleConfig resolved = resolvedObject(options).toConfig();
return resolved.asRoot(rootPath);
AbstractConfigObject resolved = resolvedObject(options);
return newRootIfObjectChanged(this, resolved);
}
@Override
public RootConfig withFallback(ConfigMergeable value) {
// this can return "this" if the withFallback does nothing
return super.withFallback(value).asRoot(rootPath);
return newRootIfObjectChanged(this, super.withFallback(value).toObject());
}
Path rootPathObject() {

View File

@ -59,6 +59,13 @@ class SimpleConfig implements Config {
return new RootConfig(underlying, newRootPath);
}
static protected RootConfig newRootIfObjectChanged(RootConfig self, AbstractConfigObject underlying) {
if (underlying == self.object)
return self;
else
return new RootConfig(underlying, self.rootPathObject());
}
protected AbstractConfigObject resolvedObject(ConfigResolveOptions options) {
AbstractConfigValue resolved = SubstitutionResolver.resolve(object,
object, options);

View File

@ -45,10 +45,8 @@ final class SimpleConfigObject extends AbstractConfigObject {
}
@Override
public SimpleConfigObject newCopy(ResolveStatus newStatus,
boolean ignoresFallbacks) {
return new SimpleConfigObject(origin(), value, newStatus,
ignoresFallbacks);
protected SimpleConfigObject newCopy(ResolveStatus newStatus, boolean newIgnoresFallbacks) {
return new SimpleConfigObject(origin(), value, newStatus, newIgnoresFallbacks);
}
@Override

View File

@ -15,6 +15,7 @@ import com.typesafe.config.ConfigResolveOptions
import java.io.File
import com.typesafe.config.ConfigParseOptions
import com.typesafe.config.ConfigFactory
import com.typesafe.config.ConfigMergeable
class ConfigTest extends TestUtils {
@ -83,7 +84,10 @@ class ConfigTest extends TestUtils {
val trees = Seq(m(allObjects: _*)) ++ makeTrees(allObjects)
for (tree <- trees) {
// if this fails, we were not associative.
assertEquals(trees(0), tree)
if (!trees(0).equals(tree))
throw new AssertionError("Merge was not associative, " +
"verify that it should not be, then don't use associativeMerge " +
"for this one.\none: " + trees(0) + "\ntwo: " + tree)
}
for (tree <- trees) {
@ -445,6 +449,61 @@ class ConfigTest extends TestUtils {
}
}
private def ignoresFallbacks(m: ConfigMergeable) = {
m match {
case v: AbstractConfigValue =>
v.ignoresFallbacks()
case c: SimpleConfig =>
c.toObject.ignoresFallbacks()
}
}
private def testIgnoredMergesDoNothing(nonEmpty: ConfigMergeable) {
// falling back to a primitive once should switch us to "ignoreFallbacks" mode
// and then twice should "return this". Falling back to an empty object should
// return this unless the empty object was ignoreFallbacks and then we should
// "catch" its ignoreFallbacks.
// some of what this tests is just optimization, not API contract (withFallback
// can return a new object anytime it likes) but want to be sure we do the
// optimizations.
val empty = SimpleConfigObject.empty(null)
val primitive = intValue(42)
val emptyIgnoringFallbacks = empty.withFallback(primitive)
val nonEmptyIgnoringFallbacks = nonEmpty.withFallback(primitive)
assertEquals(false, empty.ignoresFallbacks())
assertEquals(true, primitive.ignoresFallbacks())
assertEquals(true, emptyIgnoringFallbacks.ignoresFallbacks())
assertEquals(false, ignoresFallbacks(nonEmpty))
assertEquals(true, ignoresFallbacks(nonEmptyIgnoringFallbacks))
assertTrue(nonEmpty ne nonEmptyIgnoringFallbacks)
assertTrue(empty ne emptyIgnoringFallbacks)
// falling back from primitive just returns this
assertTrue(primitive eq primitive.withFallback(empty))
assertTrue(primitive eq primitive.withFallback(nonEmpty))
assertTrue(primitive eq primitive.withFallback(nonEmptyIgnoringFallbacks))
// falling back again from an ignoreFallbacks should be a no-op, return this
assertTrue(nonEmptyIgnoringFallbacks eq nonEmptyIgnoringFallbacks.withFallback(empty))
assertTrue(nonEmptyIgnoringFallbacks eq nonEmptyIgnoringFallbacks.withFallback(primitive))
assertTrue(emptyIgnoringFallbacks eq emptyIgnoringFallbacks.withFallback(empty))
assertTrue(emptyIgnoringFallbacks eq emptyIgnoringFallbacks.withFallback(primitive))
}
@Test
def ignoredMergesDoNothing() {
val conf = parseConfig("{ a : 1 }")
testIgnoredMergesDoNothing(conf)
// ConfigRoot mode uses a little different codepath
val root = conf.asRoot(path("whatever"))
testIgnoredMergesDoNothing(root)
}
@Test
def test01Getting() {
val conf = ConfigFactory.load("test01")