From 12329114cf32de371ecb9fe8368cbc3f9480d2b8 Mon Sep 17 00:00:00 2001 From: Eric K Richardson Date: Mon, 23 Dec 2019 17:19:22 -0800 Subject: [PATCH] Dotty update #3 (#62) Official Dotty support in master with tests omitted due to lack of serialization support. * Add Dotty 0.20.0-RC1 to build * Fix non-local return in ConfigBeanImpl * Fix non-local return in BadMap * Fix non-local return in ConfigNodeField * Fix non-local return in Path * Fix non-local return in SerializedConfigValue * Fix non-local return in ConfigNodeRoot * Fix non-local return in AbstractConfigValue * Add lines between methods and format * Fix non-local return in SimpleConfigObject * Fix non-local return in ConfigNodeObject part 1 * Fix non-local return in ConfigNodeObject part 1 - redo * Reverse addition of Dotty so CI passes * Update dotty to 0.21.0-RC1 * Fix non-local return in ConfigNodeObject part 2 * Remove default case causing dotty warning * Fix non-local return in ConfigDelayedMergeObject * Add Dotty back in with tests filtered that don't pass --- build.sbt | 22 ++- .../ekrich/config/impl/ConfigBeanImpl.scala | 21 +-- .../config/impl/AbstractConfigValue.scala | 22 ++- .../scala/org/ekrich/config/impl/BadMap.scala | 9 +- .../impl/ConfigDelayedMergeObject.scala | 153 ++++++++++-------- .../ekrich/config/impl/ConfigNodeField.scala | 21 +-- .../ekrich/config/impl/ConfigNodeObject.scala | 72 ++++----- .../ekrich/config/impl/ConfigNodeRoot.scala | 20 +-- .../scala/org/ekrich/config/impl/Path.scala | 24 +-- .../config/impl/SerializedConfigValue.scala | 7 +- .../config/impl/SimpleConfigObject.scala | 98 ++++++----- 11 files changed, 253 insertions(+), 216 deletions(-) diff --git a/build.sbt b/build.sbt index bf61c4a7..2ca3b352 100644 --- a/build.sbt +++ b/build.sbt @@ -47,10 +47,10 @@ scalacOptions in (Compile, console) --= Seq( val scala211 = "2.11.12" val scala212 = "2.12.10" val scala213 = "2.13.1" -val dotty = "0.20.0-RC1" +val dotty = "0.21.0-RC1" val versionsBase = Seq(scala211, scala212, scala213) -val versionsJVM = versionsBase //:+ dotty +val versionsJVM = versionsBase :+ dotty val versionsJS = versionsBase val versionsNative = Seq(scala211) @@ -127,6 +127,24 @@ lazy val sconfig = crossProject(JVMPlatform, NativePlatform, JSPlatform) "-g", "-Xlint:unchecked" ), + // Dotty is missing serializable support + // Can Filter based on Test name but not method name with "erializ" + // so exclude the Tests with the 19 that cannot pass + // 530 - 19 = 511 Only 346 get run this way so we lose coverage + Test / testOptions := { + if (isDotty.value) + Seq( + Tests.Exclude( + Seq( + "org.ekrich.config.impl.ValidationTest", + "org.ekrich.config.impl.PublicApiTest", + "org.ekrich.config.impl.ConfigValueTest", + "org.ekrich.config.impl.ConfigTest" + ) + ) + ) + else Seq(Tests.Exclude(Seq())) + }, // because we test some global state such as singleton caches, // we have to run tests in serial. Test / parallelExecution := false, diff --git a/sconfig/jvm/src/main/scala/org/ekrich/config/impl/ConfigBeanImpl.scala b/sconfig/jvm/src/main/scala/org/ekrich/config/impl/ConfigBeanImpl.scala index 527a7d18..f3d2c4cd 100644 --- a/sconfig/jvm/src/main/scala/org/ekrich/config/impl/ConfigBeanImpl.scala +++ b/sconfig/jvm/src/main/scala/org/ekrich/config/impl/ConfigBeanImpl.scala @@ -15,6 +15,7 @@ import java.time.Duration import scala.reflect.ClassTag import scala.jdk.CollectionConverters._ import scala.util.control.Breaks._ +import scala.util.Try import org.ekrich.config.Config import org.ekrich.config.ConfigObject import org.ekrich.config.ConfigList @@ -327,17 +328,17 @@ object ConfigBeanImpl { else null private def hasAtLeastOneBeanProperty(clazz: Class[_]): Boolean = { - var beanInfo: BeanInfo = null - try beanInfo = Introspector.getBeanInfo(clazz) - catch { - case e: IntrospectionException => - return false - } - for (beanProp <- beanInfo.getPropertyDescriptors) { - if (beanProp.getReadMethod != null && beanProp.getWriteMethod != null) - return true + val beanInfoOpt = Try(Introspector.getBeanInfo(clazz)).toOption + beanInfoOpt match { + case None => false + case Some(beanInfo) => + beanInfo + .getPropertyDescriptors() + .exists( + beanProp => + beanProp.getReadMethod != null && beanProp.getWriteMethod != null + ) } - false } private def isOptionalProperty( diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/AbstractConfigValue.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/AbstractConfigValue.scala index 0bed5d92..a8bb141c 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/AbstractConfigValue.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/AbstractConfigValue.scala @@ -7,6 +7,8 @@ import java.{lang => jl} import java.{util => ju} import ju.Collections +import scala.jdk.CollectionConverters._ + import org.ekrich.config.ConfigException import org.ekrich.config.ConfigMergeable import org.ekrich.config.ConfigObject @@ -64,19 +66,13 @@ object AbstractConfigValue { def hasDescendantInList( list: ju.List[AbstractConfigValue], descendant: AbstractConfigValue - ): Boolean = { - import scala.jdk.CollectionConverters._ - for (v <- list.asScala) { - if (v == descendant) return true - } - // now the expensive traversal - for (v <- list.asScala) { - if (v.isInstanceOf[Container] && v - .asInstanceOf[Container] - .hasDescendant(descendant)) return true - } - false - } + ): Boolean = + list.asScala.exists(_ == descendant) || + // now the expensive traversal + list.asScala.exists { v => + v.isInstanceOf[Container] && + v.asInstanceOf[Container].hasDescendant(descendant) + } def indent( sb: jl.StringBuilder, diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/BadMap.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/BadMap.scala index f9e41c48..f07f6a7a 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/BadMap.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/BadMap.scala @@ -76,13 +76,8 @@ object BadMap { 983, 991, 997, 1009, /* now we start skipping some, this is arbitrary */ 2053, 3079, 4057, 7103, 10949, 16069, 32609, 65867, 104729) - private def nextPrime(i: Int): Int = { - for (p <- primes) { - if (p > i) return p - } - /* oh well */ - primes(primes.length - 1) - } + private def nextPrime(i: Int): Int = + primes.find(p => p > i).getOrElse(primes(primes.length - 1)) } final class BadMap[K, V] private ( diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigDelayedMergeObject.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigDelayedMergeObject.scala index bd42f274..b47ea5f9 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigDelayedMergeObject.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigDelayedMergeObject.scala @@ -5,8 +5,10 @@ package org.ekrich.config.impl import java.{lang => jl} import java.{util => ju} + +import scala.annotation.tailrec import scala.jdk.CollectionConverters._ -import scala.util.control.Breaks._ + import org.ekrich.config.ConfigException import org.ekrich.config.ConfigList import org.ekrich.config.ConfigMergeable @@ -165,6 +167,9 @@ final class ConfigDelayedMergeObject( override def keySet = throw ConfigDelayedMergeObject.notResolved override def size = throw ConfigDelayedMergeObject.notResolved override def values = throw ConfigDelayedMergeObject.notResolved + + // exercised in ValidationTest.validationFailedSerializable + // and ConfigTest.test01Serializable override def attemptPeekWithPartialResolve( key: String ): AbstractConfigValue = { @@ -179,77 +184,89 @@ final class ConfigDelayedMergeObject( // spirit. // we'll be able to return a key if we have a value that ignores // fallbacks, prior to any unmergeable values. - for (layer <- stack.asScala) { - breakable { - if (layer.isInstanceOf[AbstractConfigObject]) { - val objectLayer = - layer.asInstanceOf[AbstractConfigObject] - val v = - objectLayer.attemptPeekWithPartialResolve(key) - if (v != null) { - if (v.ignoresFallbacks) { - // we know we won't need to merge anything in to this value - return v - } else { - // we can't return this value because we know there are - // unmergeable values later in the stack that may - // contain values that need to be merged with this - // value. we'll throw the exception when we get to those - // unmergeable values, so continue here. - break // continue - } - } else if (layer.isInstanceOf[Unmergeable]) { - // an unmergeable object (which would be another - // ConfigDelayedMergeObject) can't know that a key is - // missing, so it can't return null; it can only return a - // value or throw NotPossibleToResolve - throw new ConfigException.BugOrBroken( - "should not be reached: unmergeable object returned null value" + @tailrec + def loop( + layers: List[AbstractConfigValue] + ): Either[ConfigException, AbstractConfigValue] = + layers match { + case Nil => + // If we get here, then we never found anything unresolved which means + // the ConfigDelayedMergeObject should not have existed. some + // invariant was violated. + Left( + new ConfigException.BugOrBroken( + "Delayed merge stack does not contain any unmergeable values" ) - } else { - // a non-unmergeable AbstractConfigObject that returned null - // for the key in question is not relevant, we can keep - // looking for a value. - break // continue - } - } else if (layer.isInstanceOf[Unmergeable]) { - throw new ConfigException.NotResolved( - s"Key '$key' is not available at '${origin.description}' because value at '${layer.origin.description}'" + - s" has not been resolved and may turn out to contain or hide '$key'." + - " Be sure to Config#resolve() before using a config object." ) - } else if (layer.resolveStatus eq ResolveStatus.UNRESOLVED) { - // if the layer is not an object, and not a substitution or merge, - // then it's something that's unresolved because it _contains_ - // an unresolved object... i.e. it's an array - if (!layer.isInstanceOf[ConfigList]) { - throw new ConfigException.BugOrBroken( - "Expecting a list here, not " + layer - ) - } - // all later objects will be hidden so we can say we won't find - // the key - return null - } else { - // non-object, but resolved, like an integer or something. - // has no children so the one we're after won't be in it. - // we would only have this in the stack in case something - // else "looks back" to it due to a cycle. - // anyway at this point we know we can't find the key anymore. - if (!layer.ignoresFallbacks) { - throw new ConfigException.BugOrBroken( - "resolved non-object should ignore fallbacks" - ) + case head :: tl => + head match { + case layer: AbstractConfigObject => { + layer.attemptPeekWithPartialResolve(key) match { + case v if v != null => + if (v.ignoresFallbacks) { + // we know we won't need to merge anything in to this value + Right(v) + } else { + // we can't return this value because we know there are + // unmergeable values later in the stack that may + // contain values that need to be merged with this + // value. we'll throw the exception when we get to those + // unmergeable values, so continue here. + loop(tl) + } + case _: Unmergeable => + // an unmergeable object (which would be another + // ConfigDelayedMergeObject) can't know that a key is + // missing, so it can't return null; it can only return a + // value or throw NotPossibleToResolve + throw new ConfigException.BugOrBroken( + "should not be reached: unmergeable object returned null value" + ) + case _ => + // a non-unmergeable AbstractConfigObject that returned null + // for the key in question is not relevant, we can keep + // looking for a value. + loop(tl) + } + } + case _: Unmergeable => + throw new ConfigException.NotResolved( + s"Key '$key' is not available at '${origin.description}' because value at '${head.origin.description}'" + + s" has not been resolved and may turn out to contain or hide '$key'." + + " Be sure to Config#resolve() before using a config object." + ) + case layer if (layer.resolveStatus eq ResolveStatus.UNRESOLVED) => + // if the layer is not an object, and not a substitution or merge, + // then it's something that's unresolved because it _contains_ + // an unresolved object... i.e. it's an array + if (!layer.isInstanceOf[ConfigList]) { + throw new ConfigException.BugOrBroken( + "Expecting a list here, not " + layer + ) + } + // all later objects will be hidden so we can say we won't find + // the key + Right(null) // can I get null from this?? + case _ => + // non-object, but resolved, like an integer or something. + // has no children so the one we're after won't be in it. + // we would only have this in the stack in case something + // else "looks back" to it due to a cycle. + // anyway at this point we know we can't find the key anymore. + if (!head.ignoresFallbacks) { + throw new ConfigException.BugOrBroken( + "resolved non-object should ignore fallbacks" + ) + } + Right(null) } - return null - } } + // run the logic + loop(stack.asScala.toList) match { + case Right(res) => + res + case Left(ex) => + throw ex } - // If we get here, then we never found anything unresolved which means - // the ConfigDelayedMergeObject should not have existed. some - // invariant was violated. - throw new ConfigException.BugOrBroken( - "Delayed merge stack does not contain any unmergeable values" - ) } } diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeField.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeField.scala index d2f425a9..140594bd 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeField.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeField.scala @@ -18,6 +18,7 @@ final class ConfigNodeField(_children: ju.Collection[AbstractConfigNode]) } tokens } + def replaceValue(newValue: AbstractConfigNodeValue): ConfigNodeField = { val childrenCopy = new ju.ArrayList[AbstractConfigNode](children) @@ -31,6 +32,7 @@ final class ConfigNodeField(_children: ju.Collection[AbstractConfigNode]) } throw new ConfigException.BugOrBroken("Field node doesn't have a value") } + def value: AbstractConfigNodeValue = { var i = 0 while (i < children.size) { @@ -40,6 +42,7 @@ final class ConfigNodeField(_children: ju.Collection[AbstractConfigNode]) } throw new ConfigException.BugOrBroken("Field node doesn't have a value") } + def path: ConfigNodePath = { var i = 0 while (i < children.size) { @@ -49,16 +52,16 @@ final class ConfigNodeField(_children: ju.Collection[AbstractConfigNode]) } throw new ConfigException.BugOrBroken("Field node doesn't have a path") } - private[impl] def separator: Token = { - for (child <- children.asScala) { - if (child.isInstanceOf[ConfigNodeSingleToken]) { - val t = child.asInstanceOf[ConfigNodeSingleToken].token - if ((t eq Tokens.PLUS_EQUALS) || (t eq Tokens.COLON) || (t eq Tokens.EQUALS)) - return t + + private[impl] def separator: Token = + children.asScala.iterator + .filter(_.isInstanceOf[ConfigNodeSingleToken]) + .map(_.asInstanceOf[ConfigNodeSingleToken].token) + .find { t => + (t eq Tokens.PLUS_EQUALS) || (t eq Tokens.COLON) || (t eq Tokens.EQUALS) } - } - null - } + .getOrElse(null) + private[impl] def comments: ju.List[String] = { val comments = new ju.ArrayList[String] for (child <- children.asScala) { diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeObject.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeObject.scala index 2ebf3a23..e0261fce 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeObject.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeObject.scala @@ -11,24 +11,23 @@ final class ConfigNodeObject private[impl] ( override def newNode(nodes: ju.Collection[AbstractConfigNode]) = new ConfigNodeObject(nodes) - def hasValue(desiredPath: Path): Boolean = { - for (node <- children.asScala) { - if (node.isInstanceOf[ConfigNodeField]) { - val field = node.asInstanceOf[ConfigNodeField] - val key = field.path.value - if (key == desiredPath || key.startsWith(desiredPath)) return true - else if (desiredPath.startsWith(key)) { - if (field.value.isInstanceOf[ConfigNodeObject]) { - val obj = - field.value.asInstanceOf[ConfigNodeObject] - val remainingPath = desiredPath.subPath(key.length) - if (obj.hasValue(remainingPath)) return true + def hasValue(desiredPath: Path): Boolean = + children.asScala.exists { + case field: ConfigNodeField => { + val path = field.path.value + if (path == desiredPath || path.startsWith(desiredPath)) true + else if (desiredPath.startsWith(path)) { + field.value match { + case obj: ConfigNodeObject => + val remainingPath = desiredPath.subPath(path.length) + if (obj.hasValue(remainingPath)) true + else false + case _ => false } - } + } else false } + case _ => false } - false - } protected def changeValueOnPath( desiredPath: Path, @@ -227,27 +226,28 @@ final class ConfigNodeObject private[impl] ( )) // If the path is of length greater than one, see if the value needs to be added further down if (path.length > 1) { - var i = children.size - 1 - while (i >= 0) { - breakable { - if (!children.get(i).isInstanceOf[ConfigNodeField]) - break // continue - val node = children.get(i).asInstanceOf[ConfigNodeField] - val key: Path = node.path.value - if (path.startsWith(key) && node.value - .isInstanceOf[ConfigNodeObject]) { - val remainingPath = desiredPath.subPath(key.length) - val newValue = node.value.asInstanceOf[ConfigNodeObject] - childrenCopy.set( - i, - node.replaceValue( - newValue.addValueOnPath(remainingPath, value, flavor) - ) - ) - return new ConfigNodeObject(childrenCopy) - } - } // end break for continue in Java - increment needed as it was a for loop - i -= 1 + val lastIndex = children.size - 1 + val index = children.asScala.reverse.indexWhere { v => + v match { + case node: ConfigNodeField => + val key: Path = node.path.value + path.startsWith(key) && node.value.isInstanceOf[ConfigNodeObject] + case _ => false + } + } + if (index != -1) { + val i = lastIndex - index + val node = children.get(i).asInstanceOf[ConfigNodeField] + val key: Path = node.path.value + val remainingPath = desiredPath.subPath(key.length) + val newValue = node.value.asInstanceOf[ConfigNodeObject] + childrenCopy.set( + i, + node.replaceValue( + newValue.addValueOnPath(remainingPath, value, flavor) + ) + ) + return new ConfigNodeObject(childrenCopy) } } // Otherwise, construct the new setting diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeRoot.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeRoot.scala index 42847827..ff50684d 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeRoot.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/ConfigNodeRoot.scala @@ -1,9 +1,10 @@ package org.ekrich.config.impl +import java.{util => ju} +import scala.jdk.CollectionConverters._ import org.ekrich.config.ConfigException import org.ekrich.config.ConfigOrigin import org.ekrich.config.ConfigSyntax -import java.{util => ju} final class ConfigNodeRoot private[impl] ( _children: ju.Collection[AbstractConfigNode], @@ -12,16 +13,15 @@ final class ConfigNodeRoot private[impl] ( override def newNode(nodes: ju.Collection[AbstractConfigNode]) = throw new ConfigException.BugOrBroken("Tried to indent the root object") - private[impl] def value: ConfigNodeComplexValue = { - import scala.jdk.CollectionConverters._ - for (node <- children.asScala) { - if (node.isInstanceOf[ConfigNodeComplexValue]) - return node.asInstanceOf[ConfigNodeComplexValue] + private[impl] def value: ConfigNodeComplexValue = + children.asScala.find(node => node.isInstanceOf[ConfigNodeComplexValue]) match { + case Some(node) => node.asInstanceOf[ConfigNodeComplexValue] + case None => + throw new ConfigException.BugOrBroken( + "ConfigNodeRoot did not contain a value" + ) } - throw new ConfigException.BugOrBroken( - "ConfigNodeRoot did not contain a value" - ) - } + private[impl] def setValue( desiredPath: String, value: AbstractConfigNodeValue, diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/Path.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/Path.scala index f08fa255..e2499d67 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/Path.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/Path.scala @@ -5,29 +5,15 @@ package org.ekrich.config.impl import java.{util => ju} import org.ekrich.config.ConfigException -import util.control.Breaks._ object Path { // this doesn't have a very precise meaning, just to reduce // noise from quotes in the rendered path for average cases - private[impl] def hasFunkyChars(s: String): Boolean = { - val length = s.length - if (length == 0) return false - var i = 0 - while (i < length) { - breakable { - val c = s.charAt(i) - if (Character.isLetterOrDigit(c) || c == '-' || c == '_') { - break // continue - } else { - return true - } - } - i += 1 - } - return false - } - def newKey(key: String): Path = new Path(key, null: Path) + private[impl] def hasFunkyChars(s: String): Boolean = + s.exists(c => !c.isLetterOrDigit && c != '-' && c != '_') + + def newKey(key: String): Path = new Path(key, null: Path) + def newPath(path: String): Path = PathParser.parsePath(path) private def convert(i: ju.Iterator[Path]): Seq[String] = { diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/SerializedConfigValue.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/SerializedConfigValue.scala index 8b5514a3..c40f052a 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/SerializedConfigValue.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/SerializedConfigValue.scala @@ -246,10 +246,6 @@ object SerializedConfigValue { i += 1 } new SimpleConfigObject(origin, map) - case null => - throw new IOException("Unknown serialized value type: " + stb) - case _ => // warning in Dotty "Unreachable case" because enum values are a closed set - throw new IOException("Unhandled serialized value type: " + st) } } @@ -287,8 +283,7 @@ object SerializedConfigValue { throw new IOException( "No value data found in serialization of value" ) - return value - break // break - was return value + break // break - previous set value } else if (code eq SerializedField.VALUE_DATA) { if (origin == null) throw new IOException("Origin must be stored before value data") diff --git a/sconfig/shared/src/main/scala/org/ekrich/config/impl/SimpleConfigObject.scala b/sconfig/shared/src/main/scala/org/ekrich/config/impl/SimpleConfigObject.scala index e57eab36..4ff5fba5 100644 --- a/sconfig/shared/src/main/scala/org/ekrich/config/impl/SimpleConfigObject.scala +++ b/sconfig/shared/src/main/scala/org/ekrich/config/impl/SimpleConfigObject.scala @@ -51,6 +51,7 @@ object SimpleConfigObject { result.value } } + // this is only Serializable to chill out a findbugs warning @SerialVersionUID(1L) private object RenderComparator { @@ -98,17 +99,11 @@ object SimpleConfigObject { private def mapEquals( a: ju.Map[String, ConfigValue], b: ju.Map[String, ConfigValue] - ): Boolean = { - if (a eq b) return true - val aKeys = a.keySet - val bKeys = b.keySet - if (!(aKeys == bKeys)) return false - - for (key <- aKeys.asScala) { - if (!(a.get(key) == b.get(key))) return false - } - true - } + ): Boolean = + if (a eq b) true + else if (a.keySet != b.keySet) false + else !a.keySet.asScala.exists(key => a.get(key) != b.get(key)) + private def mapHash(m: ju.Map[String, ConfigValue]) = { // the keys have to be sorted, otherwise we could be equal // to another map but have a different hashcode. @@ -121,10 +116,13 @@ object SimpleConfigObject { } 41 * (41 + keys.hashCode) + valuesHash } - private val EMPTY_NAME = "empty config" + + private val EMPTY_NAME = "empty config" + private val emptyInstance = empty(SimpleConfigOrigin.newSimple(EMPTY_NAME)) - // commented out temporarily, used in Java + private[impl] def empty(): SimpleConfigObject = emptyInstance + private[impl] def empty(origin: ConfigOrigin): SimpleConfigObject = if (origin == null) empty() else @@ -132,6 +130,7 @@ object SimpleConfigObject { origin, ju.Collections.emptyMap[String, AbstractConfigValue] ) + private[impl] def emptyMissing(baseOrigin: ConfigOrigin) = new SimpleConfigObject( SimpleConfigOrigin.newSimple(baseOrigin.description + " (not found)"), @@ -167,8 +166,10 @@ final class SimpleConfigObject( override def withOnlyKey(key: String): SimpleConfigObject = withOnlyPath(Path.newKey(key)) + override def withoutKey(key: String): SimpleConfigObject = withoutPath(Path.newKey(key)) + // gets the object with only the path if the path // exists, otherwise null if it doesn't. this ensures // that if we have { a : { b : 42 } } and do @@ -196,6 +197,7 @@ final class SimpleConfigObject( ignoresFallbacks ) } + override def withOnlyPath(path: Path): SimpleConfigObject = { val o = withOnlyPathOrNull(path) if (o == null) @@ -207,6 +209,7 @@ final class SimpleConfigObject( ) else o } + override def withoutPath(path: Path): SimpleConfigObject = { val key = path.first val next = path.remainder @@ -236,6 +239,7 @@ final class SimpleConfigObject( ) } } + override def withValue(key: String, v: ConfigValue): SimpleConfigObject = { if (v == null) throw new ConfigException.BugOrBroken( @@ -256,6 +260,7 @@ final class SimpleConfigObject( ignoresFallbacks ) } + override def withValue(path: Path, v: ConfigValue): SimpleConfigObject = { val key = path.first val next = path.remainder @@ -279,58 +284,61 @@ final class SimpleConfigObject( } } } + override def attemptPeekWithPartialResolve(key: String): AbstractConfigValue = value.get(key) + private def newCopy( newStatus: ResolveStatus, newOrigin: ConfigOrigin, newIgnoresFallbacks: Boolean ) = new SimpleConfigObject(newOrigin, value, newStatus, newIgnoresFallbacks) + override def newCopy( newStatus: ResolveStatus, newOrigin: ConfigOrigin ): SimpleConfigObject = newCopy(newStatus, newOrigin, ignoresFallbacks) + override def withFallbacksIgnored(): SimpleConfigObject = if (ignoresFallbacks) this else newCopy(resolveStatus, origin, true) + override def resolveStatus: ResolveStatus = ResolveStatus.fromBoolean(resolved) + override def replaceChild( child: AbstractConfigValue, replacement: AbstractConfigValue ): SimpleConfigObject = { val newChildren = new ju.HashMap[String, AbstractConfigValue](value) - for (old <- newChildren.entrySet.asScala) { - if (old.getValue eq child) { + newChildren.entrySet.asScala.find(_.getValue() eq child) match { + case Some(old) => if (replacement != null) old.setValue(replacement) else newChildren.remove(old.getKey) - return new SimpleConfigObject( + new SimpleConfigObject( origin, newChildren, ResolveStatus.fromValues(newChildren.values), ignoresFallbacks ) - } - } - throw new ConfigException.BugOrBroken( - "SimpleConfigObject.replaceChild did not find " + child + " in " + this - ) - } - override def hasDescendant(descendant: AbstractConfigValue): Boolean = { - for (child <- value.values.asScala) { - if (child eq descendant) return true - } - // now do the expensive search - for (child <- value.values.asScala) { - if (child.isInstanceOf[Container] && child - .asInstanceOf[Container] - .hasDescendant(descendant)) return true + case None => + throw new ConfigException.BugOrBroken( + "SimpleConfigObject.replaceChild did not find " + child + " in " + this + ) } - false } + // related to AbstractConfigValue.hasDescendantInList + override def hasDescendant(descendant: AbstractConfigValue): Boolean = + value.values.asScala.exists(_ eq descendant) || + // now the expensive traversal + value.values.asScala.exists { child => + child.isInstanceOf[Container] && + child.asInstanceOf[Container].hasDescendant(descendant) + } + override def unwrapped: ju.Map[String, AnyRef] = { val m = new ju.HashMap[String, AnyRef] @@ -339,6 +347,7 @@ final class SimpleConfigObject( } m } + override def mergedWithObject( abstractFallback: AbstractConfigObject ): SimpleConfigObject = { @@ -379,6 +388,7 @@ final class SimpleConfigObject( newCopy(newResolveStatus, origin, newIgnoresFallbacks) else this } + private def modify(modifier: AbstractConfigValue.NoExceptionsModifier) = try modifyMayThrow(modifier) catch { @@ -387,6 +397,7 @@ final class SimpleConfigObject( case e: Exception => throw new ConfigException.BugOrBroken("unexpected checked exception", e) } + @throws[Exception] private def modifyMayThrow(modifier: AbstractConfigValue.Modifier) = { var changes: ju.Map[String, AbstractConfigValue] = null @@ -431,6 +442,7 @@ final class SimpleConfigObject( ) } } + @throws[NotPossibleToResolve] override def resolveSubstitutions( context: ResolveContext, @@ -453,6 +465,7 @@ final class SimpleConfigObject( throw new ConfigException.BugOrBroken("unexpected checked exception", e) } } + override def relativized(prefix: Path): SimpleConfigObject = modify(new AbstractConfigValue.NoExceptionsModifier() { override def modifyChild( @@ -461,6 +474,7 @@ final class SimpleConfigObject( ): AbstractConfigValue = v.relativized(prefix) }) + override def render( sb: jl.StringBuilder, indentVal: Int, @@ -530,8 +544,11 @@ final class SimpleConfigObject( } if (atRoot && options.getFormatted) sb.append('\n') } + override def get(key: Any): AbstractConfigValue = value.get(key) - override def canEqual(other: Any): Boolean = other.isInstanceOf[ConfigObject] + + override def canEqual(other: Any): Boolean = other.isInstanceOf[ConfigObject] + override def equals(other: Any): Boolean = { // note that "origin" is deliberately NOT part of equality. // neither are other "extras" like ignoresFallbacks or resolve status. @@ -544,13 +561,18 @@ final class SimpleConfigObject( ) } else false } + override def hashCode : Int = { // note that "origin" is deliberately NOT part of equality SimpleConfigObject.mapHash(this) } + override def containsKey(key: Any): Boolean = value.containsKey(key) - override def keySet: ju.Set[String] = value.keySet + + override def keySet: ju.Set[String] = value.keySet + override def containsValue(v: Any): Boolean = value.containsValue(v) + override def entrySet: ju.Set[ju.Map.Entry[String, ConfigValue]] = { // total bloat just to work around lack of type variance val entries = new ju.HashSet[ju.Map.Entry[String, ConfigValue]] @@ -564,9 +586,13 @@ final class SimpleConfigObject( } entries } + override def isEmpty: Boolean = value.isEmpty - override def size: Int = value.size - override def values = new ju.HashSet[ConfigValue](value.values) + + override def size: Int = value.size + + override def values = new ju.HashSet[ConfigValue](value.values) + // serialization all goes through SerializedConfigValue @throws[ObjectStreamException] private def writeReplace(): Object = new SerializedConfigValue(this)