You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							143 lines
						
					
					
						
							4.0 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							143 lines
						
					
					
						
							4.0 KiB
						
					
					
				| package weed_server | |
| 
 | |
| import ( | |
| 	"context" | |
| 	"fmt" | |
| 	"strings" | |
| 	"time" | |
| 
 | |
| 	"github.com/seaweedfs/seaweedfs/weed/glog" | |
| 	"github.com/seaweedfs/seaweedfs/weed/stats" | |
| 
 | |
| 	"github.com/seaweedfs/raft" | |
| 
 | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/master_pb" | |
| 	"github.com/seaweedfs/seaweedfs/weed/security" | |
| 	"github.com/seaweedfs/seaweedfs/weed/storage/needle" | |
| 	"github.com/seaweedfs/seaweedfs/weed/storage/super_block" | |
| 	"github.com/seaweedfs/seaweedfs/weed/storage/types" | |
| 	"github.com/seaweedfs/seaweedfs/weed/topology" | |
| ) | |
| 
 | |
| func (ms *MasterServer) StreamAssign(server master_pb.Seaweed_StreamAssignServer) error { | |
| 	for { | |
| 		req, err := server.Recv() | |
| 		if err != nil { | |
| 			glog.Errorf("StreamAssign failed to receive: %v", err) | |
| 			return err | |
| 		} | |
| 		resp, err := ms.Assign(context.Background(), req) | |
| 		if err != nil { | |
| 			glog.Errorf("StreamAssign failed to assign: %v", err) | |
| 			return err | |
| 		} | |
| 		if err = server.Send(resp); err != nil { | |
| 			glog.Errorf("StreamAssign failed to send: %v", err) | |
| 			return err | |
| 		} | |
| 	} | |
| } | |
| func (ms *MasterServer) Assign(ctx context.Context, req *master_pb.AssignRequest) (*master_pb.AssignResponse, error) { | |
| 
 | |
| 	if !ms.Topo.IsLeader() { | |
| 		return nil, raft.NotLeaderError | |
| 	} | |
| 
 | |
| 	if req.Count == 0 { | |
| 		req.Count = 1 | |
| 	} | |
| 
 | |
| 	if req.Replication == "" { | |
| 		req.Replication = ms.option.DefaultReplicaPlacement | |
| 	} | |
| 	replicaPlacement, err := super_block.NewReplicaPlacementFromString(req.Replication) | |
| 	if err != nil { | |
| 		return nil, err | |
| 	} | |
| 	ttl, err := needle.ReadTTL(req.Ttl) | |
| 	if err != nil { | |
| 		return nil, err | |
| 	} | |
| 	diskType := types.ToDiskType(req.DiskType) | |
| 
 | |
| 	ver := needle.GetCurrentVersion() | |
| 	option := &topology.VolumeGrowOption{ | |
| 		Collection:         req.Collection, | |
| 		ReplicaPlacement:   replicaPlacement, | |
| 		Ttl:                ttl, | |
| 		DiskType:           diskType, | |
| 		Preallocate:        ms.preallocateSize, | |
| 		DataCenter:         req.DataCenter, | |
| 		Rack:               req.Rack, | |
| 		DataNode:           req.DataNode, | |
| 		MemoryMapMaxSizeMb: req.MemoryMapMaxSizeMb, | |
| 		Version:            uint32(ver), | |
| 	} | |
| 
 | |
| 	if !ms.Topo.DataCenterExists(option.DataCenter) { | |
| 		return nil, fmt.Errorf("data center %v not found in topology", option.DataCenter) | |
| 	} | |
| 
 | |
| 	vl := ms.Topo.GetVolumeLayout(option.Collection, option.ReplicaPlacement, option.Ttl, option.DiskType) | |
| 	vl.SetLastGrowCount(req.WritableVolumeCount) | |
| 
 | |
| 	var ( | |
| 		lastErr    error | |
| 		maxTimeout = time.Second * 10 | |
| 		startTime  = time.Now() | |
| 	) | |
| 
 | |
| 	for time.Now().Sub(startTime) < maxTimeout { | |
| 		fid, count, dnList, shouldGrow, err := ms.Topo.PickForWrite(req.Count, option, vl) | |
| 		if shouldGrow && !vl.HasGrowRequest() && !ms.option.VolumeGrowthDisabled { | |
| 			if err != nil && ms.Topo.AvailableSpaceFor(option) <= 0 { | |
| 				err = fmt.Errorf("%s and no free volumes left for %s", err.Error(), option.String()) | |
| 			} | |
| 			vl.AddGrowRequest() | |
| 			ms.volumeGrowthRequestChan <- &topology.VolumeGrowRequest{ | |
| 				Option: option, | |
| 				Count:  req.WritableVolumeCount, | |
| 				Reason: "grpc assign", | |
| 			} | |
| 		} | |
| 		if err != nil { | |
| 			glog.V(1).Infof("assign %v %v: %v", req, option.String(), err) | |
| 			stats.MasterPickForWriteErrorCounter.Inc() | |
| 			lastErr = err | |
| 			if (req.DataCenter != "" || req.Rack != "") && strings.Contains(err.Error(), topology.NoWritableVolumes) { | |
| 				break | |
| 			} | |
| 			time.Sleep(200 * time.Millisecond) | |
| 			continue | |
| 		} | |
| 		dn := dnList.Head() | |
| 		if dn == nil { | |
| 			continue | |
| 		} | |
| 		var replicas []*master_pb.Location | |
| 		for _, r := range dnList.Rest() { | |
| 			replicas = append(replicas, &master_pb.Location{ | |
| 				Url:        r.Url(), | |
| 				PublicUrl:  r.PublicUrl, | |
| 				GrpcPort:   uint32(r.GrpcPort), | |
| 				DataCenter: r.GetDataCenterId(), | |
| 			}) | |
| 		} | |
| 		return &master_pb.AssignResponse{ | |
| 			Fid: fid, | |
| 			Location: &master_pb.Location{ | |
| 				Url:        dn.Url(), | |
| 				PublicUrl:  dn.PublicUrl, | |
| 				GrpcPort:   uint32(dn.GrpcPort), | |
| 				DataCenter: dn.GetDataCenterId(), | |
| 			}, | |
| 			Count:    count, | |
| 			Auth:     string(security.GenJwtForVolumeServer(ms.guard.SigningKey, ms.guard.ExpiresAfterSec, fid)), | |
| 			Replicas: replicas, | |
| 		}, nil | |
| 	} | |
| 	if lastErr != nil { | |
| 		glog.V(0).Infof("assign %v %v: %v", req, option.String(), lastErr) | |
| 	} | |
| 	return nil, lastErr | |
| }
 |