Skip to content

Commit

Permalink
adding Jackson implementation
Browse files Browse the repository at this point in the history
  • Loading branch information
daviddenton committed Jan 5, 2024
1 parent c6affb0 commit 827e356
Show file tree
Hide file tree
Showing 7 changed files with 281 additions and 98 deletions.
79 changes: 62 additions & 17 deletions data4k/src/main/kotlin/dev/forkhandles/data/DataContainer.kt
Original file line number Diff line number Diff line change
Expand Up @@ -9,30 +9,75 @@ import dev.forkhandles.values.ValueFactory
*/
@Suppress("UNCHECKED_CAST")
abstract class DataContainer<CONTENT>(
protected val data: CONTENT,
existsFn: (CONTENT, String) -> Boolean,
getFn: (CONTENT, String) -> Any?
val data: CONTENT,
private val existsFn: (CONTENT, String) -> Boolean,
private val getFn: (CONTENT, String) -> Any?,
private val setFn: (CONTENT, String, Any?) -> Unit
) {
private val exists: DataContainer<CONTENT>.(String) -> Boolean = { existsFn(data, it) }
private val get: DataContainer<CONTENT>.(String) -> Any? = { getFn(data, it) }

fun <OUT> field() = DataProperty<DataContainer<CONTENT>, OUT>(exists, get)
//
// CORE FUNCTIONS - these are the only ones that should call the defined mappings
//

fun <OUT : Any?, NEXT> field(mapFn: (OUT) -> NEXT) = DataProperty<DataContainer<CONTENT>, NEXT>(exists) {
(get(it) as OUT)?.let(mapFn)
}
fun <OUT : Any?, NEXT> field(mapInFn: (OUT) -> NEXT, mapOutFn: (NEXT) -> OUT?) =
property<NEXT, OUT, OUT>(mapInFn, mapOutFn)

fun <OUT : DataContainer<CONTENT>?> obj(mapInFn: (CONTENT) -> OUT, mapOutFn: (OUT) -> CONTENT?) =
property<OUT, CONTENT, CONTENT>(mapInFn, mapOutFn)

fun <OUT, IN> list(mapInFn: (IN) -> OUT, mapOutFn: (OUT) -> IN?) =
property<List<OUT>, List<IN>, List<IN>>({ it.map(mapInFn) }, { it.mapNotNull(mapOutFn) })

//
// "PRE MAPPED" FUNCTIONS
//

fun <OUT> field() = field<OUT, OUT>({ it }, { it })

fun <OUT> list() = list<OUT, OUT>({ it }, { it })

fun <OUT, NEXT> field(mapInFn: (OUT) -> NEXT) = field(mapInFn) { error("no outbound mapping defined") }

fun <IN, OUT> list(mapInFn: (IN) -> OUT) =
list(mapInFn) { error("no outbound mapping defined") }

fun <IN : Any, OUT : Value<IN>, OUT2 : OUT?> field(factory: ValueFactory<OUT, IN>) = field(factory::of)
@JvmName("listDataContainer")
fun <OUT : DataContainer<CONTENT>?> list(mapInFn: (CONTENT) -> OUT) =
list(mapInFn) { it?.data }

fun <IN, OUT> list(mapFn: (IN) -> OUT) =
DataProperty<DataContainer<CONTENT>, List<OUT>>(exists) { get(it)?.let { (it as List<IN>).map(mapFn) } }
fun <OUT : DataContainer<CONTENT>?> obj(mapInFn: (CONTENT) -> OUT) =
obj(mapInFn) { it?.data }

fun <IN : Any, OUT : Value<IN>> list(factory: ValueFactory<OUT, IN>) = list(factory::of)
//
// VALUES4K functions
//

fun <IN : Any, OUT : Value<IN>> field(factory: ValueFactory<OUT, IN>) =
field(factory::of) { it.value }

fun <IN : Any, OUT : Value<IN>> list(factory: ValueFactory<OUT, IN>) =
list(factory::of) { it.value }

//
// UTILITY FUNCTIONS - for comparisons etc
//

override fun equals(other: Any?): Boolean {
if (this === other) return true
if (javaClass != other?.javaClass) return false

other as DataContainer<*>

return data == other.data
}

fun <OUT> list() = list<OUT, OUT> { it }
override fun hashCode() = data?.hashCode() ?: 0

fun <OUT : DataContainer<CONTENT>?> obj(mapFn: (CONTENT) -> OUT) =
DataProperty<DataContainer<CONTENT>, OUT>(exists) { (get(it) as CONTENT)?.let(mapFn) }
override fun toString() = data.toString()

fun obj() = DataProperty<DataContainer<CONTENT>, CONTENT>(exists) { get(it) as CONTENT? }
private fun <IN, OUT : Any?, OUT2> property(mapInFn: (OUT) -> IN, mapOutFn: (IN) -> OUT2?) =
DataProperty<DataContainer<CONTENT>, IN>(
{ existsFn(data, it) },
{ getFn(data, it)?.let { value -> value as OUT }?.let(mapInFn) },
{ name, value -> setFn(data, name, (value as IN?)?.let(mapOutFn)) })
}
10 changes: 7 additions & 3 deletions data4k/src/main/kotlin/dev/forkhandles/data/DataProperty.kt
Original file line number Diff line number Diff line change
@@ -1,13 +1,14 @@
package dev.forkhandles.data

import kotlin.properties.ReadOnlyProperty
import kotlin.properties.ReadWriteProperty
import kotlin.reflect.KProperty
import kotlin.reflect.jvm.jvmErasure

open class DataProperty<IN, OUT : Any?>(
private val existsFn: IN.(String) -> Boolean,
private val getFn: IN.(String) -> Any?
) : ReadOnlyProperty<IN, OUT> {
private val getFn: IN.(String) -> Any?,
private val setFn: IN.(String, Any?) -> Unit
) : ReadWriteProperty<IN, OUT> {
@Suppress("UNCHECKED_CAST")
override fun getValue(thisRef: IN, property: KProperty<*>): OUT {
val result = thisRef.getFn(property.name)
Expand All @@ -18,10 +19,13 @@ open class DataProperty<IN, OUT : Any?>(
thisRef.existsFn(property.name) -> throw NoSuchElementException("Value for field <${property.name}> is null")
else -> throw NoSuchElementException("Field <${property.name}> is missing")
}

property.returnType.jvmErasure.isInstance(result) -> result as OUT

else -> throw NoSuchElementException("Value for field <${property.name}> is not a ${property.returnType.jvmErasure} but ${result.javaClass.kotlin}")
}
}

override fun setValue(thisRef: IN, property: KProperty<*>, value: OUT) = thisRef.setFn(property.name, value)
}

Original file line number Diff line number Diff line change
@@ -1,7 +1,20 @@
package dev.forkhandles.data

import com.fasterxml.jackson.databind.JsonNode
import com.fasterxml.jackson.databind.node.*
import com.fasterxml.jackson.databind.node.ArrayNode
import com.fasterxml.jackson.databind.node.BooleanNode
import com.fasterxml.jackson.databind.node.BooleanNode.FALSE
import com.fasterxml.jackson.databind.node.BooleanNode.TRUE
import com.fasterxml.jackson.databind.node.DecimalNode
import com.fasterxml.jackson.databind.node.DoubleNode
import com.fasterxml.jackson.databind.node.FloatNode
import com.fasterxml.jackson.databind.node.IntNode
import com.fasterxml.jackson.databind.node.JsonNodeFactory.instance
import com.fasterxml.jackson.databind.node.LongNode
import com.fasterxml.jackson.databind.node.NullNode
import com.fasterxml.jackson.databind.node.ObjectNode
import com.fasterxml.jackson.databind.node.TextNode
import java.math.BigDecimal

/**
* Jackson JsonNode-based implementation of the DataContainer
Expand All @@ -10,21 +23,45 @@ abstract class JacksonDataContainer(input: JsonNode) :
DataContainer<JsonNode>(
input,
{ content, it -> content.has(it) },
{ content, it -> content[it]?.let(Companion::nodeToValue) }
{ content, it -> content[it]?.let(Companion::nodeToValue) },
{ node: JsonNode, name, value ->
(node as? ObjectNode)?.also {
node.set<JsonNode>(name, value.toNode())
} ?: error("Invalid node type ${input::class.java}")
}
) {

companion object {
private fun nodeToValue(input: JsonNode): Any? = when (input) {
is BooleanNode -> input.booleanValue()
is IntNode -> input.intValue()
is LongNode -> input.longValue()
is FloatNode -> input.floatValue()
is DecimalNode -> input.decimalValue()
is DoubleNode -> input.doubleValue()
is TextNode -> input.textValue()
is ArrayNode -> input.map(::nodeToValue)
is ObjectNode -> input
is NullNode -> null
else -> error("Invalid node type $input")
else -> error("Invalid node type ${input::class.java}")
}

private fun Any?.toNode(): JsonNode? =
when (this) {
null -> NullNode.instance
is JsonNode -> this
is DataContainer<*> -> data.toNode()
is Boolean -> if (this) TRUE else FALSE
is Int -> IntNode(this)
is Long -> LongNode(this)
is Float -> FloatNode(this)
is BigDecimal -> DecimalNode(this)
is Double -> DoubleNode(this)
is String -> TextNode(this)
is Iterable<*> -> ArrayNode(instance).also {
map { if (it is JsonNode) it else it.toNode() }.forEach(it::add)
}
else -> error("Cannot set value of type ${this::class.java}")
}
}
}
Original file line number Diff line number Diff line change
@@ -1,11 +1,10 @@
package dev.forkhandles.data

import kotlin.collections.Map

/**
* Map-based implementation of the DataContainer
*/
abstract class MapDataContainer(input: Map<String, Any?>) :
DataContainer<Map<String, Any?>>(input, { content, it -> content.containsKey(it) }, { content, it -> content[it] })


DataContainer<MutableMap<String, Any?>>(input.toMutableMap(), { content, it -> content.containsKey(it) },
{ content, it -> content[it] },
{ map, name, value -> map[name] = value }
)
Loading

0 comments on commit 827e356

Please sign in to comment.