All Downloads are FREE. Search and download functionalities are using the official Maven repository.

commonMain.co.touchlab.stately.collections.SharedLruCache.kt Maven / Gradle / Ivy

/*
 * Copyright (C) 2018 Touchlab, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package co.touchlab.stately.collections

import co.touchlab.stately.concurrency.Lock
import co.touchlab.stately.freeze

/**
 * Implementation of a least recently used cache, multithreading aware for kotlin multiplatform.
 *
 * You provide a maximum number of cached entries, and (optionally) a lambda to call when something is removed.
 *
 * Operations on this collection aggressively lock, to err on the side of caution and sanity. Bear this in mind
 * if using in a high volume context.
 *
 * Each key will only retain a single value. When an entry is removed due to exceeding capacity, onRemove is called.
 * However, if adding an entry replaces an existing entry, the old entry will be returned, and onRemove WILL NOT be called.
 * If resource management depends on onRemove, keep this in mind. You'll need to handle a duplicate/replacement add.
 *
 * Along those lines, if you are managing resources with onRemove, keep in mind that if you abandon the cache and memory
 * is reclaimed, onRemove WILL NOT be called for existing entries. If you need to close resources, you'll have to implement
 * an explicit call to 'removeAll'. That will push all existing values to onRemove.
 *
 * IMPORTANT NOTE: Locking is not reentrant. As a result, onRemove is called AFTER the mutation lock is released, in
 * case the onRemove logic intends to call back into the cache. This may have unintended consequences if you're expecting
 * the mutation to be fully atomic in nature.
 */
class SharedLruCache(
    private val maxCacheSize: Int,
    private val onRemove: (MutableMap.MutableEntry) -> Unit = {}
) : LruCache {

    private var lock: Lock = Lock()
    private val cacheMap = SharedHashMap>(initialCapacity = maxCacheSize)
    private val cacheList = SharedLinkedList(20)

    init {
        freeze()
    }

    /**
     * Stores value at key.
     *
     * If replacing an existing value, that value will be returned, but onRemove will not
     * be called.
     *
     * If adding a new value, if the total number of values exceeds maxCacheSize, the last accessed
     * value will be removed and sent to onRemove.
     *
     * If adding a value with the same key and value of an existing value, the LRU cache is updated, but
     * the existing value is not returned. This effectively refreshes the entry in the LRU list.
     */
    override fun put(key: K, value: V): V? {
        var resultValue: V? = null
        val removeCollection: MutableList> = ArrayList()

        withLock {
            val cacheEntry = cacheMap.get(key)
            val node: AbstractSharedLinkedList.Node
            val result: V?
            if (cacheEntry != null) {
                result = if (value != cacheEntry.v) {
                    cacheEntry.v
                } else {
                    null
                }
                node = cacheEntry.node
                node.readd()
            } else {
                result = null
                node = cacheList.addNode(key)
            }
            cacheMap.put(key, CacheEntry(value, node))

            while (cacheList.size > maxCacheSize) {
                val key = cacheList.removeAt(0)
                val entry = cacheMap.remove(key)
                if (entry != null) {
                    removeCollection.add(LruEntry(key, entry.v))
                }
            }

            resultValue = result
        }

        removeCollection.forEach(onRemove)

        return resultValue
    }

    /**
     * Removes value at key (if it exists). If a value is found, it is passed to onRemove.
     */
    override fun remove(key: K, skipCallback: Boolean): V? {
        var removeEntry: LruEntry? = null
        withLock {
            val entry = cacheMap.remove(key)
            if (entry != null) {
                entry.node.remove()
                removeEntry = LruEntry(key, entry.v)
            }
        }
        if (!skipCallback && removeEntry != null) {
            onRemove(removeEntry!!)
        }

        return removeEntry?.value
    }

    /**
     * Returns all entries. This does not affect position in LRU cache. IE, old entries stay old.
     */
    override val entries: MutableSet>
        get() = withLock {
            return internalAll()
        }

    /**
     * Clears the cache. If skipCallback is set to true, onRemove is not called. Defaults to false.
     */
    override fun removeAll(skipCallback: Boolean) {
        var removeCollection: Collection>? = null

        withLock {
            if (!skipCallback) {
                removeCollection = internalAll()
            }
            cacheMap.clear()
            cacheList.clear()
        }

        if (removeCollection != null) {
            removeCollection!!.forEach(onRemove)
        }
    }

    /**
     * Finds and returns cache value, if it exists. If it exists, the key gets moved to the front of the
     * LRU list.
     */
    override fun get(key: K): V? = withLock {
        val cacheEntry = cacheMap.get(key)
        return if (cacheEntry != null) {
            cacheEntry!!.node.readd()
            cacheEntry.v
        } else {
            null
        }
    }

  /*
  This was OK with Intellij but kicking back an llvm error. To investigate.
  override fun get(key: K): V? = withLock {
      val cacheEntry = cacheMap.get(key)
      if(cacheEntry != null){
          cacheEntry.node.readd()
          return cacheEntry.v
      }
      else{
          return null
      }
  }
   */

    /**
     * Well...
     */
    override fun exists(key: K): Boolean = withLock { cacheMap.get(key) != null }

    override val size: Int
        get() = withLock { cacheMap.size }

    data class CacheEntry(val v: V, val node: AbstractSharedLinkedList.Node)

    class LruEntry(override val key: K, override val value: V) : MutableMap.MutableEntry {

        override fun setValue(newValue: V): V {
            throw UnsupportedOperationException()
        }

        override fun toString(): String {
            return "LruEntry(key=$key, value=$value)"
        }
    }

    private fun internalAll(): HashSet> {
        val set = HashSet>(cacheList.size)
        cacheList.iterator().forEach {
            set.add(LruEntry(it, cacheMap.get(it)!!.v))
        }
        return set
    }

    private inline fun  withLock(proc: () -> T): T {
        lock.lock()
        try {
            return proc()
        } finally {
            lock.unlock()
        }
    }

    internal fun printDebug() {
        println("CACHELIST")
        cacheList.forEach {
            println(it)
        }
        println("CACHEMAP")
        cacheMap.entries.forEach {
            println(it)
        }
    }
}

interface LruCache {
    fun put(key: K, value: V): V?
    fun remove(key: K, skipCallback: Boolean = false): V?
    val entries: MutableSet>
    fun removeAll(skipCallback: Boolean = false)
    fun get(key: K): V?
    fun exists(key: K): Boolean
    val size: Int
}

typealias LruEntry = MutableMap.MutableEntry




© 2015 - 2025 Weber Informatics LLC | Privacy Policy