Properly track resource file origins

This improves error messages to fix #254.
This commit is contained in:
Havoc Pennington 2015-03-02 15:26:49 -05:00
parent 55bd472c24
commit a528bf104a
6 changed files with 146 additions and 86 deletions

View File

@ -45,6 +45,11 @@ public abstract class Parseable implements ConfigParseable {
private ConfigParseOptions initialOptions;
private ConfigOrigin initialOrigin;
protected interface Relativizer {
ConfigParseable relativeTo(String filename);
}
private static final ThreadLocal<LinkedList<Parseable>> parseStack = new ThreadLocal<LinkedList<Parseable>>() {
@Override
protected LinkedList<Parseable> initialValue() {
@ -213,7 +218,7 @@ public abstract class Parseable implements ConfigParseable {
}
}
protected AbstractConfigValue rawParseValue(Reader reader, ConfigOrigin origin,
private AbstractConfigValue rawParseValue(Reader reader, ConfigOrigin origin,
ConfigParseOptions finalOptions) throws IOException {
if (finalOptions.getSyntax() == ConfigSyntax.PROPERTIES) {
return PropertiesParser.parse(reader, origin);
@ -412,12 +417,17 @@ public abstract class Parseable implements ConfigParseable {
return new ParseableString(input, options);
}
private final static class ParseableURL extends Parseable {
final private URL input;
private static class ParseableURL extends Parseable {
final protected URL input;
private String contentType = null;
ParseableURL(URL input, ConfigParseOptions options) {
protected ParseableURL(URL input) {
this.input = input;
// does not postConstruct (subclass does it)
}
ParseableURL(URL input, ConfigParseOptions options) {
this(input);
postConstruct(options);
}
@ -551,7 +561,35 @@ public abstract class Parseable implements ConfigParseable {
return new ParseableFile(input, options);
}
private final static class ParseableResources extends Parseable {
private final static class ParseableResourceURL extends ParseableURL {
private final Relativizer relativizer;
private final String resource;
ParseableResourceURL(URL input, ConfigParseOptions options, String resource, Relativizer relativizer) {
super(input);
this.relativizer = relativizer;
this.resource = resource;
postConstruct(options);
}
@Override
protected ConfigOrigin createOrigin() {
return SimpleConfigOrigin.newResource(resource, input);
}
@Override
ConfigParseable relativeTo(String filename) {
return relativizer.relativeTo(filename);
}
}
private static Parseable newResourceURL(URL input, ConfigParseOptions options, String resource, Relativizer relativizer) {
return new ParseableResourceURL(input, options, resource, relativizer);
}
private final static class ParseableResources extends Parseable implements Relativizer {
final private String resource;
ParseableResources(String resource, ConfigParseOptions options) {
@ -583,31 +621,12 @@ public abstract class Parseable implements ConfigParseable {
URL url = e.nextElement();
if (ConfigImpl.traceLoadsEnabled())
trace("Loading config from URL " + url.toExternalForm() + " from class loader "
trace("Loading config from resource '" + resource + "' URL " + url.toExternalForm() + " from class loader "
+ loader);
ConfigOrigin elementOrigin = ((SimpleConfigOrigin) origin).addURL(url);
Parseable element = newResourceURL(url, finalOptions, resource, this);
AbstractConfigValue v;
// it's tempting to use ParseableURL here but it would be wrong
// because the wrong relativeTo() would be used for includes.
InputStream stream = url.openStream();
try {
Reader reader = readerFromStream(stream);
stream = null; // reader now owns it
try {
// parse in "raw" mode which will throw any IOException
// from here.
v = rawParseValue(reader, elementOrigin, finalOptions);
} finally {
reader.close();
}
} finally {
// stream is null if the reader owns it
if (stream != null)
stream.close();
}
AbstractConfigValue v = element.parseValue();
merged = merged.withFallback(v);
}
@ -634,7 +653,7 @@ public abstract class Parseable implements ConfigParseable {
}
@Override
ConfigParseable relativeTo(String sibling) {
public ConfigParseable relativeTo(String sibling) {
if (sibling.startsWith("/")) {
// if it starts with "/" then don't make it relative to
// the including resource

View File

@ -65,7 +65,9 @@ class SerializedConfigValue extends AbstractConfigValue implements Externalizabl
ORIGIN_URL,
ORIGIN_COMMENTS,
ORIGIN_NULL_URL,
ORIGIN_NULL_COMMENTS;
ORIGIN_NULL_COMMENTS,
ORIGIN_RESOURCE,
ORIGIN_NULL_RESOURCE;
static SerializedField forInt(int b) {
if (b < values().length)
@ -179,6 +181,9 @@ class SerializedConfigValue extends AbstractConfigValue implements Externalizabl
case ORIGIN_URL:
out.writeUTF((String) v);
break;
case ORIGIN_RESOURCE:
out.writeUTF((String) v);
break;
case ORIGIN_COMMENTS:
@SuppressWarnings("unchecked")
List<String> list = (List<String>) v;
@ -189,6 +194,7 @@ class SerializedConfigValue extends AbstractConfigValue implements Externalizabl
}
break;
case ORIGIN_NULL_URL: // FALL THRU
case ORIGIN_NULL_RESOURCE: // FALL THRU
case ORIGIN_NULL_COMMENTS:
// nothing to write out besides code and length
break;
@ -245,6 +251,10 @@ class SerializedConfigValue extends AbstractConfigValue implements Externalizabl
in.readInt(); // discard length
v = in.readUTF();
break;
case ORIGIN_RESOURCE:
in.readInt(); // discard length
v = in.readUTF();
break;
case ORIGIN_COMMENTS:
in.readInt(); // discard length
int size = in.readInt();
@ -255,6 +265,7 @@ class SerializedConfigValue extends AbstractConfigValue implements Externalizabl
v = list;
break;
case ORIGIN_NULL_URL: // FALL THRU
case ORIGIN_NULL_RESOURCE: // FALL THRU
case ORIGIN_NULL_COMMENTS:
// nothing to read besides code and length
in.readInt(); // discard length

View File

@ -28,10 +28,11 @@ final class SimpleConfigOrigin implements ConfigOrigin {
final private int endLineNumber;
final private OriginType originType;
final private String urlOrNull;
final private String resourceOrNull;
final private List<String> commentsOrNull;
protected SimpleConfigOrigin(String description, int lineNumber, int endLineNumber,
OriginType originType, String urlOrNull, List<String> commentsOrNull) {
protected SimpleConfigOrigin(String description, int lineNumber, int endLineNumber, OriginType originType,
String urlOrNull, String resourceOrNull, List<String> commentsOrNull) {
if (description == null)
throw new ConfigException.BugOrBroken("description may not be null");
this.description = description;
@ -39,11 +40,12 @@ final class SimpleConfigOrigin implements ConfigOrigin {
this.endLineNumber = endLineNumber;
this.originType = originType;
this.urlOrNull = urlOrNull;
this.resourceOrNull = resourceOrNull;
this.commentsOrNull = commentsOrNull;
}
static SimpleConfigOrigin newSimple(String description) {
return new SimpleConfigOrigin(description, -1, -1, OriginType.GENERIC, null, null);
return new SimpleConfigOrigin(description, -1, -1, OriginType.GENERIC, null, null, null);
}
static SimpleConfigOrigin newFile(String filename) {
@ -53,17 +55,22 @@ final class SimpleConfigOrigin implements ConfigOrigin {
} catch (MalformedURLException e) {
url = null;
}
return new SimpleConfigOrigin(filename, -1, -1, OriginType.FILE, url, null);
return new SimpleConfigOrigin(filename, -1, -1, OriginType.FILE, url, null, null);
}
static SimpleConfigOrigin newURL(URL url) {
String u = url.toExternalForm();
return new SimpleConfigOrigin(u, -1, -1, OriginType.URL, u, null);
return new SimpleConfigOrigin(u, -1, -1, OriginType.URL, u, null, null);
}
static SimpleConfigOrigin newResource(String resource, URL url) {
return new SimpleConfigOrigin(resource, -1, -1, OriginType.RESOURCE,
url != null ? url.toExternalForm() : null, null);
String desc;
if (url != null)
desc = resource + " @ " + url.toExternalForm();
else
desc = resource;
return new SimpleConfigOrigin(desc, -1, -1, OriginType.RESOURCE, url != null ? url.toExternalForm() : null,
resource, null);
}
static SimpleConfigOrigin newResource(String resource) {
@ -75,14 +82,14 @@ final class SimpleConfigOrigin implements ConfigOrigin {
if (lineNumber == this.lineNumber && lineNumber == this.endLineNumber) {
return this;
} else {
return new SimpleConfigOrigin(this.description, lineNumber, lineNumber,
this.originType, this.urlOrNull, this.commentsOrNull);
return new SimpleConfigOrigin(this.description, lineNumber, lineNumber, this.originType, this.urlOrNull,
this.resourceOrNull, this.commentsOrNull);
}
}
SimpleConfigOrigin addURL(URL url) {
return new SimpleConfigOrigin(this.description, this.lineNumber, this.endLineNumber,
this.originType, url != null ? url.toExternalForm() : null, this.commentsOrNull);
return new SimpleConfigOrigin(this.description, this.lineNumber, this.endLineNumber, this.originType,
url != null ? url.toExternalForm() : null, this.resourceOrNull, this.commentsOrNull);
}
@Override
@ -90,8 +97,8 @@ final class SimpleConfigOrigin implements ConfigOrigin {
if (ConfigImplUtil.equalsHandlingNull(comments, this.commentsOrNull)) {
return this;
} else {
return new SimpleConfigOrigin(this.description, this.lineNumber, this.endLineNumber,
this.originType, this.urlOrNull, comments);
return new SimpleConfigOrigin(this.description, this.lineNumber, this.endLineNumber, this.originType,
this.urlOrNull, this.resourceOrNull, comments);
}
}
@ -101,8 +108,7 @@ final class SimpleConfigOrigin implements ConfigOrigin {
} else if (this.commentsOrNull == null) {
return withComments(comments);
} else {
List<String> merged = new ArrayList<String>(comments.size()
+ this.commentsOrNull.size());
List<String> merged = new ArrayList<String>(comments.size() + this.commentsOrNull.size());
merged.addAll(comments);
merged.addAll(this.commentsOrNull);
return withComments(merged);
@ -115,8 +121,7 @@ final class SimpleConfigOrigin implements ConfigOrigin {
} else if (this.commentsOrNull == null) {
return withComments(comments);
} else {
List<String> merged = new ArrayList<String>(comments.size()
+ this.commentsOrNull.size());
List<String> merged = new ArrayList<String>(comments.size() + this.commentsOrNull.size());
merged.addAll(this.commentsOrNull);
merged.addAll(comments);
return withComments(merged);
@ -125,8 +130,6 @@ final class SimpleConfigOrigin implements ConfigOrigin {
@Override
public String description() {
// not putting the URL in here for files and resources, because people
// parsing "file: line" syntax would hit the ":" in the URL.
if (lineNumber < 0) {
return description;
} else if (endLineNumber == lineNumber) {
@ -141,11 +144,10 @@ final class SimpleConfigOrigin implements ConfigOrigin {
if (other instanceof SimpleConfigOrigin) {
SimpleConfigOrigin otherOrigin = (SimpleConfigOrigin) other;
return this.description.equals(otherOrigin.description)
&& this.lineNumber == otherOrigin.lineNumber
&& this.endLineNumber == otherOrigin.endLineNumber
&& this.originType == otherOrigin.originType
&& ConfigImplUtil.equalsHandlingNull(this.urlOrNull, otherOrigin.urlOrNull);
return this.description.equals(otherOrigin.description) && this.lineNumber == otherOrigin.lineNumber
&& this.endLineNumber == otherOrigin.endLineNumber && this.originType == otherOrigin.originType
&& ConfigImplUtil.equalsHandlingNull(this.urlOrNull, otherOrigin.urlOrNull)
&& ConfigImplUtil.equalsHandlingNull(this.resourceOrNull, otherOrigin.resourceOrNull);
} else {
return false;
}
@ -159,17 +161,14 @@ final class SimpleConfigOrigin implements ConfigOrigin {
h = 41 * (h + originType.hashCode());
if (urlOrNull != null)
h = 41 * (h + urlOrNull.hashCode());
if (resourceOrNull != null)
h = 41 * (h + resourceOrNull.hashCode());
return h;
}
@Override
public String toString() {
// the url is only really useful on top of description for resources
if (originType == OriginType.RESOURCE && urlOrNull != null) {
return "ConfigOrigin(" + description + "," + urlOrNull + ")";
} else {
return "ConfigOrigin(" + description + ")";
}
return "ConfigOrigin(" + description + ")";
}
@Override
@ -208,11 +207,7 @@ final class SimpleConfigOrigin implements ConfigOrigin {
@Override
public String resource() {
if (originType == OriginType.RESOURCE) {
return description;
} else {
return null;
}
return resourceOrNull;
}
@Override
@ -291,6 +286,13 @@ final class SimpleConfigOrigin implements ConfigOrigin {
mergedURL = null;
}
String mergedResource;
if (ConfigImplUtil.equalsHandlingNull(a.resourceOrNull, b.resourceOrNull)) {
mergedResource = a.resourceOrNull;
} else {
mergedResource = null;
}
if (ConfigImplUtil.equalsHandlingNull(a.commentsOrNull, b.commentsOrNull)) {
mergedComments = a.commentsOrNull;
} else {
@ -301,8 +303,8 @@ final class SimpleConfigOrigin implements ConfigOrigin {
mergedComments.addAll(b.commentsOrNull);
}
return new SimpleConfigOrigin(mergedDesc, mergedStartLine, mergedEndLine, mergedType,
mergedURL, mergedComments);
return new SimpleConfigOrigin(mergedDesc, mergedStartLine, mergedEndLine, mergedType, mergedURL,
mergedResource, mergedComments);
}
private static int similarity(SimpleConfigOrigin a, SimpleConfigOrigin b) {
@ -322,6 +324,8 @@ final class SimpleConfigOrigin implements ConfigOrigin {
count += 1;
if (ConfigImplUtil.equalsHandlingNull(a.urlOrNull, b.urlOrNull))
count += 1;
if (ConfigImplUtil.equalsHandlingNull(a.resourceOrNull, b.resourceOrNull))
count += 1;
}
return count;
@ -331,8 +335,7 @@ final class SimpleConfigOrigin implements ConfigOrigin {
// common. we want to merge two lines in the same file rather than something
// else with one of the lines; because two lines in the same file can be
// better consolidated.
private static SimpleConfigOrigin mergeThree(SimpleConfigOrigin a, SimpleConfigOrigin b,
SimpleConfigOrigin c) {
private static SimpleConfigOrigin mergeThree(SimpleConfigOrigin a, SimpleConfigOrigin b, SimpleConfigOrigin c) {
if (similarity(a, b) >= similarity(b, c)) {
return mergeTwo(mergeTwo(a, b), c);
} else {
@ -397,6 +400,8 @@ final class SimpleConfigOrigin implements ConfigOrigin {
if (urlOrNull != null)
m.put(SerializedField.ORIGIN_URL, urlOrNull);
if (resourceOrNull != null)
m.put(SerializedField.ORIGIN_RESOURCE, resourceOrNull);
if (commentsOrNull != null)
m.put(SerializedField.ORIGIN_COMMENTS, commentsOrNull);
@ -422,16 +427,14 @@ final class SimpleConfigOrigin implements ConfigOrigin {
for (Map.Entry<SerializedField, Object> baseEntry : base.entrySet()) {
SerializedField f = baseEntry.getKey();
if (m.containsKey(f)
&& ConfigImplUtil.equalsHandlingNull(baseEntry.getValue(), m.get(f))) {
if (m.containsKey(f) && ConfigImplUtil.equalsHandlingNull(baseEntry.getValue(), m.get(f))) {
// if field is unchanged, just remove it so we inherit
m.remove(f);
} else if (!m.containsKey(f)) {
// if field has been removed, we have to add a deletion entry
switch (f) {
case ORIGIN_DESCRIPTION:
throw new ConfigException.BugOrBroken("origin missing description field? "
+ child);
throw new ConfigException.BugOrBroken("origin missing description field? " + child);
case ORIGIN_LINE_NUMBER:
m.put(SerializedField.ORIGIN_LINE_NUMBER, -1);
break;
@ -443,14 +446,17 @@ final class SimpleConfigOrigin implements ConfigOrigin {
case ORIGIN_URL:
m.put(SerializedField.ORIGIN_NULL_URL, "");
break;
case ORIGIN_RESOURCE:
m.put(SerializedField.ORIGIN_NULL_RESOURCE, "");
break;
case ORIGIN_COMMENTS:
m.put(SerializedField.ORIGIN_NULL_COMMENTS, "");
break;
case ORIGIN_NULL_URL: // FALL THRU
case ORIGIN_NULL_RESOURCE: // FALL THRU
case ORIGIN_NULL_COMMENTS:
throw new ConfigException.BugOrBroken(
"computing delta, base object should not contain " + f
+ " " + base);
throw new ConfigException.BugOrBroken("computing delta, base object should not contain " + f + " "
+ base);
case END_MARKER:
case ROOT_VALUE:
case ROOT_WAS_CONFIG:
@ -480,10 +486,16 @@ final class SimpleConfigOrigin implements ConfigOrigin {
throw new IOException("Missing ORIGIN_TYPE field");
OriginType originType = OriginType.values()[originTypeOrdinal.byteValue()];
String urlOrNull = (String) m.get(SerializedField.ORIGIN_URL);
String resourceOrNull = (String) m.get(SerializedField.ORIGIN_RESOURCE);
@SuppressWarnings("unchecked")
List<String> commentsOrNull = (List<String>) m.get(SerializedField.ORIGIN_COMMENTS);
// Older versions did not have a resource field, they stuffed it into
// the description.
if (originType == OriginType.RESOURCE && resourceOrNull == null) {
resourceOrNull = description;
}
return new SimpleConfigOrigin(description, lineNumber != null ? lineNumber : -1,
endLineNumber != null ? endLineNumber : -1, originType, urlOrNull, commentsOrNull);
endLineNumber != null ? endLineNumber : -1, originType, urlOrNull, resourceOrNull, commentsOrNull);
}
static Map<SerializedField, Object> applyFieldsDelta(Map<SerializedField, Object> base,
@ -510,6 +522,13 @@ final class SimpleConfigOrigin implements ConfigOrigin {
m.put(f, base.get(f));
}
break;
case ORIGIN_RESOURCE:
if (delta.containsKey(SerializedField.ORIGIN_NULL_RESOURCE)) {
m.remove(SerializedField.ORIGIN_NULL_RESOURCE);
} else {
m.put(f, base.get(f));
}
break;
case ORIGIN_COMMENTS:
if (delta.containsKey(SerializedField.ORIGIN_NULL_COMMENTS)) {
m.remove(SerializedField.ORIGIN_NULL_COMMENTS);
@ -518,11 +537,12 @@ final class SimpleConfigOrigin implements ConfigOrigin {
}
break;
case ORIGIN_NULL_URL: // FALL THRU
case ORIGIN_NULL_RESOURCE: // FALL THRU
case ORIGIN_NULL_COMMENTS: // FALL THRU
// base objects shouldn't contain these, should just
// lack the field. these are only in deltas.
throw new ConfigException.BugOrBroken(
"applying fields, base object should not contain " + f + " " + base);
throw new ConfigException.BugOrBroken("applying fields, base object should not contain " + f + " "
+ base);
case ORIGIN_END_LINE_NUMBER: // FALL THRU
case ORIGIN_LINE_NUMBER: // FALL THRU
case ORIGIN_TYPE:
@ -542,8 +562,8 @@ final class SimpleConfigOrigin implements ConfigOrigin {
return m;
}
static SimpleConfigOrigin fromBase(SimpleConfigOrigin baseOrigin,
Map<SerializedField, Object> delta) throws IOException {
static SimpleConfigOrigin fromBase(SimpleConfigOrigin baseOrigin, Map<SerializedField, Object> delta)
throws IOException {
Map<SerializedField, Object> baseFields;
if (baseOrigin != null)
baseFields = baseOrigin.toFields();

View File

@ -860,18 +860,27 @@ class ConfigTest extends TestUtils {
val conf = ConfigFactory.load("test01")
val o1 = conf.getValue("ints.fortyTwo").origin()
assertEquals("test01.conf: 3", o1.description)
// the checkout directory would be in between this startsWith and endsWith
assertTrue("description starts with resource '" + o1.description + "'", o1.description.startsWith("test01.conf @"))
assertTrue("description ends with url and line '" + o1.description + "'", o1.description.endsWith("/config/target/test-classes/test01.conf: 3"))
assertEquals("test01.conf", o1.resource)
assertTrue("url ends with resource file", o1.url.getPath.endsWith("/config/target/test-classes/test01.conf"))
assertEquals(3, o1.lineNumber)
val o2 = conf.getValue("fromJson1").origin()
assertEquals("test01.json: 2", o2.description)
// the checkout directory would be in between this startsWith and endsWith
assertTrue("description starts with json resource '" + o2.description + "'", o2.description.startsWith("test01.json @"))
assertTrue("description of json resource ends with url and line '" + o2.description + "'", o2.description.endsWith("/config/target/test-classes/test01.json: 2"))
assertEquals("test01.json", o2.resource)
assertTrue("url ends with json resource file", o2.url.getPath.endsWith("/config/target/test-classes/test01.json"))
assertEquals(2, o2.lineNumber)
val o3 = conf.getValue("fromProps.bool").origin()
assertEquals("test01.properties", o3.description)
// the checkout directory would be in between this startsWith and endsWith
assertTrue("description starts with props resource '" + o3.description + "'", o3.description.startsWith("test01.properties @"))
assertTrue("description of props resource ends with url '" + o3.description + "'", o3.description.endsWith("/config/target/test-classes/test01.properties"))
assertEquals("test01.properties", o3.resource)
assertTrue("url ends with props resource file", o3.url.getPath.endsWith("/config/target/test-classes/test01.properties"))
// we don't have line numbers for properties files
assertEquals(-1, o3.lineNumber)
}

View File

@ -896,7 +896,8 @@ class ConfigValueTest extends TestUtils {
SimpleConfigOrigin.newSimple("foo"),
SimpleConfigOrigin.newFile("/tmp/blahblah"),
SimpleConfigOrigin.newURL(new URL("http://example.com")),
SimpleConfigOrigin.newResource("myresource"))
SimpleConfigOrigin.newResource("myresource"),
SimpleConfigOrigin.newResource("myresource", new URL("file://foo/bar")))
val combos = bases.flatMap({
base =>
Seq(

View File

@ -561,8 +561,8 @@ class PublicApiTest extends TestUtils {
// check that each value has its own ConfigOrigin
val v1 = conf.getValue("ints.fortyTwo")
val v2 = conf.getValue("test-lib.fromTestLib")
assertEquals("test01.conf", v1.origin.resource)
assertEquals("test01.conf", v2.origin.resource)
assertEquals("v1 has right origin resource", "test01.conf", v1.origin.resource)
assertEquals("v2 has right origin resource", "test01.conf", v2.origin.resource)
assertEquals(v1.origin.resource, v2.origin.resource)
assertFalse("same urls in " + v1.origin + " " + v2.origin, v1.origin.url == v2.origin.url)
assertFalse(v1.origin.filename == v2.origin.filename)