com.jtransc.lang.Dynamic.kt Maven / Gradle / Ivy
package com.jtransc.lang
import com.jtransc.error.InvalidOperationException
import com.jtransc.error.noImpl
import java.lang.reflect.ParameterizedType
import java.lang.reflect.Type
import java.util.*
object Dynamic {
@Suppress("UNCHECKED_CAST")
fun createEmptyClass(clazz: Class): T {
if (clazz == java.util.List::class.java) return listOf() as T
if (clazz == java.util.Map::class.java) return mapOf() as T
if (clazz == java.lang.Iterable::class.java) return listOf() as T
val constructor = clazz.constructors.first()
val args = constructor.parameterTypes.map {
dynamicCast(null, it)
}
return constructor.newInstance(*args.toTypedArray()) as T
}
fun setField(instance: T, name: String, value: Any?) {
val field = instance.javaClass.declaredFields.find { it.name == name }
//val field = instance.javaClass.getField(name)
field?.isAccessible = true
field?.set(instance, value)
}
fun getField(instance: T?, name: String): Any? {
if (instance == null) return null
val getter = instance.javaClass.getMethod("get${name.capitalize()}")
if (getter != null) {
getter.isAccessible = true
return getter.invoke(instance)
} else {
val field = instance.javaClass.declaredFields.find { it.name == name }
//val field = instance.javaClass.getField(name)
field?.isAccessible = true
return field?.get(instance)
}
}
fun toNumber(it: Any?): Double {
return when (it) {
null -> 0.0
is Number -> it.toDouble()
else -> it.toString().toDouble()
}
}
fun toInt(it: Any?): Int {
return toNumber(it).toInt()
}
fun toBool(it: Any?): Boolean {
return when (it) {
null -> false
is Boolean -> it
is String -> it.isNotEmpty() && it != "0" && it != "false"
else -> toInt(it) != 0
}
}
fun toIterable(it: Any?): Iterable<*> {
return when (it) {
null -> listOf()
is Iterable<*> -> it
is CharSequence -> it.toList()
else -> listOf()
}
}
fun toComparable(it: Any?): Comparable {
return when (it) {
null -> 0 as Comparable
is Comparable<*> -> it as Comparable
else -> it.toString() as Comparable
}
}
fun compare(l: Any?, r: Any?): Int {
val lc = toComparable(l)
val rc = toComparable(r)
if (lc.javaClass.isAssignableFrom(rc.javaClass)) {
return lc.compareTo(rc)
} else {
return -1
}
}
fun accessAny(instance: Any?, key:Any?): Any? {
return when (instance) {
null -> null
is Map<*, *> -> instance[key]
is Iterable<*> -> instance.toList()[toInt(key)]
else -> getField(instance, key.toString())
}
}
fun setAny(instance: Any?, key:Any?, value: Any?): Any? {
return when (instance) {
null -> null
is MutableMap<*, *> -> (instance as MutableMap).set(key, value)
is MutableList<*> -> (instance as MutableList)[toInt(key)] = value
else -> setField(instance, key.toString(), value)
}
}
fun hasField(javaClass: Class, name: String): Boolean {
return javaClass.declaredFields.any { it.name == name }
}
fun getFieldType(javaClass: Class, name: String): Class<*> {
return javaClass.getField(name).type
}
inline fun dynamicCast(value: Any?): T? = dynamicCast(value, T::class.java)
@Suppress("UNCHECKED_CAST")
fun dynamicCast(value: Any?, target: Class, genericType: Type? = null): T? {
if (value != null && target.isAssignableFrom(value.javaClass)) {
return if (genericType != null && genericType is ParameterizedType) {
val typeArgs = genericType.actualTypeArguments
when(value) {
is List<*> -> value.map { dynamicCast(it, typeArgs[0] as Class) }
else -> value
} as T
} else {
value as T
}
}
val str = if (value != null) "$value" else "0"
if (target.isPrimitive) {
when (target) {
java.lang.Boolean.TYPE -> return (str == "true" || str == "1") as T
java.lang.Byte.TYPE -> return str.parseInt().toByte() as T
java.lang.Character.TYPE -> return str.parseInt().toChar() as T
java.lang.Short.TYPE -> return str.parseInt().toShort() as T
java.lang.Long.TYPE -> return str.parseLong() as T
java.lang.Float.TYPE -> return str.toFloat() as T
java.lang.Double.TYPE -> return str.parseDouble() as T
java.lang.Integer.TYPE -> return str.parseInt() as T
else -> throw InvalidOperationException("Unhandled primitive '${target.name}'")
}
}
if (target.isAssignableFrom(java.lang.Boolean::class.java)) return (str == "true" || str == "1") as T
if (target.isAssignableFrom(java.lang.Byte::class.java)) return str.parseInt().toByte() as T
if (target.isAssignableFrom(java.lang.Character::class.java)) return str.parseInt().toChar() as T
if (target.isAssignableFrom(java.lang.Short::class.java)) return str.parseShort() as T
if (target.isAssignableFrom(java.lang.Integer::class.java)) return str.parseInt() as T
if (target.isAssignableFrom(java.lang.Long::class.java)) return str.parseLong() as T
if (target.isAssignableFrom(java.lang.Float::class.java)) return str.toFloat() as T
if (target.isAssignableFrom(java.lang.Double::class.java)) return str.toDouble() as T
if (target.isAssignableFrom(java.lang.String::class.java)) {
return (if (value == null) "" else str) as T
}
if (target.isEnum) {
if (value == null) return (target.getMethod("values").invoke(null) as Array)[0] as T
return java.lang.Enum.valueOf(target as Class, str) as T
}
if (value is List<*>) {
return value.toList() as T
}
if (value is Map<*, *>) {
val map = value as Map
val resultClass = target as Class
val result = Dynamic.createEmptyClass(resultClass)
for (field in result.javaClass.declaredFields) {
if (field.name in map) {
val value = map[field.name]
field.isAccessible = true
field.set(result, dynamicCast(value, field.type, field.genericType))
}
}
return result as T
}
if (value == null) return createEmptyClass(target)
throw InvalidOperationException("Can't convert '$value' to '$target'")
}
private enum class AnyEnum {}
fun String?.parseBool(): Boolean? = when (this) {
"true", "yes", "1" -> true
"false", "no", "0" -> false
else -> null
}
fun String?.parseInt(): Int = this?.parseDouble()?.toInt() ?: 0
fun String?.parseShort(): Short = this?.parseDouble()?.toShort() ?: 0
fun String?.parseLong(): Long = try {
this?.toLong()
} catch (e: Throwable) {
this?.parseDouble()?.toLong()
} ?: 0L
fun String?.parseDouble(): Double {
if (this == null) return 0.0
try {
return this.toDouble()
} catch (e: Throwable) {
return 0.0
}
}
fun fromTyped(value: T?): Any? {
return when (value) {
null -> null
true -> true
false -> false
is Number -> value.toDouble()
is String -> value
is Map<*, *> -> value
is Iterable<*> -> value
else -> {
val clazz = value.javaClass
val out = hashMapOf()
for (field in clazz.declaredFields) {
if (field.name.startsWith('$')) continue
field.isAccessible = true
out[field.name] = fromTyped(field.get(value))
}
out
}
}
}
fun unop(r: Any?, op: String): Any? {
return when (op) {
"+" -> r
"-" -> -toNumber(r)
"~" -> toInt(r).inv()
"!" -> !toBool(r)
else -> noImpl("Not implemented unary operator $op")
}
}
//fun toFixNumber(value: Double): Any = if (value == value.toInt().toDouble()) value.toInt() else value
fun toString(value: Any?): String {
when (value) {
is Double -> {
if (value == value.toInt().toDouble()) {
return value.toInt().toString()
}
}
}
return value.toString()
}
fun binop(l: Any?, r: Any?, op: String): Any? {
return when (op) {
"+" -> {
when (l) {
is String -> l.toString() + r.toBetterString()
is Iterable<*> -> toIterable(l) + toIterable(r)
else -> toNumber(l) + toNumber(r)
}
}
"-" -> toNumber(l) - toNumber(r)
"+" -> toNumber(l) + toNumber(r)
"*" -> toNumber(l) * toNumber(r)
"/" -> toNumber(l) / toNumber(r)
"%" -> toNumber(l) % toNumber(r)
"**" -> Math.pow(toNumber(l), toNumber(r))
"&" -> toInt(l) and toInt(r)
"or" -> toInt(l) or toInt(r)
"^" -> toInt(l) xor toInt(r)
"&&" -> toBool(l) && toBool(r)
"||" -> toBool(l) || toBool(r)
"==" -> Objects.equals(l, r)
"!=" -> !Objects.equals(l, r)
"<" -> compare(l, r) < 0
"<=" -> compare(l, r) <= 0
">" -> compare(l, r) > 0
">=" -> compare(l, r) >= 0
else -> noImpl("Not implemented binary operator $op")
}
}
fun callAny(obj: Any?, key: Any?, args: List): Any? {
if (obj == null) return null
if (key == null) return null
val method = obj.javaClass.methods.first { it.name == key }
method.isAccessible = true
val result = method.invoke(obj, *args.toTypedArray())
return result
}
fun callAny(callable: Any?, args: List): Any? {
return callAny(callable, "invoke", args)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy