org.jetbrains.exposed.dao.EntityCache.kt Maven / Gradle / Ivy
package org.jetbrains.exposed.dao
import org.jetbrains.exposed.dao.id.EntityID
import org.jetbrains.exposed.dao.id.IdTable
import org.jetbrains.exposed.sql.*
import org.jetbrains.exposed.sql.transactions.transactionScope
import java.util.*
import kotlin.collections.HashMap
import kotlin.collections.HashSet
val Transaction.entityCache: EntityCache by transactionScope { EntityCache(this) }
@Suppress("UNCHECKED_CAST")
class EntityCache(private val transaction: Transaction) {
private var flushingEntities = false
private var initializingEntities: LinkedIdentityHashSet> = LinkedIdentityHashSet()
internal val pendingInitializationLambdas = IdentityHashMap, MutableList<(Entity<*>)->Unit>>()
val data = LinkedHashMap, MutableMap>>()
internal val inserts = LinkedHashMap, MutableSet>>()
private val updates = LinkedHashMap, MutableSet>>()
internal val referrers = HashMap, MutableMap, SizedIterable<*>>>()
/**
* Amount of entities to keep in a cache per an Entity class.
* On setting a new value all data stored in cache will be adjusted to a new size
*/
@Suppress("MemberVisibilityCanBePrivate")
var maxEntitiesToStore = transaction.db.config.maxEntitiesToStoreInCachePerEntity
set(value) {
val diff = value - field
field = value
if (diff < 0) {
data.values.forEach { map ->
val sizeExceed = map.size - value
if (sizeExceed > 0) {
val iterator = map.iterator()
repeat(sizeExceed) {
iterator.next()
iterator.remove()
}
}
}
}
}
private fun getMap(f: EntityClass<*, *>): MutableMap> = getMap(f.table)
private fun getMap(table: IdTable<*>): MutableMap> = data.getOrPut(table) {
LimitedHashMap()
}
fun > getReferrers(sourceId: EntityID<*>, key: Column<*>): SizedIterable? {
return referrers[key]?.get(sourceId) as? SizedIterable
}
fun > getOrPutReferrers(sourceId: EntityID<*>, key: Column<*>, refs: () -> SizedIterable): SizedIterable {
return referrers.getOrPut(key) { HashMap() }.getOrPut(sourceId) { LazySizedCollection(refs()) } as SizedIterable
}
fun , T : Entity> find(f: EntityClass, id: EntityID): T? =
getMap(f)[id.value] as T?
?: inserts[f.table]?.firstOrNull { it.id == id } as? T
?: initializingEntities.firstOrNull { it.klass == f && it.id == id } as? T
fun , T : Entity> findAll(f: EntityClass): Collection = getMap(f).values as Collection
fun , T : Entity> store(f: EntityClass, o: T) {
getMap(f)[o.id.value] = o
}
fun store(o: Entity<*>) {
getMap(o.klass.table)[o.id.value] = o
}
fun , T : Entity> remove(table: IdTable, o: T) {
getMap(table).remove(o.id.value)
}
internal fun addNotInitializedEntityToQueue(entity: Entity<*>) {
require(initializingEntities.add(entity)) { "Entity ${entity::class.simpleName} already in initialization process" }
}
internal fun finishEntityInitialization(entity: Entity<*>) {
require(initializingEntities.lastOrNull() == entity) {
"Can't finish initialization for entity ${entity::class.simpleName} - the initialization order is broken"
}
initializingEntities.remove(entity)
}
internal fun isEntityInInitializationState(entity: Entity<*>) = entity in initializingEntities
fun , T : Entity> scheduleInsert(f: EntityClass, o: T) {
inserts.getOrPut(f.table) { LinkedIdentityHashSet() }.add(o as Entity<*>)
}
fun , T : Entity> scheduleUpdate(f: EntityClass, o: T) {
updates.getOrPut(f.table) { LinkedIdentityHashSet() }.add(o as Entity<*>)
}
fun flush() {
val toFlush = when {
inserts.isEmpty() && updates.isEmpty() -> emptyList()
inserts.isNotEmpty() && updates.isNotEmpty() -> inserts.keys + updates.keys
inserts.isNotEmpty() -> inserts.keys
else -> updates.keys
}
flush(toFlush)
}
private fun updateEntities(idTable: IdTable<*>) {
updates.remove(idTable)?.takeIf { it.isNotEmpty() }?.let {
val updatedEntities = HashSet>()
val batch = EntityBatchUpdate(it.first().klass)
for (entity in it) {
if (entity.flush(batch)) {
check(entity.klass !is ImmutableEntityClass<*, *>) { "Update on immutable entity ${entity.javaClass.simpleName} ${entity.id}" }
updatedEntities.add(entity)
}
}
executeAsPartOfEntityLifecycle {
batch.execute(transaction)
}
updatedEntities.forEach {
transaction.registerChange(it.klass, it.id, EntityChangeType.Updated)
}
}
}
fun flush(tables: Iterable>) {
if (flushingEntities) return
try {
flushingEntities = true
val insertedTables = inserts.keys
val updateBeforeInsert = SchemaUtils.sortTablesByReferences(insertedTables).filterIsInstance>()
updateBeforeInsert.forEach(::updateEntities)
SchemaUtils.sortTablesByReferences(tables).filterIsInstance>().forEach(::flushInserts)
val updateTheRestTables = tables - updateBeforeInsert
for (t in updateTheRestTables) {
updateEntities(t)
}
if (insertedTables.isNotEmpty()) {
removeTablesReferrers(insertedTables, true)
}
} finally {
flushingEntities = false
}
}
internal fun removeTablesReferrers(tables: Collection, isInsert: Boolean) {
val insertedTablesSet = tables.toSet()
val columnsToInvalidate = tables.flatMapTo(hashSetOf()) { it.columns.mapNotNull { it.takeIf { it.referee != null } } }
columnsToInvalidate.forEach {
referrers.remove(it)
}
referrers.keys.filter { refColumn ->
when {
isInsert -> false
refColumn.referee?.table in insertedTablesSet -> true
refColumn.table.columns.any { it.referee?.table in tables } -> true
else -> false
}
}.forEach {
referrers.remove(it)
}
}
internal fun flushInserts(table: IdTable<*>) {
var toFlush: List> = inserts.remove(table)?.toList().orEmpty()
while (toFlush.isNotEmpty()) {
val partition = toFlush.partition {
it.writeValues.none {
val (key, value) = it
key.referee == table.id && value is EntityID<*> && value._value == null
}
}
toFlush = partition.first
val ids = executeAsPartOfEntityLifecycle {
table.batchInsert(toFlush) { entry ->
for ((c, v) in entry.writeValues) {
this[c] = v
}
}
}
for ((entry, genValues) in toFlush.zip(ids)) {
if (entry.id._value == null) {
val id = genValues[table.id]
entry.id._value = id._value
entry.writeValues[entry.klass.table.id as Column] = id
}
genValues.fieldIndex.keys.forEach { key ->
entry.writeValues[key as Column] = genValues[key]
}
entry.storeWrittenValues()
store(entry)
transaction.registerChange(entry.klass, entry.id, EntityChangeType.Created)
pendingInitializationLambdas[entry]?.forEach { it(entry) }
}
toFlush = partition.second
}
transaction.alertSubscribers()
}
fun clear(flush: Boolean = true) {
if (flush) flush()
data.clear()
inserts.clear()
updates.clear()
clearReferrersCache()
}
fun clearReferrersCache() {
referrers.clear()
}
private inner class LimitedHashMap : LinkedHashMap() {
override fun removeEldestEntry(eldest: MutableMap.MutableEntry?): Boolean {
return size > maxEntitiesToStore
}
}
companion object {
fun invalidateGlobalCaches(created: List>) {
created.asSequence().mapNotNull { it.klass as? ImmutableCachedEntityClass<*, *> }.distinct().forEach {
it.expireCache()
}
}
}
}
fun Transaction.flushCache(): List> {
with(entityCache) {
val newEntities = inserts.flatMap { it.value }
flush()
return newEntities
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy