mirror of
https://github.com/gofiber/storage.git
synced 2025-10-04 16:22:52 +08:00
Fix GC deleting values without expiry (#54)
* Test MYSQL Fix * Test * Add postgres * Fix mysql * Fix sqlite
This commit is contained in:
@@ -82,11 +82,11 @@ func New(config ...Config) *Storage {
|
|||||||
sqlInsert: fmt.Sprintf("INSERT INTO %s (k, v, e) VALUES (?,?,?) ON DUPLICATE KEY UPDATE v = ?, e = ?", cfg.Table),
|
sqlInsert: fmt.Sprintf("INSERT INTO %s (k, v, e) VALUES (?,?,?) ON DUPLICATE KEY UPDATE v = ?, e = ?", cfg.Table),
|
||||||
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=?", cfg.Table),
|
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=?", cfg.Table),
|
||||||
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
||||||
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= ?", cfg.Table),
|
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= ? AND e != 0", cfg.Table),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start garbage collector
|
// Start garbage collector
|
||||||
go store.gc()
|
go store.gcTicker()
|
||||||
|
|
||||||
return store
|
return store
|
||||||
}
|
}
|
||||||
@@ -160,8 +160,8 @@ func (s *Storage) Close() error {
|
|||||||
return s.db.Close()
|
return s.db.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Garbage collector to delete expired keys
|
// gcTicker starts the gc ticker
|
||||||
func (s *Storage) gc() {
|
func (s *Storage) gcTicker() {
|
||||||
ticker := time.NewTicker(s.gcInterval)
|
ticker := time.NewTicker(s.gcInterval)
|
||||||
defer ticker.Stop()
|
defer ticker.Stop()
|
||||||
for {
|
for {
|
||||||
@@ -169,7 +169,12 @@ func (s *Storage) gc() {
|
|||||||
case <-s.done:
|
case <-s.done:
|
||||||
return
|
return
|
||||||
case t := <-ticker.C:
|
case t := <-ticker.C:
|
||||||
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
s.gc(t)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// gc deletes all expired entries
|
||||||
|
func (s *Storage) gc(t time.Time) {
|
||||||
|
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
||||||
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
package mysql
|
package mysql
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"database/sql"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
@@ -122,6 +123,31 @@ func Test_MYSQL_Reset(t *testing.T) {
|
|||||||
utils.AssertEqual(t, true, len(result) == 0)
|
utils.AssertEqual(t, true, len(result) == 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Test_MYSQL_GC(t *testing.T) {
|
||||||
|
var (
|
||||||
|
testVal = []byte("doe")
|
||||||
|
)
|
||||||
|
|
||||||
|
// This key should expire
|
||||||
|
err := testStore.Set("john", testVal, time.Nanosecond)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
row := testStore.db.QueryRow(testStore.sqlSelect, "john")
|
||||||
|
err = row.Scan(nil, nil)
|
||||||
|
utils.AssertEqual(t, sql.ErrNoRows, err)
|
||||||
|
|
||||||
|
// This key should not expire
|
||||||
|
err = testStore.Set("john", testVal, 0)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
val, err := testStore.Get("john")
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
utils.AssertEqual(t, testVal, val)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func Test_MYSQL_Close(t *testing.T) {
|
func Test_MYSQL_Close(t *testing.T) {
|
||||||
utils.AssertEqual(t, nil, testStore.Close())
|
utils.AssertEqual(t, nil, testStore.Close())
|
||||||
}
|
}
|
||||||
|
@@ -100,11 +100,11 @@ func New(config ...Config) *Storage {
|
|||||||
sqlInsert: fmt.Sprintf("INSERT INTO %s (k, v, e) VALUES ($1, $2, $3) ON CONFLICT (k) DO UPDATE SET v = $4, e = $5", cfg.Table),
|
sqlInsert: fmt.Sprintf("INSERT INTO %s (k, v, e) VALUES ($1, $2, $3) ON CONFLICT (k) DO UPDATE SET v = $4, e = $5", cfg.Table),
|
||||||
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=$1", cfg.Table),
|
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=$1", cfg.Table),
|
||||||
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
||||||
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= $1", cfg.Table),
|
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= $1 AND e != 0", cfg.Table),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start garbage collector
|
// Start garbage collector
|
||||||
go store.gc()
|
go store.gcTicker()
|
||||||
|
|
||||||
return store
|
return store
|
||||||
}
|
}
|
||||||
@@ -175,8 +175,8 @@ func (s *Storage) Close() error {
|
|||||||
return s.db.Close()
|
return s.db.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
// GC deletes all expired entries
|
// gcTicker starts the gc ticker
|
||||||
func (s *Storage) gc() {
|
func (s *Storage) gcTicker() {
|
||||||
ticker := time.NewTicker(s.gcInterval)
|
ticker := time.NewTicker(s.gcInterval)
|
||||||
defer ticker.Stop()
|
defer ticker.Stop()
|
||||||
for {
|
for {
|
||||||
@@ -184,7 +184,12 @@ func (s *Storage) gc() {
|
|||||||
case <-s.done:
|
case <-s.done:
|
||||||
return
|
return
|
||||||
case t := <-ticker.C:
|
case t := <-ticker.C:
|
||||||
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
s.gc(t)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// gc deletes all expired entries
|
||||||
|
func (s *Storage) gc(t time.Time) {
|
||||||
|
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
||||||
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
package postgres
|
package postgres
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"database/sql"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
@@ -122,6 +123,31 @@ func Test_Postgres_Reset(t *testing.T) {
|
|||||||
utils.AssertEqual(t, true, len(result) == 0)
|
utils.AssertEqual(t, true, len(result) == 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Test_Postgres_GC(t *testing.T) {
|
||||||
|
var (
|
||||||
|
testVal = []byte("doe")
|
||||||
|
)
|
||||||
|
|
||||||
|
// This key should expire
|
||||||
|
err := testStore.Set("john", testVal, time.Nanosecond)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
row := testStore.db.QueryRow(testStore.sqlSelect, "john")
|
||||||
|
err = row.Scan(nil, nil)
|
||||||
|
utils.AssertEqual(t, sql.ErrNoRows, err)
|
||||||
|
|
||||||
|
// This key should not expire
|
||||||
|
err = testStore.Set("john", testVal, 0)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
val, err := testStore.Get("john")
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
utils.AssertEqual(t, testVal, val)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func Test_Postgres_Close(t *testing.T) {
|
func Test_Postgres_Close(t *testing.T) {
|
||||||
utils.AssertEqual(t, nil, testStore.Close())
|
utils.AssertEqual(t, nil, testStore.Close())
|
||||||
}
|
}
|
||||||
|
@@ -81,11 +81,11 @@ func New(config ...Config) *Storage {
|
|||||||
sqlInsert: fmt.Sprintf("INSERT OR REPLACE INTO %s (k, v, e) VALUES (?,?,?)", cfg.Table),
|
sqlInsert: fmt.Sprintf("INSERT OR REPLACE INTO %s (k, v, e) VALUES (?,?,?)", cfg.Table),
|
||||||
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=?", cfg.Table),
|
sqlDelete: fmt.Sprintf("DELETE FROM %s WHERE k=?", cfg.Table),
|
||||||
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
sqlReset: fmt.Sprintf("DELETE FROM %s;", cfg.Table),
|
||||||
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= ?", cfg.Table),
|
sqlGC: fmt.Sprintf("DELETE FROM %s WHERE e <= ? AND e != 0", cfg.Table),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start garbage collector
|
// Start garbage collector
|
||||||
go store.gc()
|
go store.gcTicker()
|
||||||
|
|
||||||
return store
|
return store
|
||||||
}
|
}
|
||||||
@@ -152,8 +152,8 @@ func (s *Storage) Close() error {
|
|||||||
return s.db.Close()
|
return s.db.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
// GC deletes all expired entries
|
// gcTicker starts the gc ticker
|
||||||
func (s *Storage) gc() {
|
func (s *Storage) gcTicker() {
|
||||||
ticker := time.NewTicker(s.gcInterval)
|
ticker := time.NewTicker(s.gcInterval)
|
||||||
defer ticker.Stop()
|
defer ticker.Stop()
|
||||||
for {
|
for {
|
||||||
@@ -161,7 +161,12 @@ func (s *Storage) gc() {
|
|||||||
case <-s.done:
|
case <-s.done:
|
||||||
return
|
return
|
||||||
case t := <-ticker.C:
|
case t := <-ticker.C:
|
||||||
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
s.gc(t)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// gc deletes all expired entries
|
||||||
|
func (s *Storage) gc(t time.Time) {
|
||||||
|
_, _ = s.db.Exec(s.sqlGC, t.Unix())
|
||||||
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
package sqlite3
|
package sqlite3
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"database/sql"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -119,6 +120,31 @@ func Test_SQLite3_Reset(t *testing.T) {
|
|||||||
utils.AssertEqual(t, true, len(result) == 0)
|
utils.AssertEqual(t, true, len(result) == 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Test_SQLite3_GC(t *testing.T) {
|
||||||
|
var (
|
||||||
|
testVal = []byte("doe")
|
||||||
|
)
|
||||||
|
|
||||||
|
// This key should expire
|
||||||
|
err := testStore.Set("john", testVal, time.Nanosecond)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
row := testStore.db.QueryRow(testStore.sqlSelect, "john")
|
||||||
|
err = row.Scan(nil, nil)
|
||||||
|
utils.AssertEqual(t, sql.ErrNoRows, err)
|
||||||
|
|
||||||
|
// This key should not expire
|
||||||
|
err = testStore.Set("john", testVal, 0)
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
|
||||||
|
testStore.gc(time.Now())
|
||||||
|
val, err := testStore.Get("john")
|
||||||
|
utils.AssertEqual(t, nil, err)
|
||||||
|
utils.AssertEqual(t, testVal, val)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func Test_SQLite3_Close(t *testing.T) {
|
func Test_SQLite3_Close(t *testing.T) {
|
||||||
utils.AssertEqual(t, nil, testStore.Close())
|
utils.AssertEqual(t, nil, testStore.Close())
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user