diskpacked: bug fix walkPack and add test.

Replace reindex.go deleted blob.Ref detection loop with a regexp.
Replace diskpacked.go's isDeletedRef with same regexp.

Add a test that verifies the returned blobs match expectations and verifies
multipack diskpacked works..

Tests fail without the changes to reindex.go.

Change-Id: I332c2c3c8c37ebf262ce95e1ec0628146ab5108e
This commit is contained in:
Bill Thiede 2014-05-14 20:59:13 -07:00
parent eb7f66fe28
commit 83d2546d21
5 changed files with 77 additions and 32 deletions

View File

@ -41,6 +41,7 @@ import (
"log" "log"
"os" "os"
"path/filepath" "path/filepath"
"regexp"
"strings" "strings"
"sync" "sync"
@ -468,12 +469,6 @@ func headerLength(digest string, size uint32) int {
return len(fmt.Sprintf("[%s %d]", digest, size)) return len(fmt.Sprintf("[%s %d]", digest, size))
} }
// The header of deleted blobs has a digest in which the hash type is
// set to all 'x', but the correct size.
func isDeletedRef(digest string) bool {
return strings.HasPrefix(digest, "x")
}
// Type readSeekNopCloser is an io.ReadSeeker with a no-op Close method. // Type readSeekNopCloser is an io.ReadSeeker with a no-op Close method.
type readSeekNopCloser struct { type readSeekNopCloser struct {
io.ReadSeeker io.ReadSeeker
@ -485,6 +480,10 @@ func newReadSeekNopCloser(rs io.ReadSeeker) types.ReadSeekCloser {
return readSeekNopCloser{rs} return readSeekNopCloser{rs}
} }
// The header of deleted blobs has a digest in which the hash type is
// set to all 'x', the hash value is all '0', and has the correct size.
var deletedBlobRef = regexp.MustCompile(`^x+-0+$`)
// StreamBlobs Implements the blobserver.StreamBlobs interface. // StreamBlobs Implements the blobserver.StreamBlobs interface.
func (s *storage) StreamBlobs(ctx *context.Context, dest chan<- *blob.Blob, contToken string, limitBytes int64) (nextContinueToken string, err error) { func (s *storage) StreamBlobs(ctx *context.Context, dest chan<- *blob.Blob, contToken string, limitBytes int64) (nextContinueToken string, err error) {
defer close(dest) defer close(dest)
@ -556,7 +555,7 @@ func (s *storage) StreamBlobs(ctx *context.Context, dest chan<- *blob.Blob, cont
} }
offsetToAdd += int64(headerLength(digest, size)) offsetToAdd += int64(headerLength(digest, size))
if isDeletedRef(digest) { if deletedBlobRef.MatchString(digest) {
// Skip over deletion padding // Skip over deletion padding
_, err = io.CopyN(ioutil.Discard, r, int64(size)) _, err = io.CopyN(ioutil.Discard, r, int64(size))
if err != nil { if err != nil {

View File

@ -203,36 +203,14 @@ func (s *storage) walkPack(verbose bool, packID int,
} }
size = uint32(size64) size = uint32(size64)
// maybe deleted? if deletedBlobRef.Match(chunk[:i]) {
state, deleted := 0, true
if chunk[0] == 'x' {
Loop:
for _, c := range chunk[:i] {
switch state {
case 0:
if c != 'x' {
if c == '-' {
state++
} else {
deleted = false
break Loop
}
}
case 1:
if c != '0' {
deleted = false
break Loop
}
}
}
}
if deleted {
ref = blob.Ref{} ref = blob.Ref{}
if verbose { if verbose {
log.Printf("found deleted at %d", pos) log.Printf("found deleted at %d", pos)
} }
} else { } else {
ref, ok := blob.Parse(string(chunk[:i])) var ok bool
ref, ok = blob.Parse(string(chunk[:i]))
if !ok { if !ok {
return errAt("", fmt.Sprintf("cannot parse %q as blobref", chunk[:i])) return errAt("", fmt.Sprintf("cannot parse %q as blobref", chunk[:i]))
} }

View File

@ -0,0 +1,68 @@
/*
Copyright 2014 Google Inc.
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 diskpacked
import (
"testing"
"camlistore.org/pkg/blob"
)
type blobStat struct {
id int
ref string
offset int64
size uint32
}
func TestWalkPack(t *testing.T) {
want := []blobStat{
{0, "sha1-f7ff9e8b7bb2e09b70935a5d785e0cc5d9d0abf0", 49, 5},
{0, "sha1-70c07ec18ef89c5309bbb0937f3a6342411e1fdd", 103, 5},
{0, "<invalid-blob.Ref>", 157, 7},
{0, "sha1-70c07ec18ef89c5309bbb0937f3a6342411e1fdd", 213, 6},
{1, "sha1-fe05bcdcdc4928012781a5f1a2a77cbb5398e106", 49, 3},
{1, "sha1-ad782ecdac770fc6eb9a62e44f90873fb97fb26b", 101, 3},
{1, "sha1-b802f384302cb24fbab0a44997e820bf2e8507bb", 153, 5},
}
var got []blobStat
s := storage{root: "testdata"}
walk := func(packID int, ref blob.Ref, offset int64, size uint32) error {
t.Log(packID, ref, offset, size)
got = append(got, blobStat{
id: packID,
ref: ref.String(),
offset: offset,
size: size,
})
return nil
}
if err := s.Walk(nil, walk); err != nil {
t.Fatal(err)
}
if len(got) != len(want) {
t.Errorf("Got len %q want len %q", got, want)
}
for i, g := range got {
w := want[i]
if g.id != w.id || g.ref != w.ref || g.offset != w.offset || g.size != w.size {
t.Errorf("%d: got %d, %q, %d, %d want %d, %q, %d, %d", i, g.id, g.ref, g.offset, g.size, w.id, w.ref, w.offset, w.size)
}
}
}

Binary file not shown.

Binary file not shown.