commonMain.jetbrains.datalore.plot.base.DataFrame.kt Maven / Gradle / Ivy
/*
* Copyright (c) 2019. JetBrains s.r.o.
* Use of this source code is governed by the MIT license that can be found in the LICENSE file.
*/
package jetbrains.datalore.plot.base
import jetbrains.datalore.base.gcommon.collect.ClosedRange
import jetbrains.datalore.base.logging.PortableLogging
import jetbrains.datalore.plot.common.data.SeriesUtil
import kotlin.jvm.JvmOverloads
class DataFrame private constructor(builder: Builder) {
private val myVectorByVar: Map>
private val myIsNumeric: MutableMap
// volatile variables (yet)
private val myRanges = HashMap?>()
private val myDistinctValues = HashMap>()
val isEmpty: Boolean
get() = myVectorByVar.isEmpty()
init {
assertAllSeriesAreSameSize(builder.myVectorByVar)
myVectorByVar = HashMap(builder.myVectorByVar)
myIsNumeric = HashMap(builder.myIsNumeric)
}
private fun assertAllSeriesAreSameSize(vectorByVar: Map>) {
if (vectorByVar.size > 1) {
val entries = vectorByVar.entries.iterator()
val first = entries.next()
val size = first.value.size
while (entries.hasNext()) {
val next = entries.next()
if (next.value.size != size) {
throw IllegalArgumentException(
"All data series in data frame must have equal size\n" + dumpSizes(
vectorByVar
)
)
}
}
}
}
private fun dumpSizes(vectorByVar: Map>): String {
val sb = StringBuilder()
for ((key, value) in vectorByVar) {
sb.append(key.name)
.append(" : ")
.append(value.size)
.append('\n')
}
return sb.toString()
}
fun rowCount(): Int {
return if (myVectorByVar.isEmpty()) 0 else myVectorByVar.entries.iterator().next().value.size
}
fun has(variable: Variable): Boolean {
return myVectorByVar.containsKey(variable)
}
fun isEmpty(variable: Variable): Boolean {
return get(variable).isEmpty()
}
fun hasNoOrEmpty(variable: Variable): Boolean {
return !has(variable) || isEmpty(variable)
}
operator fun get(variable: Variable): List<*> {
assertDefined(variable)
return myVectorByVar[variable]!!
}
fun getNumeric(variable: Variable): List {
assertDefined(variable)
val list = myVectorByVar[variable]
if (list!!.isEmpty()) {
return emptyList()
}
assertNumeric(variable)
@Suppress("UNCHECKED_CAST")
return list as List
}
fun distinctValues(variable: Variable): Collection {
assertDefined(variable)
return myDistinctValues.getOrPut(variable) {
val values = LinkedHashSet(get(variable)).apply {
this.remove(null)
}
@Suppress("UNCHECKED_CAST")
return values as Collection
}
}
fun variables(): Set {
return myVectorByVar.keys
}
fun isNumeric(variable: Variable): Boolean {
assertDefined(variable)
if (!myIsNumeric.containsKey(variable)) {
val checkedDoubles = SeriesUtil.checkedDoubles(get(variable))
myIsNumeric[variable] = checkedDoubles.notEmptyAndCanBeCast()
}
return myIsNumeric[variable]!!
}
fun range(variable: Variable): ClosedRange? {
if (!myRanges.containsKey(variable)) {
val v = getNumeric(variable)
val r = SeriesUtil.range(v)
myRanges[variable] = r
}
return myRanges[variable]
}
fun builder(): Builder {
return Builder(this)
}
private fun assertDefined(variable: Variable) {
if (!has(variable)) {
val e = IllegalArgumentException("Undefined variable: '$variable'")
LOG.error(e) { e.message!! }
throw e
}
}
private fun assertNumeric(variable: Variable) {
if (!isNumeric(variable)) {
val e = IllegalArgumentException("Not a numeric variable: '$variable'")
LOG.error(e) { e.message!! }
throw e
}
}
fun selectIndices(indices: List): DataFrame {
return buildModified { serie -> SeriesUtil.pickAtIndices(serie, indices) }
}
fun selectIndices(indices: Set): DataFrame {
return buildModified { serie -> SeriesUtil.pickAtIndices(serie, indices) }
}
fun dropIndices(indices: Set): DataFrame {
return if (indices.isEmpty()) this else buildModified { serie -> SeriesUtil.skipAtIndices(serie, indices) }
}
private fun buildModified(serieFun: (List<*>) -> List<*>): DataFrame {
val builder = this.builder()
for (variable in myVectorByVar.keys) {
val serie = myVectorByVar[variable]
val modifiedSerie = serieFun(serie!!)
builder.putIntern(variable, modifiedSerie)
}
return builder.build()
}
class Variable @JvmOverloads constructor(
val name: String,
val source: Source = Source.ORIGIN,
val label: String = name
) {
val isOrigin: Boolean
get() = source == Source.ORIGIN
val isStat: Boolean
get() = source == Source.STAT
override fun toString(): String {
// important
return name
}
fun toSummaryString(): String {
return "$name, '$label' [$source]"
}
enum class Source {
ORIGIN, TRANSFORM, STAT
}
companion object {
@JvmOverloads
fun createOriginal(name: String, label: String = name): Variable {
return Variable(
name,
Source.ORIGIN,
label
)
}
}
}
companion object {
private val LOG = PortableLogging.logger(DataFrame::class)
}
class Builder {
internal val myVectorByVar = HashMap>()
internal val myIsNumeric = HashMap()
constructor()
constructor(data: DataFrame) {
myVectorByVar.putAll(data.myVectorByVar)
myIsNumeric.putAll(data.myIsNumeric)
}
fun put(variable: Variable, v: List<*>): Builder {
putIntern(variable, v)
myIsNumeric.remove(variable) // unknown state
return this
}
fun putNumeric(variable: Variable, v: List): Builder {
putIntern(variable, v)
myIsNumeric[variable] = true
return this
}
fun putDiscrete(variable: Variable, v: List<*>): Builder {
putIntern(variable, v)
myIsNumeric[variable] = false
return this
}
fun putIntern(variable: Variable, v: List<*>) {
myVectorByVar[variable] = ArrayList(v)
}
fun remove(variable: Variable): Builder {
myVectorByVar.remove(variable)
myIsNumeric.remove(variable)
return this
}
fun build(): DataFrame {
return DataFrame(this)
}
companion object {
fun emptyFrame(): DataFrame {
return Builder().build()
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy