Skip to content

Commit

Permalink
Optimize revokeSalted by not calling view.List twice (#4465)
Browse files Browse the repository at this point in the history
* Optimize revokeSalted by not calling view.List twice

* Minor comment update

* Do not go through the orphaning dance if we are revoking the entire tree

* Update comment
  • Loading branch information
calvn authored May 18, 2018
1 parent 3a95aa5 commit a9daf49
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 35 deletions.
73 changes: 42 additions & 31 deletions vault/token_store.go
Original file line number Diff line number Diff line change
Expand Up @@ -1071,12 +1071,14 @@ func (ts *TokenStore) revokeOrphan(ctx context.Context, id string) error {
if err != nil {
return err
}
return ts.revokeSalted(ctx, saltedID)
return ts.revokeSalted(ctx, saltedID, false)
}

// revokeSalted is used to invalidate a given salted token,
// any child tokens will be orphaned.
func (ts *TokenStore) revokeSalted(ctx context.Context, saltedID string) (ret error) {
// revokeSalted is used to invalidate a given salted token, any child tokens
// will be orphaned unless otherwise specified. skipOrphan should be used
// whenever we are revoking the entire tree starting from a particular parent
// (e.g. revokeTreeSalted).
func (ts *TokenStore) revokeSalted(ctx context.Context, saltedID string, skipOrphan bool) (ret error) {
// Check and set the token deletion state. We only proceed with the deletion
// if we don't have a pending deletion (empty), or if the deletion previously
// failed (state is false)
Expand Down Expand Up @@ -1169,37 +1171,45 @@ func (ts *TokenStore) revokeSalted(ctx context.Context, saltedID string) (ret er
}
}

// Mark all children token as orphan by removing
// their parent index, and clear the parent entry.
//
// Marking the token as orphan is the correct behavior in here since
// revokeTreeSalted will ensure that they are deleted anyways if it's not an
// explicit call to orphan the child tokens (the delete occurs at the leaf
// node and uses parent prefix, not entry.Parent, to build the tree for
// traversal).
parentPath := parentPrefix + saltedID + "/"
children, err := ts.view.List(ctx, parentPath)
if err != nil {
return errwrap.Wrapf("failed to scan for children: {{err}}", err)
}
for _, child := range children {
entry, err := ts.lookupSalted(ctx, child, true)
if !skipOrphan {
// Mark all children token as orphan by removing
// their parent index, and clear the parent entry.
//
// Marking the token as orphan should be skipped if it's called by
// revokeTreeSalted to avoid unnecessary view.List operations. Since
// the deletion occurs in a DFS fashion we don't need to perform a delete
// on child prefixes as there will be none (as saltedID entry is a leaf node).
parentPath := parentPrefix + saltedID + "/"
children, err := ts.view.List(ctx, parentPath)
if err != nil {
return errwrap.Wrapf("failed to get child token: {{err}}", err)
return errwrap.Wrapf("failed to scan for children: {{err}}", err)
}
lock := locksutil.LockForKey(ts.tokenLocks, entry.ID)
lock.Lock()
for _, child := range children {
entry, err := ts.lookupSalted(ctx, child, true)
if err != nil {
return errwrap.Wrapf("failed to get child token: {{err}}", err)
}
lock := locksutil.LockForKey(ts.tokenLocks, entry.ID)
lock.Lock()

entry.Parent = ""
err = ts.store(ctx, entry)
if err != nil {
entry.Parent = ""
err = ts.store(ctx, entry)
if err != nil {
lock.Unlock()
return errwrap.Wrapf("failed to update child token: {{err}}", err)
}
lock.Unlock()
return errwrap.Wrapf("failed to update child token: {{err}}", err)

// Delete the the child storage entry after we update the token entry Since
// paths are not deeply nested (i.e. they are simply
// parenPrefix/<parentID>/<childID>), we can simply call view.Delete instead
// of logical.ClearView
index := parentPath + child
err = ts.view.Delete(ctx, index)
if err != nil {
return errwrap.Wrapf("failed to delete child entry: {{err}}", err)
}
}
lock.Unlock()
}
if err = logical.ClearView(ctx, ts.view.SubView(parentPath)); err != nil {
return errwrap.Wrapf("failed to delete entry: {{err}}", err)
}

return nil
Expand Down Expand Up @@ -1247,7 +1257,8 @@ func (ts *TokenStore) revokeTreeSalted(ctx context.Context, saltedID string) err
// take care of expiring them. If Vault is restarted, any revoked tokens
// would have been deleted, and any pending leases for deletion will be restored
// by the expiration manager.
if err := ts.revokeSalted(ctx, id); err != nil {
if err := ts.revokeSalted(ctx, id, true); err != nil {

return errwrap.Wrapf("failed to revoke entry: {{err}}", err)
}
// If the length of l is equal to 1, then the last token has been deleted
Expand Down
8 changes: 4 additions & 4 deletions vault/token_store_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -306,7 +306,7 @@ func TestTokenStore_HandleRequest_ListAccessors(t *testing.T) {
if err != nil {
t.Fatal(err)
}
ts.revokeSalted(context.Background(), salted)
ts.revokeSalted(context.Background(), salted, false)

req := logical.TestRequest(t, logical.ListOperation, "accessors/")

Expand Down Expand Up @@ -3498,7 +3498,7 @@ func TestTokenStore_RevokeUseCountToken(t *testing.T) {
return fmt.Errorf("keep it frosty")
}

err = ts.revokeSalted(context.Background(), saltTut)
err = ts.revokeSalted(context.Background(), saltTut, false)
if err == nil {
t.Fatalf("expected err")
}
Expand All @@ -3521,7 +3521,7 @@ func TestTokenStore_RevokeUseCountToken(t *testing.T) {

go func() {
cubbyFuncLock.RLock()
err := ts.revokeSalted(context.Background(), saltTut)
err := ts.revokeSalted(context.Background(), saltTut, false)
cubbyFuncLock.RUnlock()
if err == nil {
t.Fatalf("expected error")
Expand All @@ -3546,7 +3546,7 @@ func TestTokenStore_RevokeUseCountToken(t *testing.T) {
defer cubbyFuncLock.Unlock()
ts.cubbyholeDestroyer = origDestroyCubbyhole

err = ts.revokeSalted(context.Background(), saltTut)
err = ts.revokeSalted(context.Background(), saltTut, false)
if err != nil {
t.Fatal(err)
}
Expand Down

0 comments on commit a9daf49

Please sign in to comment.