2022-07-03 03:35:36 +00:00
|
|
|
const STRATEGIES = [
|
2022-07-03 04:08:00 +00:00
|
|
|
'LAST_GET_TIME',
|
|
|
|
'GETS_COUNT'
|
2022-07-03 03:35:36 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
class SimpleDataStore {
|
|
|
|
#store
|
2022-07-06 10:14:00 +00:00
|
|
|
#size
|
2022-07-03 03:35:36 +00:00
|
|
|
#limit
|
|
|
|
#strategy
|
|
|
|
|
|
|
|
constructor (options = {}) {
|
|
|
|
if (typeof options !== 'object') {
|
|
|
|
throw new TypeError('Missing options object.')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Number.isFinite(options.limit) || options.limit <= 1) {
|
|
|
|
throw new TypeError('Limit must be a finite number that is at least 2.')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!STRATEGIES.includes(options.strategy)) {
|
|
|
|
throw new TypeError(`Strategy must be one of these: ${STRATEGIES.map(s => `"${s}"`).join(', ')}.`)
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#store = new Map()
|
2022-07-06 10:14:00 +00:00
|
|
|
this.#size = this.#store.size
|
2022-07-03 03:35:36 +00:00
|
|
|
this.#limit = options.limit
|
|
|
|
this.#strategy = options.strategy
|
|
|
|
}
|
|
|
|
|
|
|
|
clear () {
|
2022-07-06 10:14:00 +00:00
|
|
|
this.#store.clear()
|
|
|
|
this.#size = 0
|
2022-07-03 03:35:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
delete (key) {
|
2022-07-06 10:14:00 +00:00
|
|
|
if (this.#store.delete(key)) {
|
|
|
|
this.#size--
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2022-07-03 03:35:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-06 09:37:54 +00:00
|
|
|
deleteStalest () {
|
|
|
|
const stalest = this.getStalest()
|
|
|
|
if (stalest) {
|
2022-07-06 10:14:00 +00:00
|
|
|
return this.delete(stalest)
|
2022-07-06 09:37:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-03 03:35:36 +00:00
|
|
|
get (key) {
|
|
|
|
const entry = this.#store.get(key)
|
2022-07-06 09:37:54 +00:00
|
|
|
// This may return undefined or null
|
|
|
|
// undefined should be an indicator for when the key legitimately has not been set,
|
|
|
|
// null should be an indicator for when the key is still being held via hold() function
|
|
|
|
if (!entry) return entry
|
2022-07-03 03:35:36 +00:00
|
|
|
|
|
|
|
switch (this.#strategy) {
|
|
|
|
case STRATEGIES[0]:
|
|
|
|
entry.stratval = Date.now()
|
|
|
|
break
|
|
|
|
case STRATEGIES[1]:
|
|
|
|
entry.stratval++
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#store.set(key, entry)
|
|
|
|
return entry.value
|
|
|
|
}
|
|
|
|
|
|
|
|
getStalest () {
|
2022-07-03 04:00:48 +00:00
|
|
|
let stalest = [null, { stratval: Infinity }]
|
2022-07-03 03:35:36 +00:00
|
|
|
switch (this.#strategy) {
|
|
|
|
// both "lastGetTime" and "getsCount" simply must find lowest value
|
|
|
|
// to determine the stalest entry
|
|
|
|
case STRATEGIES[0]:
|
|
|
|
case STRATEGIES[1]:
|
|
|
|
for (const entry of this.#store) {
|
2022-07-06 09:37:54 +00:00
|
|
|
if (entry[1] && entry[1].stratval < stalest[1].stratval) {
|
2022-07-03 03:35:36 +00:00
|
|
|
stalest = entry
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// return its key only
|
|
|
|
return stalest[0]
|
|
|
|
}
|
|
|
|
|
2022-07-06 09:37:54 +00:00
|
|
|
hold (key) {
|
2022-07-06 10:14:00 +00:00
|
|
|
this.#store.set(key, null)
|
|
|
|
return true
|
2022-07-06 09:37:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
set (key, value) {
|
2022-07-06 10:14:00 +00:00
|
|
|
if (this.#size >= this.#limit) {
|
2022-07-06 09:37:54 +00:00
|
|
|
this.deleteStalest()
|
2022-07-03 03:35:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let stratval
|
|
|
|
switch (this.#strategy) {
|
|
|
|
case STRATEGIES[0]:
|
|
|
|
stratval = Date.now()
|
|
|
|
break
|
|
|
|
case STRATEGIES[1]:
|
|
|
|
stratval = 0
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2022-07-06 10:14:00 +00:00
|
|
|
if (this.#store.set(key, { value, stratval })) {
|
|
|
|
this.#size++
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2022-07-03 03:35:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get limit () {
|
|
|
|
return this.#limit
|
|
|
|
}
|
|
|
|
|
|
|
|
set limit (_) {
|
|
|
|
throw Error('This property is read-only.')
|
|
|
|
}
|
|
|
|
|
|
|
|
get size () {
|
2022-07-06 10:14:00 +00:00
|
|
|
return this.#size
|
2022-07-03 03:35:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
set size (_) {
|
|
|
|
throw Error('This property is read-only.')
|
|
|
|
}
|
|
|
|
|
|
|
|
get strategy () {
|
|
|
|
return this.#strategy
|
|
|
|
}
|
|
|
|
|
|
|
|
set strategy (_) {
|
|
|
|
throw Error('This property is read-only.')
|
|
|
|
}
|
|
|
|
|
|
|
|
get store () {
|
|
|
|
// return shallow copy
|
|
|
|
return new Map(this.#store)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = SimpleDataStore
|
2022-07-03 04:08:00 +00:00
|
|
|
module.exports.STRATEGIES = STRATEGIES
|