Pureconfig将配置读取为属性映射

时间:2020-10-16 20:11:21

标签: scala pureconfig

是否可以使pureconfig读取属性为Map[String, String]?我有以下

application.conf

cfg{
  some.property.name: "value"
  some.another.property.name: "another value"
}

这是我尝试使用以下命令读取配置的应用程序:

import pureconfig.generic.auto._
import pureconfig.ConfigSource
import pureconfig.error.ConfigReaderException

object Model extends App {
  case class Config(cfg: Map[String, String])

  val result = ConfigSource.default
    .load[Config]
    .left
    .map(err => new ConfigReaderException[Config](err))
    .toTry

  val config = result.get
  println(config)
}

问题在于它引发了以下排他性行为:

Exception in thread "main" pureconfig.error.ConfigReaderException: Cannot convert configuration to a Model$Config. Failures are:
  at 'cfg.some':
    - (application.conf @ file:/home/somename/prcfg/target/classes/application.conf: 2-3) Expected type STRING. Found OBJECT instead.

    at Model$.$anonfun$result$2(Model.scala:11)
    at scala.util.Either$LeftProjection.map(Either.scala:614)
    at Model$.delayedEndpoint$Model$1(Model.scala:11)
    at Model$delayedInit$body.apply(Model.scala:5)
    at scala.Function0.apply$mcV$sp(Function0.scala:39)
    at scala.Function0.apply$mcV$sp$(Function0.scala:39)
    at scala.runtime.AbstractFunction0.apply$mcV$sp(AbstractFunction0.scala:17)
    at scala.App.$anonfun$main$1(App.scala:73)
    at scala.App.$anonfun$main$1$adapted(App.scala:73)
    at scala.collection.IterableOnceOps.foreach(IterableOnce.scala:553)
    at scala.collection.IterableOnceOps.foreach$(IterableOnce.scala:551)
    at scala.collection.AbstractIterable.foreach(Iterable.scala:920)
    at scala.App.main(App.scala:73)
    at scala.App.main$(App.scala:71)
    at Model$.main(Model.scala:5)
    at Model.main(Model.scala)

是否有解决方法?我希望Map[String, String]将包含以下映射:

some.property.name -> "value"
some.another.property.name -> "another value"

2 个答案:

答案 0 :(得分:2)

您的问题不是pureconfig。您的问题是根据HOCON spec编写的内容:

cfg {
  some.property.name: "value"
  some.another.property.name: "another value"
}

是以下语言的语法糖:

cfg {
  some {
    property {
      name = "value"
    }
  }
  
  another {
    property {
      name = "another value"
    }
  }
}

由TypeSafe Config / Lightbend Config决定您的cfg具有两个属性,并且它们都是嵌套配置。 Pureconfig仅采用这些嵌套的配置,并将它们映射到案例类中。但它无法映射结构与预期完全不同的事物。

如果您写:

cfg {
  some-property-name: "value"
  some-another-property-name: "another value"
}

您将能够将"cfg"路径解码为Map[String, String],将顶级配置解码为case class Config(cfg: Map[String, String])。如果您想将.视为键的一部分而不是嵌套的话……那么恐怕您必须自己编写ConfigReader,因为这是非标准用法。

答案 1 :(得分:2)

您可以使用以下Map[String, String]来读取ConfigReader

implicit val strMapReader: ConfigReader[Map[String, String]] = {
  implicit val r: ConfigReader[String => Map[String, String]] =
    ConfigReader[String]
      .map(v => (prefix: String) => Map(prefix -> v))
      .orElse { strMapReader.map { v =>
        (prefix: String) => v.map { case (k, v2) => s"$prefix.$k" -> v2 }
      }}
  ConfigReader[Map[String, String => Map[String, String]]].map {
    _.flatMap { case (prefix, v) => v(prefix) }
  }
}

请注意,这是递归的val定义,因为strMapReader在其自己的定义中使用。之所以起作用,是因为orElse方法按名称而不是按值获取参数。