在Kotlin阅读和处理HOCON

时间:2016-05-07 19:28:18

标签: kotlin typesafe-config hocon

我想从HOCON(Typesafe配置)文件中读取以下配置到Kotlin。

tablename: {  
  columns: [
    { item: { type: integer, key: true, null: false } }
    { desc: { type: varchar, length: 64 } }
    { quantity: { type: integer, null: false } }
    { price: { type: decimal, precision: 14, scale: 3 } }
  ]
}

实际上我想提取关键列。到目前为止,我已尝试过以下内容。

val metadata = ConfigFactory.parseFile(metafile)
val keys = metadata.getObjectList("${tablename.toLowerCase()}.columns")
                   .filter { it.unwrapped().values.first().get("key") == true }

但它失败并出现以下错误。

Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, ???>.get(key: kotlin.String): ??? defined in kotlin.collections

很明显,Kotlin无法理解Map中“value”字段的数据类型。我如何申报或让Kotlin知道?

此地图中还有不同的类型和可选键。

PS:我知道Kotlin有几种包装,如Konfig和Klutter。我希望如果这很容易写,我可以避免使用另一个库。

更新1:

我尝试了以下内容。

it.unwrapped().values.first().get<String, Boolean>("key")

获取以下编译器错误。

Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, kotlin.Boolean>.get(key: kotlin.String): kotlin.Boolean? defined in kotlin.collections

这个

it.unwrapped().values.first().get<String, Boolean?>("key")

带输出

Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, kotlin.Boolean?>.get(key: kotlin.String): kotlin.Boolean? defined in kotlin.collections

更新2:

看看它在其他地方处理的方式,我想我可能需要使用反射。用我有限的曝光试试吧。到目前为止没有运气。

2 个答案:

答案 0 :(得分:8)

考虑一下你的代码,解构如下:

val keys = metadata.getObjectList("tablename.columns")
        .filter {
            val item:ConfigObject = it
            val unwrapped:Map<String,Any?> = item.unwrapped()
            val values:Collection<Any?> = unwrapped.values
            val firstValue:Any? = values.first()
            firstValue.get("key") == true // does not compile
        }

从上面的问题应该是显而易见的。您需要使用firstValue拥有Map的信息来帮助编译器:

val firstValueMap = firstValue as Map<String,Any?>
firstValueMap["key"] == true

答案 1 :(得分:2)

即使你没有使用Klutter,我也为它创建了一个更新,使ConfigObjectConfig行为统一相同。从Klutter版本1.17.1开始(今天推送到Maven中心),您可以根据您的问题执行以下单元测试中表示的内容。

查找关键列的函数:

fun findKeyColumns(cfg: Config, tableName: String): Map<String, ConfigObject> {
    return cfg.nested(tableName).value("columns").asObjectList()
            .map { it.keys.single() to it.value(it.keys.single()).asObject() }
            .filter {
                it.second.value("key").asBoolean(false)
            }
            .toMap()
}

这是完整的单元测试:

// from http://stackoverflow.com/questions/37092808/reading-and-processing-hocon-in-kotlin
@Test fun testFromSo37092808() {
    // === mocked configuration file

    val cfg = loadConfig(StringAsConfig("""
            products: {
              columns: [
                { item: { type: integer, key: true, null: false } }
                { desc: { type: varchar, length: 64 } }
                { quantity: { type: integer, null: false } }
                { price: { type: decimal, precision: 14, scale: 3 } }
              ]
            }
          """))

    // === function to find which columns are key columns

    fun findKeyColumns(cfg: Config, tableName: String): Map<String, ConfigObject> {
        return cfg.nested(tableName).value("columns").asObjectList()
                .map { it.keys.single() to it.value(it.keys.single()).asObject() }
                .filter {
                    it.second.value("key").asBoolean(false)
                }
                .toMap()
    }

    // === sample usage

    val productKeys = findKeyColumns(cfg, "products")

    // we only have 1 in the test data, so grab the name and the values
    val onlyColumnName = productKeys.entries.first().key
    val onlyColumnObj = productKeys.entries.first().value

    assertEquals ("item", onlyColumnName)
    assertEquals (true, onlyColumnObj.value("key").asBoolean())
    assertEquals ("integer", onlyColumnObj.value("type").asString())
    assertEquals (false, onlyColumnObj.value("null").asBoolean())
}

您可以返回上面的Map或列表名Pair的列表到设置映射,因为列名不在其设置中。

也可以更改配置文件的设计,使配置处理更简单(即配置对象内的表名,而不是左侧键。列名相同,添加到对象而不是左侧键。)