efcdaa43a3
* lib/uplink: encryption context Change-Id: I5c23dca3286a46b713b30c4997e9ae6e630b2280 * lib/uplink: bucket operation examples Change-Id: Ia0f6e69f365dcff0cf11c731f51b30842bce053b * lib/uplink: encryption key sharing test cases Change-Id: I3a172d565f33f4e591402cdcb9460664a7cc7fbe * fix encrypted path prefix restriction issue Change-Id: I8f3921f9d52aaf4b84039de608b8cbbc88769554 * implement panics in libuplink encryption code todo on cipher suite selection as well as an api concern Change-Id: Ifa39eb3cc4b3443f7d96f9304df9b2ac4ec4085d * implement GetProjectInfo api call to get salt Change-Id: Ic5f6b3be9ea35df48c1aa214ab5d355fb328e2cf * some fixes and accessors for encryption store Change-Id: I3bb61f6712a037900e2a96e72ad4029ec1d3f718 * general fixes to builds/tests/etc Change-Id: I9930fa96acb3b221d9a001f8e274af5729cc8a47 * java bindings changes Change-Id: Ia2bd4c9c69739c8d3154d79616cff1f36fb403b6 * get libuplink examples passing Change-Id: I828f09a144160e0a5dd932324f78491ae2ec8a07 * fix proto.lock file Change-Id: I2fbbf4d0976a7d0473c2645e6dcb21aaa3be7651 * fix proto.lock again Change-Id: I92702cf49e1a340eef6379c2be4f7c4a268112a9 * fix golint issues Change-Id: I631ff9f43307a58e3b25a58cbb4a4cc2495f5eb6 * more linting fixes Change-Id: I51f8f30b367b5bca14c94b15417b9a4c9e7aa0ce * bug fixed by structs bump Change-Id: Ibb03c691fce7606c35c08721b3ef0781ab48a38a * retrigger Change-Id: Ieee0470b6a2d07168a1578552e8e7f271ae93a13 * retrigger Change-Id: I753d63853171e6a436c104ce176048892eb974c5 * semantic merge conflict Change-Id: I9419448496de90340569047a6a16a1b858a7978a * update total to match prod defaults Change-Id: I693d55c1ebb28b5803ee1d26e9e198decf82308b * retrigger Change-Id: I28b74d5d6202f61aa3866fe407d423f6a0a14b9e * retrigger Change-Id: I6fd054885c715f602e2cef623fd464c42e88742c * retrigger Change-Id: I6a01bae88c72406d4ed5a8f13bf8a2b3c650bd2d
247 lines
7.2 KiB
Go
247 lines
7.2 KiB
Go
// Copyright (C) 2019 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package metainfo
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/golang/protobuf/ptypes"
|
|
"github.com/golang/protobuf/ptypes/timestamp"
|
|
"github.com/skyrings/skyring-common/tools/uuid"
|
|
"github.com/zeebo/errs"
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
monkit "gopkg.in/spacemonkeygo/monkit.v2"
|
|
|
|
"storj.io/storj/pkg/auth/grpcauth"
|
|
"storj.io/storj/pkg/pb"
|
|
"storj.io/storj/pkg/storj"
|
|
"storj.io/storj/pkg/transport"
|
|
"storj.io/storj/storage"
|
|
)
|
|
|
|
var (
|
|
mon = monkit.Package()
|
|
|
|
// Error is the errs class of standard metainfo errors
|
|
Error = errs.Class("metainfo error")
|
|
)
|
|
|
|
// Client creates a grpcClient
|
|
type Client struct {
|
|
client pb.MetainfoClient
|
|
conn *grpc.ClientConn
|
|
}
|
|
|
|
// ListItem is a single item in a listing
|
|
type ListItem struct {
|
|
Path storj.Path
|
|
Pointer *pb.Pointer
|
|
IsPrefix bool
|
|
}
|
|
|
|
// New used as a public function
|
|
func New(client pb.MetainfoClient) *Client {
|
|
return &Client{
|
|
client: client,
|
|
}
|
|
}
|
|
|
|
// Dial dials to metainfo endpoint with the specified api key.
|
|
func Dial(ctx context.Context, tc transport.Client, address string, apiKey string) (*Client, error) {
|
|
apiKeyInjector := grpcauth.NewAPIKeyInjector(apiKey)
|
|
conn, err := tc.DialAddress(
|
|
ctx,
|
|
address,
|
|
grpc.WithUnaryInterceptor(apiKeyInjector),
|
|
)
|
|
if err != nil {
|
|
return nil, Error.Wrap(err)
|
|
}
|
|
|
|
return &Client{
|
|
client: pb.NewMetainfoClient(conn),
|
|
conn: conn,
|
|
}, nil
|
|
}
|
|
|
|
// Close closes the dialed connection.
|
|
func (client *Client) Close() error {
|
|
if client.conn != nil {
|
|
return Error.Wrap(client.conn.Close())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// CreateSegment requests the order limits for creating a new segment
|
|
func (client *Client) CreateSegment(ctx context.Context, bucket string, path storj.Path, segmentIndex int64, redundancy *pb.RedundancyScheme, maxEncryptedSegmentSize int64, expiration time.Time) (limits []*pb.AddressedOrderLimit, rootPieceID storj.PieceID, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
var exp *timestamp.Timestamp
|
|
if !expiration.IsZero() {
|
|
exp, err = ptypes.TimestampProto(expiration)
|
|
if err != nil {
|
|
return nil, rootPieceID, err
|
|
}
|
|
}
|
|
|
|
response, err := client.client.CreateSegment(ctx, &pb.SegmentWriteRequest{
|
|
Bucket: []byte(bucket),
|
|
Path: []byte(path),
|
|
Segment: segmentIndex,
|
|
Redundancy: redundancy,
|
|
MaxEncryptedSegmentSize: maxEncryptedSegmentSize,
|
|
Expiration: exp,
|
|
})
|
|
if err != nil {
|
|
return nil, rootPieceID, Error.Wrap(err)
|
|
}
|
|
|
|
return response.GetAddressedLimits(), response.RootPieceId, nil
|
|
}
|
|
|
|
// CommitSegment requests to store the pointer for the segment
|
|
func (client *Client) CommitSegment(ctx context.Context, bucket string, path storj.Path, segmentIndex int64, pointer *pb.Pointer, originalLimits []*pb.OrderLimit2) (savedPointer *pb.Pointer, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
response, err := client.client.CommitSegment(ctx, &pb.SegmentCommitRequest{
|
|
Bucket: []byte(bucket),
|
|
Path: []byte(path),
|
|
Segment: segmentIndex,
|
|
Pointer: pointer,
|
|
OriginalLimits: originalLimits,
|
|
})
|
|
if err != nil {
|
|
return nil, Error.Wrap(err)
|
|
}
|
|
|
|
return response.GetPointer(), nil
|
|
}
|
|
|
|
// SegmentInfo requests the pointer of a segment
|
|
func (client *Client) SegmentInfo(ctx context.Context, bucket string, path storj.Path, segmentIndex int64) (pointer *pb.Pointer, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
response, err := client.client.SegmentInfo(ctx, &pb.SegmentInfoRequest{
|
|
Bucket: []byte(bucket),
|
|
Path: []byte(path),
|
|
Segment: segmentIndex,
|
|
})
|
|
if err != nil {
|
|
if status.Code(err) == codes.NotFound {
|
|
return nil, storage.ErrKeyNotFound.Wrap(err)
|
|
}
|
|
return nil, Error.Wrap(err)
|
|
}
|
|
|
|
return response.GetPointer(), nil
|
|
}
|
|
|
|
// ReadSegment requests the order limits for reading a segment
|
|
func (client *Client) ReadSegment(ctx context.Context, bucket string, path storj.Path, segmentIndex int64) (pointer *pb.Pointer, limits []*pb.AddressedOrderLimit, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
response, err := client.client.DownloadSegment(ctx, &pb.SegmentDownloadRequest{
|
|
Bucket: []byte(bucket),
|
|
Path: []byte(path),
|
|
Segment: segmentIndex,
|
|
})
|
|
if err != nil {
|
|
if status.Code(err) == codes.NotFound {
|
|
return nil, nil, storage.ErrKeyNotFound.Wrap(err)
|
|
}
|
|
return nil, nil, Error.Wrap(err)
|
|
}
|
|
|
|
return response.GetPointer(), sortLimits(response.GetAddressedLimits(), response.GetPointer()), nil
|
|
}
|
|
|
|
// sortLimits sorts order limits and fill missing ones with nil values
|
|
func sortLimits(limits []*pb.AddressedOrderLimit, pointer *pb.Pointer) []*pb.AddressedOrderLimit {
|
|
sorted := make([]*pb.AddressedOrderLimit, pointer.GetRemote().GetRedundancy().GetTotal())
|
|
for _, piece := range pointer.GetRemote().GetRemotePieces() {
|
|
sorted[piece.GetPieceNum()] = getLimitByStorageNodeID(limits, piece.NodeId)
|
|
}
|
|
return sorted
|
|
}
|
|
|
|
func getLimitByStorageNodeID(limits []*pb.AddressedOrderLimit, storageNodeID storj.NodeID) *pb.AddressedOrderLimit {
|
|
for _, limit := range limits {
|
|
if limit.GetLimit().StorageNodeId == storageNodeID {
|
|
return limit
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// DeleteSegment requests the order limits for deleting a segment
|
|
func (client *Client) DeleteSegment(ctx context.Context, bucket string, path storj.Path, segmentIndex int64) (limits []*pb.AddressedOrderLimit, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
response, err := client.client.DeleteSegment(ctx, &pb.SegmentDeleteRequest{
|
|
Bucket: []byte(bucket),
|
|
Path: []byte(path),
|
|
Segment: segmentIndex,
|
|
})
|
|
if err != nil {
|
|
if status.Code(err) == codes.NotFound {
|
|
return nil, storage.ErrKeyNotFound.Wrap(err)
|
|
}
|
|
return nil, Error.Wrap(err)
|
|
}
|
|
|
|
return response.GetAddressedLimits(), nil
|
|
}
|
|
|
|
// ListSegments lists the available segments
|
|
func (client *Client) ListSegments(ctx context.Context, bucket string, prefix, startAfter, endBefore storj.Path, recursive bool, limit int32, metaFlags uint32) (items []ListItem, more bool, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
response, err := client.client.ListSegments(ctx, &pb.ListSegmentsRequest{
|
|
Bucket: []byte(bucket),
|
|
Prefix: []byte(prefix),
|
|
StartAfter: []byte(startAfter),
|
|
EndBefore: []byte(endBefore),
|
|
Recursive: recursive,
|
|
Limit: limit,
|
|
MetaFlags: metaFlags,
|
|
})
|
|
if err != nil {
|
|
return nil, false, Error.Wrap(err)
|
|
}
|
|
|
|
list := response.GetItems()
|
|
items = make([]ListItem, len(list))
|
|
for i, item := range list {
|
|
items[i] = ListItem{
|
|
Path: storj.Path(item.GetPath()),
|
|
Pointer: item.GetPointer(),
|
|
IsPrefix: item.IsPrefix,
|
|
}
|
|
}
|
|
|
|
return items, response.GetMore(), nil
|
|
}
|
|
|
|
// SetAttribution tries to set the attribution information on the bucket.
|
|
func (client *Client) SetAttribution(ctx context.Context, bucket string, partnerID uuid.UUID) (err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
_, err = client.client.SetAttribution(ctx, &pb.SetAttributionRequest{
|
|
PartnerId: partnerID[:], // TODO: implement storj.UUID that can be sent using pb
|
|
BucketName: []byte(bucket),
|
|
})
|
|
|
|
return err
|
|
}
|
|
|
|
// GetProjectInfo gets the ProjectInfo for the api key associated with the metainfo client.
|
|
func (client *Client) GetProjectInfo(ctx context.Context) (resp *pb.ProjectInfoResponse, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
return client.client.ProjectInfo(ctx, &pb.ProjectInfoRequest{})
|
|
}
|