forked from External/grumble
pkg/blobstore: modernize the blobstore package.
This commit is contained in:
parent
8effbbc6b3
commit
3dc3b25f57
10 changed files with 244 additions and 409 deletions
|
|
@ -13,7 +13,6 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"mumbleapp.com/grumble/pkg/blobstore"
|
|
||||||
"mumbleapp.com/grumble/pkg/cryptstate"
|
"mumbleapp.com/grumble/pkg/cryptstate"
|
||||||
"mumbleapp.com/grumble/pkg/mumbleproto"
|
"mumbleapp.com/grumble/pkg/mumbleproto"
|
||||||
"mumbleapp.com/grumble/pkg/packetdata"
|
"mumbleapp.com/grumble/pkg/packetdata"
|
||||||
|
|
@ -579,7 +578,7 @@ func (client *Client) sendChannelTree(channel *Channel) {
|
||||||
if client.Version >= 0x10202 {
|
if client.Version >= 0x10202 {
|
||||||
chanstate.DescriptionHash = channel.DescriptionBlobHashBytes()
|
chanstate.DescriptionHash = channel.DescriptionBlobHashBytes()
|
||||||
} else {
|
} else {
|
||||||
buf, err := blobstore.Get(channel.DescriptionBlob)
|
buf, err := blobStore.Get(channel.DescriptionBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic("Blobstore error.")
|
panic("Blobstore error.")
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
var servers map[int64]*Server
|
var servers map[int64]*Server
|
||||||
|
var blobStore blobstore.BlobStore
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
var err error
|
var err error
|
||||||
|
|
@ -57,10 +58,7 @@ func main() {
|
||||||
if err != nil && !os.IsExist(err) {
|
if err != nil && !os.IsExist(err) {
|
||||||
log.Fatalf("Unable to create blob directory: %v", err)
|
log.Fatalf("Unable to create blob directory: %v", err)
|
||||||
}
|
}
|
||||||
err = blobstore.Open(blobDir)
|
blobStore = blobstore.Open(blobDir)
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Unable to initialize blobstore: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check whether we should regenerate the default global keypair
|
// Check whether we should regenerate the default global keypair
|
||||||
// and corresponding certificate.
|
// and corresponding certificate.
|
||||||
|
|
|
||||||
15
message.go
15
message.go
|
|
@ -10,7 +10,6 @@ import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
"mumbleapp.com/grumble/pkg/ban"
|
"mumbleapp.com/grumble/pkg/ban"
|
||||||
"mumbleapp.com/grumble/pkg/blobstore"
|
|
||||||
"mumbleapp.com/grumble/pkg/freezer"
|
"mumbleapp.com/grumble/pkg/freezer"
|
||||||
"mumbleapp.com/grumble/pkg/mumbleproto"
|
"mumbleapp.com/grumble/pkg/mumbleproto"
|
||||||
"net"
|
"net"
|
||||||
|
|
@ -251,7 +250,7 @@ func (server *Server) handleChannelStateMessage(client *Client, msg *Message) {
|
||||||
|
|
||||||
key := ""
|
key := ""
|
||||||
if len(description) > 0 {
|
if len(description) > 0 {
|
||||||
key, err = blobstore.Put([]byte(description))
|
key, err = blobStore.Put([]byte(description))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
@ -433,7 +432,7 @@ func (server *Server) handleChannelStateMessage(client *Client, msg *Message) {
|
||||||
if len(description) == 0 {
|
if len(description) == 0 {
|
||||||
channel.DescriptionBlob = ""
|
channel.DescriptionBlob = ""
|
||||||
} else {
|
} else {
|
||||||
key, err := blobstore.Put([]byte(description))
|
key, err := blobStore.Put([]byte(description))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
@ -696,7 +695,7 @@ func (server *Server) handleUserStateMessage(client *Client, msg *Message) {
|
||||||
broadcast := false
|
broadcast := false
|
||||||
|
|
||||||
if userstate.Texture != nil && target.user != nil {
|
if userstate.Texture != nil && target.user != nil {
|
||||||
key, err := blobstore.Put(userstate.Texture)
|
key, err := blobStore.Put(userstate.Texture)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
return
|
return
|
||||||
|
|
@ -737,7 +736,7 @@ func (server *Server) handleUserStateMessage(client *Client, msg *Message) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if userstate.Comment != nil && target.user != nil {
|
if userstate.Comment != nil && target.user != nil {
|
||||||
key, err := blobstore.Put([]byte(*userstate.Comment))
|
key, err := blobStore.Put([]byte(*userstate.Comment))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
@ -1467,7 +1466,7 @@ func (server *Server) handleRequestBlob(client *Client, msg *Message) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if target.user.HasTexture() {
|
if target.user.HasTexture() {
|
||||||
buf, err := blobstore.Get(target.user.TextureBlob)
|
buf, err := blobStore.Get(target.user.TextureBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
return
|
return
|
||||||
|
|
@ -1492,7 +1491,7 @@ func (server *Server) handleRequestBlob(client *Client, msg *Message) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if target.user.HasComment() {
|
if target.user.HasComment() {
|
||||||
buf, err := blobstore.Get(target.user.CommentBlob)
|
buf, err := blobStore.Get(target.user.CommentBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
return
|
return
|
||||||
|
|
@ -1517,7 +1516,7 @@ func (server *Server) handleRequestBlob(client *Client, msg *Message) {
|
||||||
if channel, ok := server.Channels[int(cid)]; ok {
|
if channel, ok := server.Channels[int(cid)]; ok {
|
||||||
if channel.HasDescription() {
|
if channel.HasDescription() {
|
||||||
chanstate.Reset()
|
chanstate.Reset()
|
||||||
buf, err := blobstore.Get(channel.DescriptionBlob)
|
buf, err := blobStore.Get(channel.DescriptionBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err)
|
server.Panicf("Blobstore error: %v", err)
|
||||||
return
|
return
|
||||||
|
|
|
||||||
|
|
@ -14,7 +14,6 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"log"
|
"log"
|
||||||
"mumbleapp.com/grumble/pkg/ban"
|
"mumbleapp.com/grumble/pkg/ban"
|
||||||
"mumbleapp.com/grumble/pkg/blobstore"
|
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
@ -148,7 +147,7 @@ func populateChannelInfoFromDatabase(server *Server, c *Channel, db *sql.DB) err
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(description) > 0 {
|
if len(description) > 0 {
|
||||||
key, err := blobstore.Put([]byte(description))
|
key, err := blobStore.Put([]byte(description))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
@ -428,7 +427,7 @@ func populateUsers(server *Server, db *sql.DB) (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(Texture) > 0 {
|
if len(Texture) > 0 {
|
||||||
key, err := blobstore.Put(Texture)
|
key, err := blobStore.Put(Texture)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
@ -468,7 +467,7 @@ func populateUsers(server *Server, db *sql.DB) (err error) {
|
||||||
case UserInfoEmail:
|
case UserInfoEmail:
|
||||||
user.Email = Value
|
user.Email = Value
|
||||||
case UserInfoComment:
|
case UserInfoComment:
|
||||||
key, err := blobstore.Put([]byte(Value))
|
key, err := blobStore.Put([]byte(Value))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,14 +0,0 @@
|
||||||
include $(GOROOT)/src/Make.inc
|
|
||||||
|
|
||||||
TARG = grumble/blobstore
|
|
||||||
GOFILES = \
|
|
||||||
blobstore.go \
|
|
||||||
blobreader.go
|
|
||||||
|
|
||||||
ifeq ($(GOOS),windows)
|
|
||||||
GOFILES += lock_windows.go
|
|
||||||
else
|
|
||||||
GOFILES += lock_unix.go
|
|
||||||
endif
|
|
||||||
|
|
||||||
include $(GOROOT)/src/Make.pkg
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
// Copyright (c) 2011 The Grumble Authors
|
// Copyright (c) 2011-2013 The Grumble Authors
|
||||||
// The use of this source code is goverened by a BSD-style
|
// The use of this source code is goverened by a BSD-style
|
||||||
// license that can be found in the LICENSE-file.
|
// license that can be found in the LICENSE-file.
|
||||||
|
|
||||||
|
|
@ -8,46 +8,77 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"crypto/sha1"
|
"crypto/sha1"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
|
||||||
"hash"
|
"hash"
|
||||||
"io"
|
"io"
|
||||||
)
|
)
|
||||||
|
|
||||||
// blobReader is based on the principles of the checksumReader from the archive/zip
|
// EOFHashMismatchError signals that a blobReader reached EOF, but that
|
||||||
// package of the Go standard library.
|
// the calculated hash did not match the given blob key. This signals
|
||||||
|
// a successful read of the blob, but that the on-disk content is
|
||||||
|
// corrupted in some fashion.
|
||||||
|
type EOFHashMismatchError struct {
|
||||||
|
// Sum represents that was calculated during the read operation.
|
||||||
|
Sum []byte
|
||||||
|
}
|
||||||
|
|
||||||
// ErrHashMismatch is returned if a blobReader has read a file whose computed hash
|
func (hme EOFHashMismatchError) Error() string {
|
||||||
// did not match its key.
|
return "blobstore: EOF hash mismatch"
|
||||||
var ErrHashMismatch = errors.New("hash mismatch")
|
}
|
||||||
|
|
||||||
// blobReader reads a blob from disk, hashing all incoming data. On EOF, it checks
|
// blobReader implements an io.ReadCloser that reads a blob from disk
|
||||||
// whether the read data matches the key.
|
// and hashes all incoming data to ensure integrity. On EOF, it matches
|
||||||
|
// its calculated hash with the given blob key in order to detect data
|
||||||
|
// corruption.
|
||||||
|
//
|
||||||
|
// If a mismatch is detected on EOF, the blobReader will return
|
||||||
|
// the error ErrEOFHashMismatch instead of a regular io.EOF error.
|
||||||
type blobReader struct {
|
type blobReader struct {
|
||||||
rc io.ReadCloser
|
rc io.ReadCloser
|
||||||
sum []byte
|
sum []byte
|
||||||
hash hash.Hash
|
hash hash.Hash
|
||||||
}
|
}
|
||||||
|
|
||||||
func newBlobReader(rc io.ReadCloser, key string) (br *blobReader, err error) {
|
// newBlobReader returns a new blobReader reading from rc.
|
||||||
|
// The rc is expected to be a blobstore entry identified by
|
||||||
|
// the given key. (The blobstore is content addressible, and
|
||||||
|
// a blob's key represents the SHA1 of its content).
|
||||||
|
func newBlobReader(rc io.ReadCloser, key string) (*blobReader, error) {
|
||||||
sum, err := hex.DecodeString(key)
|
sum, err := hex.DecodeString(key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
return &blobReader{rc, sum, sha1.New()}, nil
|
return &blobReader{rc, sum, sha1.New()}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *blobReader) Read(b []byte) (n int, err error) {
|
// Read implements the Read method of io.ReadCloser.
|
||||||
n, err = r.rc.Read(b)
|
// This Read implementation passes on read calls to the
|
||||||
r.hash.Write(b[:n])
|
// wrapper io.ReadCloser and hashes all read content.
|
||||||
|
// When EOF is reached, the sum of the streaming hash
|
||||||
|
// hash is calculated and compared to the blob key given
|
||||||
|
// in newBlobReader. If the calculated hash does not match
|
||||||
|
// the blob key, the special error ErrEOFHashMismatch is
|
||||||
|
// returned to signal EOF, while also signalling a hash
|
||||||
|
// mismatch.
|
||||||
|
func (r *blobReader) Read(b []byte) (int, error) {
|
||||||
|
n, err := r.rc.Read(b)
|
||||||
|
_, werr := r.hash.Write(b[:n])
|
||||||
|
if werr != nil {
|
||||||
|
return 0, werr
|
||||||
|
}
|
||||||
if err != io.EOF {
|
if err != io.EOF {
|
||||||
return
|
return n, err
|
||||||
}
|
}
|
||||||
if !bytes.Equal(r.sum, r.hash.Sum(nil)) {
|
// Match the calculated digest with the expected
|
||||||
err = ErrHashMismatch
|
// digest on EOF.
|
||||||
|
calcSum := r.hash.Sum(nil)
|
||||||
|
if !bytes.Equal(r.sum, calcSum) {
|
||||||
|
return 0, EOFHashMismatchError{Sum: calcSum}
|
||||||
}
|
}
|
||||||
return
|
return n, io.EOF
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close implements the Close method of io.ReadCloser.
|
||||||
|
// This Close method simply closes the wrapped io.ReadCloser.
|
||||||
func (r *blobReader) Close() error {
|
func (r *blobReader) Close() error {
|
||||||
return r.rc.Close()
|
return r.rc.Close()
|
||||||
}
|
}
|
||||||
41
pkg/blobstore/blobreader_test.go
Normal file
41
pkg/blobstore/blobreader_test.go
Normal file
|
|
@ -0,0 +1,41 @@
|
||||||
|
// Copyright (c) 2013 The Grumble Authors
|
||||||
|
// The use of this source code is goverened by a BSD-style
|
||||||
|
// license that can be found in the LICENSE-file.
|
||||||
|
|
||||||
|
package blobstore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
type blobReaderTest struct {
|
||||||
|
Key string
|
||||||
|
ExpectedSum string
|
||||||
|
Data string
|
||||||
|
}
|
||||||
|
|
||||||
|
var blobReaderTests = []blobReaderTest{
|
||||||
|
{
|
||||||
|
Key: "a3da7877f94ad4cf58636a395fff77537cb8b919",
|
||||||
|
ExpectedSum: "a3da7877f94ad4cf58636a395fff77537cb8b919",
|
||||||
|
Data: "Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBlobReader(t *testing.T) {
|
||||||
|
for _, test := range blobReaderTests {
|
||||||
|
rc := ioutil.NopCloser(bytes.NewBufferString(test.Data))
|
||||||
|
br, err := newBlobReader(rc, test.Key)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("unable to construct blob reader: %v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
_, err = io.Copy(ioutil.Discard, br)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("got error: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -2,8 +2,7 @@
|
||||||
// The use of this source code is goverened by a BSD-style
|
// The use of this source code is goverened by a BSD-style
|
||||||
// license that can be found in the LICENSE-file.
|
// license that can be found in the LICENSE-file.
|
||||||
|
|
||||||
// This package implements a simple disk-persisted content-addressed
|
// This package implements a simple disk-persisted content-addressed blobstore.
|
||||||
// blobstore.
|
|
||||||
package blobstore
|
package blobstore
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|
@ -13,263 +12,179 @@ import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// ErrNoSuchKey signals that a blob with the given key does
|
||||||
|
// not exist in the BlobStore.
|
||||||
|
ErrNoSuchKey = errors.New("blobstore: no such key")
|
||||||
|
|
||||||
|
// ErrBadKey signals that the given key is not well formed.
|
||||||
|
ErrBadKey = errors.New("blobstore: bad key")
|
||||||
|
)
|
||||||
|
|
||||||
|
// BlobStore represents a simple disk-persisted content addressible
|
||||||
|
// blob store that uses the file system for persistence.
|
||||||
|
//
|
||||||
|
// Blobs in the blobstore are indexed by their SHA1 hash.
|
||||||
|
//
|
||||||
|
// The BlobStore is backed by a directory on the filesystem. This
|
||||||
|
// directory contains subdirectories which contain keys (SHA1 hashes).
|
||||||
|
// Each subdirectory is named according to the first hex-encoded byte
|
||||||
|
// of the keys that subdirectory contains.
|
||||||
|
//
|
||||||
|
// For example, a file that has the content 'hello world' will have
|
||||||
|
// the SHA1 hash '2aae6c35c94fcfb415dbe95f408b9ce91ee846ed'. If our
|
||||||
|
// blobstore's backing directory is called 'blobstore', the blob with
|
||||||
|
// only 'hello world' in it will be stored as follows:
|
||||||
|
//
|
||||||
|
// blobstore/2a/2aae6c35c94fcfb415dbe95f408b9ce91ee846ed
|
||||||
|
//
|
||||||
|
// The BlobStore is self-synchronizing, relying on the filesystem
|
||||||
|
// operations to ensure atomicity. Thus, accessing a single BlobStore
|
||||||
|
// from multiple goroutines should have no ill side effects.
|
||||||
type BlobStore struct {
|
type BlobStore struct {
|
||||||
dir string
|
dir string
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
// Open opens an existing BlobStore. The path parameter must
|
||||||
ErrBadFile = errors.New("a bad file exists in the blobstore directory. unable to create container directores.")
|
// point to a directory that already exists for correct
|
||||||
ErrNoSuchKey = errors.New("no such key")
|
// operation, however, the Open function does not check that
|
||||||
ErrInvalidKey = errors.New("invalid key")
|
// this is the case.
|
||||||
)
|
func Open(path string) BlobStore {
|
||||||
|
return BlobStore{dir: path}
|
||||||
var (
|
|
||||||
defaultStore *BlobStore
|
|
||||||
defaultMutex sync.Mutex
|
|
||||||
)
|
|
||||||
|
|
||||||
// Open an existing, or create a new BlobStore at path.
|
|
||||||
// Path must point to a directory, and must already exist.
|
|
||||||
// See NewBlobStore for more information.
|
|
||||||
func Open(path string) (err error) {
|
|
||||||
defaultMutex.Lock()
|
|
||||||
defer defaultMutex.Unlock()
|
|
||||||
|
|
||||||
if defaultStore != nil {
|
|
||||||
panic("Default BlobStore already open")
|
|
||||||
}
|
|
||||||
|
|
||||||
defaultStore, err = NewBlobStore(path)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close the open default BlobStore. This removes the lockfile allowing
|
// isValidKey checks whether key is a valid BlobStore key.
|
||||||
// other processes to open the BlobStore.
|
func isValidKey(key string) bool {
|
||||||
func Close() (err error) {
|
// SHA1 digests are 40 bytes long when hex-encoded.
|
||||||
if defaultStore == nil {
|
|
||||||
panic("DefaultStore not open")
|
|
||||||
}
|
|
||||||
|
|
||||||
err = defaultStore.Close()
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
defaultStore = nil
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lookup a blob by its key and return a buffer containing the contents
|
|
||||||
// of the blob.
|
|
||||||
func Get(key string) (buf []byte, err error) {
|
|
||||||
return defaultStore.Get(key)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store a blob. If the blob was successfully stored, the returned key
|
|
||||||
// can be used to retrieve the buf from the BlobStore.
|
|
||||||
func Put(buf []byte) (key string, err error) {
|
|
||||||
return defaultStore.Put(buf)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Open an existing, or create a new BlobStore residing at path.
|
|
||||||
// Path must point to a directory, and must already exist.
|
|
||||||
func NewBlobStore(path string) (bs *BlobStore, err error) {
|
|
||||||
// Does the directory exist?
|
|
||||||
dir, err := os.Open(path)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
dir.Close()
|
|
||||||
|
|
||||||
dirStructureExists := true
|
|
||||||
// Check whether a 'blobstore' file exists in the directory.
|
|
||||||
// The existence of the file signals that the directory already
|
|
||||||
// has the correct hierarchy structure.
|
|
||||||
bsf, err := os.Open(filepath.Join(path, "blobstore"))
|
|
||||||
if err != nil {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
dirStructureExists = false
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
bsf.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
if !dirStructureExists {
|
|
||||||
for i := 0; i < 256; i++ {
|
|
||||||
outer := filepath.Join(path, hex.EncodeToString([]byte{byte(i)}))
|
|
||||||
err = os.Mkdir(outer, 0700)
|
|
||||||
if os.IsExist(err) {
|
|
||||||
// The path already exists. Stat it to check whether it is indeed
|
|
||||||
// a directory.
|
|
||||||
fi, err := os.Stat(outer)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if !fi.IsDir() {
|
|
||||||
return nil, ErrBadFile
|
|
||||||
}
|
|
||||||
} else if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
for j := 0; j < 256; j++ {
|
|
||||||
inner := filepath.Join(outer, hex.EncodeToString([]byte{byte(j)}))
|
|
||||||
err = os.Mkdir(inner, 0700)
|
|
||||||
if os.IsExist(err) {
|
|
||||||
// The path already exists. Stat it to check whether it is indeed
|
|
||||||
// a directory.
|
|
||||||
fi, err := os.Stat(outer)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if !fi.IsDir() {
|
|
||||||
return nil, ErrBadFile
|
|
||||||
}
|
|
||||||
} else if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add a blobstore file to signal that a correct directory
|
|
||||||
// structure exists for this blobstore.
|
|
||||||
bsf, err = os.Create(filepath.Join(path, "blobstore"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
bsf.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
bs = &BlobStore{
|
|
||||||
dir: path,
|
|
||||||
}
|
|
||||||
return bs, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close an open BlobStore. This removes the lockfile allowing
|
|
||||||
// other processes to open the BlobStore.
|
|
||||||
func (bs *BlobStore) Close() (err error) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Checks that a given key is a valid key for the BlobStore.
|
|
||||||
// If it is, it returns the three components that make up the on-disk path
|
|
||||||
// the given key can be found or should be stored at.
|
|
||||||
func getKeyComponents(key string) (dir1, dir2, fn string, err error) {
|
|
||||||
// SHA1 digests are 40 bytes long when hex-encoded
|
|
||||||
if len(key) != 40 {
|
if len(key) != 40 {
|
||||||
err = ErrInvalidKey
|
return false
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check whether the string is valid hex-encoding.
|
// Check whether the string is valid hex-encoding.
|
||||||
_, err = hex.DecodeString(key)
|
_, err := hex.DecodeString(key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = ErrInvalidKey
|
return false
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return key[0:2], key[2:4], key[4:], nil
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// Lookup the path hat a key would have on disk.
|
// extractKeyComponents returns the directory and the filename that the
|
||||||
// Returns an error if the key is not a valid BlobStore key.
|
// blob identified by key should be stored under in the BlobStore.
|
||||||
func (bs *BlobStore) pathForKey(key string) (fn string, err error) {
|
// This function also checks whether the key is valid. If not, it returns
|
||||||
dir1, dir2, rest, err := getKeyComponents(key)
|
// ErrBadKey.
|
||||||
if err != nil {
|
func extractKeyComponents(key string) (dir string, fn string, err error) {
|
||||||
return
|
if !isValidKey(key) {
|
||||||
|
return "", "", ErrBadKey
|
||||||
}
|
}
|
||||||
|
return key[0:2], key, nil
|
||||||
fn = filepath.Join(bs.dir, dir1, dir2, rest)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Lookup a blob by its key and return a buffer containing the contents
|
// Get returns a byte slice containing the contents of
|
||||||
// of the blob.
|
// the blob identified by key. If no such blob is found,
|
||||||
func (bs *BlobStore) Get(key string) (buf []byte, err error) {
|
// Get returns ErrNoSuchKey.
|
||||||
fn, err := bs.pathForKey(key)
|
func (bs BlobStore) Get(key string) ([]byte, error) {
|
||||||
|
dir, fn, err := extractKeyComponents(key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
file, err := os.Open(fn)
|
blobfn := filepath.Join(bs.dir, dir, fn)
|
||||||
|
f, err := os.Open(blobfn)
|
||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
err = ErrNoSuchKey
|
return nil, ErrNoSuchKey
|
||||||
return
|
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
br, err := newBlobReader(file, key)
|
br, err := newBlobReader(f, key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
file.Close()
|
f.Close()
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
defer br.Close()
|
defer br.Close()
|
||||||
|
|
||||||
buf, err = ioutil.ReadAll(br)
|
buf, err := ioutil.ReadAll(br)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
return buf, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store a blob. If the blob was successfully stored, the returned key
|
// Put puts the contents of blob into the BlobStore. If
|
||||||
// can be used to retrieve the buf from the BlobStore.
|
// the blob was successfully stored, the returned key can
|
||||||
func (bs *BlobStore) Put(buf []byte) (key string, err error) {
|
// be used to retrieve the buf from the BlobStore at a
|
||||||
|
// later time.
|
||||||
|
func (bs BlobStore) Put(buf []byte) (key string, err error) {
|
||||||
// Calculate the key for the blob. We can't really delay it more than this,
|
// Calculate the key for the blob. We can't really delay it more than this,
|
||||||
// since we need to know the key for the blob to check whether it's already on
|
// since we need to know the key for the blob to check whether it's already on
|
||||||
// disk.
|
// disk.
|
||||||
h := sha1.New()
|
h := sha1.New()
|
||||||
h.Write(buf)
|
_, err = h.Write(buf)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
key = hex.EncodeToString(h.Sum(nil))
|
key = hex.EncodeToString(h.Sum(nil))
|
||||||
|
|
||||||
// Get the components that make up the on-disk
|
// Get the components that make up the on-disk
|
||||||
// path for the blob.
|
// path for the blob.
|
||||||
dir1, dir2, rest, err := getKeyComponents(key)
|
dir, fn, err := extractKeyComponents(key)
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
blobpath := filepath.Join(bs.dir, dir1, dir2, rest)
|
|
||||||
blobdir := filepath.Join(bs.dir, dir1, dir2)
|
|
||||||
|
|
||||||
// Check if the blob already exists.
|
|
||||||
file, err := os.Open(blobpath)
|
|
||||||
if err == nil {
|
|
||||||
// File exists. Job's done.
|
|
||||||
file.Close()
|
|
||||||
return
|
|
||||||
} else {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
// No such file exists on disk. Ready to rock!
|
|
||||||
} else {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a temporary file to write to. Once we're done, we
|
|
||||||
// can atomically rename the file to the correct key.
|
|
||||||
file, err = ioutil.TempFile(blobdir, rest)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
tmpfn := file.Name()
|
|
||||||
|
|
||||||
_, err = file.Write(buf)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
err = file.Sync()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
err = file.Close()
|
blobdir := filepath.Join(bs.dir, dir)
|
||||||
|
blobpath := filepath.Join(blobdir, fn)
|
||||||
|
|
||||||
|
// Check if the blob already exists.
|
||||||
|
_, err = os.Stat(blobpath)
|
||||||
|
if err == nil {
|
||||||
|
// The file already exists. Our job is done.
|
||||||
|
return key, nil
|
||||||
|
} else if os.IsNotExist(err) {
|
||||||
|
// The blob does not exist on disk yet.
|
||||||
|
// Fallthrough.
|
||||||
|
} else if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure that blobdir exist.
|
||||||
|
err = os.Mkdir(blobdir, 0750)
|
||||||
|
if err != nil && !os.IsExist(err) {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a temporary file to write to.
|
||||||
|
//
|
||||||
|
// Once we're done, we can atomically rename the file
|
||||||
|
// to the correct key.
|
||||||
|
//
|
||||||
|
// This method is racy: two callers can attempt to write
|
||||||
|
// the same blob at the same time. This shouldn't affect
|
||||||
|
// the consistency of the final blob, but worst case, we've
|
||||||
|
// done some extra work.
|
||||||
|
f, err := ioutil.TempFile(blobdir, fn)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpfn := f.Name()
|
||||||
|
_, err = f.Write(buf)
|
||||||
|
if err != nil {
|
||||||
|
f.Close()
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = f.Sync()
|
||||||
|
if err != nil {
|
||||||
|
f.Close()
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = f.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
@ -277,7 +192,7 @@ func (bs *BlobStore) Put(buf []byte) (key string, err error) {
|
||||||
err = os.Rename(tmpfn, blobpath)
|
err = os.Rename(tmpfn, blobpath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
os.Remove(tmpfn)
|
os.Remove(tmpfn)
|
||||||
return
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
return key, nil
|
return key, nil
|
||||||
|
|
|
||||||
|
|
@ -10,91 +10,9 @@ import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestMakeAllCreateAll(t *testing.T) {
|
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.RemoveAll(dir)
|
|
||||||
|
|
||||||
bs, err := NewBlobStore(dir)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer bs.Close()
|
|
||||||
|
|
||||||
// Check whether the blobstore created all the directories...
|
|
||||||
for i := 0; i < 256; i++ {
|
|
||||||
for j := 0; j < 256; j++ {
|
|
||||||
dirname := filepath.Join(dir, hex.EncodeToString([]byte{byte(i)}), hex.EncodeToString([]byte{byte(j)}))
|
|
||||||
fi, err := os.Stat(dirname)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if !fi.IsDir() {
|
|
||||||
t.Errorf("Not a directory")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAllInvalidFiles(t *testing.T) {
|
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.RemoveAll(dir)
|
|
||||||
|
|
||||||
err = ioutil.WriteFile(filepath.Join(dir, "00"), []byte{0x0f, 0x00}, 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err = NewBlobStore(dir)
|
|
||||||
if err == ErrBadFile {
|
|
||||||
// Success
|
|
||||||
} else if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
} else {
|
|
||||||
t.Error("NewBlobStore returned without error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAllInvalidFilesLevel2(t *testing.T) {
|
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.RemoveAll(dir)
|
|
||||||
|
|
||||||
err = os.Mkdir(filepath.Join(dir, "00"), 0700)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = ioutil.WriteFile(filepath.Join(dir, "00", "00"), []byte{0x0f, 0x00}, 0600)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err = NewBlobStore(dir)
|
|
||||||
if err == ErrBadFile {
|
|
||||||
// Success
|
|
||||||
} else if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
} else {
|
|
||||||
t.Error("NewBlobStore returned without error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestStoreRetrieve(t *testing.T) {
|
func TestStoreRetrieve(t *testing.T) {
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
dir, err := ioutil.TempDir("", "blobstore")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
@ -103,12 +21,7 @@ func TestStoreRetrieve(t *testing.T) {
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(dir)
|
defer os.RemoveAll(dir)
|
||||||
|
|
||||||
bs, err := NewBlobStore(dir)
|
bs := OpenBlobStore(dir)
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer bs.Close()
|
|
||||||
|
|
||||||
data := []byte{0xde, 0xad, 0xca, 0xfe, 0xba, 0xbe, 0xbe, 0xef}
|
data := []byte{0xde, 0xad, 0xca, 0xfe, 0xba, 0xbe, 0xbe, 0xef}
|
||||||
|
|
||||||
|
|
@ -136,12 +49,7 @@ func TestReadNonExistantKey(t *testing.T) {
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(dir)
|
defer os.RemoveAll(dir)
|
||||||
|
|
||||||
bs, err := NewBlobStore(dir)
|
bs := OpenBlobStore(dir)
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer bs.Close()
|
|
||||||
|
|
||||||
h := sha1.New()
|
h := sha1.New()
|
||||||
h.Write([]byte{0x42})
|
h.Write([]byte{0x42})
|
||||||
|
|
@ -160,12 +68,7 @@ func TestReadInvalidKeyLength(t *testing.T) {
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(dir)
|
defer os.RemoveAll(dir)
|
||||||
|
|
||||||
bs, err := NewBlobStore(dir)
|
bs := OpenBlobStore(dir)
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer bs.Close()
|
|
||||||
|
|
||||||
key := ""
|
key := ""
|
||||||
for i := 0; i < 5; i++ {
|
for i := 0; i < 5; i++ {
|
||||||
|
|
@ -173,13 +76,13 @@ func TestReadInvalidKeyLength(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = bs.Get(key)
|
_, err = bs.Get(key)
|
||||||
if err != ErrInvalidKey {
|
if err != ErrBadKey {
|
||||||
t.Error("Expected invalid key for %v, got %v", key, err)
|
t.Error("Expected invalid key for %v, got %v", key, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestReadInvalidKeyNonHex(t *testing.T) {
|
func TestReadBadKeyNonHex(t *testing.T) {
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
dir, err := ioutil.TempDir("", "blobstore")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
|
|
@ -187,12 +90,7 @@ func TestReadInvalidKeyNonHex(t *testing.T) {
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(dir)
|
defer os.RemoveAll(dir)
|
||||||
|
|
||||||
bs, err := NewBlobStore(dir)
|
bs := OpenBlobStore(dir)
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer bs.Close()
|
|
||||||
|
|
||||||
key := ""
|
key := ""
|
||||||
for i := 0; i < 40; i++ {
|
for i := 0; i < 40; i++ {
|
||||||
|
|
@ -200,38 +98,8 @@ func TestReadInvalidKeyNonHex(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = bs.Get(key)
|
_, err = bs.Get(key)
|
||||||
if err != ErrInvalidKey {
|
if err != ErrBadKey {
|
||||||
t.Errorf("Expected invalid key for %v, got %v", key, err)
|
t.Errorf("Expected bad key for %v, got %v", key, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultBlobStore(t *testing.T) {
|
|
||||||
dir, err := ioutil.TempDir("", "blobstore")
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.RemoveAll(dir)
|
|
||||||
|
|
||||||
err = Open(dir)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
data := []byte{0xf, 0x0, 0x0, 0xb, 0xa, 0xf}
|
|
||||||
|
|
||||||
key, err := Put(data)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
fetchedData, err := Get(key)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if !bytes.Equal(fetchedData, data) {
|
|
||||||
t.Errorf("stored data and retrieved data does not match: %v vs. %v", fetchedData, data)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -18,7 +18,6 @@ import (
|
||||||
"hash"
|
"hash"
|
||||||
"log"
|
"log"
|
||||||
"mumbleapp.com/grumble/pkg/ban"
|
"mumbleapp.com/grumble/pkg/ban"
|
||||||
"mumbleapp.com/grumble/pkg/blobstore"
|
|
||||||
"mumbleapp.com/grumble/pkg/freezer"
|
"mumbleapp.com/grumble/pkg/freezer"
|
||||||
"mumbleapp.com/grumble/pkg/htmlfilter"
|
"mumbleapp.com/grumble/pkg/htmlfilter"
|
||||||
"mumbleapp.com/grumble/pkg/logtarget"
|
"mumbleapp.com/grumble/pkg/logtarget"
|
||||||
|
|
@ -610,7 +609,7 @@ func (server *Server) finishAuthenticate(client *Client) {
|
||||||
if client.Version >= 0x10203 {
|
if client.Version >= 0x10203 {
|
||||||
userstate.TextureHash = client.user.TextureBlobHashBytes()
|
userstate.TextureHash = client.user.TextureBlobHashBytes()
|
||||||
} else {
|
} else {
|
||||||
buf, err := blobstore.Get(client.user.TextureBlob)
|
buf, err := blobStore.Get(client.user.TextureBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err.Error())
|
server.Panicf("Blobstore error: %v", err.Error())
|
||||||
}
|
}
|
||||||
|
|
@ -623,7 +622,7 @@ func (server *Server) finishAuthenticate(client *Client) {
|
||||||
if client.Version >= 0x10203 {
|
if client.Version >= 0x10203 {
|
||||||
userstate.CommentHash = client.user.CommentBlobHashBytes()
|
userstate.CommentHash = client.user.CommentBlobHashBytes()
|
||||||
} else {
|
} else {
|
||||||
buf, err := blobstore.Get(client.user.CommentBlob)
|
buf, err := blobStore.Get(client.user.CommentBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err.Error())
|
server.Panicf("Blobstore error: %v", err.Error())
|
||||||
}
|
}
|
||||||
|
|
@ -796,7 +795,7 @@ func (server *Server) sendUserList(client *Client) {
|
||||||
if client.Version >= 0x10203 {
|
if client.Version >= 0x10203 {
|
||||||
userstate.TextureHash = connectedClient.user.TextureBlobHashBytes()
|
userstate.TextureHash = connectedClient.user.TextureBlobHashBytes()
|
||||||
} else {
|
} else {
|
||||||
buf, err := blobstore.Get(connectedClient.user.TextureBlob)
|
buf, err := blobStore.Get(connectedClient.user.TextureBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err.Error())
|
server.Panicf("Blobstore error: %v", err.Error())
|
||||||
}
|
}
|
||||||
|
|
@ -809,7 +808,7 @@ func (server *Server) sendUserList(client *Client) {
|
||||||
if client.Version >= 0x10203 {
|
if client.Version >= 0x10203 {
|
||||||
userstate.CommentHash = connectedClient.user.CommentBlobHashBytes()
|
userstate.CommentHash = connectedClient.user.CommentBlobHashBytes()
|
||||||
} else {
|
} else {
|
||||||
buf, err := blobstore.Get(connectedClient.user.CommentBlob)
|
buf, err := blobStore.Get(connectedClient.user.CommentBlob)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
server.Panicf("Blobstore error: %v", err.Error())
|
server.Panicf("Blobstore error: %v", err.Error())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue