mirror of
https://github.com/ddworken/hishtory.git
synced 2025-01-12 01:09:10 +01:00
Automatically retry search queries when the DB is locked to increase reliability (especially when the backend is down, which leads to more queries, and thus more instances of the DB being locked)
This commit is contained in:
parent
f589a0b1ba
commit
4f592f4aef
@ -482,6 +482,8 @@ func normalizeEntryTimezone(entry data.HistoryEntry) data.HistoryEntry {
|
|||||||
return entry
|
return entry
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const SQLITE_LOCKED_ERR_MSG = "database is locked ("
|
||||||
|
|
||||||
func RetryingDbFunction(dbFunc func() error) error {
|
func RetryingDbFunction(dbFunc func() error) error {
|
||||||
var err error = nil
|
var err error = nil
|
||||||
i := 0
|
i := 0
|
||||||
@ -491,7 +493,7 @@ func RetryingDbFunction(dbFunc func() error) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
errMsg := err.Error()
|
errMsg := err.Error()
|
||||||
if strings.Contains(errMsg, "database is locked (5) (SQLITE_BUSY)") || strings.Contains(errMsg, "database is locked (261)") {
|
if strings.Contains(errMsg, SQLITE_LOCKED_ERR_MSG) {
|
||||||
time.Sleep(time.Duration(i*rand.Intn(100)) * time.Millisecond)
|
time.Sleep(time.Duration(i*rand.Intn(100)) * time.Millisecond)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -678,6 +680,12 @@ func MakeWhereQueryFromSearch(ctx context.Context, db *gorm.DB, query string) (*
|
|||||||
}
|
}
|
||||||
|
|
||||||
func Search(ctx context.Context, db *gorm.DB, query string, limit int) ([]*data.HistoryEntry, error) {
|
func Search(ctx context.Context, db *gorm.DB, query string, limit int) ([]*data.HistoryEntry, error) {
|
||||||
|
return retryingSearch(ctx, db, query, limit, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
const SEARCH_RETRY_COUNT = 3
|
||||||
|
|
||||||
|
func retryingSearch(ctx context.Context, db *gorm.DB, query string, limit int, currentRetryNum int) ([]*data.HistoryEntry, error) {
|
||||||
if ctx == nil && query != "" {
|
if ctx == nil && query != "" {
|
||||||
return nil, fmt.Errorf("lib.Search called with a nil context and a non-empty query (this should never happen)")
|
return nil, fmt.Errorf("lib.Search called with a nil context and a non-empty query (this should never happen)")
|
||||||
}
|
}
|
||||||
@ -697,6 +705,11 @@ func Search(ctx context.Context, db *gorm.DB, query string, limit int) ([]*data.
|
|||||||
var historyEntries []*data.HistoryEntry
|
var historyEntries []*data.HistoryEntry
|
||||||
result := tx.Find(&historyEntries)
|
result := tx.Find(&historyEntries)
|
||||||
if result.Error != nil {
|
if result.Error != nil {
|
||||||
|
if strings.Contains(result.Error.Error(), SQLITE_LOCKED_ERR_MSG) && currentRetryNum < SEARCH_RETRY_COUNT {
|
||||||
|
hctx.GetLogger().Infof("Ignoring err=%v and retrying search query, cnt=%d", result.Error, currentRetryNum)
|
||||||
|
time.Sleep(time.Duration(currentRetryNum*rand.Intn(50)) * time.Millisecond)
|
||||||
|
return retryingSearch(ctx, db, query, limit, currentRetryNum+1)
|
||||||
|
}
|
||||||
return nil, fmt.Errorf("DB query error: %w", result.Error)
|
return nil, fmt.Errorf("DB query error: %w", result.Error)
|
||||||
}
|
}
|
||||||
return historyEntries, nil
|
return historyEntries, nil
|
||||||
|
Loading…
Reference in New Issue
Block a user