mirror of
https://github.com/grafana/grafana.git
synced 2025-01-18 20:43:26 -06:00
f4f0d74838
* Add context to user * Add context for enterprise * Add context for UpdateUserLastSeenAtCommand * Remove xorm
113 lines
2.6 KiB
Go
113 lines
2.6 KiB
Go
package serverlock
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
"github.com/grafana/grafana/pkg/services/sqlstore"
|
|
)
|
|
|
|
func ProvideService(sqlStore *sqlstore.SQLStore) *ServerLockService {
|
|
return &ServerLockService{
|
|
SQLStore: sqlStore,
|
|
log: log.New("infra.lockservice"),
|
|
}
|
|
}
|
|
|
|
// ServerLockService allows servers in HA mode to claim a lock
|
|
// and execute an function if the server was granted the lock
|
|
type ServerLockService struct {
|
|
SQLStore *sqlstore.SQLStore
|
|
log log.Logger
|
|
}
|
|
|
|
// LockAndExecute try to create a lock for this server and only executes the
|
|
// `fn` function when successful. This should not be used at low internal. But services
|
|
// that needs to be run once every ex 10m.
|
|
func (sl *ServerLockService) LockAndExecute(ctx context.Context, actionName string, maxInterval time.Duration, fn func(ctx context.Context)) error {
|
|
// gets or creates a lockable row
|
|
rowLock, err := sl.getOrCreate(ctx, actionName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// avoid execution if last lock happened less than `maxInterval` ago
|
|
if rowLock.LastExecution != 0 {
|
|
lastExecutionTime := time.Unix(rowLock.LastExecution, 0)
|
|
if time.Since(lastExecutionTime) < maxInterval {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// try to get lock based on rowLow version
|
|
acquiredLock, err := sl.acquireLock(ctx, rowLock)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if acquiredLock {
|
|
fn(ctx)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (sl *ServerLockService) acquireLock(ctx context.Context, serverLock *serverLock) (bool, error) {
|
|
var result bool
|
|
|
|
err := sl.SQLStore.WithDbSession(ctx, func(dbSession *sqlstore.DBSession) error {
|
|
newVersion := serverLock.Version + 1
|
|
sql := `UPDATE server_lock SET
|
|
version = ?,
|
|
last_execution = ?
|
|
WHERE
|
|
id = ? AND version = ?`
|
|
|
|
res, err := dbSession.Exec(sql, newVersion, time.Now().Unix(), serverLock.Id, serverLock.Version)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
affected, err := res.RowsAffected()
|
|
result = affected == 1
|
|
|
|
return err
|
|
})
|
|
|
|
return result, err
|
|
}
|
|
|
|
func (sl *ServerLockService) getOrCreate(ctx context.Context, actionName string) (*serverLock, error) {
|
|
var result *serverLock
|
|
|
|
err := sl.SQLStore.WithTransactionalDbSession(ctx, func(dbSession *sqlstore.DBSession) error {
|
|
lockRows := []*serverLock{}
|
|
err := dbSession.Where("operation_uid = ?", actionName).Find(&lockRows)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(lockRows) > 0 {
|
|
result = lockRows[0]
|
|
return nil
|
|
}
|
|
|
|
lockRow := &serverLock{
|
|
OperationUID: actionName,
|
|
LastExecution: 0,
|
|
}
|
|
|
|
_, err = dbSession.Insert(lockRow)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
result = lockRow
|
|
|
|
return nil
|
|
})
|
|
|
|
return result, err
|
|
}
|