2015-11-13 19:11:28 +00:00
// Copyright 2015 clair authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package database
import (
"strconv"
"time"
"github.com/barakmich/glog"
cerrors "github.com/coreos/clair/utils/errors"
"github.com/google/cayley"
"github.com/google/cayley/graph"
"github.com/google/cayley/graph/path"
)
2015-11-20 20:03:20 +00:00
const (
fieldLockLocked = "locked"
fieldLockLockedValue = "locked"
fieldLockLockedBy = "locked_by"
fieldLockLockedUntil = "locked_until"
)
2015-11-13 19:11:28 +00:00
// Lock tries to set a temporary lock in the database.
// If a lock already exists with the given name/owner, then the lock is renewed
//
// Lock does not block, instead, it returns true and its expiration time
// is the lock has been successfully acquired or false otherwise
func Lock ( name string , duration time . Duration , owner string ) ( bool , time . Time ) {
pruneLocks ( )
until := time . Now ( ) . Add ( duration )
untilString := strconv . FormatInt ( until . Unix ( ) , 10 )
// Try to get the expiration time of a lock with the same name/owner
2015-11-20 20:03:20 +00:00
currentExpiration , err := toValue ( cayley . StartPath ( store , name ) . Has ( fieldLockLockedBy , owner ) . Out ( fieldLockLockedUntil ) )
2015-11-13 19:11:28 +00:00
if err == nil && currentExpiration != "" {
// Renew our lock
if currentExpiration == untilString {
return true , until
}
t := cayley . NewTransaction ( )
2015-11-20 20:03:20 +00:00
t . RemoveQuad ( cayley . Quad ( name , fieldLockLockedUntil , currentExpiration , "" ) )
t . AddQuad ( cayley . Quad ( name , fieldLockLockedUntil , untilString , "" ) )
2015-11-13 19:11:28 +00:00
// It is not necessary to verify if the lock is ours again in the transaction
// because if someone took it, the lock's current expiration probably changed and the transaction will fail
return store . ApplyTransaction ( t ) == nil , until
}
t := cayley . NewTransaction ( )
2015-11-20 20:03:20 +00:00
t . AddQuad ( cayley . Quad ( name , fieldLockLocked , fieldLockLockedValue , "" ) ) // Necessary to make the transaction fails if the lock already exists (and has not been pruned)
t . AddQuad ( cayley . Quad ( name , fieldLockLockedUntil , untilString , "" ) )
t . AddQuad ( cayley . Quad ( name , fieldLockLockedBy , owner , "" ) )
2015-11-13 19:11:28 +00:00
glog . SetStderrThreshold ( "FATAL" )
success := store . ApplyTransaction ( t ) == nil
glog . SetStderrThreshold ( "ERROR" )
return success , until
}
// Unlock unlocks a lock specified by its name if I own it
func Unlock ( name , owner string ) {
2015-11-15 01:14:27 +00:00
unlocked := 0
2015-11-20 20:03:20 +00:00
it , _ := cayley . StartPath ( store , name ) . Has ( fieldLockLocked , fieldLockLockedValue ) . Has ( fieldLockLockedBy , owner ) . Save ( fieldLockLockedUntil , fieldLockLockedUntil ) . BuildIterator ( ) . Optimize ( )
2015-11-13 19:11:28 +00:00
defer it . Close ( )
for cayley . RawNext ( it ) {
tags := make ( map [ string ] graph . Value )
it . TagResults ( tags )
2015-11-15 01:14:27 +00:00
t := cayley . NewTransaction ( )
2015-11-20 20:03:20 +00:00
t . RemoveQuad ( cayley . Quad ( name , fieldLockLocked , fieldLockLockedValue , "" ) )
t . RemoveQuad ( cayley . Quad ( name , fieldLockLockedUntil , store . NameOf ( tags [ fieldLockLockedUntil ] ) , "" ) )
t . RemoveQuad ( cayley . Quad ( name , fieldLockLockedBy , owner , "" ) )
2015-11-15 01:14:27 +00:00
err := store . ApplyTransaction ( t )
if err != nil {
log . Errorf ( "failed transaction (Unlock): %s" , err )
}
2015-11-13 19:11:28 +00:00
2015-11-15 01:14:27 +00:00
unlocked ++
}
if it . Err ( ) != nil {
log . Errorf ( "failed query in Unlock: %s" , it . Err ( ) )
}
if unlocked > 1 {
// We should never see this, it would mean that our database doesn't ensure quad uniqueness
// and that the entire lock system is jeopardized.
log . Errorf ( "found inconsistency in Unlock: matched %d times a locked named: %s" , unlocked , name )
}
2015-11-13 19:11:28 +00:00
}
// LockInfo returns the owner of a lock specified by its name and its
// expiration time
func LockInfo ( name string ) ( string , time . Time , error ) {
2015-11-20 20:03:20 +00:00
it , _ := cayley . StartPath ( store , name ) . Has ( fieldLockLocked , fieldLockLockedValue ) . Save ( fieldLockLockedUntil , fieldLockLockedUntil ) . Save ( fieldLockLockedBy , fieldLockLockedBy ) . BuildIterator ( ) . Optimize ( )
2015-11-13 19:11:28 +00:00
defer it . Close ( )
for cayley . RawNext ( it ) {
tags := make ( map [ string ] graph . Value )
it . TagResults ( tags )
2015-11-20 20:03:20 +00:00
tt , _ := strconv . ParseInt ( store . NameOf ( tags [ fieldLockLockedUntil ] ) , 10 , 64 )
return store . NameOf ( tags [ fieldLockLockedBy ] ) , time . Unix ( tt , 0 ) , nil
2015-11-13 19:11:28 +00:00
}
if it . Err ( ) != nil {
log . Errorf ( "failed query in LockInfo: %s" , it . Err ( ) )
return "" , time . Time { } , ErrBackendException
}
return "" , time . Time { } , cerrors . ErrNotFound
}
// pruneLocks removes every expired locks from the database
func pruneLocks ( ) {
now := time . Now ( )
// Delete every expired locks
2015-11-20 20:03:20 +00:00
it , _ := cayley . StartPath ( store , "locked" ) . In ( "locked" ) . Save ( fieldLockLockedUntil , fieldLockLockedUntil ) . Save ( fieldLockLockedBy , fieldLockLockedBy ) . BuildIterator ( ) . Optimize ( )
2015-11-13 19:11:28 +00:00
defer it . Close ( )
for cayley . RawNext ( it ) {
tags := make ( map [ string ] graph . Value )
it . TagResults ( tags )
n := store . NameOf ( it . Result ( ) )
2015-11-20 20:03:20 +00:00
t := store . NameOf ( tags [ fieldLockLockedUntil ] )
o := store . NameOf ( tags [ fieldLockLockedBy ] )
2015-11-13 19:11:28 +00:00
tt , _ := strconv . ParseInt ( t , 10 , 64 )
if now . Unix ( ) > tt {
2015-11-18 23:53:00 +00:00
log . Debugf ( "lock %s owned by %s has expired." , n , o )
2015-11-15 01:14:27 +00:00
tr := cayley . NewTransaction ( )
2015-11-20 20:03:20 +00:00
tr . RemoveQuad ( cayley . Quad ( n , fieldLockLocked , fieldLockLockedValue , "" ) )
tr . RemoveQuad ( cayley . Quad ( n , fieldLockLockedUntil , t , "" ) )
tr . RemoveQuad ( cayley . Quad ( n , fieldLockLockedBy , o , "" ) )
2015-11-15 01:14:27 +00:00
err := store . ApplyTransaction ( tr )
if err != nil {
log . Errorf ( "failed transaction (pruneLocks): %s" , err )
2015-11-18 23:53:00 +00:00
continue
2015-11-15 01:14:27 +00:00
}
2015-11-18 23:53:00 +00:00
log . Debugf ( "lock %s has been successfully pruned." , n )
2015-11-13 19:11:28 +00:00
}
}
2015-11-15 01:14:27 +00:00
if it . Err ( ) != nil {
log . Errorf ( "failed query in Unlock: %s" , it . Err ( ) )
}
2015-11-13 19:11:28 +00:00
}
// getLockedNodes returns every nodes that are currently locked
func getLockedNodes ( ) * path . Path {
return cayley . StartPath ( store , "locked" ) . In ( "locked" )
}