2020-04-18 16:12:01 +08:00
|
|
|
package broker
|
2020-04-17 17:29:38 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2020-05-08 17:47:22 +08:00
|
|
|
"io"
|
2020-04-17 17:29:38 +08:00
|
|
|
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/glog"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/operation"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/messaging_pb"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/security"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (broker *MessageBroker) appendToFile(targetFile string, topicConfig *messaging_pb.TopicConfiguration, data []byte) error {
|
|
|
|
|
|
|
|
assignResult, uploadResult, err2 := broker.assignAndUpload(topicConfig, data)
|
|
|
|
if err2 != nil {
|
|
|
|
return err2
|
|
|
|
}
|
|
|
|
|
|
|
|
dir, name := util.FullPath(targetFile).DirAndName()
|
|
|
|
|
|
|
|
// append the chunk
|
|
|
|
if err := broker.WithFilerClient(func(client filer_pb.SeaweedFilerClient) error {
|
|
|
|
|
|
|
|
request := &filer_pb.AppendToEntryRequest{
|
|
|
|
Directory: dir,
|
|
|
|
EntryName: name,
|
2020-05-01 08:20:44 +08:00
|
|
|
Chunks: []*filer_pb.FileChunk{uploadResult.ToPbFileChunk(assignResult.Fid, 0)},
|
2020-04-17 17:29:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
_, err := client.AppendToEntry(context.Background(), request)
|
|
|
|
if err != nil {
|
|
|
|
glog.V(0).Infof("append to file %v: %v", request, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return fmt.Errorf("append to file %v: %v", targetFile, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (broker *MessageBroker) assignAndUpload(topicConfig *messaging_pb.TopicConfiguration, data []byte) (*operation.AssignResult, *operation.UploadResult, error) {
|
|
|
|
|
|
|
|
var assignResult = &operation.AssignResult{}
|
|
|
|
|
|
|
|
// assign a volume location
|
|
|
|
if err := broker.WithFilerClient(func(client filer_pb.SeaweedFilerClient) error {
|
|
|
|
|
|
|
|
request := &filer_pb.AssignVolumeRequest{
|
|
|
|
Count: 1,
|
|
|
|
Replication: topicConfig.Replication,
|
|
|
|
Collection: topicConfig.Collection,
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := client.AssignVolume(context.Background(), request)
|
|
|
|
if err != nil {
|
|
|
|
glog.V(0).Infof("assign volume failure %v: %v", request, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if resp.Error != "" {
|
|
|
|
return fmt.Errorf("assign volume failure %v: %v", request, resp.Error)
|
|
|
|
}
|
|
|
|
|
|
|
|
assignResult.Auth = security.EncodedJwt(resp.Auth)
|
|
|
|
assignResult.Fid = resp.FileId
|
|
|
|
assignResult.Url = resp.Url
|
|
|
|
assignResult.PublicUrl = resp.PublicUrl
|
|
|
|
assignResult.Count = uint64(resp.Count)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// upload data
|
|
|
|
targetUrl := fmt.Sprintf("http://%s/%s", assignResult.Url, assignResult.Fid)
|
|
|
|
uploadResult, err := operation.UploadData(targetUrl, "", broker.option.Cipher, data, false, "", nil, assignResult.Auth)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("upload data %s: %v", targetUrl, err)
|
|
|
|
}
|
|
|
|
// println("uploaded to", targetUrl)
|
|
|
|
return assignResult, uploadResult, nil
|
|
|
|
}
|
|
|
|
|
2020-04-30 04:26:02 +08:00
|
|
|
var _ = filer_pb.FilerClient(&MessageBroker{})
|
|
|
|
|
2020-04-17 17:29:38 +08:00
|
|
|
func (broker *MessageBroker) WithFilerClient(fn func(filer_pb.SeaweedFilerClient) error) (err error) {
|
|
|
|
|
|
|
|
for _, filer := range broker.option.Filers {
|
|
|
|
if err = pb.WithFilerClient(filer, broker.grpcDialOption, fn); err != nil {
|
2020-05-08 17:47:22 +08:00
|
|
|
if err == io.EOF {
|
|
|
|
return
|
|
|
|
}
|
2020-04-17 17:29:38 +08:00
|
|
|
glog.V(0).Infof("fail to connect to %s: %v", filer, err)
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
2021-01-29 06:36:29 +08:00
|
|
|
|
|
|
|
func (broker *MessageBroker) AdjustedUrl(location *filer_pb.Location) string {
|
|
|
|
return location.Url
|
|
|
|
}
|