diff --git a/config/src/main/java/com/typesafe/config/impl/ConfigNodeObject.java b/config/src/main/java/com/typesafe/config/impl/ConfigNodeObject.java
index 7d1a28c5..a7403110 100644
--- a/config/src/main/java/com/typesafe/config/impl/ConfigNodeObject.java
+++ b/config/src/main/java/com/typesafe/config/impl/ConfigNodeObject.java
@@ -12,32 +12,83 @@ final class ConfigNodeObject extends ConfigNodeComplexValue {
         super(children);
     }
 
-    protected ConfigNodeObject changeValueOnPath(Path desiredPath, AbstractConfigNodeValue value) {
+    public boolean hasValue(Path desiredPath) {
+        for (AbstractConfigNode node : children) {
+            if (node instanceof ConfigNodeField) {
+                Path key = ((ConfigNodeField) node).path().value();
+                if (key.equals(desiredPath) || key.startsWith(desiredPath)) {
+                    return true;
+                } else if (desiredPath.startsWith(key)) {
+                    if (((ConfigNodeField) node).value() instanceof ConfigNodeObject) {
+                        Path remainingPath = desiredPath.subPath(key.length());
+                        if (((ConfigNodeObject) ((ConfigNodeField) node).value()).hasValue(remainingPath)) {
+                            return true;
+                        }
+                    }
+                }
+            }
+        }
+        return false;
+    }
+
+    protected ConfigNodeObject changeValueOnPath(Path desiredPath, AbstractConfigNodeValue value, ConfigSyntax flavor) {
         ArrayList<AbstractConfigNode> childrenCopy = new ArrayList<AbstractConfigNode>(super.children);
+        boolean seenNonMatching = false;
         // Copy the value so we can change it to null but not modify the original parameter
         AbstractConfigNodeValue valueCopy = value;
-        for (int i = super.children.size() - 1; i >= 0; i--) {
-            if (!(super.children.get(i) instanceof ConfigNodeField)) {
+        for (int i = childrenCopy.size() - 1; i >= 0; i--) {
+            if (childrenCopy.get(i) instanceof ConfigNodeSingleToken) {
+                Token t = ((ConfigNodeSingleToken) childrenCopy.get(i)).token();
+                // Ensure that, when we are removing settings in JSON, we don't end up with a trailing comma
+                if (flavor == ConfigSyntax.JSON && !seenNonMatching && t == Tokens.COMMA) {
+                    childrenCopy.remove(i);
+                }
+                continue;
+            }else if (!(childrenCopy.get(i) instanceof ConfigNodeField)) {
                 continue;
             }
-            ConfigNodeField node = (ConfigNodeField)super.children.get(i);
+            ConfigNodeField node = (ConfigNodeField)childrenCopy.get(i);
             Path key = node.path().value();
-            if (key.equals(desiredPath)) {
-                if (valueCopy == null)
+            if (key.equals(desiredPath) || key.startsWith(desiredPath)) {
+                if (valueCopy == null) {
                     childrenCopy.remove(i);
-                else {
+                    // Remove any whitespace or commas after the deleted setting
+                    for (int j = i; j < childrenCopy.size(); j++) {
+                        if (childrenCopy.get(j) instanceof ConfigNodeSingleToken) {
+                            Token t = ((ConfigNodeSingleToken) childrenCopy.get(j)).token();
+                            if (Tokens.isIgnoredWhitespace(t) || t == Tokens.COMMA) {
+                                childrenCopy.remove(j);
+                                j--;
+                            } else {
+                                break;
+                            }
+                        } else {
+                            break;
+                        }
+                    }
+                }
+                else if (key.equals(desiredPath)){
+                    seenNonMatching = true;
                     childrenCopy.set(i, node.replaceValue(value));
                     valueCopy = null;
                 }
             } else if (desiredPath.startsWith(key)) {
+                seenNonMatching = true;
                 if (node.value() instanceof ConfigNodeObject) {
                     Path remainingPath = desiredPath.subPath(key.length());
-                    childrenCopy.set(i, node.replaceValue(((ConfigNodeObject)node.value()).changeValueOnPath(remainingPath, valueCopy)));
+                    childrenCopy.set(i, node.replaceValue(((ConfigNodeObject)node.value()).changeValueOnPath(remainingPath, valueCopy, flavor)));
                     if (valueCopy != null && !node.equals(super.children.get(i)))
                         valueCopy = null;
                 }
+            } else {
+                seenNonMatching = true;
             }
         }
+
+        // Since we've removed values and valid JSON does not allow trailing commas, remove a comma after the final setting
+        if (flavor == ConfigSyntax.JSON) {
+
+        }
         return new ConfigNodeObject(childrenCopy);
     }
 
@@ -51,10 +102,10 @@ final class ConfigNodeObject extends ConfigNodeComplexValue {
     }
 
     private ConfigNodeObject setValueOnPath(ConfigNodePath desiredPath, AbstractConfigNodeValue value, ConfigSyntax flavor) {
-        ConfigNodeObject node = changeValueOnPath(desiredPath.value(), value);
+        ConfigNodeObject node = changeValueOnPath(desiredPath.value(), value, flavor);
 
         // If the desired Path did not exist, add it
-        if (node.render().equals(render())) {
+        if (node.equals(this)) {
             return addValueOnPath(desiredPath, value, flavor);
         }
         return node;
@@ -120,8 +171,8 @@ final class ConfigNodeObject extends ConfigNodeComplexValue {
         return new ConfigNodeObject(childrenCopy);
     }
 
-    public ConfigNodeComplexValue removeValueOnPath(String desiredPath) {
-        Path path = PathParser.parsePath(desiredPath);
-        return changeValueOnPath(path, null);
+    public ConfigNodeObject removeValueOnPath(String desiredPath, ConfigSyntax flavor) {
+        Path path = PathParser.parsePathNode(desiredPath, flavor).value();
+        return changeValueOnPath(path, null, flavor);
     }
 }
diff --git a/config/src/main/java/com/typesafe/config/impl/ConfigNodeRoot.java b/config/src/main/java/com/typesafe/config/impl/ConfigNodeRoot.java
index 34421855..72ca612d 100644
--- a/config/src/main/java/com/typesafe/config/impl/ConfigNodeRoot.java
+++ b/config/src/main/java/com/typesafe/config/impl/ConfigNodeRoot.java
@@ -30,13 +30,33 @@ final class ConfigNodeRoot extends ConfigNodeComplexValue {
             AbstractConfigNode node = childrenCopy.get(i);
             if (node instanceof ConfigNodeComplexValue) {
                 if (node instanceof ConfigNodeArray) {
-                    throw new ConfigException.WrongType(origin, "The ConfigDocument had an array at the root level, and values cannot be replaced inside an array.");
+                    throw new ConfigException.WrongType(origin, "The ConfigDocument had an array at the root level, and values cannot be modified inside an array.");
                 } else if (node instanceof ConfigNodeObject) {
-                    childrenCopy.set(i, ((ConfigNodeObject) node).setValueOnPath(desiredPath, value, flavor));
+                    if (value == null) {
+                        childrenCopy.set(i, ((ConfigNodeObject)node).removeValueOnPath(desiredPath, flavor));
+                    } else {
+                        childrenCopy.set(i, ((ConfigNodeObject) node).setValueOnPath(desiredPath, value, flavor));
+                    }
                     return new ConfigNodeRoot(childrenCopy, origin);
                 }
             }
         }
         throw new ConfigException.BugOrBroken("ConfigNodeRoot did not contain a value");
     }
+
+    protected boolean hasValue(String desiredPath) {
+        Path path = PathParser.parsePath(desiredPath);
+        ArrayList<AbstractConfigNode> childrenCopy = new ArrayList<AbstractConfigNode>(children);
+        for (int i = 0; i < childrenCopy.size(); i++) {
+            AbstractConfigNode node = childrenCopy.get(i);
+            if (node instanceof ConfigNodeComplexValue) {
+                if (node instanceof ConfigNodeArray) {
+                    throw new ConfigException.WrongType(origin, "The ConfigDocument had an array at the root level, and values cannot be modified inside an array.");
+                } else if (node instanceof ConfigNodeObject) {
+                    return ((ConfigNodeObject) node).hasValue(path);
+                }
+            }
+        }
+        throw new ConfigException.BugOrBroken("ConfigNodeRoot did not contain a value");
+    }
 }
diff --git a/config/src/main/java/com/typesafe/config/impl/SimpleConfigDocument.java b/config/src/main/java/com/typesafe/config/impl/SimpleConfigDocument.java
index f0fc5154..217ca238 100644
--- a/config/src/main/java/com/typesafe/config/impl/SimpleConfigDocument.java
+++ b/config/src/main/java/com/typesafe/config/impl/SimpleConfigDocument.java
@@ -31,7 +31,25 @@ final class SimpleConfigDocument implements ConfigDocument {
         return setValue(path, newValue.render());
     }
 
+    public ConfigDocument removeValue(String path) {
+        return new SimpleConfigDocument(configNodeTree.setValue(path, null, parseOptions.getSyntax()), parseOptions);
+    }
+
+    public boolean hasValue(String path) {
+        return configNodeTree.hasValue(path);
+    }
+
     public String render() {
         return configNodeTree.render();
     }
+
+    @Override
+    public boolean equals(Object other) {
+        return other instanceof ConfigDocument && render().equals(((ConfigDocument) other).render());
+    }
+
+    @Override
+    public int hashCode() {
+        return render().hashCode();
+    }
 }
diff --git a/config/src/main/java/com/typesafe/config/parser/ConfigDocument.java b/config/src/main/java/com/typesafe/config/parser/ConfigDocument.java
index c71e5a33..782c5021 100644
--- a/config/src/main/java/com/typesafe/config/parser/ConfigDocument.java
+++ b/config/src/main/java/com/typesafe/config/parser/ConfigDocument.java
@@ -54,6 +54,24 @@ public interface ConfigDocument {
      */
     ConfigDocument setValue(String path, ConfigValue newValue);
 
+    /**
+     * Returns a new ConfigDocument that is a copy of the current ConfigDocument, but with
+     * the value at the desired path removed. If the desired path does not exist in the document,
+     * a copy of the current document will be returned. If there is an array at the root, an exception
+     * will be thrown.
+     *
+     * @param path the path to remove from the document
+     * @return a copy of the ConfigDocument with the desired value removed from the document.
+     */
+    ConfigDocument removeValue(String path);
+
+    /**
+     * Returns a boolean indicating whether or not a ConfigDocument has a value at the desired path.
+     * @param path the path to check
+     * @return true if the path exists in the document, otherwise false
+     */
+    boolean hasValue(String path);
+
     /**
      * The original text of the input, modified if necessary with
      * any replaced or added values.
diff --git a/config/src/test/scala/com/typesafe/config/impl/ConfigDocumentTest.scala b/config/src/test/scala/com/typesafe/config/impl/ConfigDocumentTest.scala
index c8ba589c..537b13cd 100644
--- a/config/src/test/scala/com/typesafe/config/impl/ConfigDocumentTest.scala
+++ b/config/src/test/scala/com/typesafe/config/impl/ConfigDocumentTest.scala
@@ -180,7 +180,39 @@ class ConfigDocumentTest extends TestUtils {
     }
 
     @Test
-    def configDocumentArrayReplaceFailure {
+    def configDocumentHasValue {
+        val origText = "{a: b, a.b.c.d: e, c: {a: {b: c}}}"
+        val configDoc = ConfigDocumentFactory.parseString(origText)
+
+        assertTrue(configDoc.hasValue("a"))
+        assertTrue(configDoc.hasValue("a.b.c"))
+        assertTrue(configDoc.hasValue("c.a.b"))
+        assertFalse(configDoc.hasValue("c.a.b.c"))
+        assertFalse(configDoc.hasValue("a.b.c.d.e"))
+        assertFalse(configDoc.hasValue("this.does.not.exist"))
+    }
+
+    @Test
+    def configDocumentRemoveValue {
+        val origText = "{a: b, a.b.c.d: e, c: {a: {b: c}}}"
+        val configDoc = ConfigDocumentFactory.parseString(origText)
+
+        assertEquals("{c: {a: {b: c}}}", configDoc.removeValue("a").render())
+        assertEquals("{a: b, a.b.c.d: e, }", configDoc.removeValue("c").render())
+        assertEquals(configDoc, configDoc.removeValue("this.does.not.exist"))
+    }
+
+    @Test
+    def configDocumentRemoveValueJSON {
+        val origText = """{"a": "b", "c": "d"}"""
+        val configDoc = ConfigDocumentFactory.parseString(origText, ConfigParseOptions.defaults().setSyntax(ConfigSyntax.JSON))
+
+        // Ensure that removing a value in JSON does not leave us with a trailing comma
+        assertEquals("""{"a": "b" }""", configDoc.removeValue("c").render())
+    }
+
+    @Test
+    def configDocumentArrayFailures {
         // Attempting a replace on a ConfigDocument parsed from an array throws an error
         val origText = "[1, 2, 3, 4, 5]"
         val document = ConfigDocumentFactory.parseString(origText)
@@ -194,6 +226,28 @@ class ConfigDocumentTest extends TestUtils {
                 assertTrue(e.getMessage.contains("ConfigDocument had an array at the root level"))
         }
         assertTrue(exceptionThrown)
+
+        exceptionThrown = false;
+        try {
+            document.hasValue("a")
+        } catch {
+            case e: Exception =>
+                exceptionThrown = true
+                assertTrue(e.isInstanceOf[ConfigException])
+                assertTrue(e.getMessage.contains("ConfigDocument had an array at the root level"))
+        }
+        assertTrue(exceptionThrown)
+
+        exceptionThrown = false
+        try {
+            document.removeValue("a")
+        } catch {
+            case e: Exception =>
+                exceptionThrown = true
+                assertTrue(e.isInstanceOf[ConfigException])
+                assertTrue(e.getMessage.contains("ConfigDocument had an array at the root level"))
+        }
+        assertTrue(exceptionThrown)
     }
 
     @Test
diff --git a/config/src/test/scala/com/typesafe/config/impl/ConfigNodeTest.scala b/config/src/test/scala/com/typesafe/config/impl/ConfigNodeTest.scala
index 40cb76f1..08c7b093 100644
--- a/config/src/test/scala/com/typesafe/config/impl/ConfigNodeTest.scala
+++ b/config/src/test/scala/com/typesafe/config/impl/ConfigNodeTest.scala
@@ -212,7 +212,7 @@ class ConfigNodeTest extends TestUtils {
             nodeCloseBrace))
         assertEquals(origText, origNode.render())
         val finalText = "foo : bar\nbaz : {\n\t\"abc.def\" : true\n\t//This is a comment about the below setting\n\n\tabc : {\n\t\t" +
-            "def : false\n\t\n\"this.does.not.exist@@@+$#\" : {\nend : doesnotexist\n}\n}\n}\nbaz.abc.ghi : randomunquotedString\n}"
+            "def : false\n\n\"this.does.not.exist@@@+$#\" : {\nend : doesnotexist\n}\n}\n}\nbaz.abc.ghi : randomunquotedString\n}"
 
         //Can replace settings in nested maps
         // Paths with quotes in the name are treated as a single Path, rather than multiple sub-paths