Compare commits
2 Commits
375f3bdf69
...
b38e3f31ba
Author | SHA1 | Date | |
---|---|---|---|
b38e3f31ba | |||
d46e0d2ddf |
@ -1,4 +1,5 @@
|
|||||||
require "file_utils"
|
require "file_utils"
|
||||||
require "json"
|
require "json"
|
||||||
|
|
||||||
|
require "./fifo.cr" # FIFO class to implement a cache policy.
|
||||||
require "./dodb/*" # Databases and indexes (basic indexes, partitions, tags, etc.).
|
require "./dodb/*" # Databases and indexes (basic indexes, partitions, tags, etc.).
|
||||||
|
@ -57,19 +57,18 @@ class DODB::Storage::Cached(V) < DODB::Storage(V)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Getting data from the hash in RAM.
|
# Gets data from the hash in RAM.
|
||||||
def []?(key : Int32) : V?
|
def []?(key : Int32) : V?
|
||||||
@data[key]
|
self[key] rescue nil
|
||||||
rescue e
|
|
||||||
# FIXME: rescues any error the same way.
|
|
||||||
return nil
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# :inherit:
|
# Gets the data with the *key*.
|
||||||
|
# In case the data is missing, returns an exception `DODB::MissingEntry`.
|
||||||
#
|
#
|
||||||
# Data needs to be cloned in case it will be modified, otherwise it will mess with indexes.
|
# Data needs to be cloned in case it will be modified, otherwise it will mess with indexes.
|
||||||
#
|
#
|
||||||
# WARNING: data isn't cloned.
|
# WARNING: data isn't cloned.
|
||||||
|
# WARNING: may raise a MissingEntry exception.
|
||||||
def [](key : Int32) : V
|
def [](key : Int32) : V
|
||||||
@data[key] rescue raise MissingEntry.new(key)
|
@data[key] rescue raise MissingEntry.new(key)
|
||||||
end
|
end
|
||||||
|
@ -1,15 +1,53 @@
|
|||||||
# Stacked database: only recently requested entries are kept in memory.
|
# Stacked database: only recently requested entries are kept in memory.
|
||||||
|
#
|
||||||
# Most recently used entries are in cache and put on the top of the stack.
|
# Most recently used entries are in cache and put on the top of the stack.
|
||||||
# Least recently used entries may be removed from the cache in order to keep the amount of memory used reasonable.
|
# Least recently used entries may be removed from the cache in order to keep the amount of memory used reasonable.
|
||||||
#
|
#
|
||||||
|
# This database is relevant for high demand applications;
|
||||||
|
# which means both a high number of entries (data cannot fit entirely in RAM),
|
||||||
|
# and a high number of requests.
|
||||||
|
# Typically a retail website.
|
||||||
|
# In such applications, the "keep the most recently used data in cache" policy works since new users
|
||||||
|
# constantly push commonly requested data on top of the stack.
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# # Creates a DODB stacked database.
|
||||||
|
# car_database = DODB::Storage::Stacked.new "/path/to/db"
|
||||||
|
#
|
||||||
|
# # Creates a (cached) index.
|
||||||
|
# cars_by_name = car_database.new_index "name", &.name
|
||||||
|
#
|
||||||
|
# # Add a value in the database.
|
||||||
|
# car_database << Car.new "Corvet"
|
||||||
|
# ```
|
||||||
|
# On the file-system:
|
||||||
|
# ```plain
|
||||||
|
# storage
|
||||||
|
# ├── data
|
||||||
|
# │ └── 0000000000
|
||||||
|
# ├── indices
|
||||||
|
# │ └── by_name <- the "name" basic index
|
||||||
|
# │ └── Corvet -> ../../data/0000000000
|
||||||
|
# ```
|
||||||
|
#
|
||||||
# NOTE: fast for frequently requested data and requires a stable (and configurable) amount of memory.
|
# NOTE: fast for frequently requested data and requires a stable (and configurable) amount of memory.
|
||||||
# TODO: not yet implemented.
|
# TODO: not yet implemented.
|
||||||
class DODB::Storage::Stacked(V) < DODB::Storage::Cached(V)
|
class DODB::Storage::Stacked(V) < DODB::Storage::Cached(V)
|
||||||
# The maximum number of accepted entries in the cache.
|
# The *stack* a simple FIFO instance where the key of the requested data is pushed.
|
||||||
property max_entries : Int32
|
# In case the number of stored entries exceeds what is allowed, the least recently used entry is removed.
|
||||||
|
property stack : FIFO(Int32)
|
||||||
|
|
||||||
# Initializes the `StackedDataBase` with a maximum number of entries in the cache.
|
# Initializes the `StackedDataBase` with a maximum number of entries in the cache.
|
||||||
def initialize(directory_name : String, @max_entries : Int32 = 100_000)
|
def initialize(directory_name : String, @max_entries : Int32 = 100_000)
|
||||||
super directory_name
|
super directory_name
|
||||||
|
@stack = FIFO(Int32).new @max_entries
|
||||||
|
end
|
||||||
|
|
||||||
|
def [](key : Int32) : V
|
||||||
|
val = @data[key] rescue raise MissingEntry.new(key)
|
||||||
|
if entry_to_remove = @stack << key
|
||||||
|
@data.delete entry_to_remove
|
||||||
|
end
|
||||||
|
val
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
37
src/fifo.cr
Normal file
37
src/fifo.cr
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
# This class enables to keep track of used data.
|
||||||
|
#
|
||||||
|
# Each time a value is pushed, it is put on top of a FIFO stack.
|
||||||
|
# Stack size is configurable, so in case the stack size exceeds what is allowed,
|
||||||
|
# the least recently used value is removed.
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# fifo = FIFO(Int32).new 3 # Only 3 allowed entries.
|
||||||
|
#
|
||||||
|
# pp! fifo << 1 # -> nil (there is still room in the stack)
|
||||||
|
# pp! fifo << 2 # -> nil (there is still room in the stack)
|
||||||
|
# pp! fifo << 3 # -> nil (last entry without exceeding the allowed size)
|
||||||
|
# pp! fifo << 4 # -> 1 (least recently used data)
|
||||||
|
# pp! fifo << 4 # -> nil (already in the stack)
|
||||||
|
# pp! fifo << 2 # -> nil (already in the stack)
|
||||||
|
# pp! fifo << 5 # -> 3 (least recently used data)
|
||||||
|
# ```
|
||||||
|
class FIFO(V)
|
||||||
|
# This array is used as a *stack*.
|
||||||
|
property data : Array(V)
|
||||||
|
|
||||||
|
# Maximum allowed entries in the stack.
|
||||||
|
property max_entries : UInt32
|
||||||
|
|
||||||
|
def initialize(@max_entries : UInt32)
|
||||||
|
@data = Array(V).new
|
||||||
|
end
|
||||||
|
|
||||||
|
# Pushes a value in the FIFO and gets the oldest value whether it exceeds the allowed number of entries.
|
||||||
|
# NOTE: `#<<(v : V)` is the only function since it's enough for the intended use, feel free to improve this.
|
||||||
|
# WARNING: implementation is extremely simple (3 lines) and not designed to be highly efficient.
|
||||||
|
def <<(v : V) : V?
|
||||||
|
@data.unshift v # push on top of the stack
|
||||||
|
@data.uniq! # remove dups
|
||||||
|
@data.pop if @data.size > @max_entries # remove least recently used entry if `@data` is too big
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue
Block a user