2022-03-18 02:12:25 +00:00
|
|
|
package arangodb
|
|
|
|
|
|
|
|
import (
|
2022-03-19 02:51:16 +00:00
|
|
|
"context"
|
2022-03-18 02:12:25 +00:00
|
|
|
"crypto/md5"
|
|
|
|
"encoding/binary"
|
|
|
|
"encoding/hex"
|
|
|
|
"io"
|
2022-03-19 02:51:16 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/arangodb/go-driver"
|
2022-07-29 07:17:28 +00:00
|
|
|
"github.com/seaweedfs/seaweedfs/weed/util"
|
2022-03-18 02:12:25 +00:00
|
|
|
)
|
|
|
|
|
2022-09-09 16:43:42 +00:00
|
|
|
// convert a string into arango-key safe hex bytes hash
|
2022-03-18 02:12:25 +00:00
|
|
|
func hashString(dir string) string {
|
|
|
|
h := md5.New()
|
|
|
|
io.WriteString(h, dir)
|
|
|
|
b := h.Sum(nil)
|
|
|
|
return hex.EncodeToString(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// convert slice of bytes into slice of uint64
|
|
|
|
// the first uint64 indicates the length in bytes
|
|
|
|
func bytesToArray(bs []byte) []uint64 {
|
|
|
|
out := make([]uint64, 0, 2+len(bs)/8)
|
|
|
|
out = append(out, uint64(len(bs)))
|
|
|
|
for len(bs)%8 != 0 {
|
|
|
|
bs = append(bs, 0)
|
|
|
|
}
|
|
|
|
for i := 0; i < len(bs); i = i + 8 {
|
|
|
|
out = append(out, binary.BigEndian.Uint64(bs[i:]))
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
// convert from slice of uint64 back to bytes
|
|
|
|
// if input length is 0 or 1, will return nil
|
|
|
|
func arrayToBytes(xs []uint64) []byte {
|
|
|
|
if len(xs) < 2 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
first := xs[0]
|
|
|
|
out := make([]byte, len(xs)*8) // i think this can actually be len(xs)*8-8, but i dont think an extra 8 bytes hurts...
|
|
|
|
for i := 1; i < len(xs); i = i + 1 {
|
|
|
|
binary.BigEndian.PutUint64(out[((i-1)*8):], xs[i])
|
|
|
|
}
|
|
|
|
return out[:first]
|
|
|
|
}
|
2022-03-19 02:51:16 +00:00
|
|
|
|
2022-03-19 03:21:57 +00:00
|
|
|
// gets the collection the bucket points to from filepath
|
|
|
|
func (store *ArangodbStore) extractBucketCollection(ctx context.Context, fullpath util.FullPath) (c driver.Collection, err error) {
|
|
|
|
bucket, _ := extractBucket(fullpath)
|
|
|
|
if bucket == "" {
|
|
|
|
bucket = DEFAULT_COLLECTION
|
|
|
|
}
|
|
|
|
c, err = store.ensureBucket(ctx, bucket)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return c, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// called by extractBucketCollection
|
2022-03-19 02:51:16 +00:00
|
|
|
func extractBucket(fullpath util.FullPath) (string, string) {
|
|
|
|
if !strings.HasPrefix(string(fullpath), BUCKET_PREFIX+"/") {
|
|
|
|
return "", string(fullpath)
|
|
|
|
}
|
2022-03-19 03:21:57 +00:00
|
|
|
if strings.Count(string(fullpath), "/") < 3 {
|
2022-03-19 02:51:16 +00:00
|
|
|
return "", string(fullpath)
|
|
|
|
}
|
2022-03-19 03:21:57 +00:00
|
|
|
bucketAndObjectKey := string(fullpath)[len(BUCKET_PREFIX+"/"):]
|
2022-03-19 02:51:16 +00:00
|
|
|
t := strings.Index(bucketAndObjectKey, "/")
|
|
|
|
bucket := bucketAndObjectKey
|
|
|
|
shortPath := "/"
|
|
|
|
if t > 0 {
|
|
|
|
bucket = bucketAndObjectKey[:t]
|
|
|
|
shortPath = string(util.FullPath(bucketAndObjectKey[t:]))
|
|
|
|
}
|
|
|
|
return bucket, shortPath
|
|
|
|
}
|
|
|
|
|
|
|
|
// get bucket collection from cache. if not exist, creates the buckets collection and grab it
|
|
|
|
func (store *ArangodbStore) ensureBucket(ctx context.Context, bucket string) (bc driver.Collection, err error) {
|
|
|
|
var ok bool
|
|
|
|
store.mu.RLock()
|
|
|
|
bc, ok = store.buckets[bucket]
|
|
|
|
store.mu.RUnlock()
|
2022-09-05 03:32:10 +00:00
|
|
|
if ok && bc != nil {
|
2022-03-19 02:51:16 +00:00
|
|
|
return bc, nil
|
|
|
|
}
|
|
|
|
store.mu.Lock()
|
|
|
|
defer store.mu.Unlock()
|
|
|
|
store.buckets[bucket], err = store.ensureCollection(ctx, bucket)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return store.buckets[bucket], nil
|
|
|
|
}
|
|
|
|
|
2022-09-09 16:43:42 +00:00
|
|
|
// transform to an arango compliant name
|
|
|
|
func bucketToCollectionName(s string) string {
|
|
|
|
if len(s) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
// replace all "." with _
|
|
|
|
s = strings.ReplaceAll(s, ".", "_")
|
|
|
|
|
|
|
|
// if starts with number or '.' then add a special prefix
|
|
|
|
if (s[0] >= '0' && s[0] <= '9') || (s[0] == '.' || s[0] == '_' || s[0] == '-') {
|
|
|
|
s = "xN--" + s
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2022-03-19 02:51:16 +00:00
|
|
|
// creates collection if not exist, ensures indices if not exist
|
2022-09-09 16:43:42 +00:00
|
|
|
func (store *ArangodbStore) ensureCollection(ctx context.Context, bucket_name string) (c driver.Collection, err error) {
|
|
|
|
// convert the bucket to collection name
|
|
|
|
name := bucketToCollectionName(bucket_name)
|
|
|
|
|
2022-03-19 02:51:16 +00:00
|
|
|
ok, err := store.database.CollectionExists(ctx, name)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if ok {
|
|
|
|
c, err = store.database.Collection(ctx, name)
|
|
|
|
} else {
|
|
|
|
c, err = store.database.CreateCollection(ctx, name, &driver.CreateCollectionOptions{})
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// ensure indices
|
|
|
|
if _, _, err = c.EnsurePersistentIndex(ctx, []string{"directory", "name"},
|
|
|
|
&driver.EnsurePersistentIndexOptions{
|
|
|
|
Name: "directory_name_multi", Unique: true,
|
|
|
|
}); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if _, _, err = c.EnsurePersistentIndex(ctx, []string{"directory"},
|
|
|
|
&driver.EnsurePersistentIndexOptions{Name: "IDX_directory"}); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if _, _, err = c.EnsureTTLIndex(ctx, "ttl", 1,
|
|
|
|
&driver.EnsureTTLIndexOptions{Name: "IDX_TTL"}); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if _, _, err = c.EnsurePersistentIndex(ctx, []string{"name"}, &driver.EnsurePersistentIndexOptions{
|
|
|
|
Name: "IDX_name",
|
|
|
|
}); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
}
|